* [PATCH] make verify-pack a bit more useful with bad packs
@ 2008-05-29 21:34 Nicolas Pitre
2008-06-02 6:27 ` Junio C Hamano
0 siblings, 1 reply; 2+ messages in thread
From: Nicolas Pitre @ 2008-05-29 21:34 UTC (permalink / raw)
To: Junio C Hamano; +Cc: git
When a pack gets corrupted, its SHA1 checksum will fail. However, this
is more useful to let the test go on in order to find the actual
problem location than only complain about the SHA1 mismatch and
bail out.
Also, it is more useful to compare the stored pack SHA1 with the one in
the index file instead of the computed SHA1 since the computed SHA1
from a corrupted pack won't match the one stored in the index either.
Finally a few code and message cleanups were thrown in as a bonus.
Signed-off-by: Nicolas Pitre <nico@cam.org>
---
Yes, I just had my first real usage Git repository corruption...
Still investigating.
diff --git a/pack-check.c b/pack-check.c
index 0f8ad2c..c03525e 100644
--- a/pack-check.c
+++ b/pack-check.c
@@ -25,10 +25,10 @@ static int verify_packfile(struct packed_git *p,
off_t index_size = p->index_size;
const unsigned char *index_base = p->index_data;
SHA_CTX ctx;
- unsigned char sha1[20];
- off_t offset = 0, pack_sig = p->pack_size - 20;
+ unsigned char sha1[20], *pack_sig;
+ off_t offset = 0, pack_sig_ofs = p->pack_size - 20;
uint32_t nr_objects, i;
- int err;
+ int err = 0;
struct idx_entry *entries;
/* Note that the pack header checks are actually performed by
@@ -38,21 +38,22 @@ static int verify_packfile(struct packed_git *p,
*/
SHA1_Init(&ctx);
- while (offset < pack_sig) {
+ while (offset < pack_sig_ofs) {
unsigned int remaining;
unsigned char *in = use_pack(p, w_curs, offset, &remaining);
offset += remaining;
- if (offset > pack_sig)
- remaining -= (unsigned int)(offset - pack_sig);
+ if (offset > pack_sig_ofs)
+ remaining -= (unsigned int)(offset - pack_sig_ofs);
SHA1_Update(&ctx, in, remaining);
}
SHA1_Final(sha1, &ctx);
- if (hashcmp(sha1, use_pack(p, w_curs, pack_sig, NULL)))
- return error("Packfile %s SHA1 mismatch with itself",
- p->pack_name);
- if (hashcmp(sha1, index_base + index_size - 40))
- return error("Packfile %s SHA1 mismatch with idx",
- p->pack_name);
+ pack_sig = use_pack(p, w_curs, pack_sig_ofs, NULL);
+ if (hashcmp(sha1, pack_sig))
+ err = error("%s SHA1 checksum mismatch",
+ p->pack_name);
+ if (hashcmp(index_base + index_size - 40, pack_sig))
+ err = error("%s SHA1 does not match its inddex",
+ p->pack_name);
unuse_pack(w_curs);
/* Make sure everything reachable from idx is valid. Since we
@@ -72,22 +73,23 @@ static int verify_packfile(struct packed_git *p,
}
qsort(entries, nr_objects, sizeof(*entries), compare_entries);
- for (i = 0, err = 0; i < nr_objects; i++) {
+ for (i = 0; i < nr_objects; i++) {
void *data;
enum object_type type;
unsigned long size;
data = unpack_entry(p, entries[i].offset, &type, &size);
if (!data) {
- err = error("cannot unpack %s from %s",
- sha1_to_hex(entries[i].sha1), p->pack_name);
- continue;
+ err = error("cannot unpack %s from %s at offset %"PRIuMAX"",
+ sha1_to_hex(entries[i].sha1), p->pack_name,
+ (uintmax_t)entries[i].offset);
+ break;
}
if (check_sha1_signature(entries[i].sha1, data, size, typename(type))) {
err = error("packed %s from %s is corrupt",
sha1_to_hex(entries[i].sha1), p->pack_name);
free(data);
- continue;
+ break;
}
free(data);
}
@@ -158,31 +160,28 @@ int verify_pack(struct packed_git *p, int verbose)
const unsigned char *index_base;
SHA_CTX ctx;
unsigned char sha1[20];
- int ret;
+ int err = 0;
+ struct pack_window *w_curs = NULL;
if (open_pack_index(p))
return error("packfile %s index not opened", p->pack_name);
index_size = p->index_size;
index_base = p->index_data;
- ret = 0;
/* Verify SHA1 sum of the index file */
SHA1_Init(&ctx);
SHA1_Update(&ctx, index_base, (unsigned int)(index_size - 20));
SHA1_Final(sha1, &ctx);
if (hashcmp(sha1, index_base + index_size - 20))
- ret = error("Packfile index for %s SHA1 mismatch",
+ err = error("Packfile index for %s SHA1 mismatch",
p->pack_name);
- if (!ret) {
- /* Verify pack file */
- struct pack_window *w_curs = NULL;
- ret = verify_packfile(p, &w_curs);
- unuse_pack(&w_curs);
- }
+ /* Verify pack file */
+ err = verify_packfile(p, &w_curs);
+ unuse_pack(&w_curs);
if (verbose) {
- if (ret)
+ if (err)
printf("%s: bad\n", p->pack_name);
else {
show_pack_info(p);
@@ -190,5 +189,5 @@ int verify_pack(struct packed_git *p, int verbose)
}
}
- return ret;
+ return err;
}
^ permalink raw reply related [flat|nested] 2+ messages in thread
* Re: [PATCH] make verify-pack a bit more useful with bad packs
2008-05-29 21:34 [PATCH] make verify-pack a bit more useful with bad packs Nicolas Pitre
@ 2008-06-02 6:27 ` Junio C Hamano
0 siblings, 0 replies; 2+ messages in thread
From: Junio C Hamano @ 2008-06-02 6:27 UTC (permalink / raw)
To: Nicolas Pitre; +Cc: git
Nicolas Pitre <nico@cam.org> writes:
> When a pack gets corrupted, its SHA1 checksum will fail. However, this
> is more useful to let the test go on in order to find the actual
> problem location than only complain about the SHA1 mismatch and
> bail out.
> ...
> /* Verify SHA1 sum of the index file */
> SHA1_Init(&ctx);
> SHA1_Update(&ctx, index_base, (unsigned int)(index_size - 20));
> SHA1_Final(sha1, &ctx);
> if (hashcmp(sha1, index_base + index_size - 20))
> - ret = error("Packfile index for %s SHA1 mismatch",
> + err = error("Packfile index for %s SHA1 mismatch",
> p->pack_name);
>
> - if (!ret) {
> - /* Verify pack file */
> - struct pack_window *w_curs = NULL;
> - ret = verify_packfile(p, &w_curs);
> - unuse_pack(&w_curs);
> - }
> + /* Verify pack file */
> + err = verify_packfile(p, &w_curs);
This overwrites the index SHA-1 mismatch you detected earlier. An obvious
fix-up would be to:
err |= verify_packfile(p, &w_curs);
Will be in 1.5.6-rc1.
^ permalink raw reply [flat|nested] 2+ messages in thread
end of thread, other threads:[~2008-06-02 6:28 UTC | newest]
Thread overview: 2+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2008-05-29 21:34 [PATCH] make verify-pack a bit more useful with bad packs Nicolas Pitre
2008-06-02 6:27 ` Junio C Hamano
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).