summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2018-11-25 10:23:16 -0800
committerKaz Kylheku <kaz@kylheku.com>2018-11-25 10:23:16 -0800
commit6395346bee2382d0e800ce28593c48ad3bab9792 (patch)
treef4f37b49b4269bc346d60bdbfe31f707c518e8e1 /arith.c
parent2ca84b85530a886bb1d5314ca5c006529e08540b (diff)
downloadtxr-6395346bee2382d0e800ce28593c48ad3bab9792.tar.gz
txr-6395346bee2382d0e800ce28593c48ad3bab9792.tar.bz2
txr-6395346bee2382d0e800ce28593c48ad3bab9792.zip
logxor: fix seriously broken function.
Reported by Guillaume le Vaillant. * arith.c (logxor): Fix broken behavior when the arguments are the same nonzero fixnum, or the same bignum object. (logxor_old): New function: verbatim copy of previous logxor. * eval.c (eval_init): Register logxor intrinsic to the broken function if compatibility is 202 or less. * txr.1: Compat note added.
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c43
1 files changed, 43 insertions, 0 deletions
diff --git a/arith.c b/arith.c
index 90575fc5..badc5d84 100644
--- a/arith.c
+++ b/arith.c
@@ -2461,6 +2461,49 @@ val logxor(val a, val b)
{
val c;
+ switch (TYPE_PAIR(type(a), type(b))) {
+ case TYPE_PAIR(NUM, CHR):
+ case TYPE_PAIR(CHR, NUM):
+ {
+ cnum ac = c_n(a);
+ cnum bc = c_n(b);
+ return chr(ac ^ bc);
+ }
+ case TYPE_PAIR(NUM, NUM):
+ {
+ cnum ac = c_n(a);
+ cnum bc = c_n(b);
+ return num_fast(ac ^ bc);
+ }
+ case TYPE_PAIR(BGNUM, NUM):
+ {
+ val tmp = a;
+ a = b;
+ b = tmp;
+ }
+ /* fallthrough */
+ case TYPE_PAIR(NUM, BGNUM):
+ a = bignum(c_n(a));
+ /* fallthrough */
+ case TYPE_PAIR(BGNUM, BGNUM):
+ if (a == b)
+ return zero;
+ c = make_ubignum();
+ if (mp_xor(mp(a), mp(b), mp(c)) != MP_OKAY)
+ goto bad;
+ return normalize(c);
+ default:
+ uw_throwf(error_s, lit("logxor: non-integral operands ~s ~s"), a, b, nao);
+ }
+
+bad:
+ uw_throwf(error_s, lit("logxor: operation failed on ~s ~s"), a, b, nao);
+}
+
+val logxor_old(val a, val b)
+{
+ val c;
+
if (zerop(a) && zerop(b))
return zero;