summaryrefslogtreecommitdiffstats
path: root/arith.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2019-01-05 14:47:22 -0800
committerKaz Kylheku <kaz@kylheku.com>2019-01-05 14:47:22 -0800
commit0cd4fc4869d368c81436a43e3df59d0d42022783 (patch)
treedcc34b739587cdd3eb573180d0f264018cffb40d /arith.c
parent5e4c74dfd5927b3829b4f5e04a7964dbac6a4f34 (diff)
downloadtxr-0cd4fc4869d368c81436a43e3df59d0d42022783.tar.gz
txr-0cd4fc4869d368c81436a43e3df59d0d42022783.tar.bz2
txr-0cd4fc4869d368c81436a43e3df59d0d42022783.zip
New function: square.
The square function calulates (* x x) but is faster for bignum integers by taking advantage of mp_sqr. * arith.c (square): New function. * eval.c (eval_init): Register square as intrinsic. * lib.h (square): Declared. * txr.1: Documented.
Diffstat (limited to 'arith.c')
-rw-r--r--arith.c53
1 files changed, 53 insertions, 0 deletions
diff --git a/arith.c b/arith.c
index 67fa75c2..cd3416ca 100644
--- a/arith.c
+++ b/arith.c
@@ -2102,6 +2102,59 @@ negop:
uw_throw(error_s, lit("isqrt: negative operand"));
}
+val square(val anum)
+{
+ val self = lit("square");
+
+ switch (type(anum)) {
+ case NUM:
+ {
+ cnum a = c_n(anum);
+#if HAVE_DOUBLE_INTPTR_T
+ double_intptr_t product = a * convert(double_intptr_t, a);
+ if (product < NUM_MIN || product > NUM_MAX)
+ return bignum_dbl_ipt(product);
+ return num_fast(product);
+#else
+ cnum ap = ABS(a);
+ if (2 * highest_bit(ap) < CNUM_BIT - 1) {
+ cnum product = a * a;
+ if (product >= NUM_MIN && product <= NUM_MAX)
+ return num_fast(product);
+ return bignum(product);
+ } else {
+ val n = make_bignum();
+ mp_err mpe;
+ mp_set_intptr(mp(n), a);
+ mpe = mp_sqr(mp(n), mp(n));
+ if (mpe != MP_OKAY)
+ do_mp_error(self, mpe);
+ return n;
+ }
+#endif
+ }
+ case BGNUM:
+ {
+ val n = make_bignum();
+ mp_err mpe = mp_sqr(mp(anum), mp(n));
+ if (mpe != MP_OKAY)
+ do_mp_error(self, mpe);
+ return n;
+ }
+ case FLNUM:
+ {
+ double a = c_flo(anum, self);
+ return flo(a * a);
+ }
+ case RNG:
+ return rcons(square(from(anum)), square(to(anum)));
+ default:
+ break;
+ }
+
+ uw_throwf(error_s, lit("square: invalid operand ~s"), anum, nao);
+}
+
val gcd(val anum, val bnum)
{
val n;