xref: /freebsd/lib/msun/src/s_sin.c (revision 0dd5a5603e7a33d976f8e6015620bbc79839c609)
13a8617a8SJordan K. Hubbard /*
23a8617a8SJordan K. Hubbard  * ====================================================
33a8617a8SJordan K. Hubbard  * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
43a8617a8SJordan K. Hubbard  *
53a8617a8SJordan K. Hubbard  * Developed at SunPro, a Sun Microsystems, Inc. business.
63a8617a8SJordan K. Hubbard  * Permission to use, copy, modify, and distribute this
73a8617a8SJordan K. Hubbard  * software is freely granted, provided that this notice
83a8617a8SJordan K. Hubbard  * is preserved.
93a8617a8SJordan K. Hubbard  * ====================================================
103a8617a8SJordan K. Hubbard  */
113a8617a8SJordan K. Hubbard 
123a8617a8SJordan K. Hubbard /* sin(x)
133a8617a8SJordan K. Hubbard  * Return sine function of x.
143a8617a8SJordan K. Hubbard  *
153a8617a8SJordan K. Hubbard  * kernel function:
163a8617a8SJordan K. Hubbard  *	__kernel_sin		... sine function on [-pi/4,pi/4]
173a8617a8SJordan K. Hubbard  *	__kernel_cos		... cose function on [-pi/4,pi/4]
183a8617a8SJordan K. Hubbard  *	__ieee754_rem_pio2	... argument reduction routine
193a8617a8SJordan K. Hubbard  *
203a8617a8SJordan K. Hubbard  * Method.
213a8617a8SJordan K. Hubbard  *      Let S,C and T denote the sin, cos and tan respectively on
223a8617a8SJordan K. Hubbard  *	[-PI/4, +PI/4]. Reduce the argument x to y1+y2 = x-k*pi/2
233a8617a8SJordan K. Hubbard  *	in [-pi/4 , +pi/4], and let n = k mod 4.
243a8617a8SJordan K. Hubbard  *	We have
253a8617a8SJordan K. Hubbard  *
263a8617a8SJordan K. Hubbard  *          n        sin(x)      cos(x)        tan(x)
273a8617a8SJordan K. Hubbard  *     ----------------------------------------------------------
283a8617a8SJordan K. Hubbard  *	    0	       S	   C		 T
293a8617a8SJordan K. Hubbard  *	    1	       C	  -S		-1/T
303a8617a8SJordan K. Hubbard  *	    2	      -S	  -C		 T
313a8617a8SJordan K. Hubbard  *	    3	      -C	   S		-1/T
323a8617a8SJordan K. Hubbard  *     ----------------------------------------------------------
333a8617a8SJordan K. Hubbard  *
343a8617a8SJordan K. Hubbard  * Special cases:
353a8617a8SJordan K. Hubbard  *      Let trig be any of sin, cos, or tan.
363a8617a8SJordan K. Hubbard  *      trig(+-INF)  is NaN, with signals;
373a8617a8SJordan K. Hubbard  *      trig(NaN)    is that NaN;
383a8617a8SJordan K. Hubbard  *
393a8617a8SJordan K. Hubbard  * Accuracy:
403a8617a8SJordan K. Hubbard  *	TRIG(x) returns trig(x) nearly rounded
413a8617a8SJordan K. Hubbard  */
423a8617a8SJordan K. Hubbard 
438e77cc64SDavid Schultz #include <float.h>
448e77cc64SDavid Schultz 
453a8617a8SJordan K. Hubbard #include "math.h"
4638662c96SBruce Evans #define INLINE_REM_PIO2
473a8617a8SJordan K. Hubbard #include "math_private.h"
4838662c96SBruce Evans #include "e_rem_pio2.c"
493a8617a8SJordan K. Hubbard 
5059b19ff1SAlfred Perlstein double
sin(double x)513819e840SPeter Wemm sin(double x)
523a8617a8SJordan K. Hubbard {
533a8617a8SJordan K. Hubbard 	double y[2],z=0.0;
543a8617a8SJordan K. Hubbard 	int32_t n, ix;
553a8617a8SJordan K. Hubbard 
563a8617a8SJordan K. Hubbard     /* High word of x. */
573a8617a8SJordan K. Hubbard 	GET_HIGH_WORD(ix,x);
583a8617a8SJordan K. Hubbard 
593a8617a8SJordan K. Hubbard     /* |x| ~< pi/4 */
603a8617a8SJordan K. Hubbard 	ix &= 0x7fffffff;
614339c67cSBruce Evans 	if(ix <= 0x3fe921fb) {
62*e044d80dSDavid Schultz 	    if(ix<0x3e500000)			/* |x| < 2**-26 */
634339c67cSBruce Evans 	       {if((int)x==0) return x;}	/* generate inexact */
644339c67cSBruce Evans 	    return __kernel_sin(x,z,0);
654339c67cSBruce Evans 	}
663a8617a8SJordan K. Hubbard 
673a8617a8SJordan K. Hubbard     /* sin(Inf or NaN) is NaN */
683a8617a8SJordan K. Hubbard 	else if (ix>=0x7ff00000) return x-x;
693a8617a8SJordan K. Hubbard 
703a8617a8SJordan K. Hubbard     /* argument reduction needed */
713a8617a8SJordan K. Hubbard 	else {
723a8617a8SJordan K. Hubbard 	    n = __ieee754_rem_pio2(x,y);
733a8617a8SJordan K. Hubbard 	    switch(n&3) {
743a8617a8SJordan K. Hubbard 		case 0: return  __kernel_sin(y[0],y[1],1);
753a8617a8SJordan K. Hubbard 		case 1: return  __kernel_cos(y[0],y[1]);
763a8617a8SJordan K. Hubbard 		case 2: return -__kernel_sin(y[0],y[1],1);
773a8617a8SJordan K. Hubbard 		default:
783a8617a8SJordan K. Hubbard 			return -__kernel_cos(y[0],y[1]);
793a8617a8SJordan K. Hubbard 	    }
803a8617a8SJordan K. Hubbard 	}
813a8617a8SJordan K. Hubbard }
828e77cc64SDavid Schultz 
838e77cc64SDavid Schultz #if (LDBL_MANT_DIG == 53)
848e77cc64SDavid Schultz __weak_reference(sin, sinl);
858e77cc64SDavid Schultz #endif
86