1use std::io::{self, Read, Write};
41
42pub const FOOTER_SIZE: usize = 76;
43pub const MAGIC: [u8; 8] = *b"ONELF\x00\x01\x00";
44pub const END_MAGIC: [u8; 8] = *b"FLENONE\x00";
45
46bitflags! {
47 #[derive(Debug, Clone, Copy, PartialEq, Eq)]
48 pub struct Flags: u16 {
49 const HAS_DICT = 1 << 0;
50 const MEMFD_HINT = 1 << 1;
51 const SHARUN_COMPAT = 1 << 2;
52 }
53}
54
55#[derive(Debug, Clone)]
56pub struct Footer {
57 pub format_version: u16,
59 pub flags: Flags,
61 pub manifest_offset: u64,
63 pub manifest_compressed: u64,
65 pub manifest_original: u64,
67 pub payload_offset: u64,
69 pub payload_size: u64,
71 pub dict_offset: u64,
73 pub dict_size: u32,
75 pub manifest_checksum: [u8; 4],
77}
78
79impl Footer {
80 pub fn write_to<W: Write>(&self, w: &mut W) -> io::Result<()> {
81 w.write_all(&MAGIC)?; w.write_all(&self.format_version.to_le_bytes())?; w.write_all(&self.flags.bits().to_le_bytes())?; w.write_all(&self.manifest_offset.to_le_bytes())?; w.write_all(&self.manifest_compressed.to_le_bytes())?; w.write_all(&self.manifest_original.to_le_bytes())?; w.write_all(&self.payload_offset.to_le_bytes())?; w.write_all(&self.payload_size.to_le_bytes())?; w.write_all(&self.dict_offset.to_le_bytes())?; w.write_all(&self.dict_size.to_le_bytes())?; w.write_all(&self.manifest_checksum)?; w.write_all(&END_MAGIC)?; Ok(()) }
95
96 pub fn read_from<R: Read>(r: &mut R) -> io::Result<Self> {
97 let mut buf = [0u8; FOOTER_SIZE];
98 r.read_exact(&mut buf)?;
99 Self::from_bytes(&buf)
100 }
101
102 pub fn from_bytes(buf: &[u8; FOOTER_SIZE]) -> io::Result<Self> {
103 if &buf[0..8] != &MAGIC {
104 return Err(io::Error::new(
105 io::ErrorKind::InvalidData,
106 "invalid onelf magic",
107 ));
108 }
109 if &buf[68..76] != &END_MAGIC {
110 return Err(io::Error::new(
111 io::ErrorKind::InvalidData,
112 "invalid onelf end magic",
113 ));
114 }
115
116 let format_version = u16::from_le_bytes(buf[8..10].try_into().unwrap());
117 if format_version != 1 {
118 return Err(io::Error::new(
119 io::ErrorKind::InvalidData,
120 format!("unsupported format version: {}", format_version),
121 ));
122 }
123
124 let flags_raw = u16::from_le_bytes(buf[10..12].try_into().unwrap());
125 let flags = Flags::from_bits_truncate(flags_raw);
126
127 Ok(Footer {
128 format_version,
129 flags,
130 manifest_offset: u64::from_le_bytes(buf[12..20].try_into().unwrap()),
131 manifest_compressed: u64::from_le_bytes(buf[20..28].try_into().unwrap()),
132 manifest_original: u64::from_le_bytes(buf[28..36].try_into().unwrap()),
133 payload_offset: u64::from_le_bytes(buf[36..44].try_into().unwrap()),
134 payload_size: u64::from_le_bytes(buf[44..52].try_into().unwrap()),
135 dict_offset: u64::from_le_bytes(buf[52..60].try_into().unwrap()),
136 dict_size: u32::from_le_bytes(buf[60..64].try_into().unwrap()),
137 manifest_checksum: buf[64..68].try_into().unwrap(),
138 })
139 }
140}