summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2017-02-25 05:09:34 -0800
committerKaz Kylheku <kaz@kylheku.com>2017-02-25 05:09:34 -0800
commit1aff5cbf9c7c79c1deb58d6b985f2bd03c51a4ba (patch)
tree9d542f46972db880d33217b508d0e24d34c1c8f4 /arith.c
parentcf555eb22101b02bca5c0818ca4864a5b823acbc (diff)
downloadtxr-1aff5cbf9c7c79c1deb58d6b985f2bd03c51a4ba.tar.gz
txr-1aff5cbf9c7c79c1deb58d6b985f2bd03c51a4ba.tar.bz2
txr-1aff5cbf9c7c79c1deb58d6b985f2bd03c51a4ba.zip
Adding round function.
* arith.c (round1): New static function. (roundiv): New function. * configure: New test for C99 round function. * eval.c (eval_init): Register round intrinsic. * txr.1: Documented.
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c60
1 files changed, 60 insertions, 0 deletions
diff --git a/arith.c b/arith.c
index 3bded75b..36ecad20 100644
--- a/arith.c
+++ b/arith.c
@@ -1231,6 +1231,66 @@ val ceildiv(val anum, val bnum)
return neg(floordiv(neg(anum), bnum));
}
+static val round1(val num)
+{
+ switch (type(num)) {
+ case NUM:
+ case BGNUM:
+ return num;
+ case FLNUM:
+#if HAVE_ROUND
+ return flo(round(c_flo(num)));
+#else
+ {
+ double n = c_flo(num);
+ return if3(n >= 0,
+ flo(floor(0.5 + n)),
+ flo(-floor(0.5 + fabs(n))));
+ }
+#endif
+ case RNG:
+ return rcons(round1(from(num)), round1(to(num)));
+ default:
+ break;
+ }
+ uw_throwf(error_s, lit("round: invalid operand ~s"), num);
+}
+
+
+val roundiv(val anum, val bnum)
+{
+ if (missingp(bnum))
+ return round1(anum);
+
+ if (minusp(bnum)) {
+ anum = neg(anum);
+ bnum = neg(bnum);
+ }
+
+ if (rangep(anum)) {
+ return rcons(roundiv(from(anum), bnum), roundiv(to(anum), bnum));
+ } else if (floatp(anum) || floatp(bnum)) {
+ val quot = divi(anum, bnum);
+#if HAVE_ROUND
+ return flo(round(c_flo(quot)));
+#else
+ {
+ double q = c_flo(quot);
+ return if3(q >= 0,
+ flo(floor(0.5 + q)),
+ flo(-ceil(0.5 + fabs(q))));
+ }
+#endif
+ } else {
+ val quot = floordiv(anum, bnum);
+ val rem = minus(anum, mul(quot, bnum));
+ val drem = ash(rem, one);
+ return if3(eq(drem, bnum),
+ if3(minusp(quot), quot, succ(quot)),
+ if3(lt(drem, bnum), quot, succ(quot)));
+ }
+}
+
val wrap_star(val start, val end, val num)
{
val modulus = minus(end, start);