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/] [sf_llrint.c] - Diff between revs 207 and 345

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

Rev 207 Rev 345
/* lrintf adapted to be llrintf for Newlib, 2009 by Craig Howland.  */
/* lrintf adapted to be llrintf for Newlib, 2009 by Craig Howland.  */
/* @(#)sf_lrint.c 5.1 93/09/24 */
/* @(#)sf_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.
 * ====================================================
 * ====================================================
 */
 */
 
 
/*
/*
 * llrintf(x)
 * llrintf(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 llrintf(x).
 *      Inexact flag raised if x not equal to llrintf(x).
 */
 */
 
 
#include "fdlibm.h"
#include "fdlibm.h"
 
 
#ifdef __STDC__
#ifdef __STDC__
static const float
static const float
#else
#else
static float
static float
#endif
#endif
/* Adding a float, x, to 2^23 will cause the result to be rounded based on
/* Adding a float, x, to 2^23 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^23 is the smallest float that can be represented using all 23 significant
   mode.  2^23 is the smallest float that can be represented using all 23 significant
   digits. */
   digits. */
TWO23[2]={
TWO23[2]={
  8.3886080000e+06, /* 0x4b000000 */
  8.3886080000e+06, /* 0x4b000000 */
 -8.3886080000e+06, /* 0xcb000000 */
 -8.3886080000e+06, /* 0xcb000000 */
};
};
 
 
#ifdef __STDC__
#ifdef __STDC__
        long long int llrintf(float x)
        long long int llrintf(float x)
#else
#else
        long long int llrintf(x)
        long long int llrintf(x)
        float x;
        float x;
#endif
#endif
{
{
  __int32_t j0,sx;
  __int32_t j0,sx;
  __uint32_t i0;
  __uint32_t i0;
  float t;
  float t;
  volatile float w;
  volatile float w;
  long long int result;
  long long int result;
 
 
  GET_FLOAT_WORD(i0,x);
  GET_FLOAT_WORD(i0,x);
 
 
  /* Extract sign bit. */
  /* Extract sign bit. */
  sx = (i0 >> 31);
  sx = (i0 >> 31);
 
 
  /* Extract exponent field. */
  /* Extract exponent field. */
  j0 = ((i0 & 0x7f800000) >> 23) - 127;
  j0 = ((i0 & 0x7f800000) >> 23) - 127;
 
 
  if (j0 < (int)(sizeof (long long int) * 8) - 1)
  if (j0 < (int)(sizeof (long long int) * 8) - 1)
    {
    {
      if (j0 < -1)
      if (j0 < -1)
        return 0;
        return 0;
      else if (j0 >= 23)
      else if (j0 >= 23)
        result = (long long int) ((i0 & 0x7fffff) | 0x800000) << (j0 - 23);
        result = (long long int) ((i0 & 0x7fffff) | 0x800000) << (j0 - 23);
      else
      else
        {
        {
          w = TWO23[sx] + x;
          w = TWO23[sx] + x;
          t = w - TWO23[sx];
          t = w - TWO23[sx];
          GET_FLOAT_WORD (i0, t);
          GET_FLOAT_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 & ~(1 << 31)) == 0)
          if ((i0 & ~(1 << 31)) == 0)
              return 0;
              return 0;
          j0 = ((i0 >> 23) & 0xff) - 0x7f;
          j0 = ((i0 >> 23) & 0xff) - 0x7f;
          i0 &= 0x7fffff;
          i0 &= 0x7fffff;
          i0 |= 0x800000;
          i0 |= 0x800000;
          result = i0 >> (23 - j0);
          result = i0 >> (23 - j0);
        }
        }
    }
    }
  else
  else
    {
    {
      return (long long int) x;
      return (long long int) x;
    }
    }
  return sx ? -result : result;
  return sx ? -result : result;
}
}
 
 
#ifdef _DOUBLE_IS_32BITS
#ifdef _DOUBLE_IS_32BITS
 
 
#ifdef __STDC__
#ifdef __STDC__
        long long int llrint(double x)
        long long int llrint(double x)
#else
#else
        long long int llrint(x)
        long long int llrint(x)
        double x;
        double x;
#endif
#endif
{
{
  return llrintf((float) x);
  return llrintf((float) x);
}
}
 
 
#endif /* defined(_DOUBLE_IS_32BITS) */
#endif /* defined(_DOUBLE_IS_32BITS) */
 
 

powered by: WebSVN 2.1.0

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