diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2017-02-25 05:09:34 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2017-02-25 05:09:34 -0800 |
commit | 1aff5cbf9c7c79c1deb58d6b985f2bd03c51a4ba (patch) | |
tree | 9d542f46972db880d33217b508d0e24d34c1c8f4 /arith.c | |
parent | cf555eb22101b02bca5c0818ca4864a5b823acbc (diff) | |
download | txr-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.c | 60 |
1 files changed, 60 insertions, 0 deletions
@@ -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); |