mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-05-01 12:20:26 +02:00
num2integral.i.c. MAX_VALUE and MIN_VALU are no longer used, we now rely on SIZEOF_ macros that have been figured out at configure time. * num2integral.i.c: Adapt to new interface. (NUM2INTEGRAL): Test whether a fixnum can be represented in the target type by casting it and checking whether it is still the same. Do not try to handle bignums for integral types that are smaller than fixnums. When handling bignums, collect the magnituse first into a unsigned type, and correctly check for overflow. (INTEGRAL2BIG): Do not use MIN_VALUE explicitely by observing that only -MIN_VALUE can still be negative of all negative numbers (in twos-complement).
188 lines
3.8 KiB
C
188 lines
3.8 KiB
C
/* this file is #include'd (many times) by numbers.c */
|
|
|
|
#ifndef UNSIGNED_ITYPE
|
|
#ifdef UNSIGNED
|
|
#define UNSIGNED_ITYPE ITYPE
|
|
#else
|
|
#define UNSIGNED_ITYPE unsigned ITYPE
|
|
#endif
|
|
#endif
|
|
|
|
#define UNSIGNED_ITYPE_MAX (~((UNSIGNED_ITYPE)0))
|
|
|
|
#ifndef SIZEOF_ITYPE
|
|
#define SIZEOF_ITYPE (2*SIZEOF_SCM_T_BITS)
|
|
#endif
|
|
|
|
ITYPE
|
|
NUM2INTEGRAL (SCM num, unsigned long int pos, const char *s_caller)
|
|
{
|
|
if (SCM_INUMP (num))
|
|
{ /* immediate */
|
|
|
|
scm_t_signed_bits n = SCM_INUM (num);
|
|
|
|
#ifdef UNSIGNED
|
|
if (n < 0)
|
|
scm_out_of_range (s_caller, num);
|
|
#endif
|
|
|
|
if (sizeof (ITYPE) >= sizeof (scm_t_signed_bits))
|
|
/* can't fit anything too big for this type in an inum
|
|
anyway */
|
|
return (ITYPE) n;
|
|
else
|
|
{ /* an inum can be out of range, so check */
|
|
if (((ITYPE)n) != n)
|
|
scm_out_of_range (s_caller, num);
|
|
else
|
|
return (ITYPE) n;
|
|
}
|
|
}
|
|
else if (SCM_BIGP (num))
|
|
{ /* bignum */
|
|
#if SIZEOF_ITYPE >= SIZEOF_SCM_T_BITS
|
|
|
|
UNSIGNED_ITYPE pos_res = 0;
|
|
size_t l;
|
|
|
|
#ifdef UNSIGNED
|
|
if (SCM_BIGSIGN (num))
|
|
scm_out_of_range (s_caller, num);
|
|
#endif
|
|
|
|
for (l = SCM_NUMDIGS (num); l--;)
|
|
{
|
|
if (pos_res > SCM_BIGDN (UNSIGNED_ITYPE_MAX))
|
|
scm_out_of_range (s_caller, num);
|
|
pos_res = SCM_I_BIGUP (ITYPE, pos_res) + SCM_BDIGITS (num)[l];
|
|
}
|
|
|
|
#ifdef UNSIGNED
|
|
return pos_res;
|
|
#else
|
|
if (SCM_BIGSIGN (num))
|
|
{
|
|
ITYPE res = -((ITYPE)pos_res);
|
|
if (res <= 0)
|
|
return res;
|
|
else
|
|
scm_out_of_range (s_caller, num);
|
|
}
|
|
else
|
|
{
|
|
ITYPE res = (ITYPE)pos_res;
|
|
if (res >= 0)
|
|
return res;
|
|
else
|
|
scm_out_of_range (s_caller, num);
|
|
}
|
|
#endif
|
|
|
|
#else /* SIZEOF_ITYPE >= SIZEOF_SCM_T_BITS */
|
|
scm_out_of_range (s_caller, num);
|
|
#endif
|
|
|
|
}
|
|
else
|
|
scm_wrong_type_arg (s_caller, pos, num);
|
|
}
|
|
|
|
SCM
|
|
INTEGRAL2NUM (ITYPE n)
|
|
{
|
|
/* If we know the size of the type, determine at compile time
|
|
whether we need to perform the FIXABLE test or not. This is not
|
|
done to get more optimal code out of the compiler (it can figure
|
|
this out on its own already), but to avoid a spurious warning.
|
|
If we don't know the size, assume that the test must be done.
|
|
*/
|
|
|
|
#if SIZEOF_ITYPE >= SIZEOF_SCM_T_BITS
|
|
#ifndef UNSIGNED
|
|
if (SCM_FIXABLE (n))
|
|
#else
|
|
if (SCM_POSFIXABLE (n))
|
|
#endif
|
|
#endif
|
|
return SCM_MAKINUM ((scm_t_signed_bits) n);
|
|
|
|
#ifdef SCM_BIGDIG
|
|
return INTEGRAL2BIG (n);
|
|
#else
|
|
return scm_make_real ((double) n);
|
|
#endif
|
|
}
|
|
|
|
#ifdef SCM_BIGDIG
|
|
|
|
SCM
|
|
INTEGRAL2BIG (ITYPE n)
|
|
{
|
|
SCM res;
|
|
int neg_p;
|
|
unsigned int n_digits;
|
|
size_t i;
|
|
SCM_BIGDIG *digits;
|
|
|
|
#ifndef UNSIGNED
|
|
neg_p = (n < 0);
|
|
if (neg_p) n = -n;
|
|
#else
|
|
neg_p = 0;
|
|
#endif
|
|
|
|
#ifndef UNSIGNED
|
|
/* If n is still negative here, it must be the minimum value of the
|
|
type (assuming twos-complement, but we are tied to that anyway).
|
|
If this is the case, we can not count the number of digits by
|
|
right-shifting n until it is zero.
|
|
*/
|
|
if (n < 0)
|
|
{
|
|
/* special case */
|
|
n_digits =
|
|
(sizeof (ITYPE) + sizeof (SCM_BIGDIG) - 1) / sizeof (SCM_BIGDIG);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
ITYPE tn;
|
|
for (tn = n, n_digits = 0;
|
|
tn;
|
|
++n_digits, tn = SCM_BIGDN (tn))
|
|
;
|
|
}
|
|
|
|
i = 0;
|
|
res = scm_i_mkbig (n_digits, neg_p);
|
|
digits = SCM_BDIGITS (res);
|
|
|
|
while (i < n_digits)
|
|
{
|
|
digits[i++] = SCM_BIGLO (n);
|
|
n = SCM_BIGDN (n);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
#endif
|
|
|
|
/* clean up */
|
|
#undef INTEGRAL2NUM
|
|
#undef INTEGRAL2BIG
|
|
#undef NUM2INTEGRAL
|
|
#ifdef UNSIGNED
|
|
#undef UNSIGNED
|
|
#endif
|
|
#undef ITYPE
|
|
#undef SIZEOF_ITYPE
|
|
#undef UNSIGNED_ITYPE
|
|
#undef UNSIGNED_ITYPE_MAX
|
|
|
|
/*
|
|
Local Variables:
|
|
c-file-style: "gnu"
|
|
End:
|
|
*/
|