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
use crate::randfile::RandFile;
use std::io::{Read, Result, Seek};

#[cfg(all(feature = "mapped_io", not(target_arch = "wasm32")))]
use std::fs::File;

pub struct Blob<T> {
    file: RandFile<T>,
    size: usize,
    offset: u64,
}

pub struct BlobReader<'a, T> {
    cursor: u64,
    blob: &'a mut Blob<T>,
}

impl<'a, T: Read + Seek> Read for BlobReader<'a, T> {
    fn read(&mut self, buf: &mut [u8]) -> Result<usize> {
        let bytes_read = self.blob.read_block(self.cursor, buf)?;
        self.cursor += bytes_read as u64;
        Ok(bytes_read)
    }
}

impl<T: Read + Seek> Blob<T> {
    pub fn get_reader(&mut self) -> BlobReader<'_, T> {
        BlobReader {
            cursor: 0,
            blob: self,
        }
    }
    pub fn size(&self) -> usize {
        self.size
    }
    pub fn get_view(&self, offset: u64, size: usize) -> Self {
        let rel_offset = offset.min(self.size as u64);
        let abs_offset = self.offset + rel_offset;
        let size = (self.size - rel_offset as usize).min(size);

        Self {
            file: self.file.clone(),
            size,
            offset: abs_offset,
        }
    }
    pub fn read_block(&mut self, offset: u64, buf: &mut [u8]) -> Result<usize> {
        if self.size < offset as usize {
            return Ok(0);
        }

        let bytes_to_read = buf.len().min(self.size - offset as usize);

        self.file.read_block(self.offset + offset, buf)?;

        Ok(bytes_to_read)
    }
}

impl<T> Blob<T> {
    pub(crate) fn new(file: RandFile<T>, offset: u64, size: usize) -> Self {
        Self { file, size, offset }
    }
}

#[cfg(all(feature = "mapped_io", not(target_arch = "wasm32")))]
impl Blob<File> {
    pub fn mmap(&self) -> Result<impl AsRef<[u8]>> {
        self.file.mmap(self.offset, self.size)
    }
    pub fn mmap_mut(&mut self) -> Result<impl AsMut<[u8]>> {
        self.file.mmap_mut(self.offset, self.size)
    }
}