Skip to content

Commit 5c5062a

Browse files
authored
Unrolled build for rust-lang#119835
Rollup merge of rust-lang#119835 - Nadrieril:simplify-empty-logic, r=compiler-errors Exhaustiveness: simplify empty pattern logic The logic that handles empty patterns had gotten quite convoluted. This PR simplifies it a lot. I tried to make the logic as easy as possible to follow; this only does logically equivalent changes. The first commit is a drive-by comment clarification that was requested after another PR a while back. r? `@compiler-errors`
2 parents 0547c41 + d95644d commit 5c5062a

File tree

4 files changed

+30
-51
lines changed

4 files changed

+30
-51
lines changed

compiler/rustc_mir_build/src/thir/pattern/check_match.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -289,7 +289,8 @@ impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
289289
fn is_known_valid_scrutinee(&self, scrutinee: &Expr<'tcx>) -> bool {
290290
use ExprKind::*;
291291
match &scrutinee.kind {
292-
// Both pointers and references can validly point to a place with invalid data.
292+
// Pointers can validly point to a place with invalid data. It is undecided whether
293+
// references can too, so we conservatively assume they can.
293294
Deref { .. } => false,
294295
// Inherit validity of the parent place, unless the parent is an union.
295296
Field { lhs, .. } => {

compiler/rustc_pattern_analysis/src/constructor.rs

+5-14
Original file line numberDiff line numberDiff line change
@@ -861,12 +861,14 @@ impl<Cx: TypeCx> ConstructorSet<Cx> {
861861
/// any) are missing; 2/ split constructors to handle non-trivial intersections e.g. on ranges
862862
/// or slices. This can get subtle; see [`SplitConstructorSet`] for details of this operation
863863
/// and its invariants.
864-
#[instrument(level = "debug", skip(self, pcx, ctors), ret)]
864+
#[instrument(level = "debug", skip(self, ctors), ret)]
865865
pub(crate) fn split<'a>(
866866
&self,
867-
pcx: &PlaceCtxt<'a, Cx>,
868867
ctors: impl Iterator<Item = &'a Constructor<Cx>> + Clone,
869-
) -> SplitConstructorSet<Cx> {
868+
) -> SplitConstructorSet<Cx>
869+
where
870+
Cx: 'a,
871+
{
870872
let mut present: SmallVec<[_; 1]> = SmallVec::new();
871873
// Empty constructors found missing.
872874
let mut missing_empty = Vec::new();
@@ -1006,17 +1008,6 @@ impl<Cx: TypeCx> ConstructorSet<Cx> {
10061008
}
10071009
}
10081010

1009-
// We have now grouped all the constructors into 3 buckets: present, missing, missing_empty.
1010-
// In the absence of the `exhaustive_patterns` feature however, we don't count nested empty
1011-
// types as empty. Only non-nested `!` or `enum Foo {}` are considered empty.
1012-
if !pcx.mcx.tycx.is_exhaustive_patterns_feature_on()
1013-
&& !(pcx.is_scrutinee && matches!(self, Self::NoConstructors))
1014-
{
1015-
// Treat all missing constructors as nonempty.
1016-
// This clears `missing_empty`.
1017-
missing.append(&mut missing_empty);
1018-
}
1019-
10201011
SplitConstructorSet { present, missing, missing_empty }
10211012
}
10221013
}

compiler/rustc_pattern_analysis/src/lints.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@ impl<'p, 'tcx> PatternColumn<'p, 'tcx> {
5656
) -> Result<SplitConstructorSet<'p, 'tcx>, ErrorGuaranteed> {
5757
let column_ctors = self.patterns.iter().map(|p| p.ctor());
5858
let ctors_for_ty = &pcx.ctors_for_ty()?;
59-
Ok(ctors_for_ty.split(pcx, column_ctors))
59+
Ok(ctors_for_ty.split(column_ctors))
6060
}
6161

6262
/// Does specialization: given a constructor, this takes the patterns from the column that match

compiler/rustc_pattern_analysis/src/usefulness.rs

+22-35
Original file line numberDiff line numberDiff line change
@@ -737,15 +737,13 @@ pub(crate) struct PlaceCtxt<'a, Cx: TypeCx> {
737737
pub(crate) mcx: MatchCtxt<'a, Cx>,
738738
/// Type of the place under investigation.
739739
pub(crate) ty: Cx::Ty,
740-
/// Whether the place is the original scrutinee place, as opposed to a subplace of it.
741-
pub(crate) is_scrutinee: bool,
742740
}
743741

744742
impl<'a, Cx: TypeCx> PlaceCtxt<'a, Cx> {
745743
/// A `PlaceCtxt` when code other than `is_useful` needs one.
746744
#[cfg_attr(not(feature = "rustc"), allow(dead_code))]
747745
pub(crate) fn new_dummy(mcx: MatchCtxt<'a, Cx>, ty: Cx::Ty) -> Self {
748-
PlaceCtxt { mcx, ty, is_scrutinee: false }
746+
PlaceCtxt { mcx, ty }
749747
}
750748

751749
pub(crate) fn ctor_arity(&self, ctor: &Constructor<Cx>) -> usize {
@@ -768,30 +766,16 @@ impl<'a, Cx: TypeCx> PlaceCtxt<'a, Cx> {
768766
pub enum ValidityConstraint {
769767
ValidOnly,
770768
MaybeInvalid,
771-
/// Option for backwards compatibility: the place is not known to be valid but we allow omitting
772-
/// `useful && !reachable` arms anyway.
773-
MaybeInvalidButAllowOmittingArms,
774769
}
775770

776771
impl ValidityConstraint {
777772
pub fn from_bool(is_valid_only: bool) -> Self {
778773
if is_valid_only { ValidOnly } else { MaybeInvalid }
779774
}
780775

781-
fn allow_omitting_side_effecting_arms(self) -> Self {
782-
match self {
783-
MaybeInvalid | MaybeInvalidButAllowOmittingArms => MaybeInvalidButAllowOmittingArms,
784-
// There are no side-effecting empty arms here, nothing to do.
785-
ValidOnly => ValidOnly,
786-
}
787-
}
788-
789776
fn is_known_valid(self) -> bool {
790777
matches!(self, ValidOnly)
791778
}
792-
fn allows_omitting_empty_arms(self) -> bool {
793-
matches!(self, ValidOnly | MaybeInvalidButAllowOmittingArms)
794-
}
795779

796780
/// If the place has validity given by `self` and we read that the value at the place has
797781
/// constructor `ctor`, this computes what we can assume about the validity of the constructor
@@ -814,7 +798,7 @@ impl fmt::Display for ValidityConstraint {
814798
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
815799
let s = match self {
816800
ValidOnly => "✓",
817-
MaybeInvalid | MaybeInvalidButAllowOmittingArms => "?",
801+
MaybeInvalid => "?",
818802
};
819803
write!(f, "{s}")
820804
}
@@ -1447,41 +1431,44 @@ fn compute_exhaustiveness_and_usefulness<'a, 'p, Cx: TypeCx>(
14471431
};
14481432

14491433
debug!("ty: {ty:?}");
1450-
let pcx = &PlaceCtxt { mcx, ty, is_scrutinee: is_top_level };
1434+
let pcx = &PlaceCtxt { mcx, ty };
1435+
let ctors_for_ty = pcx.ctors_for_ty()?;
14511436

14521437
// Whether the place/column we are inspecting is known to contain valid data.
14531438
let place_validity = matrix.place_validity[0];
1454-
// For backwards compability we allow omitting some empty arms that we ideally shouldn't.
1455-
let place_validity = place_validity.allow_omitting_side_effecting_arms();
1439+
// We treat match scrutinees of type `!` or `EmptyEnum` differently.
1440+
let is_toplevel_exception =
1441+
is_top_level && matches!(ctors_for_ty, ConstructorSet::NoConstructors);
1442+
// Whether empty patterns can be omitted for exhaustiveness.
1443+
let can_omit_empty_arms = is_toplevel_exception || mcx.tycx.is_exhaustive_patterns_feature_on();
1444+
// Whether empty patterns are counted as useful or not.
1445+
let empty_arms_are_unreachable = place_validity.is_known_valid() && can_omit_empty_arms;
14561446

14571447
// Analyze the constructors present in this column.
14581448
let ctors = matrix.heads().map(|p| p.ctor());
1459-
let ctors_for_ty = pcx.ctors_for_ty()?;
1460-
let is_integers = matches!(ctors_for_ty, ConstructorSet::Integers { .. }); // For diagnostics.
1461-
let split_set = ctors_for_ty.split(pcx, ctors);
1449+
let mut split_set = ctors_for_ty.split(ctors);
14621450
let all_missing = split_set.present.is_empty();
1463-
14641451
// Build the set of constructors we will specialize with. It must cover the whole type.
1452+
// We need to iterate over a full set of constructors, so we add `Missing` to represent the
1453+
// missing ones. This is explained under "Constructor Splitting" at the top of this file.
14651454
let mut split_ctors = split_set.present;
1466-
if !split_set.missing.is_empty() {
1467-
// We need to iterate over a full set of constructors, so we add `Missing` to represent the
1468-
// missing ones. This is explained under "Constructor Splitting" at the top of this file.
1469-
split_ctors.push(Constructor::Missing);
1470-
} else if !split_set.missing_empty.is_empty() && !place_validity.is_known_valid() {
1471-
// The missing empty constructors are reachable if the place can contain invalid data.
1455+
if !(split_set.missing.is_empty()
1456+
&& (split_set.missing_empty.is_empty() || empty_arms_are_unreachable))
1457+
{
14721458
split_ctors.push(Constructor::Missing);
14731459
}
14741460

14751461
// Decide what constructors to report.
1462+
let is_integers = matches!(ctors_for_ty, ConstructorSet::Integers { .. });
14761463
let always_report_all = is_top_level && !is_integers;
14771464
// Whether we should report "Enum::A and Enum::C are missing" or "_ is missing".
14781465
let report_individual_missing_ctors = always_report_all || !all_missing;
14791466
// Which constructors are considered missing. We ensure that `!missing_ctors.is_empty() =>
1480-
// split_ctors.contains(Missing)`. The converse usually holds except in the
1481-
// `MaybeInvalidButAllowOmittingArms` backwards-compatibility case.
1467+
// split_ctors.contains(Missing)`. The converse usually holds except when
1468+
// `!place_validity.is_known_valid()`.
14821469
let mut missing_ctors = split_set.missing;
1483-
if !place_validity.allows_omitting_empty_arms() {
1484-
missing_ctors.extend(split_set.missing_empty);
1470+
if !can_omit_empty_arms {
1471+
missing_ctors.append(&mut split_set.missing_empty);
14851472
}
14861473

14871474
let mut ret = WitnessMatrix::empty();

0 commit comments

Comments
 (0)