summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2012-03-29 10:14:08 -0700
committerKaz Kylheku <kaz@kylheku.com>2012-03-29 10:14:08 -0700
commit9e60a829fdd3874f510ebc57d1a378d020a55fb9 (patch)
tree5b62dfc5b71484eedaa6932a08df1217ecbe1def /arith.c
parent6363875356bc050ef81d40553e573fc47aca2e28 (diff)
downloadtxr-9e60a829fdd3874f510ebc57d1a378d020a55fb9.tar.gz
txr-9e60a829fdd3874f510ebc57d1a378d020a55fb9.tar.bz2
txr-9e60a829fdd3874f510ebc57d1a378d020a55fb9.zip
* arith.c (dmod): New static function.
(mod): Use dmod instead of fmod directly, to calculate the correct semantics for combinations of negative operands in the floating point domain also. * txr.1: Documented /, trunc and mod.
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c17
1 files changed, 14 insertions, 3 deletions
diff --git a/arith.c b/arith.c
index 5dd42347..a8cfb578 100644
--- a/arith.c
+++ b/arith.c
@@ -737,6 +737,17 @@ divzero:
uw_throw(numeric_error_s, lit("trunc: division by zero"));
}
+static double dmod(double a, double b)
+{
+ if (b < 0.0) {
+ double m = fmod(-a, -b);
+ return - (m < 0.0 ? m - b : m);
+ } else {
+ double m = fmod(a, b);
+ return m < 0 ? m + b : m;
+ }
+}
+
val mod(val anum, val bnum)
{
tail:
@@ -785,7 +796,7 @@ tail:
return normalize(n);
}
case FLNUM:
- return flo(fmod(c_num(anum), c_flo(bnum)));
+ return flo(dmod(c_num(anum), c_flo(bnum)));
default:
break;
}
@@ -836,7 +847,7 @@ tail:
}
}
case FLNUM:
- return flo(fmod(c_flo(anum), c_num(bnum)));
+ return flo(dmod(c_flo(anum), c_num(bnum)));
default:
break;
}
@@ -867,7 +878,7 @@ tail:
return normalize(n);
}
case TYPE_PAIR(FLNUM, FLNUM):
- return flo(fmod(c_flo(anum), c_flo(bnum)));
+ return flo(dmod(c_flo(anum), c_flo(bnum)));
case TYPE_PAIR(BGNUM, FLNUM):
anum = flo_int(anum);
goto tail;