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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
use super::{BLOB, COMMIT, OFS_DELTA, REF_DELTA, TAG, TREE};
use std::io;
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
#[allow(missing_docs)]
pub enum Header {
Commit,
Tree,
Blob,
Tag,
RefDelta { base_id: git_hash::ObjectId },
OfsDelta { base_distance: u64 },
}
impl Header {
pub fn verified_base_pack_offset(pack_offset: u64, distance: u64) -> Option<u64> {
if distance == 0 {
return None;
}
pack_offset.checked_sub(distance)
}
pub fn as_kind(&self) -> Option<git_object::Kind> {
use git_object::Kind::*;
Some(match self {
Header::Tree => Tree,
Header::Blob => Blob,
Header::Commit => Commit,
Header::Tag => Tag,
Header::RefDelta { .. } | Header::OfsDelta { .. } => return None,
})
}
pub fn as_type_id(&self) -> u8 {
use Header::*;
match self {
Blob => BLOB,
Tree => TREE,
Commit => COMMIT,
Tag => TAG,
OfsDelta { .. } => OFS_DELTA,
RefDelta { .. } => REF_DELTA,
}
}
pub fn is_delta(&self) -> bool {
matches!(self, Header::OfsDelta { .. } | Header::RefDelta { .. })
}
pub fn is_base(&self) -> bool {
!self.is_delta()
}
}
impl Header {
pub fn write_to(&self, decompressed_size_in_bytes: u64, mut out: impl io::Write) -> io::Result<usize> {
let mut size = decompressed_size_in_bytes;
let mut written = 1;
let mut c: u8 = (self.as_type_id() << 4) | (size as u8 & 0b0000_1111);
size >>= 4;
while size != 0 {
out.write_all(&[c | 0b1000_0000])?;
written += 1;
c = size as u8 & 0b0111_1111;
size >>= 7;
}
out.write_all(&[c])?;
use Header::*;
match self {
RefDelta { base_id: oid } => {
out.write_all(oid.as_slice())?;
written += oid.as_slice().len();
}
OfsDelta { mut base_distance } => {
let mut buf = [0u8; 10];
let mut bytes_written = 1;
buf[buf.len() - 1] = base_distance as u8 & 0b0111_1111;
for out in buf.iter_mut().rev().skip(1) {
base_distance >>= 7;
if base_distance == 0 {
break;
}
base_distance -= 1;
*out = 0b1000_0000 | (base_distance as u8 & 0b0111_1111);
bytes_written += 1;
}
out.write_all(&buf[buf.len() - bytes_written..])?;
written += bytes_written;
}
Blob | Tree | Commit | Tag => {}
}
Ok(written)
}
pub fn size(&self, decompressed_size: u64) -> usize {
self.write_to(decompressed_size, io::sink())
.expect("io::sink() to never fail")
}
}