summaryrefslogtreecommitdiff
path: root/src/arch/arm/insts
diff options
context:
space:
mode:
authorGabe Black <gblack@eecs.umich.edu>2010-06-02 12:58:15 -0500
committerGabe Black <gblack@eecs.umich.edu>2010-06-02 12:58:15 -0500
commit186273e5f318786cd4db1b652b9b0332c18ea7bf (patch)
treef7a75b93f9ef222de1513fe934462b65e7c43e34 /src/arch/arm/insts
parent98e2315f1c459d8126227b3c598ba4ea9620df7a (diff)
downloadgem5-186273e5f318786cd4db1b652b9b0332c18ea7bf.tar.xz
ARM: Fix up nans to match ARM's expected behavior.
Diffstat (limited to 'src/arch/arm/insts')
-rw-r--r--src/arch/arm/insts/vfp.hh80
1 files changed, 80 insertions, 0 deletions
diff --git a/src/arch/arm/insts/vfp.hh b/src/arch/arm/insts/vfp.hh
index b0fc8b6dc..b69f93598 100644
--- a/src/arch/arm/insts/vfp.hh
+++ b/src/arch/arm/insts/vfp.hh
@@ -45,6 +45,9 @@
#include <fenv.h>
#include <cmath>
+namespace ArmISA
+{
+
enum VfpMicroMode {
VfpNotAMicroop,
VfpMicroop,
@@ -122,6 +125,81 @@ vfpFlushToZero(uint32_t &fpscr, fpType &op1, fpType &op2)
vfpFlushToZero(fpscr, op2);
}
+static inline uint32_t
+fpToBits(float fp)
+{
+ union
+ {
+ float fp;
+ uint32_t bits;
+ } val;
+ val.fp = fp;
+ return val.bits;
+}
+
+static inline uint64_t
+fpToBits(double fp)
+{
+ union
+ {
+ double fp;
+ uint64_t bits;
+ } val;
+ val.fp = fp;
+ return val.bits;
+}
+
+static inline float
+bitsToFp(uint64_t bits, float junk)
+{
+ union
+ {
+ float fp;
+ uint32_t bits;
+ } val;
+ val.bits = bits;
+ return val.fp;
+}
+
+static inline double
+bitsToFp(uint64_t bits, double junk)
+{
+ union
+ {
+ double fp;
+ uint64_t bits;
+ } val;
+ val.bits = bits;
+ return val.fp;
+}
+
+template <class fpType>
+static inline fpType
+fixNan(FPSCR fpscr, fpType val, fpType op1, fpType op2)
+{
+ if (std::isnan(val)) {
+ const bool single = (sizeof(val) == sizeof(float));
+ const uint64_t qnan = single ? 0x7fc00000 : ULL(0x7ff8000000000000);
+ const bool nan1 = std::isnan(op1);
+ const bool nan2 = std::isnan(op2);
+ const bool signal1 = nan1 && ((fpToBits(op1) & qnan) != qnan);
+ const bool signal2 = nan2 && ((fpToBits(op2) & qnan) != qnan);
+ fpType junk = 0.0;
+ if ((!nan1 && !nan2) || (fpscr.dn == 1)) {
+ val = bitsToFp(qnan, junk);
+ } else if (signal1) {
+ val = bitsToFp(fpToBits(op1) | qnan, junk);
+ } else if (signal2) {
+ val = bitsToFp(fpToBits(op2) | qnan, junk);
+ } else if (nan1) {
+ val = op1;
+ } else if (nan2) {
+ val = op2;
+ }
+ }
+ return val;
+}
+
static inline uint64_t
vfpFpSToFixed(float val, bool isSigned, bool half, uint8_t imm)
{
@@ -481,4 +559,6 @@ class VfpRegRegRegOp : public RegRegRegOp
}
};
+}
+
#endif //__ARCH_ARM_INSTS_VFP_HH__