summaryrefslogtreecommitdiffstats
path: root/lib.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2019-03-12 06:23:11 -0700
committerKaz Kylheku <kaz@kylheku.com>2019-03-12 06:23:11 -0700
commitcbc0bbb556f281b219ddbc9a6728afc71fbeaca3 (patch)
treed52f0a3da24922590b9c0722831f71742362ebd2 /lib.c
parentd0b35b7d272eaf4fbcf7754e25801739ddb2410d (diff)
downloadtxr-cbc0bbb556f281b219ddbc9a6728afc71fbeaca3.tar.gz
txr-cbc0bbb556f281b219ddbc9a6728afc71fbeaca3.tar.bz2
txr-cbc0bbb556f281b219ddbc9a6728afc71fbeaca3.zip
float: turn out-of-range calculations into exceptions.
On platforms that have the C99 fpclassify, we can use it to banish infinity and NaN representations. If such a thing arises, we throw an exception rather than producing an object that prints as #<bad-float>. * configure: add detection for fpclassify. * lib.c (bad_float): New inline function and macro. (flo): If the argument is other than zero, a normal value or a subnormal, then throw an exception. We thereby refuse to admit such objects into our numeric object system.
Diffstat (limited to 'lib.c')
-rw-r--r--lib.c28
1 files changed, 24 insertions, 4 deletions
diff --git a/lib.c b/lib.c
index f23e40df..658a17a6 100644
--- a/lib.c
+++ b/lib.c
@@ -3268,12 +3268,32 @@ cnum c_fixnum(val num, val self)
}
}
+#if HAVE_FPCLASSIFY
+INLINE int bad_float(double d)
+{
+ switch fpclassify(d) {
+ case FP_ZERO:
+ case FP_NORMAL:
+ case FP_SUBNORMAL:
+ return 0;
+ default:
+ return 1;
+ }
+}
+#else
+#define bad_float(d) (0)
+#endif
+
val flo(double n)
{
- val obj = make_obj();
- obj->fl.type = FLNUM;
- obj->fl.n = n;
- return obj;
+ if (bad_float(n)) {
+ uw_throw(numeric_error_s, lit("out-of-range floating-point result"));
+ } else {
+ val obj = make_obj();
+ obj->fl.type = FLNUM;
+ obj->fl.n = n;
+ return obj;
+ }
}
double c_flo(val num, val self)