diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2019-01-25 06:59:29 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2019-01-25 06:59:29 -0800 |
commit | 4b789435f8e4bb6ec9ea4bab3d0a109682adafa6 (patch) | |
tree | 65f91721ae512480ea4df6b9f7f493b35eab6660 /arith.c | |
parent | 4c71a77f2c39a8158c80b40af1a5cc2358251a00 (diff) | |
download | txr-4b789435f8e4bb6ec9ea4bab3d0a109682adafa6.tar.gz txr-4b789435f8e4bb6ec9ea4bab3d0a109682adafa6.tar.bz2 txr-4b789435f8e4bb6ec9ea4bab3d0a109682adafa6.zip |
Extend infrastructure for double_intptr_t.
We support an unsigned version of the type, and add functions
for converting between Lisp values and both types.
* arith.c (bignum_dbl_uipt): New function, unsigned companion
to existing bignum_dbl_ipt.
(c_dbl_num, c_dbl_unum): New functions.
* arith.h (bignum_dbl_uipt, c_dbl_num, c_dbl_unum): Declared.
* configure (superulong_t, SIZEOF_DOUBLE_INTPTR,
DOUBLE_INTPTR_MAX, DOUBLE_INTPTR_MIN, DOUBLE_UINTPTR_MAX,
double_uintptr_t): New definitions going into config.h.
* lib.h (dbl_cnum, dbl_ucnum): New typedefs: double-sized
analogs of cnum and ucnum.
* mpi/mpi.c (mp_set_double_uintptr, mp_get_double_uintptr,
mp_get_double_intptr): New functions.
(s_mp_in_big_range): New static function.
(mp_in_double_intptr_range, mp_in_double_uintptr_range): New
functions.
* mpi/mpi.h (mp_set_double_uintptr,
mp_get_double_intptr, mp_get_double_uintptr,
mp_in_double_intptr_range, mp_in_double_uintptr_range):
Declared.
Diffstat (limited to 'arith.c')
-rw-r--r-- | arith.c | 47 |
1 files changed, 47 insertions, 0 deletions
@@ -158,6 +158,13 @@ val bignum_dbl_ipt(double_intptr_t di) return n; } +val bignum_dbl_uipt(double_uintptr_t dui) +{ + val n = make_bignum(); + mp_set_double_uintptr(mp(n), dui); + return n; +} + #endif val normalize(val bignum) @@ -207,6 +214,46 @@ val unum(ucnum u) } } +#if HAVE_DOUBLE_INTPTR_T + +dbl_cnum c_dbl_num(val n) +{ + switch (type(n)) { + case CHR: case NUM: + return coerce(cnum, n) >> TAG_SHIFT; + case BGNUM: + if (mp_in_double_intptr_range(mp(n))) { + double_intptr_t out; + mp_get_double_intptr(mp(n), &out); + return out; + } + uw_throwf(error_s, lit("~s is out of signed ~a bit range"), + n, num_fast(SIZEOF_DOUBLE_INTPTR * CHAR_BIT), nao); + default: + type_mismatch(lit("~s is not an integer"), n, nao); + } +} + +dbl_ucnum c_dbl_unum(val n) +{ + switch (type(n)) { + case CHR: case NUM: + return coerce(cnum, n) >> TAG_SHIFT; + case BGNUM: + if (mp_in_double_uintptr_range(mp(n))) { + double_uintptr_t out; + mp_get_double_uintptr(mp(n), &out); + return out; + } + uw_throwf(error_s, lit("~s is out of unsigned ~a bit range"), + n, num_fast(SIZEOF_DOUBLE_INTPTR * CHAR_BIT), nao); + default: + type_mismatch(lit("~s is not an integer"), n, nao); + } +} + +#endif + val bignum_len(val num) { switch (type(num)) { |