123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- TYPE
- SCM_TO_TYPE_PROTO (SCM val)
- {
- if (SCM_I_INUMP (val))
- {
- scm_t_signed_bits n = SCM_I_INUM (val);
- #if SIZEOF_TYPE != 0 && SIZEOF_TYPE > SIZEOF_UINTPTR_T
- return n;
- #else
- if (n >= TYPE_MIN && n <= TYPE_MAX)
- return n;
- else
- {
- goto out_of_range;
- }
- #endif
- }
- else if (SCM_BIGP (val))
- {
- if (TYPE_MIN >= SCM_MOST_NEGATIVE_FIXNUM
- && TYPE_MAX <= SCM_MOST_POSITIVE_FIXNUM)
- goto out_of_range;
- else if (TYPE_MIN >= LONG_MIN && TYPE_MAX <= LONG_MAX)
- {
- if (mpz_fits_slong_p (SCM_I_BIG_MPZ (val)))
- {
- long n = mpz_get_si (SCM_I_BIG_MPZ (val));
- #if SIZEOF_TYPE != 0 && SIZEOF_TYPE > SCM_SIZEOF_LONG
- return n;
- #else
- if (n >= TYPE_MIN && n <= TYPE_MAX)
- return n;
- else
- goto out_of_range;
- #endif
- }
- else
- goto out_of_range;
- }
- else
- {
- uintmax_t abs_n;
- TYPE n;
- size_t count;
- if (mpz_sizeinbase (SCM_I_BIG_MPZ (val), 2)
- > CHAR_BIT*sizeof (uintmax_t))
- goto out_of_range;
-
- mpz_export (&abs_n, &count, 1, sizeof (uintmax_t), 0, 0,
- SCM_I_BIG_MPZ (val));
- if (mpz_sgn (SCM_I_BIG_MPZ (val)) >= 0)
- {
- if (abs_n <= TYPE_MAX)
- n = abs_n;
- else
- goto out_of_range;
- }
- else
- {
-
- if (TYPE_MIN < 0 && abs_n - 1 <= -(TYPE_MIN + 1))
- n = -1 - (TYPE)(abs_n - 1);
- else
- goto out_of_range;
- }
- if (n >= TYPE_MIN && n <= TYPE_MAX)
- return n;
- else
- {
- out_of_range:
- scm_i_range_error (val,
- scm_from_signed_integer (TYPE_MIN),
- scm_from_signed_integer (TYPE_MAX));
- return 0;
- }
- }
- }
- else
- {
- scm_wrong_type_arg_msg (NULL, 0, val, "exact integer");
- return 0;
- }
- }
- SCM
- SCM_FROM_TYPE_PROTO (TYPE val)
- {
- #if SIZEOF_TYPE != 0 && SIZEOF_TYPE < SIZEOF_UINTPTR_T
- return SCM_I_MAKINUM (val);
- #else
- if (SCM_FIXABLE (val))
- return SCM_I_MAKINUM (val);
- else if (val >= LONG_MIN && val <= LONG_MAX)
- return scm_i_long2big (val);
- else
- {
- SCM z = make_bignum ();
- mpz_init (SCM_I_BIG_MPZ (z));
- if (val < 0)
- {
- val = -val;
- mpz_import (SCM_I_BIG_MPZ (z), 1, 1, sizeof (TYPE), 0, 0,
- &val);
- mpz_neg (SCM_I_BIG_MPZ (z), SCM_I_BIG_MPZ (z));
- }
- else
- mpz_import (SCM_I_BIG_MPZ (z), 1, 1, sizeof (TYPE), 0, 0,
- &val);
- return z;
- }
- #endif
- }
- /* clean up */
- #undef TYPE
- #undef TYPE_MIN
- #undef TYPE_MAX
- #undef SIZEOF_TYPE
- #undef SCM_TO_TYPE_PROTO
- #undef SCM_FROM_TYPE_PROTO
|