aboutsummaryrefslogtreecommitdiff
path: root/tests/tcg/i386/test-i386-fscale.c
diff options
context:
space:
mode:
authorJoseph Myers <joseph@codesourcery.com>2020-05-07 00:46:28 +0000
committerPaolo Bonzini <pbonzini@redhat.com>2020-06-10 12:10:21 -0400
commitc535d68755576bfa33be7aef7bd294a601f776e0 (patch)
treea38e6c956706bdc5786520ade140301433ef73e2 /tests/tcg/i386/test-i386-fscale.c
parentc1c5fb8f9067c830e36830c2b82c0ec146c03d7b (diff)
downloadqemu-c535d68755576bfa33be7aef7bd294a601f776e0.zip
qemu-c535d68755576bfa33be7aef7bd294a601f776e0.tar.gz
qemu-c535d68755576bfa33be7aef7bd294a601f776e0.tar.bz2
target/i386: fix fscale handling of rounding precision
The fscale implementation uses floatx80_scalbn for the final scaling operation. floatx80_scalbn ends up rounding the result using the dynamic rounding precision configured for the FPU. But only a limited set of x87 floating-point instructions are supposed to respect the dynamic rounding precision, and fscale is not in that set. Fix the implementation to save and restore the rounding precision around the call to floatx80_scalbn. Signed-off-by: Joseph Myers <joseph@codesourcery.com> Message-Id: <alpine.DEB.2.21.2005070045430.18350@digraph.polyomino.org.uk> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Diffstat (limited to 'tests/tcg/i386/test-i386-fscale.c')
-rw-r--r--tests/tcg/i386/test-i386-fscale.c13
1 files changed, 13 insertions, 0 deletions
diff --git a/tests/tcg/i386/test-i386-fscale.c b/tests/tcg/i386/test-i386-fscale.c
index b953e7c..d23b3cf 100644
--- a/tests/tcg/i386/test-i386-fscale.c
+++ b/tests/tcg/i386/test-i386-fscale.c
@@ -8,6 +8,8 @@ union u {
long double ld;
};
+volatile long double ld_third = 1.0L / 3.0L;
+volatile long double ld_four_thirds = 4.0L / 3.0L;
volatile union u ld_invalid_1 = { .s = { 1, 1234 } };
volatile union u ld_invalid_2 = { .s = { 0, 1234 } };
volatile union u ld_invalid_3 = { .s = { 0, 0x7fff } };
@@ -91,5 +93,16 @@ int main(void)
printf("FAIL: fscale finite down inf\n");
ret = 1;
}
+ /* Set round-to-nearest with single-precision rounding. */
+ cw = cw & ~0xf00;
+ __asm__ volatile ("fldcw %0" : : "m" (cw));
+ __asm__ volatile ("fscale" : "=t" (ld_res) :
+ "0" (ld_third), "u" (2.0L));
+ cw = cw | 0x300;
+ __asm__ volatile ("fldcw %0" : : "m" (cw));
+ if (ld_res != ld_four_thirds) {
+ printf("FAIL: fscale single-precision\n");
+ ret = 1;
+ }
return ret;
}