4
0
mirror of git://sourceware.org/git/newlib-cygwin.git synced 2025-01-21 05:49:19 +08:00
Matt Joyce f3b8138239 Add _REENT_ERRNO(ptr)
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.
2022-07-13 06:55:41 +02:00

100 lines
2.0 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 32-bit value.
*
* Ignores `locale' stuff.
*/
__uint32_t
_strtoufix32_r (struct _reent *rptr,
const char *nptr,
char **endptr)
{
union double_union dbl;
int exp, negexp;
__uint32_t tmp, tmp2, 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 0;
return ULONG_MAX;
}
/* check for normal saturation */
if (dbl.d >= 1.0)
{
_REENT_ERRNO(rptr) = ERANGE;
return ULONG_MAX;
}
else if (dbl.d < 0)
{
_REENT_ERRNO(rptr) = ERANGE;
return 0;
}
/* otherwise we have normal positive number in range */
/* strip off exponent */
exp = ((word0(dbl) & Exp_mask) >> Exp_shift) - Bias;
negexp = -exp;
if (negexp > 32)
return 0;
word0(dbl) &= ~(Exp_mask | Sign_bit);
/* add in implicit normalized bit */
word0(dbl) |= Exp_msk1;
/* shift so result is contained left-justified in word */
tmp = word0(dbl) << Ebits;
tmp |= ((unsigned long)word1(dbl) >> (32 - Ebits));
/* perform rounding */
if (negexp > 1)
{
tmp2 = tmp + (1 << (negexp - 2));
result = (tmp2 >> (negexp - 1));
/* if rounding causes carry, add carry bit in */
if (tmp2 < tmp)
result += 1 << (32 - negexp);
}
else
{
result = tmp + ((word1(dbl) & (1 << (32 - Ebits - 1))) != 0);
/* if rounding causes carry, then saturation has occurred */
if (result < tmp)
{
_REENT_ERRNO(rptr) = ERANGE;
return ULONG_MAX;
}
}
return result;
}
#ifndef _REENT_ONLY
__uint32_t
strtoufix32 (const char *s,
char **ptr)
{
return _strtoufix32_r (_REENT, s, ptr);
}
#endif
#endif /* __SPE__ */