gix_pack/data/entry/
header.rs1use std::io;
2
3use super::{BLOB, COMMIT, OFS_DELTA, REF_DELTA, TAG, TREE};
4use crate::data;
5
6#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
8#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
9#[allow(missing_docs)]
10pub enum Header {
11 Commit,
13 Tree,
15 Blob,
17 Tag,
19 RefDelta { base_id: gix_hash::ObjectId },
30 OfsDelta { base_distance: u64 },
38}
39
40impl Header {
41 pub fn verified_base_pack_offset(pack_offset: data::Offset, distance: u64) -> Option<data::Offset> {
43 if distance == 0 {
44 return None;
45 }
46 pack_offset.checked_sub(distance)
47 }
48 pub fn as_kind(&self) -> Option<gix_object::Kind> {
50 use gix_object::Kind::*;
51 Some(match self {
52 Header::Tree => Tree,
53 Header::Blob => Blob,
54 Header::Commit => Commit,
55 Header::Tag => Tag,
56 Header::RefDelta { .. } | Header::OfsDelta { .. } => return None,
57 })
58 }
59 pub fn as_type_id(&self) -> u8 {
61 use Header::*;
62 match self {
63 Blob => BLOB,
64 Tree => TREE,
65 Commit => COMMIT,
66 Tag => TAG,
67 OfsDelta { .. } => OFS_DELTA,
68 RefDelta { .. } => REF_DELTA,
69 }
70 }
71 pub fn is_delta(&self) -> bool {
73 matches!(self, Header::OfsDelta { .. } | Header::RefDelta { .. })
74 }
75 pub fn is_base(&self) -> bool {
77 !self.is_delta()
78 }
79}
80
81impl Header {
82 pub fn write_to(&self, decompressed_size_in_bytes: u64, out: &mut dyn io::Write) -> io::Result<usize> {
87 let mut size = decompressed_size_in_bytes;
88 let mut written = 1;
89 let mut c: u8 = (self.as_type_id() << 4) | (size as u8 & 0b0000_1111);
90 size >>= 4;
91 while size != 0 {
92 out.write_all(&[c | 0b1000_0000])?;
93 written += 1;
94 c = size as u8 & 0b0111_1111;
95 size >>= 7;
96 }
97 out.write_all(&[c])?;
98
99 use Header::*;
100 match self {
101 RefDelta { base_id: oid } => {
102 out.write_all(oid.as_slice())?;
103 written += oid.as_slice().len();
104 }
105 OfsDelta { base_distance } => {
106 let mut buf = [0u8; 10];
107 let buf = leb64_encode(*base_distance, &mut buf);
108 out.write_all(buf)?;
109 written += buf.len();
110 }
111 Blob | Tree | Commit | Tag => {}
112 }
113 Ok(written)
114 }
115
116 pub fn size(&self, decompressed_size: u64) -> usize {
118 self.write_to(decompressed_size, &mut io::sink())
119 .expect("io::sink() to never fail")
120 }
121}
122
123#[inline]
124fn leb64_encode(mut n: u64, buf: &mut [u8; 10]) -> &[u8] {
125 let mut bytes_written = 1;
126 buf[buf.len() - 1] = n as u8 & 0b0111_1111;
127 for out in buf.iter_mut().rev().skip(1) {
128 n >>= 7;
129 if n == 0 {
130 break;
131 }
132 n -= 1;
133 *out = 0b1000_0000 | (n as u8 & 0b0111_1111);
134 bytes_written += 1;
135 }
136 debug_assert_eq!(n, 0, "BUG: buffer must be large enough to hold a 64 bit integer");
137 &buf[buf.len() - bytes_written..]
138}
139
140#[cfg(test)]
141mod tests {
142 use super::*;
143
144 #[test]
145 fn leb64_encode_max_int() {
146 let mut buf = [0u8; 10];
147 let buf = leb64_encode(u64::MAX, &mut buf);
148 assert_eq!(buf.len(), 10, "10 bytes should be used when 64bits are encoded");
149 }
150}