Skip to content

Commit 9ea462f

Browse files
Rollup merge of #77368 - est31:apfloat_fix, r=varkor
Backport LLVM apfloat commit to rustc_apfloat Backports LLVM commit: llvm/llvm-project@e34bd1e Fixes #69532
2 parents 4ae7710 + d010809 commit 9ea462f

File tree

3 files changed

+59
-14
lines changed

3 files changed

+59
-14
lines changed

compiler/rustc_apfloat/src/ieee.rs

Lines changed: 10 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -1511,11 +1511,16 @@ impl<S: Semantics, T: Semantics> FloatConvert<IeeeFloat<T>> for IeeeFloat<S> {
15111511
sig::set_bit(&mut r.sig, T::PRECISION - 1);
15121512
}
15131513

1514-
// gcc forces the Quiet bit on, which means (float)(double)(float_sNan)
1515-
// does not give you back the same bits. This is dubious, and we
1516-
// don't currently do it. You're really supposed to get
1517-
// an invalid operation signal at runtime, but nobody does that.
1518-
status = Status::OK;
1514+
// Convert of sNaN creates qNaN and raises an exception (invalid op).
1515+
// This also guarantees that a sNaN does not become Inf on a truncation
1516+
// that loses all payload bits.
1517+
if self.is_signaling() {
1518+
// Quiet signaling NaN.
1519+
sig::set_bit(&mut r.sig, T::QNAN_BIT);
1520+
status = Status::INVALID_OP;
1521+
} else {
1522+
status = Status::OK;
1523+
}
15191524
} else {
15201525
*loses_info = false;
15211526
status = Status::OK;

compiler/rustc_apfloat/tests/ieee.rs

Lines changed: 25 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -566,6 +566,17 @@ fn fma() {
566566
}
567567
}
568568

569+
#[test]
570+
fn issue_69532() {
571+
let f = Double::from_bits(0x7FF0_0000_0000_0001u64 as u128);
572+
let mut loses_info = false;
573+
let sta = f.convert(&mut loses_info);
574+
let r: Single = sta.value;
575+
assert!(loses_info);
576+
assert!(r.is_nan());
577+
assert_eq!(sta.status, Status::INVALID_OP);
578+
}
579+
569580
#[test]
570581
fn min_num() {
571582
let f1 = Double::from_f64(1.0);
@@ -1492,27 +1503,32 @@ fn convert() {
14921503
assert_eq!(4294967295.0, test.to_f64());
14931504
assert!(!loses_info);
14941505

1495-
let test = Single::snan(None);
1496-
let x87_snan = X87DoubleExtended::snan(None);
1497-
let test: X87DoubleExtended = test.convert(&mut loses_info).value;
1498-
assert!(test.bitwise_eq(x87_snan));
1499-
assert!(!loses_info);
1500-
15011506
let test = Single::qnan(None);
15021507
let x87_qnan = X87DoubleExtended::qnan(None);
15031508
let test: X87DoubleExtended = test.convert(&mut loses_info).value;
15041509
assert!(test.bitwise_eq(x87_qnan));
15051510
assert!(!loses_info);
15061511

1507-
let test = X87DoubleExtended::snan(None);
1508-
let test: X87DoubleExtended = test.convert(&mut loses_info).value;
1509-
assert!(test.bitwise_eq(x87_snan));
1512+
let test = Single::snan(None);
1513+
let sta = test.convert(&mut loses_info);
1514+
let test: X87DoubleExtended = sta.value;
1515+
assert!(test.is_nan());
1516+
assert!(!test.is_signaling());
15101517
assert!(!loses_info);
1518+
assert_eq!(sta.status, Status::INVALID_OP);
15111519

15121520
let test = X87DoubleExtended::qnan(None);
15131521
let test: X87DoubleExtended = test.convert(&mut loses_info).value;
15141522
assert!(test.bitwise_eq(x87_qnan));
15151523
assert!(!loses_info);
1524+
1525+
let test = X87DoubleExtended::snan(None);
1526+
let sta = test.convert(&mut loses_info);
1527+
let test: X87DoubleExtended = sta.value;
1528+
assert!(test.is_nan());
1529+
assert!(!test.is_signaling());
1530+
assert!(!loses_info);
1531+
assert_eq!(sta.status, Status::INVALID_OP);
15161532
}
15171533

15181534
#[test]

src/test/ui/issues/issue-69532.rs

Lines changed: 24 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// run-pass
2+
#![feature(const_fn_transmute)]
3+
4+
const fn make_nans() -> (f64, f64, f32, f32) {
5+
let nan1: f64 = unsafe { std::mem::transmute(0x7FF0_0001_0000_0001u64) };
6+
let nan2: f64 = unsafe { std::mem::transmute(0x7FF0_0000_0000_0001u64) };
7+
8+
let nan1_32 = nan1 as f32;
9+
let nan2_32 = nan2 as f32;
10+
11+
(nan1, nan2, nan1_32, nan2_32)
12+
}
13+
14+
static NANS: (f64, f64, f32, f32) = make_nans();
15+
16+
fn main() {
17+
let (nan1, nan2, nan1_32, nan2_32) = NANS;
18+
19+
assert!(nan1.is_nan());
20+
assert!(nan2.is_nan());
21+
22+
assert!(nan1_32.is_nan());
23+
assert!(nan2_32.is_nan());
24+
}

0 commit comments

Comments
 (0)