Skip to content

Commit f1299bf

Browse files
peffgitster
authored andcommitted
index-pack, unpack-objects: use get_be32() for reading pack header
Both of these commands read the incoming pack into a static unsigned char buffer in BSS, and then parse it by casting the start of the buffer to a struct pack_header. This can result in SIGBUS on some platforms if the compiler doesn't place the buffer in a position that is properly aligned for 4-byte integers. This reportedly happens with unpack-objects (but not index-pack) on sparc64 when compiled with clang (but not gcc). But we are definitely in the wrong in both spots; since the buffer's type is unsigned char, we can't depend on larger alignment. When it works it is only because we are lucky. We'll fix this by switching to get_be32() to read the headers (just like the last few commits similarly switched us to put_be32() for writing into the same buffer). It would be nice to factor this out into a common helper function, but the interface ends up quite awkward. Either the caller needs to hardcode how many bytes we'll need, or it needs to pass us its fill()/use() functions as pointers. So I've just fixed both spots in the same way; this is not code that is likely to be repeated a third time (most of the pack reading code uses an mmap'd buffer, which should be properly aligned). I did make one tweak to the shared code: our pack_version_ok() macro expects us to pass the big-endian value we'd get by casting. We can introduce a "native" variant which uses the host integer ordering. Reported-by: Koakuma <[email protected]> Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 4f02f4d commit f1299bf

File tree

3 files changed

+16
-12
lines changed

3 files changed

+16
-12
lines changed

builtin/index-pack.c

Lines changed: 7 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -363,16 +363,18 @@ static const char *open_pack_file(const char *pack_name)
363363

364364
static void parse_pack_header(void)
365365
{
366-
struct pack_header *hdr = fill(sizeof(struct pack_header));
366+
unsigned char *hdr = fill(sizeof(struct pack_header));
367367

368368
/* Header consistency check */
369-
if (hdr->hdr_signature != htonl(PACK_SIGNATURE))
369+
if (get_be32(hdr) != PACK_SIGNATURE)
370370
die(_("pack signature mismatch"));
371-
if (!pack_version_ok(hdr->hdr_version))
371+
hdr += 4;
372+
if (!pack_version_ok_native(get_be32(hdr)))
372373
die(_("pack version %"PRIu32" unsupported"),
373-
ntohl(hdr->hdr_version));
374+
get_be32(hdr));
375+
hdr += 4;
374376

375-
nr_objects = ntohl(hdr->hdr_entries);
377+
nr_objects = get_be32(hdr);
376378
use(sizeof(struct pack_header));
377379
}
378380

builtin/unpack-objects.c

Lines changed: 7 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -576,15 +576,16 @@ static void unpack_one(unsigned nr)
576576
static void unpack_all(void)
577577
{
578578
int i;
579-
struct pack_header *hdr = fill(sizeof(struct pack_header));
579+
unsigned char *hdr = fill(sizeof(struct pack_header));
580580

581-
nr_objects = ntohl(hdr->hdr_entries);
582-
583-
if (ntohl(hdr->hdr_signature) != PACK_SIGNATURE)
581+
if (get_be32(hdr) != PACK_SIGNATURE)
584582
die("bad pack file");
585-
if (!pack_version_ok(hdr->hdr_version))
583+
hdr += 4;
584+
if (!pack_version_ok_native(get_be32(hdr)))
586585
die("unknown pack file version %"PRIu32,
587-
ntohl(hdr->hdr_version));
586+
get_be32(hdr));
587+
hdr += 4;
588+
nr_objects = get_be32(hdr);
588589
use(sizeof(struct pack_header));
589590

590591
if (!quiet)

pack.h

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,8 @@ struct repository;
1313
*/
1414
#define PACK_SIGNATURE 0x5041434b /* "PACK" */
1515
#define PACK_VERSION 2
16-
#define pack_version_ok(v) ((v) == htonl(2) || (v) == htonl(3))
16+
#define pack_version_ok(v) pack_version_ok_native(ntohl(v))
17+
#define pack_version_ok_native(v) ((v) == 2 || (v) == 3)
1718
struct pack_header {
1819
uint32_t hdr_signature;
1920
uint32_t hdr_version;

0 commit comments

Comments
 (0)