Skip to content

Commit d46ff64

Browse files
committed
Fix a subtle regression
Before, the SwitchInt cases were computed in two passes: if the first pass accepted e.g. 0..=5 and then 1, the second pass would not accept 0..=5 anymore because 1 would be listed in the SwitchInt options. Now there's a single pass, so if we sort 0..=5 we must take care to not sort a subsequent 1.
1 parent edea739 commit d46ff64

File tree

3 files changed

+44
-3
lines changed

3 files changed

+44
-3
lines changed

compiler/rustc_mir_build/src/build/matches/mod.rs

+6
Original file line numberDiff line numberDiff line change
@@ -1091,6 +1091,12 @@ enum TestCase<'pat, 'tcx> {
10911091
Or { pats: Box<[FlatPat<'pat, 'tcx>]> },
10921092
}
10931093

1094+
impl<'pat, 'tcx> TestCase<'pat, 'tcx> {
1095+
fn as_range(&self) -> Option<&'pat PatRange<'tcx>> {
1096+
if let Self::Range(v) = self { Some(*v) } else { None }
1097+
}
1098+
}
1099+
10941100
#[derive(Debug, Clone)]
10951101
pub(crate) struct MatchPair<'pat, 'tcx> {
10961102
/// This place...

compiler/rustc_mir_build/src/build/matches/test.rs

+24-3
Original file line numberDiff line numberDiff line change
@@ -510,9 +510,30 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
510510
(TestKind::SwitchInt, &TestCase::Constant { value })
511511
if is_switch_ty(match_pair.pattern.ty) =>
512512
{
513-
fully_matched = true;
514-
let bits = value.eval_bits(self.tcx, self.param_env);
515-
Some(TestBranch::Constant(value, bits))
513+
// Beware: there might be some ranges sorted into the failure case; we must not add
514+
// a success case that could be matched by one of these ranges.
515+
let is_covering_range = |test_case: &TestCase<'_, 'tcx>| {
516+
test_case.as_range().is_some_and(|range| {
517+
matches!(range.contains(value, self.tcx, self.param_env), None | Some(true))
518+
})
519+
};
520+
let is_conflicting_candidate = |candidate: &&mut Candidate<'_, 'tcx>| {
521+
candidate
522+
.match_pairs
523+
.iter()
524+
.any(|mp| mp.place == *test_place && is_covering_range(&mp.test_case))
525+
};
526+
if sorted_candidates
527+
.get(&TestBranch::Failure)
528+
.is_some_and(|candidates| candidates.iter().any(is_conflicting_candidate))
529+
{
530+
fully_matched = false;
531+
None
532+
} else {
533+
fully_matched = true;
534+
let bits = value.eval_bits(self.tcx, self.param_env);
535+
Some(TestBranch::Constant(value, bits))
536+
}
516537
}
517538
(TestKind::SwitchInt, TestCase::Range(range)) => {
518539
fully_matched = false;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
//@ run-pass
2+
//
3+
// Regression test for match lowering to MIR: when gathering candidates, by the time we get to the
4+
// range we know the range will only match on the failure case of the switchint. Hence we mustn't
5+
// add the `1` to the switchint or the range would be incorrectly sorted.
6+
#![allow(unreachable_patterns)]
7+
fn main() {
8+
match 1 {
9+
10 => unreachable!(),
10+
0..=5 => {}
11+
1 => unreachable!(),
12+
_ => unreachable!(),
13+
}
14+
}

0 commit comments

Comments
 (0)