Blame sysdeps/ieee754/dbl-64/sincos32.c

Packit Service 82fcde
/*
Packit Service 82fcde
 * IBM Accurate Mathematical Library
Packit Service 82fcde
 * written by International Business Machines Corp.
Packit Service 82fcde
 * Copyright (C) 2001-2018 Free Software Foundation, Inc.
Packit Service 82fcde
 *
Packit Service 82fcde
 * This program is free software; you can redistribute it and/or modify
Packit Service 82fcde
 * it under the terms of the GNU Lesser General Public License as published by
Packit Service 82fcde
 * the Free Software Foundation; either version 2.1 of the License, or
Packit Service 82fcde
 * (at your option) any later version.
Packit Service 82fcde
 *
Packit Service 82fcde
 * This program is distributed in the hope that it will be useful,
Packit Service 82fcde
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 82fcde
 * GNU Lesser General Public License for more details.
Packit Service 82fcde
 *
Packit Service 82fcde
 * You should have received a copy of the GNU Lesser General Public License
Packit Service 82fcde
 * along with this program; if not, see <http://www.gnu.org/licenses/>.
Packit Service 82fcde
 */
Packit Service 82fcde
/****************************************************************/
Packit Service 82fcde
/*  MODULE_NAME: sincos32.c                                     */
Packit Service 82fcde
/*                                                              */
Packit Service 82fcde
/*  FUNCTIONS: ss32                                             */
Packit Service 82fcde
/*             cc32                                             */
Packit Service 82fcde
/*             c32                                              */
Packit Service 82fcde
/*             sin32                                            */
Packit Service 82fcde
/*             cos32                                            */
Packit Service 82fcde
/*             mpsin                                            */
Packit Service 82fcde
/*             mpcos                                            */
Packit Service 82fcde
/*             mpranred                                         */
Packit Service 82fcde
/*             mpsin1                                           */
Packit Service 82fcde
/*             mpcos1                                           */
Packit Service 82fcde
/*                                                              */
Packit Service 82fcde
/* FILES NEEDED: endian.h mpa.h sincos32.h                      */
Packit Service 82fcde
/*               mpa.c                                          */
Packit Service 82fcde
/*                                                              */
Packit Service 82fcde
/* Multi Precision sin() and cos() function with p=32  for sin()*/
Packit Service 82fcde
/* cos() arcsin() and arccos() routines                         */
Packit Service 82fcde
/* In addition mpranred() routine  performs range  reduction of */
Packit Service 82fcde
/* a double number x into multi precision number   y,           */
Packit Service 82fcde
/* such that y=x-n*pi/2, abs(y)
Packit Service 82fcde
/****************************************************************/
Packit Service 82fcde
#include "endian.h"
Packit Service 82fcde
#include "mpa.h"
Packit Service 82fcde
#include "sincos32.h"
Packit Service 82fcde
#include <math.h>
Packit Service 82fcde
#include <math_private.h>
Packit Service 82fcde
#include <stap-probe.h>
Packit Service 82fcde
Packit Service 82fcde
#ifndef SECTION
Packit Service 82fcde
# define SECTION
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
/* Compute Multi-Precision sin() function for given p.  Receive Multi Precision
Packit Service 82fcde
   number x and result stored at y.  */
Packit Service 82fcde
static void
Packit Service 82fcde
SECTION
Packit Service 82fcde
ss32 (mp_no *x, mp_no *y, int p)
Packit Service 82fcde
{
Packit Service 82fcde
  int i;
Packit Service 82fcde
  double a;
Packit Service 82fcde
  mp_no mpt1, x2, gor, sum, mpk = {1, {1.0}};
Packit Service 82fcde
  for (i = 1; i <= p; i++)
Packit Service 82fcde
    mpk.d[i] = 0;
Packit Service 82fcde
Packit Service 82fcde
  __sqr (x, &x2, p);
Packit Service 82fcde
  __cpy (&oofac27, &gor, p);
Packit Service 82fcde
  __cpy (&gor, &sum, p);
Packit Service 82fcde
  for (a = 27.0; a > 1.0; a -= 2.0)
Packit Service 82fcde
    {
Packit Service 82fcde
      mpk.d[1] = a * (a - 1.0);
Packit Service 82fcde
      __mul (&gor, &mpk, &mpt1, p);
Packit Service 82fcde
      __cpy (&mpt1, &gor, p);
Packit Service 82fcde
      __mul (&x2, &sum, &mpt1, p);
Packit Service 82fcde
      __sub (&gor, &mpt1, &sum, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  __mul (x, &sum, y, p);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Compute Multi-Precision cos() function for given p. Receive Multi Precision
Packit Service 82fcde
   number x and result stored at y.  */
Packit Service 82fcde
static void
Packit Service 82fcde
SECTION
Packit Service 82fcde
cc32 (mp_no *x, mp_no *y, int p)
Packit Service 82fcde
{
Packit Service 82fcde
  int i;
Packit Service 82fcde
  double a;
Packit Service 82fcde
  mp_no mpt1, x2, gor, sum, mpk = {1, {1.0}};
Packit Service 82fcde
  for (i = 1; i <= p; i++)
Packit Service 82fcde
    mpk.d[i] = 0;
Packit Service 82fcde
Packit Service 82fcde
  __sqr (x, &x2, p);
Packit Service 82fcde
  mpk.d[1] = 27.0;
Packit Service 82fcde
  __mul (&oofac27, &mpk, &gor, p);
Packit Service 82fcde
  __cpy (&gor, &sum, p);
Packit Service 82fcde
  for (a = 26.0; a > 2.0; a -= 2.0)
Packit Service 82fcde
    {
Packit Service 82fcde
      mpk.d[1] = a * (a - 1.0);
Packit Service 82fcde
      __mul (&gor, &mpk, &mpt1, p);
Packit Service 82fcde
      __cpy (&mpt1, &gor, p);
Packit Service 82fcde
      __mul (&x2, &sum, &mpt1, p);
Packit Service 82fcde
      __sub (&gor, &mpt1, &sum, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  __mul (&x2, &sum, y, p);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Compute both sin(x), cos(x) as Multi precision numbers.  */
Packit Service 82fcde
void
Packit Service 82fcde
SECTION
Packit Service 82fcde
__c32 (mp_no *x, mp_no *y, mp_no *z, int p)
Packit Service 82fcde
{
Packit Service 82fcde
  mp_no u, t, t1, t2, c, s;
Packit Service 82fcde
  int i;
Packit Service 82fcde
  __cpy (x, &u, p);
Packit Service 82fcde
  u.e = u.e - 1;
Packit Service 82fcde
  cc32 (&u, &c, p);
Packit Service 82fcde
  ss32 (&u, &s, p);
Packit Service 82fcde
  for (i = 0; i < 24; i++)
Packit Service 82fcde
    {
Packit Service 82fcde
      __mul (&c, &s, &t, p);
Packit Service 82fcde
      __sub (&s, &t, &t1, p);
Packit Service 82fcde
      __add (&t1, &t1, &s, p);
Packit Service 82fcde
      __sub (&__mptwo, &c, &t1, p);
Packit Service 82fcde
      __mul (&t1, &c, &t2, p);
Packit Service 82fcde
      __add (&t2, &t2, &c, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  __sub (&__mpone, &c, y, p);
Packit Service 82fcde
  __cpy (&s, z, p);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Receive double x and two double results of sin(x) and return result which is
Packit Service 82fcde
   more accurate, computing sin(x) with multi precision routine c32.  */
Packit Service 82fcde
double
Packit Service 82fcde
SECTION
Packit Service 82fcde
__sin32 (double x, double res, double res1)
Packit Service 82fcde
{
Packit Service 82fcde
  int p;
Packit Service 82fcde
  mp_no a, b, c;
Packit Service 82fcde
  p = 32;
Packit Service 82fcde
  __dbl_mp (res, &a, p);
Packit Service 82fcde
  __dbl_mp (0.5 * (res1 - res), &b, p);
Packit Service 82fcde
  __add (&a, &b, &c, p);
Packit Service 82fcde
  if (x > 0.8)
Packit Service 82fcde
    {
Packit Service 82fcde
      __sub (&hp, &c, &a, p);
Packit Service 82fcde
      __c32 (&a, &b, &c, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    __c32 (&c, &a, &b, p);	/* b=sin(0.5*(res+res1))  */
Packit Service 82fcde
  __dbl_mp (x, &c, p);		/* c = x  */
Packit Service 82fcde
  __sub (&b, &c, &a, p);
Packit Service 82fcde
  /* if a > 0 return min (res, res1), otherwise return max (res, res1).  */
Packit Service 82fcde
  if ((a.d[0] > 0 && res >= res1) || (a.d[0] <= 0 && res <= res1))
Packit Service 82fcde
    res = res1;
Packit Service 82fcde
  LIBC_PROBE (slowasin, 2, &res, &x);
Packit Service 82fcde
  return res;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Receive double x and two double results of cos(x) and return result which is
Packit Service 82fcde
   more accurate, computing cos(x) with multi precision routine c32.  */
Packit Service 82fcde
double
Packit Service 82fcde
SECTION
Packit Service 82fcde
__cos32 (double x, double res, double res1)
Packit Service 82fcde
{
Packit Service 82fcde
  int p;
Packit Service 82fcde
  mp_no a, b, c;
Packit Service 82fcde
  p = 32;
Packit Service 82fcde
  __dbl_mp (res, &a, p);
Packit Service 82fcde
  __dbl_mp (0.5 * (res1 - res), &b, p);
Packit Service 82fcde
  __add (&a, &b, &c, p);
Packit Service 82fcde
  if (x > 2.4)
Packit Service 82fcde
    {
Packit Service 82fcde
      __sub (&pi, &c, &a, p);
Packit Service 82fcde
      __c32 (&a, &b, &c, p);
Packit Service 82fcde
      b.d[0] = -b.d[0];
Packit Service 82fcde
    }
Packit Service 82fcde
  else if (x > 0.8)
Packit Service 82fcde
    {
Packit Service 82fcde
      __sub (&hp, &c, &a, p);
Packit Service 82fcde
      __c32 (&a, &c, &b, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    __c32 (&c, &b, &a, p);	/* b=cos(0.5*(res+res1))  */
Packit Service 82fcde
  __dbl_mp (x, &c, p);		/* c = x                  */
Packit Service 82fcde
  __sub (&b, &c, &a, p);
Packit Service 82fcde
  /* if a > 0 return max (res, res1), otherwise return min (res, res1).  */
Packit Service 82fcde
  if ((a.d[0] > 0 && res <= res1) || (a.d[0] <= 0 && res >= res1))
Packit Service 82fcde
    res = res1;
Packit Service 82fcde
  LIBC_PROBE (slowacos, 2, &res, &x);
Packit Service 82fcde
  return res;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Compute sin() of double-length number (X + DX) as Multi Precision number and
Packit Service 82fcde
   return result as double.  If REDUCE_RANGE is true, X is assumed to be the
Packit Service 82fcde
   original input and DX is ignored.  */
Packit Service 82fcde
double
Packit Service 82fcde
SECTION
Packit Service 82fcde
__mpsin (double x, double dx, bool reduce_range)
Packit Service 82fcde
{
Packit Service 82fcde
  double y;
Packit Service 82fcde
  mp_no a, b, c, s;
Packit Service 82fcde
  int n;
Packit Service 82fcde
  int p = 32;
Packit Service 82fcde
Packit Service 82fcde
  if (reduce_range)
Packit Service 82fcde
    {
Packit Service 82fcde
      n = __mpranred (x, &a, p);	/* n is 0, 1, 2 or 3.  */
Packit Service 82fcde
      __c32 (&a, &c, &s, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      n = -1;
Packit Service 82fcde
      __dbl_mp (x, &b, p);
Packit Service 82fcde
      __dbl_mp (dx, &c, p);
Packit Service 82fcde
      __add (&b, &c, &a, p);
Packit Service 82fcde
      if (x > 0.8)
Packit Service 82fcde
        {
Packit Service 82fcde
          __sub (&hp, &a, &b, p);
Packit Service 82fcde
          __c32 (&b, &s, &c, p);
Packit Service 82fcde
        }
Packit Service 82fcde
      else
Packit Service 82fcde
        __c32 (&a, &c, &s, p);	/* b = sin(x+dx)  */
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Convert result based on which quarter of unit circle y is in.  */
Packit Service 82fcde
  switch (n)
Packit Service 82fcde
    {
Packit Service 82fcde
    case 1:
Packit Service 82fcde
      __mp_dbl (&c, &y, p);
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case 3:
Packit Service 82fcde
      __mp_dbl (&c, &y, p);
Packit Service 82fcde
      y = -y;
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case 2:
Packit Service 82fcde
      __mp_dbl (&s, &y, p);
Packit Service 82fcde
      y = -y;
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    /* Quadrant not set, so the result must be sin (X + DX), which is also in
Packit Service 82fcde
       S.  */
Packit Service 82fcde
    case 0:
Packit Service 82fcde
    default:
Packit Service 82fcde
      __mp_dbl (&s, &y, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  LIBC_PROBE (slowsin, 3, &x, &dx, &y);
Packit Service 82fcde
  return y;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Compute cos() of double-length number (X + DX) as Multi Precision number and
Packit Service 82fcde
   return result as double.  If REDUCE_RANGE is true, X is assumed to be the
Packit Service 82fcde
   original input and DX is ignored.  */
Packit Service 82fcde
double
Packit Service 82fcde
SECTION
Packit Service 82fcde
__mpcos (double x, double dx, bool reduce_range)
Packit Service 82fcde
{
Packit Service 82fcde
  double y;
Packit Service 82fcde
  mp_no a, b, c, s;
Packit Service 82fcde
  int n;
Packit Service 82fcde
  int p = 32;
Packit Service 82fcde
Packit Service 82fcde
  if (reduce_range)
Packit Service 82fcde
    {
Packit Service 82fcde
      n = __mpranred (x, &a, p);	/* n is 0, 1, 2 or 3.  */
Packit Service 82fcde
      __c32 (&a, &c, &s, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      n = -1;
Packit Service 82fcde
      __dbl_mp (x, &b, p);
Packit Service 82fcde
      __dbl_mp (dx, &c, p);
Packit Service 82fcde
      __add (&b, &c, &a, p);
Packit Service 82fcde
      if (x > 0.8)
Packit Service 82fcde
        {
Packit Service 82fcde
          __sub (&hp, &a, &b, p);
Packit Service 82fcde
          __c32 (&b, &s, &c, p);
Packit Service 82fcde
        }
Packit Service 82fcde
      else
Packit Service 82fcde
        __c32 (&a, &c, &s, p);	/* a = cos(x+dx)     */
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Convert result based on which quarter of unit circle y is in.  */
Packit Service 82fcde
  switch (n)
Packit Service 82fcde
    {
Packit Service 82fcde
    case 1:
Packit Service 82fcde
      __mp_dbl (&s, &y, p);
Packit Service 82fcde
      y = -y;
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case 3:
Packit Service 82fcde
      __mp_dbl (&s, &y, p);
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case 2:
Packit Service 82fcde
      __mp_dbl (&c, &y, p);
Packit Service 82fcde
      y = -y;
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    /* Quadrant not set, so the result must be cos (X + DX), which is also
Packit Service 82fcde
       stored in C.  */
Packit Service 82fcde
    case 0:
Packit Service 82fcde
    default:
Packit Service 82fcde
      __mp_dbl (&c, &y, p);
Packit Service 82fcde
    }
Packit Service 82fcde
  LIBC_PROBE (slowcos, 3, &x, &dx, &y);
Packit Service 82fcde
  return y;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Perform range reduction of a double number x into multi precision number y,
Packit Service 82fcde
   such that y = x - n * pi / 2, abs (y) < pi / 4, n = 0, +-1, +-2, ...
Packit Service 82fcde
   Return int which indicates in which quarter of circle x is.  */
Packit Service 82fcde
int
Packit Service 82fcde
SECTION
Packit Service 82fcde
__mpranred (double x, mp_no *y, int p)
Packit Service 82fcde
{
Packit Service 82fcde
  number v;
Packit Service 82fcde
  double t, xn;
Packit Service 82fcde
  int i, k, n;
Packit Service 82fcde
  mp_no a, b, c;
Packit Service 82fcde
Packit Service 82fcde
  if (fabs (x) < 2.8e14)
Packit Service 82fcde
    {
Packit Service 82fcde
      t = (x * hpinv.d + toint.d);
Packit Service 82fcde
      xn = t - toint.d;
Packit Service 82fcde
      v.d = t;
Packit Service 82fcde
      n = v.i[LOW_HALF] & 3;
Packit Service 82fcde
      __dbl_mp (xn, &a, p);
Packit Service 82fcde
      __mul (&a, &hp, &b, p);
Packit Service 82fcde
      __dbl_mp (x, &c, p);
Packit Service 82fcde
      __sub (&c, &b, y, p);
Packit Service 82fcde
      return n;
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      /* If x is very big more precision required.  */
Packit Service 82fcde
      __dbl_mp (x, &a, p);
Packit Service 82fcde
      a.d[0] = 1.0;
Packit Service 82fcde
      k = a.e - 5;
Packit Service 82fcde
      if (k < 0)
Packit Service 82fcde
	k = 0;
Packit Service 82fcde
      b.e = -k;
Packit Service 82fcde
      b.d[0] = 1.0;
Packit Service 82fcde
      for (i = 0; i < p; i++)
Packit Service 82fcde
	b.d[i + 1] = toverp[i + k];
Packit Service 82fcde
      __mul (&a, &b, &c, p);
Packit Service 82fcde
      t = c.d[c.e];
Packit Service 82fcde
      for (i = 1; i <= p - c.e; i++)
Packit Service 82fcde
	c.d[i] = c.d[i + c.e];
Packit Service 82fcde
      for (i = p + 1 - c.e; i <= p; i++)
Packit Service 82fcde
	c.d[i] = 0;
Packit Service 82fcde
      c.e = 0;
Packit Service 82fcde
      if (c.d[1] >= HALFRAD)
Packit Service 82fcde
	{
Packit Service 82fcde
	  t += 1.0;
Packit Service 82fcde
	  __sub (&c, &__mpone, &b, p);
Packit Service 82fcde
	  __mul (&b, &hp, y, p);
Packit Service 82fcde
	}
Packit Service 82fcde
      else
Packit Service 82fcde
	__mul (&c, &hp, y, p);
Packit Service 82fcde
      n = (int) t;
Packit Service 82fcde
      if (x < 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  y->d[0] = -y->d[0];
Packit Service 82fcde
	  n = -n;
Packit Service 82fcde
	}
Packit Service 82fcde
      return (n & 3);
Packit Service 82fcde
    }
Packit Service 82fcde
}