summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2011-12-11 23:16:44 -0800
committerKaz Kylheku <kaz@kylheku.com>2011-12-11 23:16:44 -0800
commit770b69a7495f5e1f83eaf0c5de5782a3db90ad7b (patch)
treea600dd018c53ac6394932f9c1b96007ce77e7eaf /arith.c
parent98d7a0cf623fb0e34ee00017909a315a32a8de38 (diff)
downloadtxr-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.c112
1 files changed, 112 insertions, 0 deletions
diff --git a/arith.c b/arith.c
index bd9a5d2c..e1ee9fe1 100644
--- a/arith.c
+++ b/arith.c
@@ -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);