close another possibility for propagating pack corruption
Abstract -------- With index v2 we have a per object CRC to allow quick and safe reuse of pack data when repacking. This, however, doesn't currently prevent a stealth corruption from being propagated into a new pack when _not_ reusing pack data as demonstrated by the modification to t5302 included here. The Context ----------- The Git database is all checksummed with SHA1 hashes. Any kind of corruption can be confirmed by verifying this per object hash against corresponding data. However this can be costly to perform systematically and therefore this check is often not performed at run time when accessing the object database. First, the loose object format is entirely compressed with zlib which already provide a CRC verification of its own when inflating data. Any disk corruption would be caught already in this case. Then, packed objects are also compressed with zlib but only for their actual payload. The object headers and delta base references are not deflated for obvious performance reasons, however this leave them vulnerable to potentially undetected disk corruptions. Object types are often validated against the expected type when they're requested, and deflated size must always match the size recorded in the object header, so those cases are pretty much covered as well. Where corruptions could go unnoticed is in the delta base reference. Of course, in the OBJ_REF_DELTA case, the odds for a SHA1 reference to get corrupted so it actually matches the SHA1 of another object with the same size (the delta header stores the expected size of the base object to apply against) are virtually zero. In the OBJ_OFS_DELTA case, the reference is a pack offset which would have to match the start boundary of a different base object but still with the same size, and although this is relatively much more "probable" than in the OBJ_REF_DELTA case, the probability is also about zero in absolute terms. Still, the possibility exists as demonstrated in t5302 and is certainly greater than a SHA1 collision, especially in the OBJ_OFS_DELTA case which is now the default when repacking. Again, repacking by reusing existing pack data is OK since the per object CRC provided by index v2 guards against any such corruptions. What t5302 failed to test is a full repack in such case. The Solution ------------ As unlikely as this kind of stealth corruption can be in practice, it certainly isn't acceptable to propagate it into a freshly created pack. But, because this is so unlikely, we don't want to pay the run time cost associated with extra validation checks all the time either. Furthermore, consequences of such corruption in anything but repacking should be rather visible, and even if it could be quite unpleasant, it still has far less severe consequences than actively creating bad packs. So the best compromize is to check packed object CRC when unpacking objects, and only during the compression/writing phase of a repack, and only when not streaming the result. The cost of this is minimal (less than 1% CPU time), and visible only with a full repack. Someone with a stats background could provide an objective evaluation of this, but I suspect that it's bad RAM that has more potential for data corruptions at this point, even in those cases where this extra check is not performed. Still, it is best to prevent a known hole for corruption when recreating object data into a new pack. What about the streamed pack case? Well, any client receiving a pack must always consider that pack as untrusty and perform full validation anyway, hence no such stealth corruption could be propagated to remote repositoryes already. It is therefore worthless doing local validation in that case. Signed-off-by: Nicolas Pitre <nico@cam.org> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
6a509a6f7f
commit
0e8189e270
@ -1698,6 +1698,16 @@ static void prepare_pack(int window, int depth)
|
|||||||
|
|
||||||
get_object_details();
|
get_object_details();
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we're locally repacking then we need to be doubly careful
|
||||||
|
* from now on in order to make sure no stealth corruption gets
|
||||||
|
* propagated to the new pack. Clients receiving streamed packs
|
||||||
|
* should validate everything they get anyway so no need to incur
|
||||||
|
* the additional cost here in that case.
|
||||||
|
*/
|
||||||
|
if (!pack_to_stdout)
|
||||||
|
do_check_packed_object_crc = 1;
|
||||||
|
|
||||||
if (!nr_objects || !window || !depth)
|
if (!nr_objects || !window || !depth)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
3
cache.h
3
cache.h
@ -565,6 +565,9 @@ extern int force_object_loose(const unsigned char *sha1, time_t mtime);
|
|||||||
/* just like read_sha1_file(), but non fatal in presence of bad objects */
|
/* just like read_sha1_file(), but non fatal in presence of bad objects */
|
||||||
extern void *read_object(const unsigned char *sha1, enum object_type *type, unsigned long *size);
|
extern void *read_object(const unsigned char *sha1, enum object_type *type, unsigned long *size);
|
||||||
|
|
||||||
|
/* global flag to enable extra checks when accessing packed objects */
|
||||||
|
extern int do_check_packed_object_crc;
|
||||||
|
|
||||||
extern int check_sha1_signature(const unsigned char *sha1, void *buf, unsigned long size, const char *type);
|
extern int check_sha1_signature(const unsigned char *sha1, void *buf, unsigned long size, const char *type);
|
||||||
|
|
||||||
extern int move_temp_to_file(const char *tmpfile, const char *filename);
|
extern int move_temp_to_file(const char *tmpfile, const char *filename);
|
||||||
|
15
sha1_file.c
15
sha1_file.c
@ -1694,6 +1694,8 @@ static void *unpack_delta_entry(struct packed_git *p,
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int do_check_packed_object_crc;
|
||||||
|
|
||||||
void *unpack_entry(struct packed_git *p, off_t obj_offset,
|
void *unpack_entry(struct packed_git *p, off_t obj_offset,
|
||||||
enum object_type *type, unsigned long *sizep)
|
enum object_type *type, unsigned long *sizep)
|
||||||
{
|
{
|
||||||
@ -1701,6 +1703,19 @@ void *unpack_entry(struct packed_git *p, off_t obj_offset,
|
|||||||
off_t curpos = obj_offset;
|
off_t curpos = obj_offset;
|
||||||
void *data;
|
void *data;
|
||||||
|
|
||||||
|
if (do_check_packed_object_crc && p->index_version > 1) {
|
||||||
|
struct revindex_entry *revidx = find_pack_revindex(p, obj_offset);
|
||||||
|
unsigned long len = revidx[1].offset - obj_offset;
|
||||||
|
if (check_pack_crc(p, &w_curs, obj_offset, len, revidx->nr)) {
|
||||||
|
const unsigned char *sha1 =
|
||||||
|
nth_packed_object_sha1(p, revidx->nr);
|
||||||
|
error("bad packed object CRC for %s",
|
||||||
|
sha1_to_hex(sha1));
|
||||||
|
mark_bad_packed_object(p, sha1);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
*type = unpack_object_header(p, &w_curs, &curpos, sizep);
|
*type = unpack_object_header(p, &w_curs, &curpos, sizep);
|
||||||
switch (*type) {
|
switch (*type) {
|
||||||
case OBJ_OFS_DELTA:
|
case OBJ_OFS_DELTA:
|
||||||
|
@ -160,7 +160,8 @@ test_expect_success \
|
|||||||
|
|
||||||
test_expect_success \
|
test_expect_success \
|
||||||
'[index v2] 5) pack-objects refuses to reuse corrupted data' \
|
'[index v2] 5) pack-objects refuses to reuse corrupted data' \
|
||||||
'test_must_fail git pack-objects test-5 <obj-list'
|
'test_must_fail git pack-objects test-5 <obj-list &&
|
||||||
|
test_must_fail git pack-objects --no-reuse-object test-6 <obj-list'
|
||||||
|
|
||||||
test_expect_success \
|
test_expect_success \
|
||||||
'[index v2] 6) verify-pack detects CRC mismatch' \
|
'[index v2] 6) verify-pack detects CRC mismatch' \
|
||||||
|
Loading…
Reference in New Issue
Block a user