diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2018-11-25 10:23:16 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2018-11-25 10:23:16 -0800 |
commit | 6395346bee2382d0e800ce28593c48ad3bab9792 (patch) | |
tree | f4f37b49b4269bc346d60bdbfe31f707c518e8e1 /arith.c | |
parent | 2ca84b85530a886bb1d5314ca5c006529e08540b (diff) | |
download | txr-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.c | 43 |
1 files changed, 43 insertions, 0 deletions
@@ -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; |