shithub: mc

Download patch

ref: 96cb72770eb9c9bf3259ebb4647ef88092438808
parent: 755433e998180ba049bc22975b8ecffa23bf0357
author: S. Gilles <[email protected]>
date: Tue Feb 27 15:19:32 EST 2018

Implement a few simple floating-point functions

Proof of concept. Trunc/floor/ceil are probably the simplest FP
functions that require bit-poking.

--- a/lib/bld.sub
+++ b/lib/bld.sub
@@ -8,6 +8,7 @@
 	inifile
 	iter
 	json
+	math
 	regex
 	std
 	sys
--- /dev/null
+++ b/lib/math/bld.sub
@@ -1,0 +1,8 @@
+lib math =
+	fpmath.myr
+
+	# trunc
+	fpmath-trunc-impl.myr
+
+	lib ../std:std
+;;
--- /dev/null
+++ b/lib/math/fpmath-trunc-impl.myr
@@ -1,0 +1,160 @@
+use std
+
+pkg math =
+	pkglocal const trunc32 : (f : flt32 -> flt32)
+	pkglocal const floor32 : (f : flt32 -> flt32)
+	pkglocal const ceil32  : (f : flt32 -> flt32)
+	pkglocal const trunc64 : (f : flt64 -> flt64)
+	pkglocal const floor64 : (f : flt64 -> flt64)
+	pkglocal const ceil64  : (f : flt64 -> flt64)
+;;
+
+pkglocal const trunc32 = {f : flt32
+	var u : uint32 = std.flt32bits(f)
+	var e : uint32 = (((u >> 23) & 0xff) : uint32) - 127
+	var e_lt_zero : uint32 = ((e >> 31) : uint32) & 0x1
+	var e_ge_zero : uint32 = 1 - e_lt_zero
+	var e_ge_23 : uint32 = 1 - ((e - 23) >> 31)
+	/*
+	   The significand 1 . m1 m2 ... m23 needs to be
+	   truncated, which corresponds to zeroing all mi
+	   bits where i is beyond the exponent e (they are
+	   the actual sub-integer portion).
+	 */
+	var m : uint32 = ~(((1 << 23) - 1) >> e)
+	m |= (-1 : uint32) * e_ge_23
+	var v_ge_zero : uint32 = (u & m) * e_ge_zero
+
+	/*
+	   On the other hand, if the exponent is < 0, "23
+	   - e" is garbage, and we should just return +/-
+	   zero
+	 */
+	var v_lt_zero : uint32 = (u & (1 << 31)) * e_lt_zero
+
+	/* Try to save a branch */
+	var v : uint32 = v_ge_zero + v_lt_zero
+	-> std.flt32frombits(v)
+}
+
+pkglocal const floor32 = {f : flt32
+	var u : uint32 = std.flt32bits(f)
+	var e : int32 = (((u >> 23) & 0xff) : int32) - 127
+	var shift_e : uint32 = (e : uint32)
+
+	/* Many special cases */
+	if e >= 23 || u == 0x80000000
+		-> f
+	elif (e < 0) && (u & (1 << 31) != 0)
+		-> -1.0
+	elif e < 0
+		-> 0.0
+	;;
+
+	if u & (1 << 31) != 0
+		var fractional_mask : uint32 = (((1 << 23) - 1) >> shift_e)
+		var v : uint32 = u & ~fractional_mask
+		if (u & fractional_mask) != 0
+			v += ((1 << 23) >> shift_e)
+		;;
+		-> std.flt32frombits(v)
+	;;
+
+	var m : uint32 = ~(((1 << 23) - 1) >> shift_e)
+	var v : uint32 = u & m
+	-> std.flt32frombits(v)
+}
+
+pkglocal const ceil32 = {f;
+	var u : uint32 = std.flt32bits(f)
+	var e : int32 = (((u >> 23) & 0xff) : int32) - 127
+	var shift_e : uint32 = (e : uint32)
+	if e >= 23 || u == 0x0
+		-> f
+	elif (e < 0) && (u & (1 << 31) == 0)
+		-> 1.0
+	elif e < 0
+		-> -0.0
+	;;
+
+	if u & (1 << 31) == 0
+		var fractional_mask : uint32 = (((1 << 23) - 1) >> shift_e)
+		var v : uint32 = u & ~fractional_mask
+		if (u & fractional_mask) != 0
+			v += ((1 << 23) >> shift_e)
+		;;
+		-> std.flt32frombits(v)
+	;;
+
+	var m : uint32 = ~(((1 << 23) - 1) >> shift_e)
+	var v : uint32 = u & m
+	-> std.flt32frombits(v)
+}
+
+pkglocal const trunc64 = {f : flt64
+	var u : uint64 = std.flt64bits(f)
+	var e : uint64 = (((u >> 52) & 0x7ff) : uint64) - 1023
+	var e_lt_zero : uint64 = ((e >> 63) : uint64) & 0x1
+	var e_ge_zero : uint64 = 1 - e_lt_zero
+	var e_ge_52 : uint64 = 1 - ((e - 52) >> 63)
+	var m : uint64 = ~(((1 << 52) - 1) >> e)
+	m |= (-1 : uint64) * e_ge_52
+	var v_ge_zero : uint64 = (u & m) * e_ge_zero
+	var v_lt_zero : uint64 = (u & (1 << 63)) * e_lt_zero
+	var v : uint64 = v_ge_zero + v_lt_zero
+	-> std.flt64frombits(v)
+}
+
+pkglocal const floor64 = {f : flt64
+	var u : uint64 = std.flt64bits(f)
+	var e : int64 = (((u >> 52) & 0x7ff) : int64) - 1023
+	var shift_e : uint64 = (e : uint64)
+
+	if e >= 52 || u == 0x8000000000000000ul
+		-> f
+	elif (e < 0) && (u & (1 << 63) != 0)
+			-> -1.0
+	elif e < 0
+		-> 0.0
+	;;
+
+	if u & (1 << 63) != 0
+		var fractional_mask : uint64 = (((1 << 52) - 1) >> shift_e)
+		var v : uint64 = u & ~fractional_mask
+		if (u & fractional_mask) != 0
+			v += ((1 << 52) >> shift_e)
+		;;
+		-> std.flt64frombits(v)
+	;;
+
+	var m : uint64 = ~(((1 << 52) - 1) >> shift_e)
+	var v : uint64 = u & m
+	-> std.flt64frombits(v)
+}
+
+pkglocal const ceil64 = {f;
+	var u : uint64 = std.flt64bits(f)
+	var e : int64 = (((u >> 52) & 0x7ff) : int64) - 1023
+	var shift_e : uint64 = (e : uint64)
+
+	if e >= 52 || u == 0x0ul
+		-> f
+	elif (e < 0) && (u & (1 << 63) == 0)
+			-> 1.0
+	elif e < 0
+		-> -0.0
+	;;
+
+	if u & (1 << 63) == 0
+		var fractional_mask : uint64 = (((1 << 52) - 1) >> shift_e)
+		var v : uint64 = u & ~fractional_mask
+		if (u & fractional_mask) != 0
+			v += ((1 << 52) >> shift_e)
+		;;
+		-> std.flt64frombits(v)
+	;;
+
+	var m : uint64 = ~(((1 << 52) - 1) >> shift_e)
+	var v : uint64 = u & m
+	-> std.flt64frombits(v)
+}
--- /dev/null
+++ b/lib/math/fpmath.myr
@@ -1,0 +1,36 @@
+use std
+
+pkg math =
+	trait fpmath @f =
+
+		/* fpmath-trunc-impl */
+		trunc : (f : @f -> @f)
+		ceil  : (f : @f -> @f)
+		floor : (f : @f -> @f)
+
+		/* compute (s, t) with s = round-nearest(a+b), s + t = a + b */
+//		fast2sum : (a : @f, b : @f -> (@f, @f))
+	;;
+
+	impl fpmath flt32
+	impl fpmath flt64
+;;
+
+impl fpmath flt32 =
+	trunc = {f; -> trunc32(f)}
+	floor = {f; -> floor32(f)}
+	ceil  = {f; -> ceil32(f)}
+;;
+
+impl fpmath flt64 =
+	trunc = {f; -> trunc64(f)}
+	floor = {f; -> floor64(f)}
+	ceil  = {f; -> ceil64(f)}
+;;
+
+extern const trunc32 : (f : flt32 -> flt32)
+extern const floor32 : (f : flt32 -> flt32)
+extern const ceil32  : (f : flt32 -> flt32)
+extern const trunc64 : (f : flt64 -> flt64)
+extern const floor64 : (f : flt64 -> flt64)
+extern const ceil64  : (f : flt64 -> flt64)
--- /dev/null
+++ b/lib/math/test/fpmath-sum-impl.myr
@@ -1,0 +1,163 @@
+use std
+use math
+use testr
+
+const main = {
+	testr.run([
+		[.name = "trunc-01",    .fn = trunc01],
+		[.name = "trunc-02",    .fn = trunc02],
+		[.name = "floor-01",    .fn = floor01],
+		[.name = "floor-02",    .fn = floor02],
+		[.name = "ceil-01",     .fn = ceil01],
+		[.name = "ceil-02",     .fn = ceil02],
+		[.name = "fast2sum-01", .fn = fast2sum01],
+	][:])
+}
+
+impl std.equatable flt32 =
+        eq = {a : flt32, b : flt32; -> std.flt32bits(a) == std.flt32bits(b)}
+;;
+
+impl std.equatable flt64 =
+        eq = {a : flt64, b : flt64; -> std.flt64bits(a) == std.flt64bits(b)}
+;;
+
+const trunc01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(1.0, 1.0),
+		(1.1, 1.0),
+		(0.9, 0.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -3.0),
+		(101.999, 101.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.trunc(f), g)
+	;;
+}
+
+const trunc02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(1.0, 1.0),
+		(1.1, 1.0),
+		(0.9, 0.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -3.0),
+		(101.999, 101.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.trunc(f), g)
+	;;
+}
+
+const floor01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 0.0),
+		(1.1, 1.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -4.0),
+		(-101.999, -102.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.floor(f), g)
+	;;
+}
+
+const floor02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 0.0),
+		(1.1, 1.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -4.0),
+		(-101.999, -102.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.floor(f), g)
+	;;
+}
+
+const ceil01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 1.0),
+		(-0.1, -0.0),
+		(1.1, 2.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -3.0),
+		(-101.999, -101.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.ceil(f), g)
+	;;
+}
+
+const ceil02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 1.0),
+		(-0.1, -0.0),
+		(1.1, 2.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -3.0),
+		(-101.999, -101.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.ceil(f), g)
+	;;
+}
+
+const fast2sum01 = {c
+        var flt32s : (flt32, flt32, flt32, flt32)[:] = [
+                (1.0, 1.0, 2.0, 0.0),
+                (10664524000000000000.0, 1.11842, 10664524000000000000.0, 1.11842),
+                (1.11843, 10664524000000000000.0, 10664524000000000000.0, 1.11843),
+                (-21897.1324, 17323.22, -4573.912, 0.0),
+        ][:]
+
+        for (a, b, s1, t1) : flt32s
+                var s2, t2
+                (s2, t2) = math.fast2sum(a, b)
+                testr.eq(c, s1, s2)
+                testr.eq(c, t1, t2)
+        ;;
+
+        var flt64s : (flt64, flt64, flt64, flt64)[:] = [
+                (1.0, 1.0, 2.0, 0.0),
+                (-21897.1324, 17323.22, -4573.912399999997, 0.0),
+                (std.flt64frombits(0x78591b0672a81284), std.flt64frombits(0x6a8c3190e27a1884),
+                 std.flt64frombits(0x78591b0672a81284), std.flt64frombits(0x6a8c3190e27a1884)),
+                (std.flt64frombits(0x6a8c3190e27a1884), std.flt64frombits(0x78591b0672a81284),
+                 std.flt64frombits(0x78591b0672a81284), std.flt64frombits(0x6a8c3190e27a1884)),
+                (std.flt64frombits(0x78591b0672a81284), std.flt64frombits(0x7858273672ca19a0),
+                 std.flt64frombits(0x7868a11e72b91612), 0.0),
+        ][:]
+
+        for (a, b, s1, t1) : flt64s
+                var s2, t2
+                (s2, t2) = math.fast2sum(a, b)
+                testr.eq(c, s1, s2)
+                testr.eq(c, std.flt64bits(t1), std.flt64bits(t2))
+        ;;
+}
--- /dev/null
+++ b/lib/math/test/fpmath-trunc-impl.myr
@@ -1,0 +1,128 @@
+use std
+use math
+use testr
+
+const main = {
+	testr.run([
+		[.name = "trunc-01",    .fn = trunc01],
+		[.name = "trunc-02",    .fn = trunc02],
+		[.name = "floor-01",    .fn = floor01],
+		[.name = "floor-02",    .fn = floor02],
+		[.name = "ceil-01",     .fn = ceil01],
+		[.name = "ceil-02",     .fn = ceil02],
+	][:])
+}
+
+impl std.equatable flt32 =
+        eq = {a : flt32, b : flt32; -> std.flt32bits(a) == std.flt32bits(b)}
+;;
+
+impl std.equatable flt64 =
+        eq = {a : flt64, b : flt64; -> std.flt64bits(a) == std.flt64bits(b)}
+;;
+
+const trunc01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(1.0, 1.0),
+		(1.1, 1.0),
+		(0.9, 0.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -3.0),
+		(101.999, 101.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.trunc(f), g)
+	;;
+}
+
+const trunc02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(1.0, 1.0),
+		(1.1, 1.0),
+		(0.9, 0.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -3.0),
+		(101.999, 101.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.trunc(f), g)
+	;;
+}
+
+const floor01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 0.0),
+		(1.1, 1.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -4.0),
+		(-101.999, -102.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.floor(f), g)
+	;;
+}
+
+const floor02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 0.0),
+		(1.1, 1.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -4.0),
+		(-101.999, -102.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.floor(f), g)
+	;;
+}
+
+const ceil01 = {c
+	var flt32s : (flt32, flt32)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 1.0),
+		(-0.1, -0.0),
+		(1.1, 2.0),
+		(10664524000000000000.0, 10664524000000000000.0),
+		(-3.5, -3.0),
+		(-101.999, -101.0),
+		(std.flt32nan(), std.flt32nan()),
+	][:]
+
+	for (f, g) : flt32s
+		testr.eq(c, math.ceil(f), g)
+	;;
+}
+
+const ceil02 = {c
+	var flt64s : (flt64, flt64)[:] = [
+		(0.0, 0.0),
+		(-0.0, -0.0),
+		(0.5, 1.0),
+		(-0.1, -0.0),
+		(1.1, 2.0),
+		(13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0, 13809453812721350000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0),
+		(-3.5, -3.0),
+		(-101.999, -101.0),
+		(std.flt64nan(), std.flt64nan()),
+	][:]
+
+	for (f, g) : flt64s
+		testr.eq(c, math.ceil(f), g)
+	;;
+}