OpenCores
URL https://opencores.org/ocsvn/openrisc_2011-10-31/openrisc_2011-10-31/trunk

Subversion Repositories openrisc_2011-10-31

[/] [openrisc/] [tags/] [gnu-src/] [newlib-1.18.0/] [newlib-1.18.0-or32-1.0rc1/] [newlib/] [libm/] [common/] [s_lrint.c] - Diff between revs 207 and 345

Only display areas with differences | Details | Blame | View Log

Rev 207 Rev 345
 
 
/* @(#)s_lrint.c 5.1 93/09/24 */
/* @(#)s_lrint.c 5.1 93/09/24 */
/*
/*
 * ====================================================
 * ====================================================
 * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
 * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
 *
 *
 * Developed at SunPro, a Sun Microsystems, Inc. business.
 * Developed at SunPro, a Sun Microsystems, Inc. business.
 * Permission to use, copy, modify, and distribute this
 * Permission to use, copy, modify, and distribute this
 * software is freely granted, provided that this notice
 * software is freely granted, provided that this notice
 * is preserved.
 * is preserved.
 * ====================================================
 * ====================================================
 */
 */
/*
/*
FUNCTION
FUNCTION
<<lrint>>, <<lrintf>>, <<llrint>>, <<llrintf>>--round to integer
<<lrint>>, <<lrintf>>, <<llrint>>, <<llrintf>>--round to integer
INDEX
INDEX
        lrint
        lrint
INDEX
INDEX
        lrintf
        lrintf
INDEX
INDEX
        llrint
        llrint
INDEX
INDEX
        llrintf
        llrintf
 
 
ANSI_SYNOPSIS
ANSI_SYNOPSIS
        #include <math.h>
        #include <math.h>
        long int lrint(double <[x]>);
        long int lrint(double <[x]>);
        long int lrintf(float <[x]>);
        long int lrintf(float <[x]>);
        long long int llrint(double <[x]>);
        long long int llrint(double <[x]>);
        long long int llrintf(float <[x]>);
        long long int llrintf(float <[x]>);
 
 
DESCRIPTION
DESCRIPTION
The <<lrint>> and <<llrint>> functions round their argument to the nearest
The <<lrint>> and <<llrint>> functions round their argument to the nearest
integer value, using the current rounding direction.  If the rounded value is
integer value, using the current rounding direction.  If the rounded value is
outside the range of the return type, the numeric result is unspecified.  A
outside the range of the return type, the numeric result is unspecified.  A
range error may occur if the magnitude of <[x]> is too large.
range error may occur if the magnitude of <[x]> is too large.
The "inexact" floating-point exception is raised in implementations that
The "inexact" floating-point exception is raised in implementations that
support it when the result differs in value from the argument (i.e., when
support it when the result differs in value from the argument (i.e., when
a fraction actually has been truncated).
a fraction actually has been truncated).
 
 
RETURNS
RETURNS
<[x]> rounded to an integral value, using the current rounding direction.
<[x]> rounded to an integral value, using the current rounding direction.
 
 
SEEALSO
SEEALSO
<<lround>>
<<lround>>
 
 
PORTABILITY
PORTABILITY
ANSI C, POSIX
ANSI C, POSIX
 
 
*/
*/
 
 
/*
/*
 * lrint(x)
 * lrint(x)
 * Return x rounded to integral value according to the prevailing
 * Return x rounded to integral value according to the prevailing
 * rounding mode.
 * rounding mode.
 * Method:
 * Method:
 *      Using floating addition.
 *      Using floating addition.
 * Exception:
 * Exception:
 *      Inexact flag raised if x not equal to lrint(x).
 *      Inexact flag raised if x not equal to lrint(x).
 */
 */
 
 
#include "fdlibm.h"
#include "fdlibm.h"
 
 
#ifndef _DOUBLE_IS_32BITS
#ifndef _DOUBLE_IS_32BITS
 
 
#ifdef __STDC__
#ifdef __STDC__
static const double
static const double
#else
#else
static double
static double
#endif
#endif
 
 
/* Adding a double, x, to 2^52 will cause the result to be rounded based on
/* Adding a double, x, to 2^52 will cause the result to be rounded based on
   the fractional part of x, according to the implementation's current rounding
   the fractional part of x, according to the implementation's current rounding
   mode.  2^52 is the smallest double that can be represented using all 52 significant
   mode.  2^52 is the smallest double that can be represented using all 52 significant
   digits. */
   digits. */
TWO52[2]={
TWO52[2]={
  4.50359962737049600000e+15, /* 0x43300000, 0x00000000 */
  4.50359962737049600000e+15, /* 0x43300000, 0x00000000 */
 -4.50359962737049600000e+15, /* 0xC3300000, 0x00000000 */
 -4.50359962737049600000e+15, /* 0xC3300000, 0x00000000 */
};
};
 
 
#ifdef __STDC__
#ifdef __STDC__
        long int lrint(double x)
        long int lrint(double x)
#else
#else
        long int lrint(x)
        long int lrint(x)
        double x;
        double x;
#endif
#endif
{
{
  __int32_t i0,j0,sx;
  __int32_t i0,j0,sx;
  __uint32_t i1;
  __uint32_t i1;
  double t;
  double t;
  volatile double w;
  volatile double w;
  long int result;
  long int result;
 
 
  EXTRACT_WORDS(i0,i1,x);
  EXTRACT_WORDS(i0,i1,x);
 
 
  /* Extract sign bit. */
  /* Extract sign bit. */
  sx = (i0>>31)&1;
  sx = (i0>>31)&1;
 
 
  /* Extract exponent field. */
  /* Extract exponent field. */
  j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
  j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
 
 
  if(j0 < 20)
  if(j0 < 20)
    {
    {
      if(j0 < -1)
      if(j0 < -1)
        return 0;
        return 0;
      else
      else
        {
        {
          w = TWO52[sx] + x;
          w = TWO52[sx] + x;
          t = w - TWO52[sx];
          t = w - TWO52[sx];
          GET_HIGH_WORD(i0, t);
          GET_HIGH_WORD(i0, t);
          /* Detect the all-zeros representation of plus and
          /* Detect the all-zeros representation of plus and
             minus zero, which fails the calculation below. */
             minus zero, which fails the calculation below. */
          if ((i0 & ~(1L << 31)) == 0)
          if ((i0 & ~(1L << 31)) == 0)
              return 0;
              return 0;
          j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
          j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
          i0 &= 0x000fffff;
          i0 &= 0x000fffff;
          i0 |= 0x00100000;
          i0 |= 0x00100000;
          result = i0 >> (20 - j0);
          result = i0 >> (20 - j0);
        }
        }
    }
    }
  else if (j0 < (int)(8 * sizeof (long int)) - 1)
  else if (j0 < (int)(8 * sizeof (long int)) - 1)
    {
    {
      if (j0 >= 52)
      if (j0 >= 52)
        result = ((long int) ((i0 & 0x000fffff) | 0x0010000) << (j0 - 20)) |
        result = ((long int) ((i0 & 0x000fffff) | 0x0010000) << (j0 - 20)) |
                   (i1 << (j0 - 52));
                   (i1 << (j0 - 52));
      else
      else
        {
        {
          w = TWO52[sx] + x;
          w = TWO52[sx] + x;
          t = w - TWO52[sx];
          t = w - TWO52[sx];
          EXTRACT_WORDS (i0, i1, t);
          EXTRACT_WORDS (i0, i1, t);
          j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
          j0 = ((i0 & 0x7ff00000) >> 20) - 1023;
          i0 &= 0x000fffff;
          i0 &= 0x000fffff;
          i0 |= 0x00100000;
          i0 |= 0x00100000;
          result = ((long int) i0 << (j0 - 20)) | (i1 >> (52 - j0));
          result = ((long int) i0 << (j0 - 20)) | (i1 >> (52 - j0));
        }
        }
    }
    }
  else
  else
    {
    {
      return (long int) x;
      return (long int) x;
    }
    }
 
 
  return sx ? -result : result;
  return sx ? -result : result;
}
}
 
 
#endif /* _DOUBLE_IS_32BITS */
#endif /* _DOUBLE_IS_32BITS */
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.