Skip to content

Commit 5fe6b58

Browse files
committed
Auto merge of #49949 - oli-obk:const_signed_pat, r=eddyb
Sign extend constants in range patterns fixes #49940 r? @Mark-Simulacrum
2 parents 78fc510 + b22c9c0 commit 5fe6b58

File tree

5 files changed

+69
-23
lines changed

5 files changed

+69
-23
lines changed

src/librustc_mir/hair/pattern/_match.rs

+11-7
Original file line numberDiff line numberDiff line change
@@ -985,15 +985,17 @@ fn slice_pat_covered_by_constructor(tcx: TyCtxt, _span: Span,
985985
Ok(true)
986986
}
987987

988-
fn constructor_covered_by_range(ctor: &Constructor,
989-
from: &ConstVal, to: &ConstVal,
990-
end: RangeEnd,
991-
ty: Ty)
992-
-> Result<bool, ErrorReported> {
988+
fn constructor_covered_by_range<'a, 'tcx>(
989+
tcx: TyCtxt<'a, 'tcx, 'tcx>,
990+
ctor: &Constructor,
991+
from: &ConstVal, to: &ConstVal,
992+
end: RangeEnd,
993+
ty: Ty<'tcx>,
994+
) -> Result<bool, ErrorReported> {
993995
trace!("constructor_covered_by_range {:#?}, {:#?}, {:#?}, {}", ctor, from, to, ty);
994-
let cmp_from = |c_from| compare_const_vals(c_from, from, ty)
996+
let cmp_from = |c_from| compare_const_vals(tcx, c_from, from, ty)
995997
.map(|res| res != Ordering::Less);
996-
let cmp_to = |c_to| compare_const_vals(c_to, to, ty);
998+
let cmp_to = |c_to| compare_const_vals(tcx, c_to, to, ty);
997999
macro_rules! some_or_ok {
9981000
($e:expr) => {
9991001
match $e {
@@ -1105,6 +1107,7 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
11051107
},
11061108
_ => {
11071109
match constructor_covered_by_range(
1110+
cx.tcx,
11081111
constructor, &value.val, &value.val, RangeEnd::Included,
11091112
value.ty,
11101113
) {
@@ -1118,6 +1121,7 @@ fn specialize<'p, 'a: 'p, 'tcx: 'a>(
11181121

11191122
PatternKind::Range { lo, hi, ref end } => {
11201123
match constructor_covered_by_range(
1124+
cx.tcx,
11211125
constructor, &lo.val, &hi.val, end.clone(), lo.ty,
11221126
) {
11231127
Ok(true) => Some(vec![]),

src/librustc_mir/hair/pattern/mod.rs

+13-4
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ mod check_match;
1616
pub use self::check_match::check_crate;
1717
pub(crate) use self::check_match::check_match;
1818

19-
use interpret::{const_val_field, const_discr};
19+
use interpret::{const_val_field, const_discr, self};
2020

2121
use rustc::middle::const_val::ConstVal;
2222
use rustc::mir::{Field, BorrowKind, Mutability};
@@ -372,7 +372,7 @@ impl<'a, 'tcx> PatternContext<'a, 'tcx> {
372372
(PatternKind::Constant { value: lo },
373373
PatternKind::Constant { value: hi }) => {
374374
use std::cmp::Ordering;
375-
match (end, compare_const_vals(&lo.val, &hi.val, ty).unwrap()) {
375+
match (end, compare_const_vals(self.tcx, &lo.val, &hi.val, ty).unwrap()) {
376376
(RangeEnd::Excluded, Ordering::Less) =>
377377
PatternKind::Range { lo, hi, end },
378378
(RangeEnd::Excluded, _) => {
@@ -1076,7 +1076,12 @@ impl<'tcx> PatternFoldable<'tcx> for PatternKind<'tcx> {
10761076
}
10771077
}
10781078

1079-
pub fn compare_const_vals(a: &ConstVal, b: &ConstVal, ty: Ty) -> Option<Ordering> {
1079+
pub fn compare_const_vals<'a, 'tcx>(
1080+
tcx: TyCtxt<'a, 'tcx, 'tcx>,
1081+
a: &ConstVal,
1082+
b: &ConstVal,
1083+
ty: Ty<'tcx>,
1084+
) -> Option<Ordering> {
10801085
use rustc_const_math::ConstFloat;
10811086
trace!("compare_const_vals: {:?}, {:?}", a, b);
10821087
use rustc::mir::interpret::{Value, PrimVal};
@@ -1096,7 +1101,11 @@ pub fn compare_const_vals(a: &ConstVal, b: &ConstVal, ty: Ty) -> Option<Ordering
10961101
// FIXME(oli-obk): report cmp errors?
10971102
l.try_cmp(r).ok()
10981103
},
1099-
ty::TyInt(_) => Some((a as i128).cmp(&(b as i128))),
1104+
ty::TyInt(_) => {
1105+
let a = interpret::sign_extend(tcx, a, ty).expect("layout error for TyInt");
1106+
let b = interpret::sign_extend(tcx, b, ty).expect("layout error for TyInt");
1107+
Some((a as i128).cmp(&(b as i128)))
1108+
},
11001109
_ => Some(a.cmp(&b)),
11011110
}
11021111
},

src/librustc_mir/interpret/eval_context.rs

+2-12
Original file line numberDiff line numberDiff line change
@@ -1679,21 +1679,11 @@ impl<'a, 'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M
16791679
}
16801680

16811681
pub fn sign_extend(&self, value: u128, ty: Ty<'tcx>) -> EvalResult<'tcx, u128> {
1682-
let layout = self.layout_of(ty)?;
1683-
let size = layout.size.bits();
1684-
assert!(layout.abi.is_signed());
1685-
// sign extend
1686-
let amt = 128 - size;
1687-
// shift the unsigned value to the left
1688-
// and back to the right as signed (essentially fills with FF on the left)
1689-
Ok((((value << amt) as i128) >> amt) as u128)
1682+
super::sign_extend(self.tcx.tcx, value, ty)
16901683
}
16911684

16921685
pub fn truncate(&self, value: u128, ty: Ty<'tcx>) -> EvalResult<'tcx, u128> {
1693-
let size = self.layout_of(ty)?.size.bits();
1694-
let amt = 128 - size;
1695-
// truncate (shift left to drop out leftover values, shift right to fill with zeroes)
1696-
Ok((value << amt) >> amt)
1686+
super::truncate(self.tcx.tcx, value, ty)
16971687
}
16981688
}
16991689

src/librustc_mir/interpret/mod.rs

+24
Original file line numberDiff line numberDiff line change
@@ -31,3 +31,27 @@ pub use self::const_eval::{
3131
pub use self::machine::Machine;
3232

3333
pub use self::memory::{write_target_uint, write_target_int, read_target_uint};
34+
35+
use rustc::mir::interpret::{EvalResult, EvalErrorKind};
36+
use rustc::ty::{Ty, TyCtxt, ParamEnv};
37+
38+
pub fn sign_extend<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>, value: u128, ty: Ty<'tcx>) -> EvalResult<'tcx, u128> {
39+
let param_env = ParamEnv::empty();
40+
let layout = tcx.layout_of(param_env.and(ty)).map_err(|layout| EvalErrorKind::Layout(layout))?;
41+
let size = layout.size.bits();
42+
assert!(layout.abi.is_signed());
43+
// sign extend
44+
let amt = 128 - size;
45+
// shift the unsigned value to the left
46+
// and back to the right as signed (essentially fills with FF on the left)
47+
Ok((((value << amt) as i128) >> amt) as u128)
48+
}
49+
50+
pub fn truncate<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>, value: u128, ty: Ty<'tcx>) -> EvalResult<'tcx, u128> {
51+
let param_env = ParamEnv::empty();
52+
let layout = tcx.layout_of(param_env.and(ty)).map_err(|layout| EvalErrorKind::Layout(layout))?;
53+
let size = layout.size.bits();
54+
let amt = 128 - size;
55+
// truncate (shift left to drop out leftover values, shift right to fill with zeroes)
56+
Ok((value << amt) >> amt)
57+
}
+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
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+
// compile-pass
12+
13+
fn main() {
14+
const MIN: i8 = -5;
15+
match 5i8 {
16+
MIN...-1 => {},
17+
_ => {},
18+
}
19+
}

0 commit comments

Comments
 (0)