Fix big left-shifts of unsigned char

Shifting 'unsigned char' or 'unsigned short' left can result in sign
extension errors, since the C integer promotion rules means that the
unsigned char/short will get implicitly promoted to a signed 'int' due to
the shift (or due to other operations).

This normally doesn't matter, but if you shift things up sufficiently, it
will now set the sign bit in 'int', and a subsequent cast to a bigger type
(eg 'long' or 'unsigned long') will now sign-extend the value despite the
original expression being unsigned.

One example of this would be something like

	unsigned long size;
	unsigned char c;

	size += c << 24;

where despite all the variables being unsigned, 'c << 24' ends up being a
signed entity, and will get sign-extended when then doing the addition in
an 'unsigned long' type.

Since git uses 'unsigned char' pointers extensively, we actually have this
bug in a couple of places.

I may have missed some, but this is the result of looking at

	git grep '[^0-9 	][ 	]*<<[ 	][a-z]' -- '*.c' '*.h'
	git grep '<<[   ]*24'

which catches at least the common byte cases (shifting variables by a
variable amount, and shifting by 24 bits).

I also grepped for just 'unsigned char' variables in general, and
converted the ones that most obviously ended up getting implicitly cast
immediately anyway (eg hash_name(), encode_85()).

In addition to just avoiding 'unsigned char', this patch also tries to use
a common idiom for the delta header size thing. We had three different
variations on it: "& 0x7fUL" in one place (getting the sign extension
right), and "& ~0x80" and "& 0x7f" in two other places (not getting it
right). Apart from making them all just avoid using "unsigned char" at
all, I also unified them to then use a simple "& 0x7f".

I considered making a sparse extension which warns about doing implicit
casts from unsigned types to signed types, but it gets rather complex very
quickly, so this is just a hack.

Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Linus Torvalds 2009-06-17 17:22:27 -07:00 коммит произвёл Junio C Hamano
Родитель 50a991ec46
Коммит 48fb7deb5b
8 изменённых файлов: 12 добавлений и 16 удалений

3
attr.c
Просмотреть файл

@ -35,8 +35,7 @@ static struct git_attr *(git_attr_hash[HASHSIZE]);
static unsigned hash_name(const char *name, int namelen) static unsigned hash_name(const char *name, int namelen)
{ {
unsigned val = 0; unsigned val = 0, c;
unsigned char c;
while (namelen--) { while (namelen--) {
c = *name++; c = *name++;

Просмотреть файл

@ -91,7 +91,7 @@ void encode_85(char *buf, const unsigned char *data, int bytes)
unsigned acc = 0; unsigned acc = 0;
int cnt; int cnt;
for (cnt = 24; cnt >= 0; cnt -= 8) { for (cnt = 24; cnt >= 0; cnt -= 8) {
int ch = *data++; unsigned ch = *data++;
acc |= ch << cnt; acc |= ch << cnt;
if (--bytes == 0) if (--bytes == 0)
break; break;

Просмотреть файл

@ -653,8 +653,7 @@ static void rehash_objects(void)
static unsigned name_hash(const char *name) static unsigned name_hash(const char *name)
{ {
unsigned char c; unsigned c, hash = 0;
unsigned hash = 0;
if (!name) if (!name)
return 0; return 0;

Просмотреть файл

@ -422,8 +422,8 @@ static void unpack_delta_entry(enum object_type type, unsigned long delta_size,
static void unpack_one(unsigned nr) static void unpack_one(unsigned nr)
{ {
unsigned shift; unsigned shift;
unsigned char *pack, c; unsigned char *pack;
unsigned long size; unsigned long size, c;
enum object_type type; enum object_type type;
obj_list[nr].offset = consumed_bytes; obj_list[nr].offset = consumed_bytes;

Просмотреть файл

@ -90,12 +90,11 @@ static inline unsigned long get_delta_hdr_size(const unsigned char **datap,
const unsigned char *top) const unsigned char *top)
{ {
const unsigned char *data = *datap; const unsigned char *data = *datap;
unsigned char cmd; unsigned long cmd, size = 0;
unsigned long size = 0;
int i = 0; int i = 0;
do { do {
cmd = *data++; cmd = *data++;
size |= (cmd & ~0x80) << i; size |= (cmd & 0x7f) << i;
i += 7; i += 7;
} while (cmd & 0x80 && data < top); } while (cmd & 0x80 && data < top);
*datap = data; *datap = data;

Просмотреть файл

@ -293,8 +293,8 @@ static void *unpack_entry_data(unsigned long offset, unsigned long size)
static void *unpack_raw_entry(struct object_entry *obj, union delta_base *delta_base) static void *unpack_raw_entry(struct object_entry *obj, union delta_base *delta_base)
{ {
unsigned char *p, c; unsigned char *p;
unsigned long size; unsigned long size, c;
off_t base_offset; off_t base_offset;
unsigned shift; unsigned shift;
void *data; void *data;
@ -312,7 +312,7 @@ static void *unpack_raw_entry(struct object_entry *obj, union delta_base *delta_
p = fill(1); p = fill(1);
c = *p; c = *p;
use(1); use(1);
size += (c & 0x7fUL) << shift; size += (c & 0x7f) << shift;
shift += 7; shift += 7;
} }
obj->size = size; obj->size = size;

Просмотреть файл

@ -44,7 +44,7 @@ void *patch_delta(const void *src_buf, unsigned long src_size,
if (cmd & 0x01) cp_off = *data++; if (cmd & 0x01) cp_off = *data++;
if (cmd & 0x02) cp_off |= (*data++ << 8); if (cmd & 0x02) cp_off |= (*data++ << 8);
if (cmd & 0x04) cp_off |= (*data++ << 16); if (cmd & 0x04) cp_off |= (*data++ << 16);
if (cmd & 0x08) cp_off |= (*data++ << 24); if (cmd & 0x08) cp_off |= ((unsigned) *data++ << 24);
if (cmd & 0x10) cp_size = *data++; if (cmd & 0x10) cp_size = *data++;
if (cmd & 0x20) cp_size |= (*data++ << 8); if (cmd & 0x20) cp_size |= (*data++ << 8);
if (cmd & 0x40) cp_size |= (*data++ << 16); if (cmd & 0x40) cp_size |= (*data++ << 16);

Просмотреть файл

@ -1162,8 +1162,7 @@ unsigned long unpack_object_header_buffer(const unsigned char *buf,
unsigned long len, enum object_type *type, unsigned long *sizep) unsigned long len, enum object_type *type, unsigned long *sizep)
{ {
unsigned shift; unsigned shift;
unsigned char c; unsigned long size, c;
unsigned long size;
unsigned long used = 0; unsigned long used = 0;
c = buf[used++]; c = buf[used++];