1996-03-05 21:41:30 +00:00
|
|
|
/* @(#)e_log10.c 5.1 93/09/24 */
|
|
|
|
/*
|
|
|
|
* ====================================================
|
|
|
|
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Developed at SunPro, a Sun Microsystems, Inc. business.
|
|
|
|
* Permission to use, copy, modify, and distribute this
|
1997-07-22 00:10:33 +00:00
|
|
|
* software is freely granted, provided that this notice
|
1996-03-05 21:41:30 +00:00
|
|
|
* is preserved.
|
|
|
|
* ====================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* __ieee754_log10(x)
|
|
|
|
* Return the base 10 logarithm of x
|
1997-07-22 00:10:33 +00:00
|
|
|
*
|
1996-03-05 21:41:30 +00:00
|
|
|
* Method :
|
|
|
|
* Let log10_2hi = leading 40 bits of log10(2) and
|
|
|
|
* log10_2lo = log10(2) - log10_2hi,
|
|
|
|
* ivln10 = 1/log(10) rounded.
|
|
|
|
* Then
|
1997-07-22 00:10:33 +00:00
|
|
|
* n = ilogb(x),
|
1996-03-05 21:41:30 +00:00
|
|
|
* if(n<0) n = n+1;
|
|
|
|
* x = scalbn(x,-n);
|
|
|
|
* log10(x) := n*log10_2hi + (n*log10_2lo + ivln10*log(x))
|
|
|
|
*
|
|
|
|
* Note 1:
|
1997-07-22 00:10:33 +00:00
|
|
|
* To guarantee log10(10**n)=n, where 10**n is normal, the rounding
|
1996-03-05 21:41:30 +00:00
|
|
|
* mode must set to Round-to-Nearest.
|
|
|
|
* Note 2:
|
|
|
|
* [1/log(10)] rounded to 53 bits has error .198 ulps;
|
|
|
|
* log10 is monotonic at all binary break points.
|
|
|
|
*
|
|
|
|
* Special cases:
|
1997-07-22 00:10:33 +00:00
|
|
|
* log10(x) is NaN with signal if x < 0;
|
1996-03-05 21:41:30 +00:00
|
|
|
* log10(+INF) is +INF with no signal; log10(0) is -INF with signal;
|
|
|
|
* log10(NaN) is that NaN with no signal;
|
|
|
|
* log10(10**N) = N for N=0,1,...,22.
|
|
|
|
*
|
|
|
|
* Constants:
|
|
|
|
* The hexadecimal values are the intended ones for the following constants.
|
|
|
|
* The decimal values may be used, provided that the compiler will convert
|
|
|
|
* from decimal to binary accurately enough to produce the hexadecimal values
|
|
|
|
* shown.
|
|
|
|
*/
|
|
|
|
|
2012-03-09 19:29:16 +00:00
|
|
|
#include <math.h>
|
Work around powerpc32 integer 0 converting to -0 (bug 887, bug 19049, bug 19050).
On powerpc32 hard-float, older processors (ones where fcfid is not
available for 32-bit code), GCC generates conversions from integers to
floating point that wrongly convert integer 0 to -0 instead of +0 in
FE_DOWNWARD mode. This in turn results in logb and a few other
functions wrongly returning -0 when they should return +0.
This patch works around this issue in glibc as I proposed in
<https://sourceware.org/ml/libc-alpha/2015-09/msg00728.html>, so that
the affected functions can be correct and the affected tests pass in
the absence of a GCC fix for this longstanding issue (GCC bug 67771 -
if fixed, of course we can put in GCC version conditionals, and
eventually phase out the workarounds). A new macro
FIX_INT_FP_CONVERT_ZERO is added in a new sysdeps header
fix-int-fp-convert-zero.h, and the powerpc32/fpu version of that
header defines the macro based on the results of a configure test for
whether such conversions use the fcfid instruction.
Tested for x86_64 (that installed stripped shared libraries are
unchanged by the patch) and powerpc (that HAVE_PPC_FCFID comes out to
0 as expected and that the relevant tests are fixed). Also tested a
build with GCC configured for -mcpu=power4 and verified that
HAVE_PPC_FCFID comes out to 1 in that case.
There are still some other issues to fix to get test-float and
test-double passing cleanly for older powerpc32 processors (apart from
the need for an ulps regeneration for powerpc). (test-ldouble will be
harder to get passing cleanly, but with a combination of selected
fixes to ldbl-128ibm code that don't involve significant performance
issues, allowing spurious underflow and inexact exceptions for that
format, and lots of XFAILing for the default case of unpatched libgcc,
it should be doable.)
[BZ #887]
[BZ #19049]
[BZ #19050]
* sysdeps/generic/fix-int-fp-convert-zero.h: New file.
* sysdeps/ieee754/dbl-64/e_log10.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log10): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/e_log2.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log2): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/s_erf.c: Include
<fix-int-fp-convert-zero.h>.
(__erfc): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/s_logb.c: Include
<fix-int-fp-convert-zero.h>.
(__logb): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/e_log10f.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log10f): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/e_log2f.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log2f): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/s_erff.c: Include
<fix-int-fp-convert-zero.h>.
(__erfcf): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/s_logbf.c: Include
<fix-int-fp-convert-zero.h>.
(__logbf): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/ldbl-128ibm/s_erfl.c: Include
<fix-int-fp-convert-zero.h>.
(__erfcl): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/ldbl-128ibm/s_logbl.c: Include
<fix-int-fp-convert-zero.h>.
(__logbl): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/powerpc/powerpc32/fpu/configure.ac: New file.
* sysdeps/powerpc/powerpc32/fpu/configure: New generated file.
* sysdeps/powerpc/powerpc32/fpu/fix-int-fp-convert-zero.h: New
file.
* config.h.in [_LIBC] (HAVE_PPC_FCFID): New macro.
2015-10-05 17:46:50 +00:00
|
|
|
#include <fix-int-fp-convert-zero.h>
|
2021-01-07 15:26:26 +00:00
|
|
|
#include <math_private.h>
|
|
|
|
#include <stdint.h>
|
2019-07-16 15:17:22 +00:00
|
|
|
#include <libm-alias-finite.h>
|
1996-03-05 21:41:30 +00:00
|
|
|
|
2021-01-07 15:26:26 +00:00
|
|
|
static const double two54 = 1.80143985094819840000e+16; /* 0x4350000000000000 */
|
|
|
|
static const double ivln10 = 4.34294481903251816668e-01; /* 0x3FDBCB7B1526E50E */
|
|
|
|
static const double log10_2hi = 3.01029995663611771306e-01; /* 0x3FD34413509F6000 */
|
|
|
|
static const double log10_2lo = 3.69423907715893078616e-13; /* 0x3D59FEF311F12B36 */
|
1996-03-05 21:41:30 +00:00
|
|
|
|
2011-10-12 15:27:51 +00:00
|
|
|
double
|
2012-05-14 19:49:42 +00:00
|
|
|
__ieee754_log10 (double x)
|
1996-03-05 21:41:30 +00:00
|
|
|
{
|
2012-05-14 19:49:42 +00:00
|
|
|
double y, z;
|
2021-01-07 15:26:26 +00:00
|
|
|
int64_t i, hx;
|
|
|
|
int32_t k;
|
1996-03-05 21:41:30 +00:00
|
|
|
|
2021-01-07 15:26:26 +00:00
|
|
|
EXTRACT_WORDS64 (hx, x);
|
1996-03-05 21:41:30 +00:00
|
|
|
|
2012-05-14 19:49:42 +00:00
|
|
|
k = 0;
|
2021-01-07 15:26:26 +00:00
|
|
|
if (hx < INT64_C(0x0010000000000000))
|
|
|
|
{ /* x < 2**-1022 */
|
|
|
|
if (__glibc_unlikely ((hx & UINT64_C(0x7fffffffffffffff)) == 0))
|
|
|
|
return -two54 / fabs (x); /* log(+-0)=-inf */
|
2014-02-10 13:45:42 +00:00
|
|
|
if (__glibc_unlikely (hx < 0))
|
2021-01-07 15:26:26 +00:00
|
|
|
return (x - x) / (x - x); /* log(-#) = NaN */
|
2012-05-14 19:49:42 +00:00
|
|
|
k -= 54;
|
2021-01-07 15:26:26 +00:00
|
|
|
x *= two54; /* subnormal number, scale up x */
|
|
|
|
EXTRACT_WORDS64 (hx, x);
|
2012-05-14 19:49:42 +00:00
|
|
|
}
|
2021-01-07 15:26:26 +00:00
|
|
|
/* scale up resulted in a NaN number */
|
|
|
|
if (__glibc_unlikely (hx >= UINT64_C(0x7ff0000000000000)))
|
2012-05-14 19:49:42 +00:00
|
|
|
return x + x;
|
2021-01-07 15:26:26 +00:00
|
|
|
k += (hx >> 52) - 1023;
|
|
|
|
i = ((uint64_t) k & UINT64_C(0x8000000000000000)) >> 63;
|
|
|
|
hx = (hx & UINT64_C(0x000fffffffffffff)) | ((0x3ff - i) << 52);
|
2012-05-14 19:49:42 +00:00
|
|
|
y = (double) (k + i);
|
Work around powerpc32 integer 0 converting to -0 (bug 887, bug 19049, bug 19050).
On powerpc32 hard-float, older processors (ones where fcfid is not
available for 32-bit code), GCC generates conversions from integers to
floating point that wrongly convert integer 0 to -0 instead of +0 in
FE_DOWNWARD mode. This in turn results in logb and a few other
functions wrongly returning -0 when they should return +0.
This patch works around this issue in glibc as I proposed in
<https://sourceware.org/ml/libc-alpha/2015-09/msg00728.html>, so that
the affected functions can be correct and the affected tests pass in
the absence of a GCC fix for this longstanding issue (GCC bug 67771 -
if fixed, of course we can put in GCC version conditionals, and
eventually phase out the workarounds). A new macro
FIX_INT_FP_CONVERT_ZERO is added in a new sysdeps header
fix-int-fp-convert-zero.h, and the powerpc32/fpu version of that
header defines the macro based on the results of a configure test for
whether such conversions use the fcfid instruction.
Tested for x86_64 (that installed stripped shared libraries are
unchanged by the patch) and powerpc (that HAVE_PPC_FCFID comes out to
0 as expected and that the relevant tests are fixed). Also tested a
build with GCC configured for -mcpu=power4 and verified that
HAVE_PPC_FCFID comes out to 1 in that case.
There are still some other issues to fix to get test-float and
test-double passing cleanly for older powerpc32 processors (apart from
the need for an ulps regeneration for powerpc). (test-ldouble will be
harder to get passing cleanly, but with a combination of selected
fixes to ldbl-128ibm code that don't involve significant performance
issues, allowing spurious underflow and inexact exceptions for that
format, and lots of XFAILing for the default case of unpatched libgcc,
it should be doable.)
[BZ #887]
[BZ #19049]
[BZ #19050]
* sysdeps/generic/fix-int-fp-convert-zero.h: New file.
* sysdeps/ieee754/dbl-64/e_log10.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log10): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/e_log2.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log2): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/s_erf.c: Include
<fix-int-fp-convert-zero.h>.
(__erfc): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/dbl-64/s_logb.c: Include
<fix-int-fp-convert-zero.h>.
(__logb): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/e_log10f.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log10f): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/e_log2f.c: Include
<fix-int-fp-convert-zero.h>.
(__ieee754_log2f): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/s_erff.c: Include
<fix-int-fp-convert-zero.h>.
(__erfcf): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/flt-32/s_logbf.c: Include
<fix-int-fp-convert-zero.h>.
(__logbf): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/ldbl-128ibm/s_erfl.c: Include
<fix-int-fp-convert-zero.h>.
(__erfcl): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/ieee754/ldbl-128ibm/s_logbl.c: Include
<fix-int-fp-convert-zero.h>.
(__logbl): Adjust signs as needed if FIX_INT_FP_CONVERT_ZERO.
* sysdeps/powerpc/powerpc32/fpu/configure.ac: New file.
* sysdeps/powerpc/powerpc32/fpu/configure: New generated file.
* sysdeps/powerpc/powerpc32/fpu/fix-int-fp-convert-zero.h: New
file.
* config.h.in [_LIBC] (HAVE_PPC_FCFID): New macro.
2015-10-05 17:46:50 +00:00
|
|
|
if (FIX_INT_FP_CONVERT_ZERO && y == 0.0)
|
|
|
|
y = 0.0;
|
2021-01-07 15:26:26 +00:00
|
|
|
INSERT_WORDS64 (x, hx);
|
2012-05-14 19:49:42 +00:00
|
|
|
z = y * log10_2lo + ivln10 * __ieee754_log (x);
|
|
|
|
return z + y * log10_2hi;
|
1996-03-05 21:41:30 +00:00
|
|
|
}
|
2019-07-16 15:17:22 +00:00
|
|
|
libm_alias_finite (__ieee754_log10, __log10)
|