diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2011-12-11 23:16:44 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2011-12-11 23:16:44 -0800 |
commit | 770b69a7495f5e1f83eaf0c5de5782a3db90ad7b (patch) | |
tree | a600dd018c53ac6394932f9c1b96007ce77e7eaf /arith.c | |
parent | 98d7a0cf623fb0e34ee00017909a315a32a8de38 (diff) | |
download | txr-770b69a7495f5e1f83eaf0c5de5782a3db90ad7b.tar.gz txr-770b69a7495f5e1f83eaf0c5de5782a3db90ad7b.tar.bz2 txr-770b69a7495f5e1f83eaf0c5de5782a3db90ad7b.zip |
* arith.c (zerop, gt, lt, ge, le): Functions from lib.c reimplemented
with bignum support.
* eval.c (eval_init): Added bignump and zerop as intrinsic function.
Renamed numberp to fixnump.
* lib.c (zerop, gt, lt, ge, le): Functions removed.
(numeq): Unused function removed.
* lib.h (numeq): Declaration removed.
* txr.1: Sections for zerop and bignump created. Changed reference
to numberp to fixnump.
Diffstat (limited to 'arith.c')
-rw-r--r-- | arith.c | 112 |
1 files changed, 112 insertions, 0 deletions
@@ -689,6 +689,118 @@ val mod(val anum, val bnum) abort(); } +val zerop(val num) +{ + if (num == zero) + return t; + + if (!fixnump(num) && !bignump(num)) + uw_throwf(error_s, lit("zerof: ~s is not a number"), num, nao); + return nil; +} + +val gt(val anum, val bnum) +{ + int tag_a = tag(anum); + int tag_b = tag(bnum); + + switch (TAG_PAIR(tag_a, tag_b)) { + case TAG_PAIR(TAG_NUM, TAG_NUM): + return c_num(anum) > c_num(bnum) ? t : nil; + case TAG_PAIR(TAG_NUM, TAG_PTR): + type_check(bnum, BGNUM); + return mp_cmp_z(mp(bnum)) == MP_LT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_NUM): + type_check(anum, BGNUM); + return mp_cmp_z(mp(anum)) == MP_GT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_PTR): + type_check(anum, BGNUM); + return mp_cmp(mp(anum), mp(bnum)) == MP_GT ? t : nil; + } + + uw_throwf(error_s, lit("gt: invalid operands ~s ~s"), anum, bnum, nao); + abort(); +} + +val lt(val anum, val bnum) +{ + int tag_a = tag(anum); + int tag_b = tag(bnum); + + switch (TAG_PAIR(tag_a, tag_b)) { + case TAG_PAIR(TAG_NUM, TAG_NUM): + return c_num(anum) < c_num(bnum) ? t : nil; + case TAG_PAIR(TAG_NUM, TAG_PTR): + type_check(bnum, BGNUM); + return mp_cmp_z(mp(bnum)) == MP_GT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_NUM): + type_check(anum, BGNUM); + return mp_cmp_z(mp(anum)) == MP_LT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_PTR): + type_check(anum, BGNUM); + return mp_cmp(mp(anum), mp(bnum)) == MP_LT ? t : nil; + } + + uw_throwf(error_s, lit("lt: invalid operands ~s ~s"), anum, bnum, nao); + abort(); +} + +val ge(val anum, val bnum) +{ + int tag_a = tag(anum); + int tag_b = tag(bnum); + + switch (TAG_PAIR(tag_a, tag_b)) { + case TAG_PAIR(TAG_NUM, TAG_NUM): + return c_num(anum) >= c_num(bnum) ? t : nil; + case TAG_PAIR(TAG_NUM, TAG_PTR): + type_check(bnum, BGNUM); + return mp_cmp_z(mp(bnum)) == MP_LT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_NUM): + type_check(anum, BGNUM); + return mp_cmp_z(mp(anum)) == MP_GT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_PTR): + type_check(anum, BGNUM); + switch (mp_cmp(mp(anum), mp(bnum))) { + case MP_GT: case MP_EQ: + return t; + default: + return nil; + } + } + + uw_throwf(error_s, lit("ge: invalid operands ~s ~s"), anum, bnum, nao); + abort(); +} + +val le(val anum, val bnum) +{ + int tag_a = tag(anum); + int tag_b = tag(bnum); + + switch (TAG_PAIR(tag_a, tag_b)) { + case TAG_PAIR(TAG_NUM, TAG_NUM): + return c_num(anum) <= c_num(bnum) ? t : nil; + case TAG_PAIR(TAG_NUM, TAG_PTR): + type_check(bnum, BGNUM); + return mp_cmp_z(mp(bnum)) == MP_GT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_NUM): + type_check(anum, BGNUM); + return mp_cmp_z(mp(anum)) == MP_LT ? t : nil; + case TAG_PAIR(TAG_PTR, TAG_PTR): + type_check(anum, BGNUM); + switch (mp_cmp(mp(anum), mp(bnum))) { + case MP_LT: case MP_EQ: + return t; + default: + return nil; + } + } + + uw_throwf(error_s, lit("lt: invalid operands ~s ~s"), anum, bnum, nao); + abort(); +} + void arith_init(void) { mp_init(&NUM_MAX_MP); |