7 static const char *base_name
;
8 static const char unpack_usage
[] = "git-unpack-objects basename";
11 unsigned int offset
; /* network byte order */
12 unsigned char sha1
[20];
15 static void *pack_base
;
16 static unsigned long pack_size
;
17 static void *index_base
;
18 static unsigned long index_size
;
20 static struct pack_entry
**pack_list
;
22 static void *map_file(const char *suffix
, unsigned long *sizep
)
24 static char pathname
[PATH_MAX
];
30 len
= snprintf(pathname
, PATH_MAX
, "%s.%s", base_name
, suffix
);
32 die("bad pack base-name");
33 fd
= open(pathname
, O_RDONLY
);
34 if (fd
< 0 || fstat(fd
, &st
))
35 die("unable to open '%s'", pathname
);
38 die("bad pack file '%s'", pathname
);
39 map
= mmap(NULL
, len
, PROT_READ
, MAP_PRIVATE
, fd
, 0);
40 if (-1 == (int)(long)map
)
41 die("unable to mmap '%s'", pathname
);
47 static int sort_by_offset(const void *_a
, const void *_b
)
49 struct pack_entry
*a
= *(struct pack_entry
**)_a
;
50 struct pack_entry
*b
= *(struct pack_entry
**)_b
;
53 o1
= ntohl(a
->offset
);
54 o2
= ntohl(b
->offset
);
55 return o1
< o2
? -1 : 1;
58 static int check_index(void)
60 unsigned int *array
= index_base
;
64 if (index_size
< 4*256 + 20)
65 return error("index file too small");
67 for (i
= 0; i
< 256; i
++) {
68 unsigned int n
= ntohl(array
[i
]);
70 return error("non-monotonic index");
75 * - 256 index entries 4 bytes each
76 * - 24-byte entries * nr (20-byte sha1 + 4-byte offset)
77 * - 20-byte SHA1 of the packfile
78 * - 20-byte SHA1 file checksum
80 if (index_size
!= 4*256 + nr
* 24 + 20 + 20)
81 return error("wrong index file size");
84 pack_list
= xmalloc(nr
* sizeof(struct pack_entry
*));
85 for (i
= 0; i
< nr
; i
++)
86 pack_list
[i
] = index_base
+ 4*256 + i
*24;
88 qsort(pack_list
, nr
, sizeof(*pack_list
), sort_by_offset
);
90 printf("%d entries\n", nr
);
94 static int unpack_non_delta_entry(struct pack_entry
*entry
,
103 unsigned char sha1
[20];
106 printf("%s %c %lu\n", sha1_to_hex(entry
->sha1
), kind
, size
);
110 buffer
= xmalloc(size
+ 1);
112 memset(&stream
, 0, sizeof(stream
));
113 stream
.next_in
= data
;
114 stream
.avail_in
= left
;
115 stream
.next_out
= buffer
;
116 stream
.avail_out
= size
;
118 inflateInit(&stream
);
119 st
= inflate(&stream
, Z_FINISH
);
121 if ((st
!= Z_STREAM_END
) || stream
.total_out
!= size
)
124 case 'C': type_s
= "commit"; break;
125 case 'T': type_s
= "tree"; break;
126 case 'B': type_s
= "blob"; break;
127 case 'G': type_s
= "tag"; break;
128 default: goto err_finish
;
130 if (write_sha1_file(buffer
, size
, type_s
, sha1
) < 0)
131 die("failed to write %s (%s)",
132 sha1_to_hex(entry
->sha1
), type_s
);
133 printf("%s %s\n", sha1_to_hex(sha1
), type_s
);
134 if (memcmp(sha1
, entry
->sha1
, 20))
135 die("resulting %s have wrong SHA1", type_s
);
146 static int find_pack_entry(unsigned char *sha1
, struct pack_entry
**ent
)
148 int *level1_ofs
= index_base
;
149 int hi
= ntohl(level1_ofs
[*sha1
]);
150 int lo
= ((*sha1
== 0x0) ? 0 : ntohl(level1_ofs
[*sha1
- 1]));
151 void *index
= index_base
+ 4*256;
154 int mi
= (lo
+ hi
) / 2;
155 int cmp
= memcmp(index
+ 24 * mi
+ 4, sha1
, 20);
157 *ent
= index
+ 24 * mi
;
168 /* forward declaration for a mutually recursive function */
169 static void unpack_entry(struct pack_entry
*);
171 static int unpack_delta_entry(struct pack_entry
*entry
,
172 unsigned char *base_sha1
,
173 unsigned long delta_size
,
176 void *data
, *delta_data
, *result
, *base
;
177 unsigned long data_size
, result_size
, base_size
;
181 unsigned char sha1
[20];
184 die("truncated pack file");
185 data
= base_sha1
+ 20;
186 data_size
= left
- 20;
187 printf("%s D %lu", sha1_to_hex(entry
->sha1
), delta_size
);
188 printf(" %s\n", sha1_to_hex(base_sha1
));
193 /* pack+5 is the base sha1, unless we have it, we need to
196 if (!has_sha1_file(base_sha1
)) {
197 struct pack_entry
*base
;
198 if (!find_pack_entry(base_sha1
, &base
))
199 die("cannot find delta-pack base object");
202 delta_data
= xmalloc(delta_size
);
204 memset(&stream
, 0, sizeof(stream
));
206 stream
.next_in
= data
;
207 stream
.avail_in
= data_size
;
208 stream
.next_out
= delta_data
;
209 stream
.avail_out
= delta_size
;
211 inflateInit(&stream
);
212 st
= inflate(&stream
, Z_FINISH
);
214 if ((st
!= Z_STREAM_END
) || stream
.total_out
!= delta_size
)
215 die("delta data unpack failed");
217 base
= read_sha1_file(base_sha1
, type
, &base_size
);
219 die("failed to read delta-pack base object %s", sha1_to_hex(base_sha1
));
220 result
= patch_delta(base
, base_size
,
221 delta_data
, delta_size
,
224 die("failed to apply delta");
227 if (write_sha1_file(result
, result_size
, type
, sha1
) < 0)
228 die("failed to write %s (%s)",
229 sha1_to_hex(entry
->sha1
), type
);
231 printf("%s %s\n", sha1_to_hex(sha1
), type
);
232 if (memcmp(sha1
, entry
->sha1
, 20))
233 die("resulting %s have wrong SHA1", type
);
237 static void unpack_entry(struct pack_entry
*entry
)
239 unsigned long offset
, size
, left
;
242 /* Have we done this one already due to deltas based on it? */
243 if (lookup_object(entry
->sha1
))
246 offset
= ntohl(entry
->offset
);
247 if (offset
> pack_size
- 5)
248 die("object offset outside of pack file");
249 pack
= pack_base
+ offset
;
250 size
= (pack
[1] << 24) + (pack
[2] << 16) + (pack
[3] << 8) + pack
[4];
251 left
= pack_size
- offset
- 5;
253 case 'C': case 'T': case 'B': case 'G':
254 unpack_non_delta_entry(entry
, *pack
, pack
+5, size
, left
);
257 unpack_delta_entry(entry
, pack
+5, size
, left
);
260 die("corrupted pack file");
265 * We unpack from the end, older files first. Now, usually
266 * there are deltas etc, so we'll not actually write the
267 * objects in that order, but we might as well try..
269 static void unpack_all(void)
274 struct pack_entry
*entry
= pack_list
[i
];
279 int main(int argc
, char **argv
)
283 for (i
= 1 ; i
< argc
; i
++) {
284 const char *arg
= argv
[i
];
287 if (!strcmp(arg
, "-n")) {
299 index_base
= map_file("idx", &index_size
);
300 pack_base
= map_file("pack", &pack_size
);
301 if (check_index() < 0)
302 die("bad index file");