unpack_sha1_file(): detect corrupt loose object files.
We did not detect broken loose object files, either when underlying inflate() signalled the breakage, nor inflate() finished and we had garbage trailing at the end. We do better now. We also make unpack_sha1_file() a static function to sha1_file.c, since it is not used by anybody outside. Signed-off-by: Junio C Hamano <junkio@cox.net>
This commit is contained in:
parent
efec43c028
commit
7efbff7531
1
cache.h
1
cache.h
@ -281,7 +281,6 @@ char *enter_repo(char *path, int strict);
|
|||||||
|
|
||||||
/* Read and unpack a sha1 file into memory, write memory to a sha1 file */
|
/* Read and unpack a sha1 file into memory, write memory to a sha1 file */
|
||||||
extern int sha1_object_info(const unsigned char *, unsigned long *);
|
extern int sha1_object_info(const unsigned char *, unsigned long *);
|
||||||
extern void * unpack_sha1_file(void *map, unsigned long mapsize, enum object_type *type, unsigned long *size);
|
|
||||||
extern void * read_sha1_file(const unsigned char *sha1, enum object_type *type, unsigned long *size);
|
extern void * read_sha1_file(const unsigned char *sha1, enum object_type *type, unsigned long *size);
|
||||||
extern int hash_sha1_file(void *buf, unsigned long len, const char *type, unsigned char *sha1);
|
extern int hash_sha1_file(void *buf, unsigned long len, const char *type, unsigned char *sha1);
|
||||||
extern int write_sha1_file(void *buf, unsigned long len, const char *type, unsigned char *return_sha1);
|
extern int write_sha1_file(void *buf, unsigned long len, const char *type, unsigned char *return_sha1);
|
||||||
|
23
sha1_file.c
23
sha1_file.c
@ -956,11 +956,12 @@ static int unpack_sha1_header(z_stream *stream, unsigned char *map, unsigned lon
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *unpack_sha1_rest(z_stream *stream, void *buffer, unsigned long size)
|
static void *unpack_sha1_rest(z_stream *stream, void *buffer, unsigned long size, const unsigned char *sha1)
|
||||||
{
|
{
|
||||||
int bytes = strlen(buffer) + 1;
|
int bytes = strlen(buffer) + 1;
|
||||||
unsigned char *buf = xmalloc(1+size);
|
unsigned char *buf = xmalloc(1+size);
|
||||||
unsigned long n;
|
unsigned long n;
|
||||||
|
int status = Z_OK;
|
||||||
|
|
||||||
n = stream->total_out - bytes;
|
n = stream->total_out - bytes;
|
||||||
if (n > size)
|
if (n > size)
|
||||||
@ -970,12 +971,22 @@ static void *unpack_sha1_rest(z_stream *stream, void *buffer, unsigned long size
|
|||||||
if (bytes < size) {
|
if (bytes < size) {
|
||||||
stream->next_out = buf + bytes;
|
stream->next_out = buf + bytes;
|
||||||
stream->avail_out = size - bytes;
|
stream->avail_out = size - bytes;
|
||||||
while (inflate(stream, Z_FINISH) == Z_OK)
|
while (status == Z_OK)
|
||||||
/* nothing */;
|
status = inflate(stream, Z_FINISH);
|
||||||
}
|
}
|
||||||
buf[size] = 0;
|
buf[size] = 0;
|
||||||
|
if ((status == Z_OK || status == Z_STREAM_END) && !stream->avail_in) {
|
||||||
inflateEnd(stream);
|
inflateEnd(stream);
|
||||||
return buf;
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status < 0)
|
||||||
|
error("corrupt loose object '%s'", sha1_to_hex(sha1));
|
||||||
|
else if (stream->avail_in)
|
||||||
|
error("garbage at end of loose object '%s'",
|
||||||
|
sha1_to_hex(sha1));
|
||||||
|
free(buf);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1029,7 +1040,7 @@ static int parse_sha1_header(const char *hdr, unsigned long *sizep)
|
|||||||
return *hdr ? -1 : type_from_string(type);
|
return *hdr ? -1 : type_from_string(type);
|
||||||
}
|
}
|
||||||
|
|
||||||
void * unpack_sha1_file(void *map, unsigned long mapsize, enum object_type *type, unsigned long *size)
|
static void *unpack_sha1_file(void *map, unsigned long mapsize, enum object_type *type, unsigned long *size, const unsigned char *sha1)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
z_stream stream;
|
z_stream stream;
|
||||||
@ -1039,7 +1050,7 @@ void * unpack_sha1_file(void *map, unsigned long mapsize, enum object_type *type
|
|||||||
if (ret < Z_OK || (*type = parse_sha1_header(hdr, size)) < 0)
|
if (ret < Z_OK || (*type = parse_sha1_header(hdr, size)) < 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
return unpack_sha1_rest(&stream, hdr, *size);
|
return unpack_sha1_rest(&stream, hdr, *size, sha1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned long get_delta_base(struct packed_git *p,
|
static unsigned long get_delta_base(struct packed_git *p,
|
||||||
@ -1555,7 +1566,7 @@ void *read_sha1_file(const unsigned char *sha1, enum object_type *type,
|
|||||||
return buf;
|
return buf;
|
||||||
map = map_sha1_file(sha1, &mapsize);
|
map = map_sha1_file(sha1, &mapsize);
|
||||||
if (map) {
|
if (map) {
|
||||||
buf = unpack_sha1_file(map, mapsize, type, size);
|
buf = unpack_sha1_file(map, mapsize, type, size, sha1);
|
||||||
munmap(map, mapsize);
|
munmap(map, mapsize);
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user