From 6d75d138093e582e00871e937578cf697f3808be Mon Sep 17 00:00:00 2001 From: matz Date: Wed, 13 Feb 2002 09:53:17 +0000 Subject: [PATCH] * bignum.c: forget to check in DIGSPERLONGLONG. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@2066 b2dd03c8-39d4-4d8f-98ff-823fe69b080e --- bignum.c | 20 ++++++++++++++++++-- 1 file changed, 18 insertions(+), 2 deletions(-) diff --git a/bignum.c b/bignum.c index c959950e7d..d59969cdd9 100644 --- a/bignum.c +++ b/bignum.c @@ -187,6 +187,8 @@ rb_int2inum(n) #ifdef HAVE_LONG_LONG +#define DIGSPERLONGLONG ((unsigned int)(sizeof(LONG_LONG)/sizeof(BDIGIT))) + void rb_quad_pack(buf, val) char *buf; @@ -267,13 +269,21 @@ rb_quad_pack(buf, val) memset(buf, 0, QUAD_SIZE); val = rb_to_int(val); if (FIXNUM_P(val)) { - val = rb_uint2big(FIX2LONG(val)); + val = rb_int2big(FIX2LONG(val)); } len = RBIGNUM(val)->len * sizeof(BDIGIT); if (len > QUAD_SIZE) len = QUAD_SIZE; memcpy(buf, (char*)BDIGITS(val), len); + if (!RBIGNUM(val)->sign) { + len = QUAD_SIZE; + while (len--) { + *buf = ~*buf++; + } + } } +#define BNEG(b) (RSHIFT(((BDIGIT*)b)[QUAD_SIZE/sizeof(BDIGIT)-1],BITSPERDIG-1) != 0) + VALUE rb_quad_unpack(buf, sign) const char *buf; @@ -282,8 +292,14 @@ rb_quad_unpack(buf, sign) VALUE big = bignew(QUAD_SIZE/sizeof(BDIGIT), 1); memcpy((char*)BDIGITS(big), buf, QUAD_SIZE); - if (sign && (buf[7] & 0x80)) { + if (sign && BNEG(buf)) { + long len = QUAD_SIZE; + char *tmp = (char*)BDIGITS(big); + RBIGNUM(big)->sign = 0; + while (len--) { + *tmp = ~*tmp++; + } } return bignorm(big);