xref: /inferno-os/libkern/sin.c (revision 37da2899f40661e3e9631e497da8dc59b971cbd0)
1*37da2899SCharles.Forsyth /*
2*37da2899SCharles.Forsyth 	C program for floating point sin/cos.
3*37da2899SCharles.Forsyth 	Calls modf.
4*37da2899SCharles.Forsyth 	There are no error exits.
5*37da2899SCharles.Forsyth 	Coefficients are #3370 from Hart & Cheney (18.80D).
6*37da2899SCharles.Forsyth */
7*37da2899SCharles.Forsyth 
8*37da2899SCharles.Forsyth #include <lib9.h>
9*37da2899SCharles.Forsyth 
10*37da2899SCharles.Forsyth #define p0      .1357884097877375669092680e8
11*37da2899SCharles.Forsyth #define p1     -.4942908100902844161158627e7
12*37da2899SCharles.Forsyth #define p2      .4401030535375266501944918e6
13*37da2899SCharles.Forsyth #define p3     -.1384727249982452873054457e5
14*37da2899SCharles.Forsyth #define p4      .1459688406665768722226959e3
15*37da2899SCharles.Forsyth #define q0      .8644558652922534429915149e7
16*37da2899SCharles.Forsyth #define q1      .4081792252343299749395779e6
17*37da2899SCharles.Forsyth #define q2      .9463096101538208180571257e4
18*37da2899SCharles.Forsyth #define q3      .1326534908786136358911494e3
19*37da2899SCharles.Forsyth 
20*37da2899SCharles.Forsyth static
21*37da2899SCharles.Forsyth double
sinus(double arg,int quad)22*37da2899SCharles.Forsyth sinus(double arg, int quad)
23*37da2899SCharles.Forsyth {
24*37da2899SCharles.Forsyth 	double e, f, ysq, x, y, temp1, temp2;
25*37da2899SCharles.Forsyth 	int k;
26*37da2899SCharles.Forsyth 
27*37da2899SCharles.Forsyth 	x = arg;
28*37da2899SCharles.Forsyth 	if(x < 0) {
29*37da2899SCharles.Forsyth 		x = -x;
30*37da2899SCharles.Forsyth 		quad += 2;
31*37da2899SCharles.Forsyth 	}
32*37da2899SCharles.Forsyth 	x *= 1/PIO2;	/* underflow? */
33*37da2899SCharles.Forsyth 	if(x > 32764) {
34*37da2899SCharles.Forsyth 		y = modf(x, &e);
35*37da2899SCharles.Forsyth 		e += quad;
36*37da2899SCharles.Forsyth 		modf(0.25*e, &f);
37*37da2899SCharles.Forsyth 		quad = e - 4*f;
38*37da2899SCharles.Forsyth 	} else {
39*37da2899SCharles.Forsyth 		k = x;
40*37da2899SCharles.Forsyth 		y = x - k;
41*37da2899SCharles.Forsyth 		quad += k;
42*37da2899SCharles.Forsyth 		quad &= 3;
43*37da2899SCharles.Forsyth 	}
44*37da2899SCharles.Forsyth 	if(quad & 1)
45*37da2899SCharles.Forsyth 		y = 1-y;
46*37da2899SCharles.Forsyth 	if(quad > 1)
47*37da2899SCharles.Forsyth 		y = -y;
48*37da2899SCharles.Forsyth 
49*37da2899SCharles.Forsyth 	ysq = y*y;
50*37da2899SCharles.Forsyth 	temp1 = ((((p4*ysq+p3)*ysq+p2)*ysq+p1)*ysq+p0)*y;
51*37da2899SCharles.Forsyth 	temp2 = ((((ysq+q3)*ysq+q2)*ysq+q1)*ysq+q0);
52*37da2899SCharles.Forsyth 	return temp1/temp2;
53*37da2899SCharles.Forsyth }
54*37da2899SCharles.Forsyth 
55*37da2899SCharles.Forsyth double
cos(double arg)56*37da2899SCharles.Forsyth cos(double arg)
57*37da2899SCharles.Forsyth {
58*37da2899SCharles.Forsyth 	if(arg < 0)
59*37da2899SCharles.Forsyth 		arg = -arg;
60*37da2899SCharles.Forsyth 	return sinus(arg, 1);
61*37da2899SCharles.Forsyth }
62*37da2899SCharles.Forsyth 
63*37da2899SCharles.Forsyth double
sin(double arg)64*37da2899SCharles.Forsyth sin(double arg)
65*37da2899SCharles.Forsyth {
66*37da2899SCharles.Forsyth 	return sinus(arg, 0);
67*37da2899SCharles.Forsyth }
68