ack/lang/cem/libcc.ansi/math/atan.c

78 lines
1.4 KiB
C
Raw Normal View History

1989-05-10 16:08:14 +00:00
/*
* (c) copyright 1988 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*
* Author: Ceriel J.H. Jacobs
*/
1994-06-24 14:02:31 +00:00
/* $Id$ */
1989-05-10 16:08:14 +00:00
2018-06-21 20:33:47 +00:00
#include <float.h>
#include <math.h>
#include <errno.h>
#include "localmath.h"
1989-05-10 16:08:14 +00:00
double
atan(double x)
{
/* Algorithm and coefficients from:
"Software manual for the elementary functions"
by W.J. Cody and W. Waite, Prentice-Hall, 1980
1989-05-10 16:08:14 +00:00
*/
static double p[] = {
-0.13688768894191926929e+2,
-0.20505855195861651981e+2,
-0.84946240351320683534e+1,
-0.83758299368150059274e+0
1989-05-10 16:08:14 +00:00
};
static double q[] = {
2018-06-21 20:33:47 +00:00
0.41066306682575781263e+2,
0.86157349597130242515e+2,
0.59578436142597344465e+2,
0.15024001160028576121e+2,
1.0
};
static double a[] = {
0.0,
2018-06-21 20:33:47 +00:00
0.52359877559829887307710723554658381, /* pi/6 */
M_PI_2,
2018-06-21 20:33:47 +00:00
1.04719755119659774615421446109316763 /* pi/3 */
1989-05-10 16:08:14 +00:00
};
2018-06-21 20:33:47 +00:00
int neg = x < 0;
int n;
double g;
1989-05-10 16:08:14 +00:00
2018-06-21 20:33:47 +00:00
if (__IsNan(x))
{
1991-03-19 16:39:40 +00:00
errno = EDOM;
return x;
}
2018-06-21 20:33:47 +00:00
if (neg)
{
1989-05-10 16:08:14 +00:00
x = -x;
}
2018-06-21 20:33:47 +00:00
if (x > 1.0)
{
x = 1.0 / x;
n = 2;
1989-05-10 16:08:14 +00:00
}
2018-06-21 20:33:47 +00:00
else
n = 0;
1989-05-10 16:08:14 +00:00
2018-06-21 20:33:47 +00:00
if (x > 0.26794919243112270647)
{ /* 2-sqtr(3) */
n = n + 1;
2018-06-21 20:33:47 +00:00
x = (((0.73205080756887729353 * x - 0.5) - 0.5) + x) / (1.73205080756887729353 + x);
1989-05-10 16:08:14 +00:00
}
/* ??? avoid underflow ??? */
g = x * x;
x += x * g * POLYNOM3(g, p) / POLYNOM4(g, q);
2018-06-21 20:33:47 +00:00
if (n > 1)
x = -x;
x += a[n];
return neg ? -x : x;
}