diff options
author | Bruno Haible <bruno@clisp.org> | 2007-10-04 03:22:14 +0200 |
---|---|---|
committer | Bruno Haible <bruno@clisp.org> | 2007-10-04 03:22:14 +0200 |
commit | ec0af5d579654eb83aafc76940412b3bc106d558 (patch) | |
tree | 0a2d1c7647928c6f83b8baba9d9a174858628589 /lib | |
parent | 606dd130bdce412935eca78598923720a7e346c4 (diff) | |
download | gnulib-ec0af5d579654eb83aafc76940412b3bc106d558.tar.gz |
New module 'trunc'.
Diffstat (limited to 'lib')
-rw-r--r-- | lib/math.in.h | 13 | ||||
-rw-r--r-- | lib/trunc.c | 68 |
2 files changed, 81 insertions, 0 deletions
diff --git a/lib/math.in.h b/lib/math.in.h index 5138c95c7c..2b6c11be15 100644 --- a/lib/math.in.h +++ b/lib/math.in.h @@ -211,6 +211,19 @@ extern long double tanl (long double x); tanl (x)) #endif +#if @GNULIB_TRUNC@ +# if !@HAVE_DECL_TRUNC@ +# define trunc rpl_trunc +extern double trunc (double x); +# endif +#elif defined GNULIB_POSIXCHECK +# undef trunc +# define trunc(x) \ + (GL_LINK_WARNING ("trunc is unportable - " \ + "use gnulib module trunc for portability"), \ + trunc (x)) +#endif + #if @GNULIB_SIGNBIT@ # if @REPLACE_SIGNBIT@ diff --git a/lib/trunc.c b/lib/trunc.c new file mode 100644 index 0000000000..b74021ee1f --- /dev/null +++ b/lib/trunc.c @@ -0,0 +1,68 @@ +/* Round towards zero. + Copyright (C) 2007 Free Software Foundation, Inc. + + This program 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 2, or (at your option) + any later version. + + This program 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. + + You should have received a copy of the GNU General Public License along + with this program; if not, write to the Free Software Foundation, + Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ + +/* Written by Bruno Haible <bruno@clisp.org>, 2007. */ + +#include <config.h> + +/* Specification. */ +#include <math.h> + +#include <float.h> + +/* 2^(DBL_MANT_DIG-1). */ +static const double TWO_MANT_DIG = + /* Assume DBL_MANT_DIG <= 4 * 31. + Use the identity + n = floor(n/4) + floor((n+1)/4) + floor((n+2)/4) + floor((n+3)/4). */ + (double) (1U << ((DBL_MANT_DIG - 1) / 4)) + * (double) (1U << ((DBL_MANT_DIG - 1 + 1) / 4)) + * (double) (1U << ((DBL_MANT_DIG - 1 + 2) / 4)) + * (double) (1U << ((DBL_MANT_DIG - 1 + 3) / 4)); + +double +trunc (double x) +{ + /* The use of 'volatile' guarantees that excess precision bits are dropped + at each addition step and before the following comparison at the caller's + site. It is necessary on x86 systems where double-floats are not IEEE + compliant by default, to avoid that the results become platform and compiler + option dependent. 'volatile' is a portable alternative to gcc's + -ffloat-store option. */ + volatile double y = x; + volatile double z = y; + + if (z > 0) + { + /* Round to the next integer (nearest or up or down, doesn't matter). */ + z += TWO_MANT_DIG; + z -= TWO_MANT_DIG; + /* Enforce rounding down. */ + if (z > y) + z -= 1.0; + } + else if (z < 0) + { + /* Round to the next integer (nearest or up or down, doesn't matter). */ + z -= TWO_MANT_DIG; + z += TWO_MANT_DIG; + /* Enforce rounding up. */ + if (z < y) + z += 1.0; + } + return z; +} |