Skip to content

Commit 756f1bc

Browse files
derrickstoleegitster
authored andcommitted
fsck: verify checksums of all .bitmap files
If a filesystem-level corruption occurs in a .bitmap file, Git can react poorly. This could take the form of a run-time error due to failing to parse an EWAH bitmap or be more subtle such as returning the wrong set of objects to a fetch or clone. A natural first response to either of these kinds of errors is to run 'git fsck' to see if any files are corrupt. This currently ignores all .bitmap files. Add checks to 'git fsck' for all .bitmap files that are currently associated with a multi-pack-index or pack file. Verify their checksums using the hashfile API. We iterate through all multi-pack-indexes and pack-files to be sure to check all .bitmap files, not just the one that would be read by the process. For example, a multi-pack-index bitmap overrules a pack-bitmap. However, if the multi-pack-index is removed, the pack-bitmap may be selected instead. Be thorough to include every file that could become active in such a way. This includes checking files in alternates. There is potential that we could extend this effort to check the structure of the reachability bitmaps themselves, but it is very expensive to do so. At minimum, it's as expensive as generating the bitmaps in the first place, and that's assuming that we don't use the trivial algorithm of verifying each bitmap individually. The trivial algorithm will result in quadratic behavior (number of objects times number of bitmapped commits) while the bitmap building operation constructs a lattice of commits to build bitmaps incrementally and then generate the final bitmaps from a subset of those commits. If we were to extend 'git fsck' to check .bitmap file contents more closely like this, then we would likely want to hide it behind an option that signals the user is more willing to do expensive operations such as this. For testing, set up a repository with a pack-bitmap _and_ a multi-pack-index bitmap. This requires some file movement to avoid deleting the pack-bitmap during the repack that creates the multi-pack-index bitmap. We can then verify that 'git fsck' is checking all files, not just the "active" bitmap. Signed-off-by: Derrick Stolee <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 48d89b5 commit 756f1bc

File tree

4 files changed

+95
-0
lines changed

4 files changed

+95
-0
lines changed

builtin/fsck.c

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
#include "run-command.h"
2828
#include "worktree.h"
2929
#include "pack-revindex.h"
30+
#include "pack-bitmap.h"
3031

3132
#define REACHABLE 0x0001
3233
#define SEEN 0x0002
@@ -57,6 +58,7 @@ static int name_objects;
5758
#define ERROR_COMMIT_GRAPH 020
5859
#define ERROR_MULTI_PACK_INDEX 040
5960
#define ERROR_PACK_REV_INDEX 0100
61+
#define ERROR_BITMAP 0200
6062

6163
static const char *describe_object(const struct object_id *oid)
6264
{
@@ -1056,6 +1058,8 @@ int cmd_fsck(int argc, const char **argv, const char *prefix)
10561058
}
10571059

10581060
errors_found |= check_pack_rev_indexes(the_repository, show_progress);
1061+
if (verify_bitmap_files(the_repository))
1062+
errors_found |= ERROR_BITMAP;
10591063

10601064
check_connectivity();
10611065

pack-bitmap.c

Lines changed: 45 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2346,3 +2346,48 @@ int bitmap_is_preferred_refname(struct repository *r, const char *refname)
23462346

23472347
return 0;
23482348
}
2349+
2350+
static int verify_bitmap_file(const char *name)
2351+
{
2352+
struct stat st;
2353+
unsigned char *data;
2354+
int fd = git_open(name);
2355+
int res = 0;
2356+
2357+
/* It is OK to not have the file. */
2358+
if (fd < 0 || fstat(fd, &st)) {
2359+
if (fd >= 0)
2360+
close(fd);
2361+
return 0;
2362+
}
2363+
2364+
data = xmmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
2365+
close(fd);
2366+
if (!hashfile_checksum_valid(data, st.st_size))
2367+
res = error(_("bitmap file '%s' has invalid checksum"),
2368+
name);
2369+
2370+
munmap(data, st.st_size);
2371+
return res;
2372+
}
2373+
2374+
int verify_bitmap_files(struct repository *r)
2375+
{
2376+
int res = 0;
2377+
2378+
for (struct multi_pack_index *m = get_multi_pack_index(r);
2379+
m; m = m->next) {
2380+
char *midx_bitmap_name = midx_bitmap_filename(m);
2381+
res |= verify_bitmap_file(midx_bitmap_name);
2382+
free(midx_bitmap_name);
2383+
}
2384+
2385+
for (struct packed_git *p = get_all_packs(r);
2386+
p; p = p->next) {
2387+
char *pack_bitmap_name = pack_bitmap_filename(p);
2388+
res |= verify_bitmap_file(pack_bitmap_name);
2389+
free(pack_bitmap_name);
2390+
}
2391+
2392+
return res;
2393+
}

pack-bitmap.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -111,4 +111,6 @@ int bitmap_is_midx(struct bitmap_index *bitmap_git);
111111
const struct string_list *bitmap_preferred_tips(struct repository *r);
112112
int bitmap_is_preferred_refname(struct repository *r, const char *refname);
113113

114+
int verify_bitmap_files(struct repository *r);
115+
114116
#endif

t/t5326-multi-pack-bitmaps.sh

Lines changed: 44 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -434,4 +434,48 @@ test_expect_success 'tagged commits are selected for bitmapping' '
434434
)
435435
'
436436

437+
corrupt_file () {
438+
chmod a+w "$1" &&
439+
printf "bogus" | dd of="$1" bs=1 seek="12" conv=notrunc
440+
}
441+
442+
test_expect_success 'git fsck correctly identifies good and bad bitmaps' '
443+
git init valid &&
444+
test_when_finished rm -rf valid &&
445+
446+
test_commit_bulk 20 &&
447+
git repack -adbf &&
448+
449+
# Move pack-bitmap aside so it is not deleted
450+
# in next repack.
451+
packbitmap=$(ls .git/objects/pack/pack-*.bitmap) &&
452+
mv "$packbitmap" "$packbitmap.bak" &&
453+
454+
test_commit_bulk 10 &&
455+
git repack -b --write-midx &&
456+
midxbitmap=$(ls .git/objects/pack/multi-pack-index-*.bitmap) &&
457+
458+
# Copy MIDX bitmap to backup. Copy pack bitmap from backup.
459+
cp "$midxbitmap" "$midxbitmap.bak" &&
460+
cp "$packbitmap.bak" "$packbitmap" &&
461+
462+
# fsck works at first
463+
git fsck 2>err &&
464+
test_must_be_empty err &&
465+
466+
corrupt_file "$packbitmap" &&
467+
test_must_fail git fsck 2>err &&
468+
grep "bitmap file '\''$packbitmap'\'' has invalid checksum" err &&
469+
470+
cp "$packbitmap.bak" "$packbitmap" &&
471+
corrupt_file "$midxbitmap" &&
472+
test_must_fail git fsck 2>err &&
473+
grep "bitmap file '\''$midxbitmap'\'' has invalid checksum" err &&
474+
475+
corrupt_file "$packbitmap" &&
476+
test_must_fail git fsck 2>err &&
477+
grep "bitmap file '\''$midxbitmap'\'' has invalid checksum" err &&
478+
grep "bitmap file '\''$packbitmap'\'' has invalid checksum" err
479+
'
480+
437481
test_done

0 commit comments

Comments
 (0)