summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2014-06-28 08:53:45 -0700
committerKaz Kylheku <kaz@kylheku.com>2014-06-28 08:53:45 -0700
commitf69bc0426a3f94318ef89dba18cdad3cbed180e7 (patch)
tree5f5094f936576ac668941774681b3bb58c0b1188 /arith.c
parent4e927689cb21336212f1aea7cb61433d5372b87a (diff)
downloadtxr-f69bc0426a3f94318ef89dba18cdad3cbed180e7.tar.gz
txr-f69bc0426a3f94318ef89dba18cdad3cbed180e7.tar.bz2
txr-f69bc0426a3f94318ef89dba18cdad3cbed180e7.zip
* arith.c (bit): New function.
* eval.c (eval_init): Register bit as intrinsic. * lib.h (bit): Declared. * mpi-patches/add-bitops (mp_bit): New function. * txr.1: Documented bit
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c50
1 files changed, 50 insertions, 0 deletions
diff --git a/arith.c b/arith.c
index c52cdb6c..76d00d1d 100644
--- a/arith.c
+++ b/arith.c
@@ -1826,6 +1826,56 @@ bad3:
uw_throwf(error_s, lit("ashift: non-integral operand ~s"), a, nao);
}
+val bit(val a, val bit)
+{
+ cnum bn;
+
+ if (!fixnump(bit))
+ goto bad;
+
+ bn = c_num(bit);
+
+ if (bn < 0)
+ goto bad2;
+
+ switch (type(a)) {
+ case NUM:
+ {
+ cnum an = c_num(a);
+ if (bn < (SIZEOF_PTR * CHAR_BIT))
+ return (an & ((cnum) 1 << bn)) ? t : nil;
+ return an < 0 ? t : nil;
+ }
+ case BGNUM:
+ {
+ mp_err res = mp_bit(mp(a), bn);
+
+ switch (res) {
+ case MP_YES:
+ return t;
+ case MP_NO:
+ return nil;
+ default:
+ goto bad4;
+ }
+ }
+ default:
+ goto bad3;
+ }
+
+bad:
+ uw_throwf(error_s, lit("bit: bit position ~s is not a fixnum"), bit, nao);
+
+bad2:
+ uw_throwf(error_s, lit("bit: bit position ~s is negative"), bit, nao);
+
+bad3:
+ uw_throwf(error_s, lit("bit: non-integral operand ~s"), a, nao);
+
+bad4:
+ uw_throwf(error_s, lit("bit: operation failed on ~s, bit ~s"), a, bit, nao);
+}
+
val maskv(val bits)
{
val accum = zero;