diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2019-01-05 14:47:22 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2019-01-05 14:47:22 -0800 |
commit | 0cd4fc4869d368c81436a43e3df59d0d42022783 (patch) | |
tree | dcc34b739587cdd3eb573180d0f264018cffb40d /arith.c | |
parent | 5e4c74dfd5927b3829b4f5e04a7964dbac6a4f34 (diff) | |
download | txr-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.c | 53 |
1 files changed, 53 insertions, 0 deletions
@@ -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; |