From: Nick Bowler Date: Thu, 16 Jul 2009 21:10:22 +0000 (-0400) Subject: libupkg: Add signed integer unpacking to fix incorrect signedness issue. X-Git-Url: https://git.draconx.ca/gitweb/upkg.git/commitdiff_plain/a6f1f69cc4874e2f2f878289bfc0f677220dfda9 libupkg: Add signed integer unpacking to fix incorrect signedness issue. --- diff --git a/src/libupkg.c b/src/libupkg.c index 6012f09..3f07095 100644 --- a/src/libupkg.c +++ b/src/libupkg.c @@ -251,7 +251,7 @@ static int pkg_init_exports(struct upkg *pkg) len += rc; if (nbuf-len < 4) goto err; - export->package = unpack_32_le(buf+len); + export->package = unpack_s32_le(buf+len); len += 4; rc = upkg_decode_index(&tmp, buf+len, nbuf-len); @@ -323,7 +323,7 @@ static int pkg_init_imports(struct upkg *pkg) len += rc; if (nbuf-len < 4) goto err; - import->package = unpack_32_le(buf+len); + import->package = unpack_s32_le(buf+len); len += 4; rc = upkg_decode_index(&tmp, buf+len, nbuf-len); diff --git a/src/pack.c b/src/pack.c index b87ac34..e711de1 100644 --- a/src/pack.c +++ b/src/pack.c @@ -22,7 +22,7 @@ #include "pack.h" -/* Integer packing. */ +/* Unsigned integer packing. */ #define DEFPACK_BE(bits, type) void pack_ ## bits ## _be ( \ unsigned char *out, type v \ ) { \ @@ -90,3 +90,48 @@ DEFUNPACK_LE(32, unsigned long) #ifdef ULLONG_MAX DEFUNPACK_LE(64, unsigned long long) #endif + +/* + * Two's complement signed integer packing. This is unlikely to work on + * systems that don't themselves use two's complement. + */ + +#define DEFUNPACK_SBE(bits, max, type) type unpack_s ## bits ## _be ( \ + unsigned char *in \ +) { \ + type v = 0; \ + unsigned i; \ + int sign = (in[0] & 0x80) ? 1 : 0; \ + in[0] &= 0x7f; \ + for (i = 0; i < bits/8; i++) { \ + v *= 256; \ + v += in[i]; \ + } \ + return sign*(-max-1) + v; \ +} + +#define DEFUNPACK_SLE(bits, max, type) type unpack_s ## bits ## _le ( \ + unsigned char *in \ +) { \ + type v = 0; \ + unsigned i; \ + int sign = (in[bits/8 - 1] & 0x80) ? 1 : 0; \ + in[bits/8 - 1] &= 0x7f; \ + for (i = 1; i <= bits/8; i++) { \ + v *= 256; \ + v += in[bits/8 - i]; \ + } \ + return sign*(-max-1) + v; \ +} + +DEFUNPACK_SBE(16, 32767, short) +DEFUNPACK_SBE(32, 2147483647l, long) +#ifdef LLONG_MAX +DEFUNPACK_SBE(64, 9223372036854775807ll, long long) +#endif + +DEFUNPACK_SLE(16, 32767, short) +DEFUNPACK_SLE(32, 2147483647l, long) +#ifdef LLONG_MAX +DEFUNPACK_SLE(64, 9223372036854775807ll, long long) +#endif diff --git a/src/pack.h b/src/pack.h index d6a5e65..8b4874d 100644 --- a/src/pack.h +++ b/src/pack.h @@ -46,4 +46,16 @@ unsigned long unpack_32_le(unsigned char *); unsigned long long unpack_64_le(unsigned char *); #endif +short unpack_s16_be(unsigned char *); +long unpack_s32_be(unsigned char *); +#ifdef LLONG_MAX +long long unpack_s64_be(unsigned char *); +#endif + +short unpack_s16_le(unsigned char *); +long unpack_s32_le(unsigned char *); +#ifdef LLONG_MAX +long long unpack_s64_le(unsigned char *); +#endif + #endif