1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-06-13 23:20:32 +02:00

(scm_i_divide): For big/big wanting inexact, use mpq_get_d

rather than converting to doubles, to avoid inf or nan when the inputs
are too big for a double but the quotient does fit.  This affects
conversions exact->inexact of big fractions.
This commit is contained in:
Kevin Ryde 2006-05-09 00:25:11 +00:00
parent 101bd61cf5
commit 65581bc84d

View file

@ -4779,6 +4779,19 @@ scm_i_divide (SCM x, SCM y, int inexact)
else else
{ {
/* big_x / big_y */ /* big_x / big_y */
if (inexact)
{
/* It's easily possible for the ratio x/y to fit a double
but one or both x and y be too big to fit a double,
hence the use of mpq_get_d rather than converting and
dividing. */
mpq_t q;
*mpq_numref(q) = *SCM_I_BIG_MPZ (x);
*mpq_denref(q) = *SCM_I_BIG_MPZ (y);
return scm_from_double (mpq_get_d (q));
}
else
{
int divisible_p = mpz_divisible_p (SCM_I_BIG_MPZ (x), int divisible_p = mpz_divisible_p (SCM_I_BIG_MPZ (x),
SCM_I_BIG_MPZ (y)); SCM_I_BIG_MPZ (y));
if (divisible_p) if (divisible_p)
@ -4791,15 +4804,7 @@ scm_i_divide (SCM x, SCM y, int inexact)
return scm_i_normbig (result); return scm_i_normbig (result);
} }
else else
{ return scm_i_make_ratio (x, y);
if (inexact)
{
double dbx = mpz_get_d (SCM_I_BIG_MPZ (x));
double dby = mpz_get_d (SCM_I_BIG_MPZ (y));
scm_remember_upto_here_2 (x, y);
return scm_from_double (dbx / dby);
}
else return scm_i_make_ratio (x, y);
} }
} }
} }