1ec601bf9f
/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 * Makefile.def: Add libquadmath; build it with language=fortran. * configure.ac: Add libquadmath. * Makefile.tpl: Handle multiple libs in check-[+language+]. * Makefile.in: Regenerate. * configure: Regenerate. libquadmath/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 Initial implementation and checkin. gcc/fortran/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 * gfortranspec.c (find_spec_file): New function. (lang_specific_driver): Try to find .spec file and use it. * trans-io.c (iocall): Define * IOCALL_X_REAL128/COMPLEX128(,write). (gfc_build_io_library_fndecls): Build decl for __float128 I/O. (transfer_expr): Call __float128 I/O functions. * trans-types.c (gfc_init_kinds): Allow kind-16 belonging to __float128. gcc/testsuite/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 * gfortran.dg/quad_1.f90: New. * lib/gcc-defs.exp (gcc-set-multilib-library-path): Use also compiler arguments. * lib/gfortran.exp (gfortran_link_flags): Add libquadmath to library search path; call gcc-set-multilib-library-path with arguments such that libgfortran.spec is found. (gfortran_init): Add path for libgfortran.spec to GFORTRAN_UNDER_TEST. libgomp/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 * configure.ac: * configure: Regenerate. libgfortran/ 2010-11-13 Francois-Xavier Coudert <fxcoudert@gcc.gnu.org> Tobias Burnus <burnus@net-b.de> PR fortran/32049 * Makefile.am: Add missing pow_r16_i4.c, add transfer128.c, link libquadmath, if used. * acinclude.m4 (LIBGFOR_CHECK_FLOAT128): Add. * configure.ac: Use it, touch spec file. * gfortran.map: Add pow_r16_i4 and transfer_(real,complex)128(,write) functions. * intrinsics/cshift0.c (cshift0): Handle __float128 type. * intrinsics/erfc_scaled_inc.c: Ditto. * intrinsics/pack_generic.c (pack): Ditto * intrinsics/spread_generic.c (spread): Ditto. * intrinsics/unpack_generic.c (unpack1): Ditto. * io/read.c (convert_real): Ditto. * io/transfer.c: Update comments. * io/transfer128.c: New file. * io/write_float.def (write_float): Handle __float128 type. * libgfortran.h: #include quadmath_weak.h, define __builtin_infq and nanq. * m4/mtype.m4: Handle __float128 type. * runtime/in_pack_generic.c (internal_pack): Ditto. * runtime/in_unpack_generic.c (internal_unpack): Ditto. * kinds-override.h: New file. * libgfortran.spec.in: Ditto. * generated/pow_r16_i4.c: Generated. * Makefile.in: Regenerate. * configure: Regenerate. * config.h: Regenerate. * bessel_r10.c: Regenerate. * bessel_r16.c: Regenerate. * bessel_r4.c: Regenerate. * bessel_r8.c: Regenerate. * exponent_r16.c: Regenerate. * fraction_r16.c: Regenerate. * nearest_r16.c: Regenerate. * norm2_r10.c: Regenerate. * norm2_r16.c: Regenerate. * norm2_r4.c: Regenerate. * norm2_r8.c: Regenerate. * rrspacing_r16.c: Regenerate. * set_exponent_r16.c: Regenerate. * spacing_r16.c: Regenerate. Co-Authored-By: Tobias Burnus <burnus@net-b.de> From-SVN: r166825
194 lines
5.2 KiB
C
194 lines
5.2 KiB
C
/* Implementation of the ERFC_SCALED intrinsic, to be included by erfc_scaled.c
|
|
Copyright (c) 2008 Free Software Foundation, Inc.
|
|
|
|
This file is part of the GNU Fortran runtime library (libgfortran).
|
|
|
|
Libgfortran is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 3 of the License, or (at your option) any later version.
|
|
|
|
Libgfortran is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR a PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
Under Section 7 of GPL version 3, you are granted additional
|
|
permissions described in the GCC Runtime Library Exception, version
|
|
3.1, as published by the Free Software Foundation.
|
|
|
|
You should have received a copy of the GNU General Public License and
|
|
a copy of the GCC Runtime Library Exception along with this program;
|
|
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
/* This implementation of ERFC_SCALED is based on the netlib algorithm
|
|
available at http://www.netlib.org/specfun/erf */
|
|
|
|
#define TYPE KIND_SUFFIX(GFC_REAL_,KIND)
|
|
#define CONCAT(x,y) x ## y
|
|
#define KIND_SUFFIX(x,y) CONCAT(x,y)
|
|
|
|
#if (KIND == 4)
|
|
|
|
# define EXP(x) expf(x)
|
|
# define TRUNC(x) truncf(x)
|
|
|
|
#elif (KIND == 8)
|
|
|
|
# define EXP(x) exp(x)
|
|
# define TRUNC(x) trunc(x)
|
|
|
|
#elif (KIND == 10) || (KIND == 16 && defined(GFC_REAL_16_IS_LONG_DOUBLE))
|
|
|
|
# ifdef HAVE_EXPL
|
|
# define EXP(x) expl(x)
|
|
# endif
|
|
# ifdef HAVE_TRUNCL
|
|
# define TRUNC(x) truncl(x)
|
|
# endif
|
|
|
|
#elif (KIND == 16 && defined(GFC_REAL_16_IS_FLOAT128))
|
|
|
|
# define EXP(x) expq(x)
|
|
# define TRUNC(x) truncq(x)
|
|
|
|
#else
|
|
|
|
# error "What exactly is it that you want me to do?"
|
|
|
|
#endif
|
|
|
|
#if defined(EXP) && defined(TRUNC)
|
|
|
|
extern TYPE KIND_SUFFIX(erfc_scaled_r,KIND) (TYPE);
|
|
export_proto(KIND_SUFFIX(erfc_scaled_r,KIND));
|
|
|
|
TYPE
|
|
KIND_SUFFIX(erfc_scaled_r,KIND) (TYPE x)
|
|
{
|
|
/* The main computation evaluates near-minimax approximations
|
|
from "Rational Chebyshev approximations for the error function"
|
|
by W. J. Cody, Math. Comp., 1969, PP. 631-638. This
|
|
transportable program uses rational functions that theoretically
|
|
approximate erf(x) and erfc(x) to at least 18 significant
|
|
decimal digits. The accuracy achieved depends on the arithmetic
|
|
system, the compiler, the intrinsic functions, and proper
|
|
selection of the machine-dependent constants. */
|
|
|
|
int i;
|
|
TYPE del, res, xden, xnum, y, ysq;
|
|
|
|
#if (KIND == 4)
|
|
static TYPE xneg = -9.382, xsmall = 5.96e-8,
|
|
xbig = 9.194, xhuge = 2.90e+3, xmax = 4.79e+37;
|
|
#else
|
|
static TYPE xneg = -26.628, xsmall = 1.11e-16,
|
|
xbig = 26.543, xhuge = 6.71e+7, xmax = 2.53e+307;
|
|
#endif
|
|
|
|
#define SQRPI ((TYPE) 0.56418958354775628695L)
|
|
#define THRESH ((TYPE) 0.46875L)
|
|
|
|
static TYPE a[5] = { 3.16112374387056560l, 113.864154151050156l,
|
|
377.485237685302021l, 3209.37758913846947l, 0.185777706184603153l };
|
|
|
|
static TYPE b[4] = { 23.6012909523441209l, 244.024637934444173l,
|
|
1282.61652607737228l, 2844.23683343917062l };
|
|
|
|
static TYPE c[9] = { 0.564188496988670089l, 8.88314979438837594l,
|
|
66.1191906371416295l, 298.635138197400131l, 881.952221241769090l,
|
|
1712.04761263407058l, 2051.07837782607147l, 1230.33935479799725l,
|
|
2.15311535474403846e-8l };
|
|
|
|
static TYPE d[8] = { 15.7449261107098347l, 117.693950891312499l,
|
|
537.181101862009858l, 1621.38957456669019l, 3290.79923573345963l,
|
|
4362.61909014324716l, 3439.36767414372164l, 1230.33935480374942l };
|
|
|
|
static TYPE p[6] = { 0.305326634961232344l, 0.360344899949804439l,
|
|
0.125781726111229246l, 0.0160837851487422766l,
|
|
0.000658749161529837803l, 0.0163153871373020978l };
|
|
|
|
static TYPE q[5] = { 2.56852019228982242l, 1.87295284992346047l,
|
|
0.527905102951428412l, 0.0605183413124413191l,
|
|
0.00233520497626869185l };
|
|
|
|
y = (x > 0 ? x : -x);
|
|
if (y <= THRESH)
|
|
{
|
|
ysq = 0;
|
|
if (y > xsmall)
|
|
ysq = y * y;
|
|
xnum = a[4]*ysq;
|
|
xden = ysq;
|
|
for (i = 0; i <= 2; i++)
|
|
{
|
|
xnum = (xnum + a[i]) * ysq;
|
|
xden = (xden + b[i]) * ysq;
|
|
}
|
|
res = x * (xnum + a[3]) / (xden + b[3]);
|
|
res = 1 - res;
|
|
res = EXP(ysq) * res;
|
|
return res;
|
|
}
|
|
else if (y <= 4)
|
|
{
|
|
xnum = c[8]*y;
|
|
xden = y;
|
|
for (i = 0; i <= 6; i++)
|
|
{
|
|
xnum = (xnum + c[i]) * y;
|
|
xden = (xden + d[i]) * y;
|
|
}
|
|
res = (xnum + c[7]) / (xden + d[7]);
|
|
}
|
|
else
|
|
{
|
|
res = 0;
|
|
if (y >= xbig)
|
|
{
|
|
if (y >= xmax)
|
|
goto finish;
|
|
if (y >= xhuge)
|
|
{
|
|
res = SQRPI / y;
|
|
goto finish;
|
|
}
|
|
}
|
|
ysq = ((TYPE) 1) / (y * y);
|
|
xnum = p[5]*ysq;
|
|
xden = ysq;
|
|
for (i = 0; i <= 3; i++)
|
|
{
|
|
xnum = (xnum + p[i]) * ysq;
|
|
xden = (xden + q[i]) * ysq;
|
|
}
|
|
res = ysq *(xnum + p[4]) / (xden + q[4]);
|
|
res = (SQRPI - res) / y;
|
|
}
|
|
|
|
finish:
|
|
if (x < 0)
|
|
{
|
|
if (x < xneg)
|
|
res = __builtin_inf ();
|
|
else
|
|
{
|
|
ysq = TRUNC (x*((TYPE) 16))/((TYPE) 16);
|
|
del = (x-ysq)*(x+ysq);
|
|
y = EXP(ysq*ysq) * EXP(del);
|
|
res = (y+y) - res;
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
#endif
|
|
|
|
#undef EXP
|
|
#undef TRUNC
|
|
|
|
#undef CONCAT
|
|
#undef TYPE
|
|
#undef KIND_SUFFIX
|