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
use git_object as object;
use object::SHA1_SIZE;
const _TYPE_EXT1: u8 = 0;
const COMMIT: u8 = 1;
const TREE: u8 = 2;
const BLOB: u8 = 3;
const TAG: u8 = 4;
const _TYPE_EXT2: u8 = 5;
const OFS_DELTA: u8 = 6;
const REF_DELTA: u8 = 7;
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub struct Entry {
pub header: Header,
pub decompressed_size: u64,
pub data_offset: u64,
}
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub enum Header {
Commit,
Tree,
Blob,
Tag,
RefDelta {
oid: object::Id,
},
OfsDelta {
pack_offset: u64,
},
}
impl Header {
pub fn to_kind(&self) -> Option<object::Kind> {
use 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 is_delta(&self) -> bool {
match self {
Header::OfsDelta { .. } | Header::RefDelta { .. } => true,
_ => false,
}
}
}
#[inline]
fn leb64decode(d: &[u8]) -> (u64, usize) {
let mut i = 0;
let mut c = d[i];
i += 1;
let mut value = c as u64 & 0x7f;
while c & 0x80 != 0 {
c = d[i];
i += 1;
value += 1;
value = (value << 7) + (c as u64 & 0x7f)
}
(value, i)
}
fn parse_header_info(data: &[u8]) -> (u8, u64, usize) {
let mut c = data[0];
let mut i = 1;
let type_id = (c >> 4) & 0b0000_0111;
let mut size = c as u64 & 0b0000_1111;
let mut s = 4;
while c & 0b1000_0000 != 0 {
c = data[i];
i += 1;
size += ((c & 0b0111_1111) as u64) << s;
s += 7
}
(type_id, size, i)
}
impl Header {
pub fn from_bytes(d: &[u8], pack_offset: u64) -> (Header, u64, u64) {
let (type_id, size, mut consumed) = parse_header_info(d);
use self::Header::*;
let object = match type_id {
OFS_DELTA => {
let (offset, leb_bytes) = leb64decode(&d[consumed..]);
let delta = OfsDelta {
pack_offset: pack_offset - offset,
};
consumed += leb_bytes;
delta
}
REF_DELTA => {
let delta = RefDelta {
oid: object::Id::from_20_bytes(&d[consumed..consumed + SHA1_SIZE]),
};
consumed += SHA1_SIZE;
delta
}
BLOB => Blob,
TREE => Tree,
COMMIT => Commit,
TAG => Tag,
_ => panic!("We currently don't support any V3 features or extensions"),
};
(object, size, consumed as u64)
}
}