|
Packit |
6c4009 |
/*
|
|
Packit |
6c4009 |
* ====================================================
|
|
Packit |
6c4009 |
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
|
|
Packit |
6c4009 |
*
|
|
Packit |
6c4009 |
* Developed at SunPro, a Sun Microsystems, Inc. business.
|
|
Packit |
6c4009 |
* Permission to use, copy, modify, and distribute this
|
|
Packit |
6c4009 |
* software is freely granted, provided that this notice
|
|
Packit |
6c4009 |
* is preserved.
|
|
Packit |
6c4009 |
* ====================================================
|
|
Packit |
6c4009 |
*/
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
/*
|
|
Packit |
6c4009 |
* modf(double x, double *iptr)
|
|
Packit |
6c4009 |
* return fraction part of x, and return x's integral part in *iptr.
|
|
Packit |
6c4009 |
* Method:
|
|
Packit |
6c4009 |
* Bit twiddling.
|
|
Packit |
6c4009 |
*
|
|
Packit |
6c4009 |
* Exception:
|
|
Packit |
6c4009 |
* No exception.
|
|
Packit |
6c4009 |
*/
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
#include <math.h>
|
|
Packit |
6c4009 |
#include <math_private.h>
|
|
Packit |
6c4009 |
#include <libm-alias-double.h>
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
static const double one = 1.0;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
double
|
|
Packit |
6c4009 |
__modf (double x, double *iptr)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
int32_t i0, i1, j0;
|
|
Packit |
6c4009 |
uint32_t i;
|
|
Packit |
6c4009 |
EXTRACT_WORDS (i0, i1, x);
|
|
Packit |
6c4009 |
j0 = ((i0 >> 20) & 0x7ff) - 0x3ff; /* exponent of x */
|
|
Packit |
6c4009 |
if (j0 < 20) /* integer part in high x */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (j0 < 0) /* |x|<1 */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
INSERT_WORDS (*iptr, i0 & 0x80000000, 0); /* *iptr = +-0 */
|
|
Packit |
6c4009 |
return x;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
i = (0x000fffff) >> j0;
|
|
Packit |
6c4009 |
if (((i0 & i) | i1) == 0) /* x is integral */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*iptr = x;
|
|
Packit |
6c4009 |
INSERT_WORDS (x, i0 & 0x80000000, 0); /* return +-0 */
|
|
Packit |
6c4009 |
return x;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
INSERT_WORDS (*iptr, i0 & (~i), 0);
|
|
Packit |
6c4009 |
return x - *iptr;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else if (__glibc_unlikely (j0 > 51)) /* no fraction part */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*iptr = x * one;
|
|
Packit |
6c4009 |
/* We must handle NaNs separately. */
|
|
Packit |
6c4009 |
if (j0 == 0x400 && ((i0 & 0xfffff) | i1))
|
|
Packit |
6c4009 |
return x * one;
|
|
Packit |
6c4009 |
INSERT_WORDS (x, i0 & 0x80000000, 0); /* return +-0 */
|
|
Packit |
6c4009 |
return x;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else /* fraction part in low x */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
i = ((uint32_t) (0xffffffff)) >> (j0 - 20);
|
|
Packit |
6c4009 |
if ((i1 & i) == 0) /* x is integral */
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*iptr = x;
|
|
Packit |
6c4009 |
INSERT_WORDS (x, i0 & 0x80000000, 0); /* return +-0 */
|
|
Packit |
6c4009 |
return x;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
INSERT_WORDS (*iptr, i0, i1 & (~i));
|
|
Packit |
6c4009 |
return x - *iptr;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
#ifndef __modf
|
|
Packit |
6c4009 |
libm_alias_double (__modf, modf)
|
|
Packit |
6c4009 |
#endif
|