103 lines
2.5 KiB
Rust
Raw Normal View History

2024-01-15 19:09:34 -08:00
//file format "sniff"
2024-01-18 19:28:01 -08:00
use binrw::{binrw, BinReaderExt, io::TakeSeekExt};
2024-01-18 16:59:00 -08:00
2024-01-15 19:09:34 -08:00
pub enum Error{
2024-01-18 16:59:00 -08:00
InvalidHeader(binrw::Error),
2024-01-15 19:30:26 -08:00
UnexpectedEOF,
InvalidBlockId(BlockId),
2024-01-18 19:28:01 -08:00
Seek(std::io::Error),
2024-01-15 19:09:34 -08:00
}
/* spec
//begin global header
//global metadata (32 bytes)
b"SNFB"
u32 format_version
u64 priming_bytes
//how many bytes of the file must be read to guarantee all of the expected
//format-specific metadata is available to facilitate streaming the remaining contents
//used by the database to guarantee that it serves at least the bare minimum
u128 resource_uuid
//identifies the file from anywhere for any other file
//global block layout (variable size)
u64 num_blocks
//the start of the first block is implicitly after the global header (32)
//num_blocks+1 used in Header.block_location is implicitly the end of the file
for block_id in 1..num_blocks{
2024-01-15 19:09:34 -08:00
u64 first_byte
}
//end global header
//begin blocks
//each block is compressed with zstd or gz or something
*/
2024-01-18 16:59:00 -08:00
#[binrw]
#[brw(little)]
2024-01-18 16:39:58 -08:00
#[derive(Clone,Copy)]
pub(crate) enum FourCC{
2024-01-18 16:59:00 -08:00
#[brw(magic=b"SNFM")]
2024-01-18 16:39:58 -08:00
Map,
2024-01-18 16:59:00 -08:00
#[brw(magic=b"SNFB")]
2024-01-18 16:39:58 -08:00
Bot,
2024-01-18 16:59:00 -08:00
#[brw(magic=b"SNFD")]
2024-01-18 16:39:58 -08:00
Demo,
2024-01-15 19:09:34 -08:00
}
2024-01-18 16:59:00 -08:00
#[binrw]
#[brw(little)]
2024-01-18 16:39:58 -08:00
struct Header{
2024-01-15 19:09:34 -08:00
/// Type of file
2024-01-18 16:39:58 -08:00
fourcc:FourCC,
2024-01-15 19:09:34 -08:00
/// Type version
version:u32,
/// Minimum data required to know the location of all streamable resources for this specific file
priming:u64,
/// uuid for this file
resource:u128,
2024-07-23 17:18:10 -07:00
//don't need try_calc: the struct will force field initialization anyways and it can be calculated there
block_count:u32,
2024-01-18 19:28:01 -08:00
#[br(count=block_count+1)]
2024-01-18 16:59:00 -08:00
block_location:Vec<u64>,
2024-01-15 19:09:34 -08:00
}
#[binrw]
#[brw(little)]
#[derive(Clone,Copy,Hash,id::Id,Eq,PartialEq)]
pub struct BlockId(u32);
2024-01-16 20:07:11 -08:00
2024-01-18 19:28:01 -08:00
pub(crate) struct File<R:BinReaderExt>{
2024-01-18 16:39:58 -08:00
header:Header,
//reference to the data
2024-01-18 19:28:01 -08:00
data:R,
2024-01-18 16:39:58 -08:00
}
2024-01-18 19:28:01 -08:00
impl<R:BinReaderExt> File<R>{
pub(crate) fn new(mut input:R)->Result<File<R>,Error>{
Ok(File{
header:input.read_le().map_err(Error::InvalidHeader)?,
2024-01-18 19:28:01 -08:00
data:input,
2024-01-18 16:59:00 -08:00
})
2024-01-18 16:39:58 -08:00
}
2024-07-25 10:48:04 -07:00
pub(crate) fn as_mut(&mut self)->&mut R{
&mut self.data
}
2024-07-23 17:18:03 -07:00
pub(crate) fn block_reader(&mut self,block_id:BlockId)->Result<binrw::io::TakeSeek<&mut R>,Error>{
if self.header.block_location.len() as u32<=block_id.get(){
return Err(Error::InvalidBlockId(block_id))
2024-01-18 19:28:01 -08:00
}
let block_start=self.header.block_location[block_id.get() as usize];
let block_end=self.header.block_location[block_id.get() as usize+1];
self.data.seek(std::io::SeekFrom::Start(block_start)).map_err(Error::Seek)?;
2024-07-25 10:48:04 -07:00
Ok(self.as_mut().take_seek(block_end-block_start))
2024-01-18 16:39:58 -08:00
}
pub(crate) fn fourcc(&self)->FourCC{
self.header.fourcc
}
2024-01-16 20:07:11 -08:00
}