aboutsummaryrefslogtreecommitdiffstats
path: root/lib/Support
diff options
context:
space:
mode:
authorDale Johannesen <dalej@apple.com>2008-10-06 22:59:10 +0000
committerDale Johannesen <dalej@apple.com>2008-10-06 22:59:10 +0000
commit2df5eec2ff25cbb7e5fd4848e667fb9f854d3339 (patch)
treeb98d1b559d7f59f2a5a8dca054f290c62233a31b /lib/Support
parente93f5db6e5f4cf87c5a6cb9d8f25d9808b9a82d6 (diff)
downloadexternal_llvm-2df5eec2ff25cbb7e5fd4848e667fb9f854d3339.zip
external_llvm-2df5eec2ff25cbb7e5fd4848e667fb9f854d3339.tar.gz
external_llvm-2df5eec2ff25cbb7e5fd4848e667fb9f854d3339.tar.bz2
Be more precise about which conversions of NaNs
are Inexact. (These are not Inexact as defined by IEEE754, but that seems like a reasonable way to abstract what happens: information is lost.) git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@57218 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/Support')
-rw-r--r--lib/Support/APFloat.cpp21
1 files changed, 18 insertions, 3 deletions
diff --git a/lib/Support/APFloat.cpp b/lib/Support/APFloat.cpp
index 3e37d79..b93f1d1 100644
--- a/lib/Support/APFloat.cpp
+++ b/lib/Support/APFloat.cpp
@@ -1721,17 +1721,32 @@ APFloat::convert(const fltSemantics &toSemantics,
} else if (category == fcNaN) {
int shift = toSemantics.precision - semantics->precision;
// Do this now so significandParts gets the right answer
+ const fltSemantics *oldSemantics = semantics;
semantics = &toSemantics;
+ fs = opOK;
// No normalization here, just truncate
if (shift>0)
APInt::tcShiftLeft(significandParts(), newPartCount, shift);
- else if (shift < 0)
- APInt::tcShiftRight(significandParts(), newPartCount, -shift);
+ else if (shift < 0) {
+ unsigned ushift = -shift;
+ // We mark this as Inexact if we are losing information. This happens
+ // if are shifting out something other than 0s, or if the x87 long
+ // double input did not have its integer bit set (pseudo-NaN), or if the
+ // x87 long double input did not have its QNan bit set (because the x87
+ // hardware sets this bit when converting a lower-precision NaN to
+ // x87 long double).
+ if (APInt::tcLSB(significandParts(), newPartCount) < ushift)
+ fs = opInexact;
+ if (oldSemantics == &APFloat::x87DoubleExtended &&
+ (!(*significandParts() & 0x8000000000000000ULL) ||
+ !(*significandParts() & 0x4000000000000000ULL)))
+ fs = opInexact;
+ APInt::tcShiftRight(significandParts(), newPartCount, ushift);
+ }
// gcc forces the Quiet bit on, which means (float)(double)(float_sNan)
// does not give you back the same bits. This is dubious, and we
// don't currently do it. You're really supposed to get
// an invalid operation signal at runtime, but nobody does that.
- fs = opOK;
} else {
semantics = &toSemantics;
fs = opOK;