Transcendental functions for 'long double', from Paolo Bonzini.
[gnulib.git] / lib / ldexpl.c
1 /* Emulation for ldexpl.
2    Contributed by Paolo Bonzini
3
4    Copyright 2002, 2003 Free Software Foundation, Inc.
5
6    This file is part of gnulib.
7
8    gnulib is free software; you can redistribute it and/or modify it
9    under the terms of the GNU Lesser General Public License as published
10    by the Free Software Foundation; either version 2.1, or (at your option)
11    any later version.
12
13    gnulib is distributed in the hope that it will be useful, but WITHOUT
14    ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
15    FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
16    License for more details.
17
18    You should have received a copy of the GNU Lesser General Public License
19    along with gnulib; see the file COPYING.LIB.  If not, write to the Free
20    Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307,
21    USA.
22  */
23
24 #include <float.h>
25 #include <math.h>
26
27 #include "mathl.h"
28
29 long double
30 ldexpl(long double x, int exp)
31 {
32   long double factor;
33   int bit;
34
35   /* Check for zero, nan and infinity. */
36   if (x != x || x + x == x )
37     return x;
38
39   if (exp < 0)
40     {
41       exp = -exp;
42       factor = 0.5L;
43     }
44   else
45     factor = 2.0L;
46
47   for (bit = 1; bit <= exp; bit <<= 1, factor *= factor)
48     if (exp & bit)
49       x *= factor;
50
51   return x;
52 }
53
54 #if 0
55 int
56 main()
57 {
58   long double x;
59   int y;
60   for (y = 0; y < 29; y++)
61     printf ("%5d %.16Lg %.16Lg\n", y, ldexpl(0.8L, y), ldexpl(0.8L, -y) * ldexpl(0.8L, y));
62 }
63 #endif