xref: /freebsd/lib/msun/src/s_sin.c (revision 38662c9698c40b6b5adf72db4c3c0c4f5d62f6a0)
13a8617a8SJordan K. Hubbard /* @(#)s_sin.c 5.1 93/09/24 */
23a8617a8SJordan K. Hubbard /*
33a8617a8SJordan K. Hubbard  * ====================================================
43a8617a8SJordan K. Hubbard  * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
53a8617a8SJordan K. Hubbard  *
63a8617a8SJordan K. Hubbard  * Developed at SunPro, a Sun Microsystems, Inc. business.
73a8617a8SJordan K. Hubbard  * Permission to use, copy, modify, and distribute this
83a8617a8SJordan K. Hubbard  * software is freely granted, provided that this notice
93a8617a8SJordan K. Hubbard  * is preserved.
103a8617a8SJordan K. Hubbard  * ====================================================
113a8617a8SJordan K. Hubbard  */
123a8617a8SJordan K. Hubbard 
1338662c96SBruce Evans #include <sys/cdefs.h>
1438662c96SBruce Evans __FBSDID("$FreeBSD$");
153a8617a8SJordan K. Hubbard 
163a8617a8SJordan K. Hubbard /* sin(x)
173a8617a8SJordan K. Hubbard  * Return sine function of x.
183a8617a8SJordan K. Hubbard  *
193a8617a8SJordan K. Hubbard  * kernel function:
203a8617a8SJordan K. Hubbard  *	__kernel_sin		... sine function on [-pi/4,pi/4]
213a8617a8SJordan K. Hubbard  *	__kernel_cos		... cose function on [-pi/4,pi/4]
223a8617a8SJordan K. Hubbard  *	__ieee754_rem_pio2	... argument reduction routine
233a8617a8SJordan K. Hubbard  *
243a8617a8SJordan K. Hubbard  * Method.
253a8617a8SJordan K. Hubbard  *      Let S,C and T denote the sin, cos and tan respectively on
263a8617a8SJordan K. Hubbard  *	[-PI/4, +PI/4]. Reduce the argument x to y1+y2 = x-k*pi/2
273a8617a8SJordan K. Hubbard  *	in [-pi/4 , +pi/4], and let n = k mod 4.
283a8617a8SJordan K. Hubbard  *	We have
293a8617a8SJordan K. Hubbard  *
303a8617a8SJordan K. Hubbard  *          n        sin(x)      cos(x)        tan(x)
313a8617a8SJordan K. Hubbard  *     ----------------------------------------------------------
323a8617a8SJordan K. Hubbard  *	    0	       S	   C		 T
333a8617a8SJordan K. Hubbard  *	    1	       C	  -S		-1/T
343a8617a8SJordan K. Hubbard  *	    2	      -S	  -C		 T
353a8617a8SJordan K. Hubbard  *	    3	      -C	   S		-1/T
363a8617a8SJordan K. Hubbard  *     ----------------------------------------------------------
373a8617a8SJordan K. Hubbard  *
383a8617a8SJordan K. Hubbard  * Special cases:
393a8617a8SJordan K. Hubbard  *      Let trig be any of sin, cos, or tan.
403a8617a8SJordan K. Hubbard  *      trig(+-INF)  is NaN, with signals;
413a8617a8SJordan K. Hubbard  *      trig(NaN)    is that NaN;
423a8617a8SJordan K. Hubbard  *
433a8617a8SJordan K. Hubbard  * Accuracy:
443a8617a8SJordan K. Hubbard  *	TRIG(x) returns trig(x) nearly rounded
453a8617a8SJordan K. Hubbard  */
463a8617a8SJordan K. Hubbard 
478e77cc64SDavid Schultz #include <float.h>
488e77cc64SDavid Schultz 
493a8617a8SJordan K. Hubbard #include "math.h"
5038662c96SBruce Evans #define INLINE_REM_PIO2
513a8617a8SJordan K. Hubbard #include "math_private.h"
5238662c96SBruce Evans #include "e_rem_pio2.c"
533a8617a8SJordan K. Hubbard 
5459b19ff1SAlfred Perlstein double
553819e840SPeter Wemm sin(double x)
563a8617a8SJordan K. Hubbard {
573a8617a8SJordan K. Hubbard 	double y[2],z=0.0;
583a8617a8SJordan K. Hubbard 	int32_t n, ix;
593a8617a8SJordan K. Hubbard 
603a8617a8SJordan K. Hubbard     /* High word of x. */
613a8617a8SJordan K. Hubbard 	GET_HIGH_WORD(ix,x);
623a8617a8SJordan K. Hubbard 
633a8617a8SJordan K. Hubbard     /* |x| ~< pi/4 */
643a8617a8SJordan K. Hubbard 	ix &= 0x7fffffff;
654339c67cSBruce Evans 	if(ix <= 0x3fe921fb) {
664339c67cSBruce Evans 	    if(ix<0x3e400000)			/* |x| < 2**-27 */
674339c67cSBruce Evans 	       {if((int)x==0) return x;}	/* generate inexact */
684339c67cSBruce Evans 	    return __kernel_sin(x,z,0);
694339c67cSBruce Evans 	}
703a8617a8SJordan K. Hubbard 
713a8617a8SJordan K. Hubbard     /* sin(Inf or NaN) is NaN */
723a8617a8SJordan K. Hubbard 	else if (ix>=0x7ff00000) return x-x;
733a8617a8SJordan K. Hubbard 
743a8617a8SJordan K. Hubbard     /* argument reduction needed */
753a8617a8SJordan K. Hubbard 	else {
763a8617a8SJordan K. Hubbard 	    n = __ieee754_rem_pio2(x,y);
773a8617a8SJordan K. Hubbard 	    switch(n&3) {
783a8617a8SJordan K. Hubbard 		case 0: return  __kernel_sin(y[0],y[1],1);
793a8617a8SJordan K. Hubbard 		case 1: return  __kernel_cos(y[0],y[1]);
803a8617a8SJordan K. Hubbard 		case 2: return -__kernel_sin(y[0],y[1],1);
813a8617a8SJordan K. Hubbard 		default:
823a8617a8SJordan K. Hubbard 			return -__kernel_cos(y[0],y[1]);
833a8617a8SJordan K. Hubbard 	    }
843a8617a8SJordan K. Hubbard 	}
853a8617a8SJordan K. Hubbard }
868e77cc64SDavid Schultz 
878e77cc64SDavid Schultz #if (LDBL_MANT_DIG == 53)
888e77cc64SDavid Schultz __weak_reference(sin, sinl);
898e77cc64SDavid Schultz #endif
90