1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
use std::io;
use std::os::raw::{c_int, c_void};
use std::slice;
use libheif_sys as lh;
use crate::enums::ReaderGrowStatus;
pub trait Reader {
fn position(&mut self) -> u64;
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize>;
fn seek(&mut self, position: u64) -> io::Result<u64>;
fn wait_for_file_size(&mut self, target_size: u64) -> ReaderGrowStatus;
}
#[derive(Debug)]
pub struct StreamReader<T>
where
T: io::Read + io::Seek,
{
stream: T,
total_size: u64,
}
impl<T> StreamReader<T>
where
T: io::Read + io::Seek,
{
pub fn new(stream: T, total_size: u64) -> StreamReader<T> {
StreamReader { stream, total_size }
}
}
impl<T> Reader for StreamReader<T>
where
T: io::Read + io::Seek,
{
fn position(&mut self) -> u64 {
self.stream.stream_position().unwrap_or(self.total_size)
}
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
self.stream.read(buf)
}
fn seek(&mut self, position: u64) -> io::Result<u64> {
self.stream.seek(io::SeekFrom::Start(position as _))
}
fn wait_for_file_size(&mut self, target_size: u64) -> ReaderGrowStatus {
if self.stream.stream_position().is_err() {
ReaderGrowStatus::Timeout
} else if target_size > self.total_size {
ReaderGrowStatus::SizeBeyondEof
} else {
ReaderGrowStatus::SizeReached
}
}
}
unsafe extern "C" fn get_position(user_data: *mut c_void) -> i64 {
let reader = &mut *(user_data as *mut Box<dyn Reader>);
reader.position() as _
}
unsafe extern "C" fn read(data: *mut c_void, size: usize, user_data: *mut c_void) -> c_int {
let reader = &mut *(user_data as *mut Box<dyn Reader>);
let buf = slice::from_raw_parts_mut(data as *mut u8, size);
match reader.read(buf) {
Ok(real_size) if real_size == buf.len() => 0,
_ => 1,
}
}
unsafe extern "C" fn seek(position: i64, user_data: *mut c_void) -> c_int {
let reader = &mut *(user_data as *mut Box<dyn Reader>);
match reader.seek(position as _) {
Ok(_) => 0,
Err(_) => 1,
}
}
unsafe extern "C" fn wait_for_file_size(
target_size: i64,
user_data: *mut c_void,
) -> lh::heif_reader_grow_status {
let reader = &mut *(user_data as *mut Box<dyn Reader>);
let target_size = target_size as u64;
reader.wait_for_file_size(target_size) as _
}
pub(crate) static HEIF_READER: lh::heif_reader = lh::heif_reader {
reader_api_version: 1,
get_position: Some(get_position),
read: Some(read),
seek: Some(seek),
wait_for_file_size: Some(wait_for_file_size),
};