Skip to content

Commit cb20f68

Browse files
committed
Auto merge of #50812 - kennytm:fix-50756-miri-bad-float-behavior, r=oli-obk
Fix issue #50811 (`NaN > NaN` was true). Fix #50811 Make sure the float comparison output is consistent with the expected behavior when NaN is involved. ---- Note: This PR is a **BREAKING CHANGE**. If you have used `>` or `>=` to compare floats, and make the result as the length of a fixed array type, like: ```rust use std::f64::NAN; let x: [u8; (NAN > NAN) as usize] = [1]; ``` then the code will no longer compile. Previously, all float comparison involving NaN will just return "Greater", i.e. `NAN > NAN` would wrongly return `true` during const evaluation. If you need to retain the old behavior (why), you may replace `a > b` with `a != a || b != b || a > b`.
2 parents 56e541d + 50bc72d commit cb20f68

File tree

2 files changed

+71
-7
lines changed

2 files changed

+71
-7
lines changed

src/librustc_mir/interpret/operator.rs

+6-7
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
use rustc::mir;
22
use rustc::ty::{self, Ty};
33
use syntax::ast::FloatTy;
4-
use std::cmp::Ordering;
54
use rustc::ty::layout::LayoutOf;
65
use rustc_apfloat::ieee::{Double, Single};
76
use rustc_apfloat::Float;
@@ -181,12 +180,12 @@ impl<'a, 'mir, 'tcx, M: Machine<'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M> {
181180
let l = <$ty>::from_bits(l);
182181
let r = <$ty>::from_bits(r);
183182
let val = match bin_op {
184-
Eq => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) == Ordering::Equal),
185-
Ne => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) != Ordering::Equal),
186-
Lt => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) == Ordering::Less),
187-
Le => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) != Ordering::Greater),
188-
Gt => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) == Ordering::Greater),
189-
Ge => PrimVal::from_bool(l.partial_cmp(&r).unwrap_or(Ordering::Greater) != Ordering::Less),
183+
Eq => PrimVal::from_bool(l == r),
184+
Ne => PrimVal::from_bool(l != r),
185+
Lt => PrimVal::from_bool(l < r),
186+
Le => PrimVal::from_bool(l <= r),
187+
Gt => PrimVal::from_bool(l > r),
188+
Ge => PrimVal::from_bool(l >= r),
190189
Add => PrimVal::Bytes((l + r).value.to_bits()),
191190
Sub => PrimVal::Bytes((l - r).value.to_bits()),
192191
Mul => PrimVal::Bytes((l * r).value.to_bits()),

src/test/run-pass/issue-50811.rs

+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(test)]
12+
13+
extern crate test;
14+
15+
use std::f64::{NAN, NEG_INFINITY, INFINITY, MAX};
16+
use std::mem::size_of;
17+
use test::black_box;
18+
19+
// Ensure the const-eval result and runtime result of float comparison are equivalent.
20+
21+
macro_rules! compare {
22+
($op:tt) => {
23+
compare!(
24+
[NEG_INFINITY, -MAX, -1.0, -0.0, 0.0, 1.0, MAX, INFINITY, NAN],
25+
$op
26+
);
27+
};
28+
([$($lhs:expr),+], $op:tt) => {
29+
$(compare!(
30+
$lhs,
31+
$op,
32+
[NEG_INFINITY, -MAX, -1.0, -0.0, 0.0, 1.0, MAX, INFINITY, NAN]
33+
);)+
34+
};
35+
($lhs:expr, $op:tt, [$($rhs:expr),+]) => {
36+
$({
37+
// Wrap the check in its own function to reduce time needed to borrowck.
38+
fn check() {
39+
static CONST_EVAL: bool = $lhs $op $rhs;
40+
let runtime_eval = black_box($lhs) $op black_box($rhs);
41+
assert_eq!(CONST_EVAL, runtime_eval, stringify!($lhs $op $rhs));
42+
assert_eq!(
43+
size_of::<[u8; ($lhs $op $rhs) as usize]>(),
44+
runtime_eval as usize,
45+
stringify!($lhs $op $rhs (forced const eval))
46+
);
47+
}
48+
check();
49+
})+
50+
};
51+
}
52+
53+
fn main() {
54+
assert_eq!(0.0/0.0 < 0.0/0.0, false);
55+
assert_eq!(0.0/0.0 > 0.0/0.0, false);
56+
assert_eq!(NAN < NAN, false);
57+
assert_eq!(NAN > NAN, false);
58+
59+
compare!(==);
60+
compare!(!=);
61+
compare!(<);
62+
compare!(<=);
63+
compare!(>);
64+
compare!(>=);
65+
}

0 commit comments

Comments
 (0)