summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2012-09-16 22:46:14 -0700
committerKaz Kylheku <kaz@kylheku.com>2012-09-16 22:46:14 -0700
commitcce735e189bfcfd4a211fd856bc1850dce9b9d8c (patch)
tree65a9d2862ef61606e592fe9c516efc70b2d0e340 /arith.c
parent20a8ba5e5d546d8962802447d40636ec0cf299c8 (diff)
downloadtxr-cce735e189bfcfd4a211fd856bc1850dce9b9d8c.tar.gz
txr-cce735e189bfcfd4a211fd856bc1850dce9b9d8c.tar.bz2
txr-cce735e189bfcfd4a211fd856bc1850dce9b9d8c.zip
* arith.c (logand, logior, logxor): Bugfix: result needs to be
normalized, otherwise we end up with fixnum-range bignums. (comp_clamp): New function. (logcomp): Changed to two argument form. If second argument is present (not nil) then call comp_clamp. * eval.c (eval_init): Change registration of logcomp to allow optional argument. * lib.h (logcomp): Declaration updated. * mpi-patches/add-bitops: New mp_clamp_comp function implemented.
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c51
1 files changed, 47 insertions, 4 deletions
diff --git a/arith.c b/arith.c
index 0202f121..ff15daff 100644
--- a/arith.c
+++ b/arith.c
@@ -1520,7 +1520,7 @@ val logand(val a, val b)
c = make_bignum();
if (mp_and(mp(a), mp(b), mp(c)) != MP_OKAY)
goto bad;
- return c;
+ return normalize(c);
default:
uw_throwf(error_s, lit("logand: non-integral operands ~s ~s"), a, b, nao);
}
@@ -1561,7 +1561,7 @@ val logior(val a, val b)
c = make_bignum();
if (mp_or(mp(a), mp(b), mp(c)) != MP_OKAY)
goto bad;
- return c;
+ return normalize(c);
default:
uw_throwf(error_s, lit("logior: non-integral operands ~s ~s"), a, b, nao);
}
@@ -1602,7 +1602,7 @@ val logxor(val a, val b)
c = make_bignum();
if (mp_xor(mp(a), mp(b), mp(c)) != MP_OKAY)
goto bad;
- return c;
+ return normalize(c);
default:
uw_throwf(error_s, lit("logxor: non-integral operands ~s ~s"), a, b, nao);
}
@@ -1611,9 +1611,52 @@ bad:
uw_throwf(error_s, lit("logxor: operation failed on ~s ~s"), a, b, nao);
}
-val logcomp(val a)
+static val comp_clamp(val a, val bits)
{
+ cnum an, bn;
val b;
+ const cnum num_mask = (NUM_MAX << 1) | 1;
+ const int num_bits = CHAR_BIT * sizeof (cnum) - 2;
+
+ if (!fixnump(bits))
+ goto bad2;
+
+ bn = c_num(bits);
+
+ switch (type(a)) {
+ case NUM:
+ an = c_num(a);
+ if (bn <= num_bits) {
+ cnum mask = num_mask >> (num_bits - bn);
+ return num_fast((an & mask) ^ mask);
+ }
+ a = bignum(an);
+ /* fallthrough */
+ case BGNUM:
+ b = make_bignum();
+ if (mp_clamp_comp(mp(a), mp(b), bn) != MP_OKAY)
+ goto bad;
+ return normalize(b);
+ default:
+ goto bad3;
+ }
+
+bad:
+ uw_throwf(error_s, lit("logcomp: operation failed on ~s"), a, nao);
+
+bad2:
+ uw_throwf(error_s, lit("logcomp: bits value ~s is not a fixnum"), bits, nao);
+
+bad3:
+ uw_throwf(error_s, lit("logcomp: non-integral operand ~s"), a, nao);
+}
+
+val logcomp(val a, val bits)
+{
+ val b;
+
+ if (bits)
+ return comp_clamp(a, bits);
switch (type(a)) {
case NUM: