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