mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-18 12:24:38 +08:00
6df01ab8ab
The defs.h header will take care of including the various config.h headers. For now, it's just config.h, but we'll add more when we integrate gnulib in. This header should be used instead of config.h, and should be the first include in every .c file. We won't rely on the old behavior where we expected files to include the port's sim-main.h which then includes the common sim-basics.h which then includes config.h. We have a ton of code that includes things before sim-main.h, and it sometimes needs to be that way. Creating a dedicated header avoids the ordering mess and implicit inclusion that shows up otherwise.
795 lines
15 KiB
C
795 lines
15 KiB
C
/* fpu.c --- FPU emulator for stand-alone RX simulator.
|
|
|
|
Copyright (C) 2008-2021 Free Software Foundation, Inc.
|
|
Contributed by Red Hat, Inc.
|
|
|
|
This file is part of the GNU simulators.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
/* This must come before any other includes. */
|
|
#include "defs.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "cpu.h"
|
|
#include "fpu.h"
|
|
|
|
/* FPU encodings are as follows:
|
|
|
|
S EXPONENT MANTISSA
|
|
1 12345678 12345678901234567890123
|
|
|
|
0 00000000 00000000000000000000000 +0
|
|
1 00000000 00000000000000000000000 -0
|
|
|
|
X 00000000 00000000000000000000001 Denormals
|
|
X 00000000 11111111111111111111111
|
|
|
|
X 00000001 XXXXXXXXXXXXXXXXXXXXXXX Normals
|
|
X 11111110 XXXXXXXXXXXXXXXXXXXXXXX
|
|
|
|
0 11111111 00000000000000000000000 +Inf
|
|
1 11111111 00000000000000000000000 -Inf
|
|
|
|
X 11111111 0XXXXXXXXXXXXXXXXXXXXXX SNaN (X != 0)
|
|
X 11111111 1XXXXXXXXXXXXXXXXXXXXXX QNaN (X != 0)
|
|
|
|
*/
|
|
|
|
#define trace 0
|
|
#define tprintf if (trace) printf
|
|
|
|
/* Some magic numbers. */
|
|
#define PLUS_MAX 0x7f7fffffUL
|
|
#define MINUS_MAX 0xff7fffffUL
|
|
#define PLUS_INF 0x7f800000UL
|
|
#define MINUS_INF 0xff800000UL
|
|
#define PLUS_ZERO 0x00000000UL
|
|
#define MINUS_ZERO 0x80000000UL
|
|
|
|
#define FP_RAISE(e) fp_raise(FPSWBITS_C##e)
|
|
static void
|
|
fp_raise (int mask)
|
|
{
|
|
regs.r_fpsw |= mask;
|
|
if (mask != FPSWBITS_CE)
|
|
{
|
|
if (regs.r_fpsw & (mask << FPSW_CESH))
|
|
regs.r_fpsw |= (mask << FPSW_CFSH);
|
|
if (regs.r_fpsw & FPSWBITS_FMASK)
|
|
regs.r_fpsw |= FPSWBITS_FSUM;
|
|
else
|
|
regs.r_fpsw &= ~FPSWBITS_FSUM;
|
|
}
|
|
}
|
|
|
|
/* We classify all numbers as one of these. They correspond to the
|
|
rows/colums in the exception tables. */
|
|
typedef enum {
|
|
FP_NORMAL,
|
|
FP_PZERO,
|
|
FP_NZERO,
|
|
FP_PINFINITY,
|
|
FP_NINFINITY,
|
|
FP_DENORMAL,
|
|
FP_QNAN,
|
|
FP_SNAN
|
|
} FP_Type;
|
|
|
|
#if defined DEBUG0
|
|
static const char *fpt_names[] = {
|
|
"Normal", "+0", "-0", "+Inf", "-Inf", "Denormal", "QNaN", "SNaN"
|
|
};
|
|
#endif
|
|
|
|
#define EXP_BIAS 127
|
|
#define EXP_ZERO -127
|
|
#define EXP_INF 128
|
|
|
|
#define MANT_BIAS 0x00080000UL
|
|
|
|
typedef struct {
|
|
int exp;
|
|
unsigned int mant; /* 24 bits */
|
|
char type;
|
|
char sign;
|
|
fp_t orig_value;
|
|
} FP_Parts;
|
|
|
|
static void
|
|
fp_explode (fp_t f, FP_Parts *p)
|
|
{
|
|
int exp, mant, sign;
|
|
|
|
exp = ((f & 0x7f800000UL) >> 23);
|
|
mant = f & 0x007fffffUL;
|
|
sign = f & 0x80000000UL;
|
|
/*printf("explode: %08x %x %2x %6x\n", f, sign, exp, mant);*/
|
|
|
|
p->sign = sign ? -1 : 1;
|
|
p->exp = exp - EXP_BIAS;
|
|
p->orig_value = f;
|
|
p->mant = mant | 0x00800000UL;
|
|
|
|
if (p->exp == EXP_ZERO)
|
|
{
|
|
if (regs.r_fpsw & FPSWBITS_DN)
|
|
mant = 0;
|
|
if (mant)
|
|
p->type = FP_DENORMAL;
|
|
else
|
|
{
|
|
p->mant = 0;
|
|
p->type = sign ? FP_NZERO : FP_PZERO;
|
|
}
|
|
}
|
|
else if (p->exp == EXP_INF)
|
|
{
|
|
if (mant == 0)
|
|
p->type = sign ? FP_NINFINITY : FP_PINFINITY;
|
|
else if (mant & 0x00400000UL)
|
|
p->type = FP_QNAN;
|
|
else
|
|
p->type = FP_SNAN;
|
|
}
|
|
else
|
|
p->type = FP_NORMAL;
|
|
}
|
|
|
|
static fp_t
|
|
fp_implode (FP_Parts *p)
|
|
{
|
|
int exp, mant;
|
|
|
|
exp = p->exp + EXP_BIAS;
|
|
mant = p->mant;
|
|
/*printf("implode: exp %d mant 0x%x\n", exp, mant);*/
|
|
if (p->type == FP_NORMAL)
|
|
{
|
|
while (mant
|
|
&& exp > 0
|
|
&& mant < 0x00800000UL)
|
|
{
|
|
mant <<= 1;
|
|
exp --;
|
|
}
|
|
while (mant > 0x00ffffffUL)
|
|
{
|
|
mant >>= 1;
|
|
exp ++;
|
|
}
|
|
if (exp < 0)
|
|
{
|
|
/* underflow */
|
|
exp = 0;
|
|
mant = 0;
|
|
FP_RAISE (E);
|
|
}
|
|
if (exp >= 255)
|
|
{
|
|
/* overflow */
|
|
exp = 255;
|
|
mant = 0;
|
|
FP_RAISE (O);
|
|
}
|
|
}
|
|
mant &= 0x007fffffUL;
|
|
exp &= 0xff;
|
|
mant |= exp << 23;
|
|
if (p->sign < 0)
|
|
mant |= 0x80000000UL;
|
|
|
|
return mant;
|
|
}
|
|
|
|
typedef union {
|
|
unsigned long long ll;
|
|
double d;
|
|
} U_d_ll;
|
|
|
|
static int checked_format = 0;
|
|
|
|
/* We assume a double format like this:
|
|
S[1] E[11] M[52]
|
|
*/
|
|
|
|
static double
|
|
fp_to_double (FP_Parts *p)
|
|
{
|
|
U_d_ll u;
|
|
|
|
if (!checked_format)
|
|
{
|
|
u.d = 1.5;
|
|
if (u.ll != 0x3ff8000000000000ULL)
|
|
abort ();
|
|
u.d = -225;
|
|
if (u.ll != 0xc06c200000000000ULL)
|
|
abort ();
|
|
u.d = 10.1;
|
|
if (u.ll != 0x4024333333333333ULL)
|
|
abort ();
|
|
checked_format = 1;
|
|
}
|
|
|
|
u.ll = 0;
|
|
if (p->sign < 0)
|
|
u.ll |= (1ULL << 63);
|
|
/* Make sure a zero encoding stays a zero. */
|
|
if (p->exp != -EXP_BIAS)
|
|
u.ll |= ((unsigned long long)p->exp + 1023ULL) << 52;
|
|
u.ll |= (unsigned long long) (p->mant & 0x007fffffUL) << (52 - 23);
|
|
return u.d;
|
|
}
|
|
|
|
static void
|
|
double_to_fp (double d, FP_Parts *p)
|
|
{
|
|
int exp;
|
|
U_d_ll u;
|
|
int sign;
|
|
|
|
u.d = d;
|
|
|
|
sign = (u.ll & 0x8000000000000000ULL) ? 1 : 0;
|
|
exp = u.ll >> 52;
|
|
exp = (exp & 0x7ff);
|
|
|
|
if (exp == 0)
|
|
{
|
|
/* A generated denormal should show up as an underflow, not
|
|
here. */
|
|
if (sign)
|
|
fp_explode (MINUS_ZERO, p);
|
|
else
|
|
fp_explode (PLUS_ZERO, p);
|
|
return;
|
|
}
|
|
|
|
exp = exp - 1023;
|
|
if ((exp + EXP_BIAS) > 254)
|
|
{
|
|
FP_RAISE (O);
|
|
switch (regs.r_fpsw & FPSWBITS_RM)
|
|
{
|
|
case FPRM_NEAREST:
|
|
if (sign)
|
|
fp_explode (MINUS_INF, p);
|
|
else
|
|
fp_explode (PLUS_INF, p);
|
|
break;
|
|
case FPRM_ZERO:
|
|
if (sign)
|
|
fp_explode (MINUS_MAX, p);
|
|
else
|
|
fp_explode (PLUS_MAX, p);
|
|
break;
|
|
case FPRM_PINF:
|
|
if (sign)
|
|
fp_explode (MINUS_MAX, p);
|
|
else
|
|
fp_explode (PLUS_INF, p);
|
|
break;
|
|
case FPRM_NINF:
|
|
if (sign)
|
|
fp_explode (MINUS_INF, p);
|
|
else
|
|
fp_explode (PLUS_MAX, p);
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
if ((exp + EXP_BIAS) < 1)
|
|
{
|
|
if (sign)
|
|
fp_explode (MINUS_ZERO, p);
|
|
else
|
|
fp_explode (PLUS_ZERO, p);
|
|
FP_RAISE (U);
|
|
}
|
|
|
|
p->sign = sign ? -1 : 1;
|
|
p->exp = exp;
|
|
p->mant = u.ll >> (52-23) & 0x007fffffUL;
|
|
p->mant |= 0x00800000UL;
|
|
p->type = FP_NORMAL;
|
|
|
|
if (u.ll & 0x1fffffffULL)
|
|
{
|
|
switch (regs.r_fpsw & FPSWBITS_RM)
|
|
{
|
|
case FPRM_NEAREST:
|
|
if (u.ll & 0x10000000ULL)
|
|
p->mant ++;
|
|
break;
|
|
case FPRM_ZERO:
|
|
break;
|
|
case FPRM_PINF:
|
|
if (sign == 1)
|
|
p->mant ++;
|
|
break;
|
|
case FPRM_NINF:
|
|
if (sign == -1)
|
|
p->mant ++;
|
|
break;
|
|
}
|
|
FP_RAISE (X);
|
|
}
|
|
|
|
}
|
|
|
|
typedef enum {
|
|
eNR, /* Use the normal result. */
|
|
ePZ, eNZ, /* +- zero */
|
|
eSZ, /* signed zero - XOR signs of ops together. */
|
|
eRZ, /* +- zero depending on rounding mode. */
|
|
ePI, eNI, /* +- Infinity */
|
|
eSI, /* signed infinity - XOR signs of ops together. */
|
|
eQN, eSN, /* Quiet/Signalling NANs */
|
|
eIn, /* Invalid. */
|
|
eUn, /* Unimplemented. */
|
|
eDZ, /* Divide-by-zero. */
|
|
eLT, /* less than */
|
|
eGT, /* greater than */
|
|
eEQ, /* equal to */
|
|
} FP_ExceptionCases;
|
|
|
|
#if defined DEBUG0
|
|
static const char *ex_names[] = {
|
|
"NR", "PZ", "NZ", "SZ", "RZ", "PI", "NI", "SI", "QN", "SN", "IN", "Un", "DZ", "LT", "GT", "EQ"
|
|
};
|
|
#endif
|
|
|
|
/* This checks for all exceptional cases (not all FP exceptions) and
|
|
returns TRUE if it is providing the result in *c. If it returns
|
|
FALSE, the caller should do the "normal" operation. */
|
|
static int
|
|
check_exceptions (FP_Parts *a, FP_Parts *b, fp_t *c,
|
|
FP_ExceptionCases ex_tab[5][5],
|
|
FP_ExceptionCases *case_ret)
|
|
{
|
|
FP_ExceptionCases fpec;
|
|
|
|
if (a->type == FP_SNAN
|
|
|| b->type == FP_SNAN)
|
|
fpec = eIn;
|
|
else if (a->type == FP_QNAN
|
|
|| b->type == FP_QNAN)
|
|
fpec = eQN;
|
|
else if (a->type == FP_DENORMAL
|
|
|| b->type == FP_DENORMAL)
|
|
fpec = eUn;
|
|
else
|
|
fpec = ex_tab[(int)(a->type)][(int)(b->type)];
|
|
|
|
/*printf("%s %s -> %s\n", fpt_names[(int)(a->type)], fpt_names[(int)(b->type)], ex_names[(int)(fpec)]);*/
|
|
|
|
if (case_ret)
|
|
*case_ret = fpec;
|
|
|
|
switch (fpec)
|
|
{
|
|
case eNR: /* Use the normal result. */
|
|
return 0;
|
|
|
|
case ePZ: /* + zero */
|
|
*c = 0x00000000;
|
|
return 1;
|
|
|
|
case eNZ: /* - zero */
|
|
*c = 0x80000000;
|
|
return 1;
|
|
|
|
case eSZ: /* signed zero */
|
|
*c = (a->sign == b->sign) ? PLUS_ZERO : MINUS_ZERO;
|
|
return 1;
|
|
|
|
case eRZ: /* +- zero depending on rounding mode. */
|
|
if ((regs.r_fpsw & FPSWBITS_RM) == FPRM_NINF)
|
|
*c = 0x80000000;
|
|
else
|
|
*c = 0x00000000;
|
|
return 1;
|
|
|
|
case ePI: /* + Infinity */
|
|
*c = 0x7F800000;
|
|
return 1;
|
|
|
|
case eNI: /* - Infinity */
|
|
*c = 0xFF800000;
|
|
return 1;
|
|
|
|
case eSI: /* sign Infinity */
|
|
*c = (a->sign == b->sign) ? PLUS_INF : MINUS_INF;
|
|
return 1;
|
|
|
|
case eQN: /* Quiet NANs */
|
|
if (a->type == FP_QNAN)
|
|
*c = a->orig_value;
|
|
else
|
|
*c = b->orig_value;
|
|
return 1;
|
|
|
|
case eSN: /* Signalling NANs */
|
|
if (a->type == FP_SNAN)
|
|
*c = a->orig_value;
|
|
else
|
|
*c = b->orig_value;
|
|
FP_RAISE (V);
|
|
return 1;
|
|
|
|
case eIn: /* Invalid. */
|
|
FP_RAISE (V);
|
|
if (a->type == FP_SNAN)
|
|
*c = a->orig_value | 0x00400000;
|
|
else if (b->type == FP_SNAN)
|
|
*c = b->orig_value | 0x00400000;
|
|
else
|
|
*c = 0x7fc00000;
|
|
return 1;
|
|
|
|
case eUn: /* Unimplemented. */
|
|
FP_RAISE (E);
|
|
return 1;
|
|
|
|
case eDZ: /* Division-by-zero. */
|
|
*c = (a->sign == b->sign) ? PLUS_INF : MINUS_INF;
|
|
FP_RAISE (Z);
|
|
return 1;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
#define CHECK_EXCEPTIONS(FPPa, FPPb, fpc, ex_tab) \
|
|
if (check_exceptions (&FPPa, &FPPb, &fpc, ex_tab, 0)) \
|
|
return fpc;
|
|
|
|
/* For each operation, we have two tables of how nonnormal cases are
|
|
handled. The DN=0 case is first, followed by the DN=1 case, with
|
|
each table using the following layout: */
|
|
|
|
static FP_ExceptionCases ex_add_tab[5][5] = {
|
|
/* N +0 -0 +In -In */
|
|
{ eNR, eNR, eNR, ePI, eNI }, /* Normal */
|
|
{ eNR, ePZ, eRZ, ePI, eNI }, /* +0 */
|
|
{ eNR, eRZ, eNZ, ePI, eNI }, /* -0 */
|
|
{ ePI, ePI, ePI, ePI, eIn }, /* +Inf */
|
|
{ eNI, eNI, eNI, eIn, eNI }, /* -Inf */
|
|
};
|
|
|
|
fp_t
|
|
rxfp_add (fp_t fa, fp_t fb)
|
|
{
|
|
FP_Parts a, b, c;
|
|
fp_t rv;
|
|
double da, db;
|
|
|
|
fp_explode (fa, &a);
|
|
fp_explode (fb, &b);
|
|
CHECK_EXCEPTIONS (a, b, rv, ex_add_tab);
|
|
|
|
da = fp_to_double (&a);
|
|
db = fp_to_double (&b);
|
|
tprintf("%g + %g = %g\n", da, db, da+db);
|
|
|
|
double_to_fp (da+db, &c);
|
|
rv = fp_implode (&c);
|
|
return rv;
|
|
}
|
|
|
|
static FP_ExceptionCases ex_sub_tab[5][5] = {
|
|
/* N +0 -0 +In -In */
|
|
{ eNR, eNR, eNR, eNI, ePI }, /* Normal */
|
|
{ eNR, eRZ, ePZ, eNI, ePI }, /* +0 */
|
|
{ eNR, eNZ, eRZ, eNI, ePI }, /* -0 */
|
|
{ ePI, ePI, ePI, eIn, ePI }, /* +Inf */
|
|
{ eNI, eNI, eNI, eNI, eIn }, /* -Inf */
|
|
};
|
|
|
|
fp_t
|
|
rxfp_sub (fp_t fa, fp_t fb)
|
|
{
|
|
FP_Parts a, b, c;
|
|
fp_t rv;
|
|
double da, db;
|
|
|
|
fp_explode (fa, &a);
|
|
fp_explode (fb, &b);
|
|
CHECK_EXCEPTIONS (a, b, rv, ex_sub_tab);
|
|
|
|
da = fp_to_double (&a);
|
|
db = fp_to_double (&b);
|
|
tprintf("%g - %g = %g\n", da, db, da-db);
|
|
|
|
double_to_fp (da-db, &c);
|
|
rv = fp_implode (&c);
|
|
|
|
return rv;
|
|
}
|
|
|
|
static FP_ExceptionCases ex_mul_tab[5][5] = {
|
|
/* N +0 -0 +In -In */
|
|
{ eNR, eNR, eNR, eSI, eSI }, /* Normal */
|
|
{ eNR, ePZ, eNZ, eIn, eIn }, /* +0 */
|
|
{ eNR, eNZ, ePZ, eIn, eIn }, /* -0 */
|
|
{ eSI, eIn, eIn, ePI, eNI }, /* +Inf */
|
|
{ eSI, eIn, eIn, eNI, ePI }, /* -Inf */
|
|
};
|
|
|
|
fp_t
|
|
rxfp_mul (fp_t fa, fp_t fb)
|
|
{
|
|
FP_Parts a, b, c;
|
|
fp_t rv;
|
|
double da, db;
|
|
|
|
fp_explode (fa, &a);
|
|
fp_explode (fb, &b);
|
|
CHECK_EXCEPTIONS (a, b, rv, ex_mul_tab);
|
|
|
|
da = fp_to_double (&a);
|
|
db = fp_to_double (&b);
|
|
tprintf("%g x %g = %g\n", da, db, da*db);
|
|
|
|
double_to_fp (da*db, &c);
|
|
rv = fp_implode (&c);
|
|
|
|
return rv;
|
|
}
|
|
|
|
static FP_ExceptionCases ex_div_tab[5][5] = {
|
|
/* N +0 -0 +In -In */
|
|
{ eNR, eDZ, eDZ, eSZ, eSZ }, /* Normal */
|
|
{ eSZ, eIn, eIn, ePZ, eNZ }, /* +0 */
|
|
{ eSZ, eIn, eIn, eNZ, ePZ }, /* -0 */
|
|
{ eSI, ePI, eNI, eIn, eIn }, /* +Inf */
|
|
{ eSI, eNI, ePI, eIn, eIn }, /* -Inf */
|
|
};
|
|
|
|
fp_t
|
|
rxfp_div (fp_t fa, fp_t fb)
|
|
{
|
|
FP_Parts a, b, c;
|
|
fp_t rv;
|
|
double da, db;
|
|
|
|
fp_explode (fa, &a);
|
|
fp_explode (fb, &b);
|
|
CHECK_EXCEPTIONS (a, b, rv, ex_div_tab);
|
|
|
|
da = fp_to_double (&a);
|
|
db = fp_to_double (&b);
|
|
tprintf("%g / %g = %g\n", da, db, da/db);
|
|
|
|
double_to_fp (da/db, &c);
|
|
rv = fp_implode (&c);
|
|
|
|
return rv;
|
|
}
|
|
|
|
static FP_ExceptionCases ex_cmp_tab[5][5] = {
|
|
/* N +0 -0 +In -In */
|
|
{ eNR, eNR, eNR, eLT, eGT }, /* Normal */
|
|
{ eNR, eEQ, eEQ, eLT, eGT }, /* +0 */
|
|
{ eNR, eEQ, eEQ, eLT, eGT }, /* -0 */
|
|
{ eGT, eGT, eGT, eEQ, eGT }, /* +Inf */
|
|
{ eLT, eLT, eLT, eLT, eEQ }, /* -Inf */
|
|
};
|
|
|
|
void
|
|
rxfp_cmp (fp_t fa, fp_t fb)
|
|
{
|
|
FP_Parts a, b;
|
|
fp_t c;
|
|
FP_ExceptionCases reason;
|
|
int flags = 0;
|
|
double da, db;
|
|
|
|
fp_explode (fa, &a);
|
|
fp_explode (fb, &b);
|
|
|
|
if (check_exceptions (&a, &b, &c, ex_cmp_tab, &reason))
|
|
{
|
|
if (reason == eQN)
|
|
{
|
|
/* Special case - incomparable. */
|
|
set_flags (FLAGBIT_Z | FLAGBIT_S | FLAGBIT_O, FLAGBIT_O);
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
|
|
switch (reason)
|
|
{
|
|
case eEQ:
|
|
flags = FLAGBIT_Z;
|
|
break;
|
|
case eLT:
|
|
flags = FLAGBIT_S;
|
|
break;
|
|
case eGT:
|
|
flags = 0;
|
|
break;
|
|
case eNR:
|
|
da = fp_to_double (&a);
|
|
db = fp_to_double (&b);
|
|
tprintf("fcmp: %g cmp %g\n", da, db);
|
|
if (da < db)
|
|
flags = FLAGBIT_S;
|
|
else if (da == db)
|
|
flags = FLAGBIT_Z;
|
|
else
|
|
flags = 0;
|
|
break;
|
|
default:
|
|
abort();
|
|
}
|
|
|
|
set_flags (FLAGBIT_Z | FLAGBIT_S | FLAGBIT_O, flags);
|
|
}
|
|
|
|
long
|
|
rxfp_ftoi (fp_t fa, int round_mode)
|
|
{
|
|
FP_Parts a;
|
|
fp_t rv;
|
|
int sign;
|
|
int whole_bits, frac_bits;
|
|
|
|
fp_explode (fa, &a);
|
|
sign = fa & 0x80000000UL;
|
|
|
|
switch (a.type)
|
|
{
|
|
case FP_NORMAL:
|
|
break;
|
|
case FP_PZERO:
|
|
case FP_NZERO:
|
|
return 0;
|
|
case FP_PINFINITY:
|
|
FP_RAISE (V);
|
|
return 0x7fffffffL;
|
|
case FP_NINFINITY:
|
|
FP_RAISE (V);
|
|
return 0x80000000L;
|
|
case FP_DENORMAL:
|
|
FP_RAISE (E);
|
|
return 0;
|
|
case FP_QNAN:
|
|
case FP_SNAN:
|
|
FP_RAISE (V);
|
|
return sign ? 0x80000000U : 0x7fffffff;
|
|
}
|
|
|
|
if (a.exp >= 31)
|
|
{
|
|
FP_RAISE (V);
|
|
return sign ? 0x80000000U : 0x7fffffff;
|
|
}
|
|
|
|
a.exp -= 23;
|
|
|
|
if (a.exp <= -25)
|
|
{
|
|
/* Less than 0.49999 */
|
|
frac_bits = a.mant;
|
|
whole_bits = 0;
|
|
}
|
|
else if (a.exp < 0)
|
|
{
|
|
frac_bits = a.mant << (32 + a.exp);
|
|
whole_bits = a.mant >> (-a.exp);
|
|
}
|
|
else
|
|
{
|
|
frac_bits = 0;
|
|
whole_bits = a.mant << a.exp;
|
|
}
|
|
|
|
if (frac_bits)
|
|
{
|
|
switch (round_mode & 3)
|
|
{
|
|
case FPRM_NEAREST:
|
|
if (frac_bits & 0x80000000UL)
|
|
whole_bits ++;
|
|
break;
|
|
case FPRM_ZERO:
|
|
break;
|
|
case FPRM_PINF:
|
|
if (!sign)
|
|
whole_bits ++;
|
|
break;
|
|
case FPRM_NINF:
|
|
if (sign)
|
|
whole_bits ++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
rv = sign ? -whole_bits : whole_bits;
|
|
|
|
return rv;
|
|
}
|
|
|
|
fp_t
|
|
rxfp_itof (long fa, int round_mode)
|
|
{
|
|
fp_t rv;
|
|
int sign = 0;
|
|
unsigned int frac_bits;
|
|
volatile unsigned int whole_bits;
|
|
FP_Parts a = {0};
|
|
|
|
if (fa == 0)
|
|
return PLUS_ZERO;
|
|
|
|
if (fa < 0)
|
|
{
|
|
fa = -fa;
|
|
sign = 1;
|
|
a.sign = -1;
|
|
}
|
|
else
|
|
a.sign = 1;
|
|
|
|
whole_bits = fa;
|
|
a.exp = 31;
|
|
|
|
while (! (whole_bits & 0x80000000UL))
|
|
{
|
|
a.exp --;
|
|
whole_bits <<= 1;
|
|
}
|
|
frac_bits = whole_bits & 0xff;
|
|
whole_bits = whole_bits >> 8;
|
|
|
|
if (frac_bits)
|
|
{
|
|
/* We must round */
|
|
switch (round_mode & 3)
|
|
{
|
|
case FPRM_NEAREST:
|
|
if (frac_bits & 0x80)
|
|
whole_bits ++;
|
|
break;
|
|
case FPRM_ZERO:
|
|
break;
|
|
case FPRM_PINF:
|
|
if (!sign)
|
|
whole_bits ++;
|
|
break;
|
|
case FPRM_NINF:
|
|
if (sign)
|
|
whole_bits ++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
a.mant = whole_bits;
|
|
if (whole_bits & 0xff000000UL)
|
|
{
|
|
a.mant >>= 1;
|
|
a.exp ++;
|
|
}
|
|
|
|
rv = fp_implode (&a);
|
|
return rv;
|
|
}
|
|
|