diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2014-06-28 08:53:45 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2014-06-28 08:53:45 -0700 |
commit | f69bc0426a3f94318ef89dba18cdad3cbed180e7 (patch) | |
tree | 5f5094f936576ac668941774681b3bb58c0b1188 /arith.c | |
parent | 4e927689cb21336212f1aea7cb61433d5372b87a (diff) | |
download | txr-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.c | 50 |
1 files changed, 50 insertions, 0 deletions
@@ -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; |