mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-23 23:47:22 +08:00
f3b8138239
Add a _REENT_ERRNO() macro to encapsulate the access to the _errno member of struct reent. This will help to replace the structure member with a thread-local storage object in a follow up patch. Replace uses of __errno_r() with _REENT_ERRNO(). Keep __errno_r() macro for potential users outside of Newlib.
103 lines
2.1 KiB
C
103 lines
2.1 KiB
C
#ifdef __SPE__
|
|
|
|
#include <_ansi.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <reent.h>
|
|
#include "vfieeefp.h"
|
|
|
|
/*
|
|
* Convert a string to a fixed-point (sign + 31-bits) value.
|
|
*
|
|
* Ignores `locale' stuff.
|
|
*/
|
|
__int32_t
|
|
_strtosfix32_r (struct _reent *rptr,
|
|
const char *nptr,
|
|
char **endptr)
|
|
{
|
|
union double_union dbl;
|
|
int exp, negexp, sign;
|
|
unsigned long tmp, tmp2;
|
|
long result = 0;
|
|
|
|
dbl.d = _strtod_r (rptr, nptr, endptr);
|
|
|
|
/* treat NAN as domain error, +/- infinity as saturation */
|
|
if (!finite(dbl.d))
|
|
{
|
|
if (isnan (dbl.d))
|
|
{
|
|
_REENT_ERRNO(rptr) = EDOM;
|
|
return 0;
|
|
}
|
|
_REENT_ERRNO(rptr) = ERANGE;
|
|
if (word0(dbl) & Sign_bit)
|
|
return LONG_MIN;
|
|
return LONG_MAX;
|
|
}
|
|
|
|
/* check for normal saturation */
|
|
if (dbl.d >= 1.0)
|
|
{
|
|
_REENT_ERRNO(rptr) = ERANGE;
|
|
return LONG_MAX;
|
|
}
|
|
else if (dbl.d < -1.0)
|
|
{
|
|
_REENT_ERRNO(rptr) = ERANGE;
|
|
return LONG_MIN;
|
|
}
|
|
|
|
/* otherwise we have normal number in range */
|
|
|
|
/* strip off sign and exponent */
|
|
sign = word0(dbl) & Sign_bit;
|
|
exp = ((word0(dbl) & Exp_mask) >> Exp_shift) - Bias;
|
|
negexp = -exp;
|
|
if (negexp > 31)
|
|
return 0;
|
|
word0(dbl) &= ~(Exp_mask | Sign_bit);
|
|
/* add in implicit normalized bit */
|
|
word0(dbl) |= Exp_msk1;
|
|
/* shift so result is contained in single word */
|
|
tmp = word0(dbl) << Ebits;
|
|
tmp |= ((unsigned long)word1(dbl) >> (32 - Ebits));
|
|
if (negexp != 0)
|
|
{
|
|
/* perform rounding */
|
|
tmp2 = tmp + (1 << (negexp - 1));
|
|
result = (long)(tmp2 >> negexp);
|
|
/* check if rounding caused carry bit which must be added into result */
|
|
if (tmp2 < tmp)
|
|
result |= (1 << (32 - negexp));
|
|
/* check if positive saturation has occurred because of rounding */
|
|
if (!sign && result < 0)
|
|
{
|
|
_REENT_ERRNO(rptr) = ERANGE;
|
|
return LONG_MAX;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* we have -1.0, no rounding necessary */
|
|
return LONG_MIN;
|
|
}
|
|
|
|
return sign ? -result : result;
|
|
}
|
|
|
|
#ifndef _REENT_ONLY
|
|
|
|
__int32_t
|
|
strtosfix32 (const char *s,
|
|
char **ptr)
|
|
{
|
|
return _strtosfix32_r (_REENT, s, ptr);
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif /* __SPE__ */
|