Skip to content

Commit ee05efd

Browse files
authored
Rollup merge of rust-lang#67439 - Centril:clean-hair-slice, r=matthewjasper
Cleanup `lower_pattern_unadjusted` & Improve slice pat typeck Following up on rust-lang#67318, in this PR, the HAIR lowering of patterns (`lower_pattern_unadjusted`) is cleaned up with a focus on slice patterns (in particular, some dead code is removed). Moreover, `check_pat_slice` is refactored some more. r? @matthewjasper
2 parents a698791 + 7353ff2 commit ee05efd

File tree

3 files changed

+96
-162
lines changed

3 files changed

+96
-162
lines changed

src/librustc_mir/hair/pattern/_match.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -620,10 +620,11 @@ impl<'a, 'tcx> MatchCheckCtxt<'a, 'tcx> {
620620
enum SliceKind {
621621
/// Patterns of length `n` (`[x, y]`).
622622
FixedLen(u64),
623-
/// Patterns using the `..` notation (`[x, .., y]`). Captures any array constructor of `length
624-
/// >= i + j`. In the case where `array_len` is `Some(_)`, this indicates that we only care
625-
/// about the first `i` and the last `j` values of the array, and everything in between is a
626-
/// wildcard `_`.
623+
/// Patterns using the `..` notation (`[x, .., y]`).
624+
/// Captures any array constructor of `length >= i + j`.
625+
/// In the case where `array_len` is `Some(_)`,
626+
/// this indicates that we only care about the first `i` and the last `j` values of the array,
627+
/// and everything in between is a wildcard `_`.
627628
VarLen(u64, u64),
628629
}
629630

src/librustc_mir/hair/pattern/mod.rs

+41-121
Original file line numberDiff line numberDiff line change
@@ -445,6 +445,11 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
445445
fn lower_pattern_unadjusted(&mut self, pat: &'tcx hir::Pat) -> Pat<'tcx> {
446446
let mut ty = self.tables.node_type(pat.hir_id);
447447

448+
if let ty::Error = ty.kind {
449+
// Avoid ICEs (e.g., #50577 and #50585).
450+
return Pat { span: pat.span, ty, kind: Box::new(PatKind::Wild) };
451+
}
452+
448453
let kind = match pat.kind {
449454
hir::PatKind::Wild => PatKind::Wild,
450455

@@ -544,57 +549,19 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
544549
}
545550

546551
hir::PatKind::Slice(ref prefix, ref slice, ref suffix) => {
547-
match ty.kind {
548-
ty::Ref(_, ty, _) =>
549-
PatKind::Deref {
550-
subpattern: Pat {
551-
ty,
552-
span: pat.span,
553-
kind: Box::new(self.slice_or_array_pattern(
554-
pat.span, ty, prefix, slice, suffix))
555-
},
556-
},
557-
ty::Slice(..) |
558-
ty::Array(..) =>
559-
self.slice_or_array_pattern(pat.span, ty, prefix, slice, suffix),
560-
ty::Error => { // Avoid ICE
561-
return Pat { span: pat.span, ty, kind: Box::new(PatKind::Wild) };
562-
}
563-
_ =>
564-
span_bug!(
565-
pat.span,
566-
"unexpanded type for vector pattern: {:?}",
567-
ty),
568-
}
552+
self.slice_or_array_pattern(pat.span, ty, prefix, slice, suffix)
569553
}
570554

571-
hir::PatKind::Tuple(ref subpatterns, ddpos) => {
572-
match ty.kind {
573-
ty::Tuple(ref tys) => {
574-
let subpatterns =
575-
subpatterns.iter()
576-
.enumerate_and_adjust(tys.len(), ddpos)
577-
.map(|(i, subpattern)| FieldPat {
578-
field: Field::new(i),
579-
pattern: self.lower_pattern(subpattern)
580-
})
581-
.collect();
582-
583-
PatKind::Leaf { subpatterns }
584-
}
585-
ty::Error => { // Avoid ICE (#50577)
586-
return Pat { span: pat.span, ty, kind: Box::new(PatKind::Wild) };
587-
}
555+
hir::PatKind::Tuple(ref pats, ddpos) => {
556+
let tys = match ty.kind {
557+
ty::Tuple(ref tys) => tys,
588558
_ => span_bug!(pat.span, "unexpected type for tuple pattern: {:?}", ty),
589-
}
559+
};
560+
let subpatterns = self.lower_tuple_subpats(pats, tys.len(), ddpos);
561+
PatKind::Leaf { subpatterns }
590562
}
591563

592564
hir::PatKind::Binding(_, id, ident, ref sub) => {
593-
let var_ty = self.tables.node_type(pat.hir_id);
594-
if let ty::Error = var_ty.kind {
595-
// Avoid ICE
596-
return Pat { span: pat.span, ty, kind: Box::new(PatKind::Wild) };
597-
};
598565
let bm = *self.tables.pat_binding_modes().get(pat.hir_id)
599566
.expect("missing binding mode");
600567
let (mutability, mode) = match bm {
@@ -609,13 +576,14 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
609576

610577
// A ref x pattern is the same node used for x, and as such it has
611578
// x's type, which is &T, where we want T (the type being matched).
579+
let var_ty = ty;
612580
if let ty::BindByReference(_) = bm {
613581
if let ty::Ref(_, rty, _) = ty.kind {
614582
ty = rty;
615583
} else {
616584
bug!("`ref {}` has wrong type {}", ident, ty);
617585
}
618-
}
586+
};
619587

620588
PatKind::Binding {
621589
mutability,
@@ -627,28 +595,14 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
627595
}
628596
}
629597

630-
hir::PatKind::TupleStruct(ref qpath, ref subpatterns, ddpos) => {
598+
hir::PatKind::TupleStruct(ref qpath, ref pats, ddpos) => {
631599
let res = self.tables.qpath_res(qpath, pat.hir_id);
632600
let adt_def = match ty.kind {
633601
ty::Adt(adt_def, _) => adt_def,
634-
ty::Error => { // Avoid ICE (#50585)
635-
return Pat { span: pat.span, ty, kind: Box::new(PatKind::Wild) };
636-
}
637-
_ => span_bug!(pat.span,
638-
"tuple struct pattern not applied to an ADT {:?}",
639-
ty),
602+
_ => span_bug!(pat.span, "tuple struct pattern not applied to an ADT {:?}", ty),
640603
};
641604
let variant_def = adt_def.variant_of_res(res);
642-
643-
let subpatterns =
644-
subpatterns.iter()
645-
.enumerate_and_adjust(variant_def.fields.len(), ddpos)
646-
.map(|(i, field)| FieldPat {
647-
field: Field::new(i),
648-
pattern: self.lower_pattern(field),
649-
})
650-
.collect();
651-
605+
let subpatterns = self.lower_tuple_subpats(pats, variant_def.fields.len(), ddpos);
652606
self.lower_variant_or_leaf(res, pat.hir_id, pat.span, ty, subpatterns)
653607
}
654608

@@ -668,11 +622,7 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
668622
self.lower_variant_or_leaf(res, pat.hir_id, pat.span, ty, subpatterns)
669623
}
670624

671-
hir::PatKind::Or(ref pats) => {
672-
PatKind::Or {
673-
pats: pats.iter().map(|p| self.lower_pattern(p)).collect(),
674-
}
675-
}
625+
hir::PatKind::Or(ref pats) => PatKind::Or { pats: self.lower_patterns(pats) },
676626
};
677627

678628
Pat {
@@ -682,80 +632,50 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
682632
}
683633
}
684634

635+
fn lower_tuple_subpats(
636+
&mut self,
637+
pats: &'tcx [P<hir::Pat>],
638+
expected_len: usize,
639+
gap_pos: Option<usize>,
640+
) -> Vec<FieldPat<'tcx>> {
641+
pats.iter()
642+
.enumerate_and_adjust(expected_len, gap_pos)
643+
.map(|(i, subpattern)| FieldPat {
644+
field: Field::new(i),
645+
pattern: self.lower_pattern(subpattern)
646+
})
647+
.collect()
648+
}
649+
685650
fn lower_patterns(&mut self, pats: &'tcx [P<hir::Pat>]) -> Vec<Pat<'tcx>> {
686651
pats.iter().map(|p| self.lower_pattern(p)).collect()
687652
}
688653

689-
fn lower_opt_pattern(&mut self, pat: &'tcx Option<P<hir::Pat>>) -> Option<Pat<'tcx>>
690-
{
654+
fn lower_opt_pattern(&mut self, pat: &'tcx Option<P<hir::Pat>>) -> Option<Pat<'tcx>> {
691655
pat.as_ref().map(|p| self.lower_pattern(p))
692656
}
693657

694-
fn flatten_nested_slice_patterns(
695-
&mut self,
696-
prefix: Vec<Pat<'tcx>>,
697-
slice: Option<Pat<'tcx>>,
698-
suffix: Vec<Pat<'tcx>>)
699-
-> (Vec<Pat<'tcx>>, Option<Pat<'tcx>>, Vec<Pat<'tcx>>)
700-
{
701-
let orig_slice = match slice {
702-
Some(orig_slice) => orig_slice,
703-
None => return (prefix, slice, suffix)
704-
};
705-
let orig_prefix = prefix;
706-
let orig_suffix = suffix;
707-
708-
// dance because of intentional borrow-checker stupidity.
709-
let kind = *orig_slice.kind;
710-
match kind {
711-
PatKind::Slice { prefix, slice, mut suffix } |
712-
PatKind::Array { prefix, slice, mut suffix } => {
713-
let mut orig_prefix = orig_prefix;
714-
715-
orig_prefix.extend(prefix);
716-
suffix.extend(orig_suffix);
717-
718-
(orig_prefix, slice, suffix)
719-
}
720-
_ => {
721-
(orig_prefix, Some(Pat {
722-
kind: box kind, ..orig_slice
723-
}), orig_suffix)
724-
}
725-
}
726-
}
727-
728658
fn slice_or_array_pattern(
729659
&mut self,
730660
span: Span,
731661
ty: Ty<'tcx>,
732662
prefix: &'tcx [P<hir::Pat>],
733663
slice: &'tcx Option<P<hir::Pat>>,
734-
suffix: &'tcx [P<hir::Pat>])
735-
-> PatKind<'tcx>
736-
{
664+
suffix: &'tcx [P<hir::Pat>],
665+
) -> PatKind<'tcx> {
737666
let prefix = self.lower_patterns(prefix);
738667
let slice = self.lower_opt_pattern(slice);
739668
let suffix = self.lower_patterns(suffix);
740-
let (prefix, slice, suffix) =
741-
self.flatten_nested_slice_patterns(prefix, slice, suffix);
742-
743669
match ty.kind {
744-
ty::Slice(..) => {
745-
// matching a slice or fixed-length array
746-
PatKind::Slice { prefix: prefix, slice: slice, suffix: suffix }
747-
}
748-
670+
// Matching a slice, `[T]`.
671+
ty::Slice(..) => PatKind::Slice { prefix, slice, suffix },
672+
// Fixed-length array, `[T; len]`.
749673
ty::Array(_, len) => {
750-
// fixed-length array
751674
let len = len.eval_usize(self.tcx, self.param_env);
752675
assert!(len >= prefix.len() as u64 + suffix.len() as u64);
753-
PatKind::Array { prefix: prefix, slice: slice, suffix: suffix }
754-
}
755-
756-
_ => {
757-
span_bug!(span, "bad slice pattern type {:?}", ty);
676+
PatKind::Array { prefix, slice, suffix }
758677
}
678+
_ => span_bug!(span, "bad slice pattern type {:?}", ty),
759679
}
760680
}
761681

src/librustc_typeck/check/pat.rs

+50-37
Original file line numberDiff line numberDiff line change
@@ -1174,47 +1174,23 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
11741174
def_bm: BindingMode,
11751175
discrim_span: Option<Span>,
11761176
) -> Ty<'tcx> {
1177-
let tcx = self.tcx;
1178-
let expected_ty = self.structurally_resolved_type(span, expected);
1179-
let (inner_ty, slice_ty) = match expected_ty.kind {
1177+
let err = self.tcx.types.err;
1178+
let expected = self.structurally_resolved_type(span, expected);
1179+
let (inner_ty, slice_ty, expected) = match expected.kind {
11801180
// An array, so we might have something like `let [a, b, c] = [0, 1, 2];`.
1181-
ty::Array(inner_ty, size) => {
1182-
let slice_ty = if let Some(size) = size.try_eval_usize(tcx, self.param_env) {
1183-
// Now we know the length...
1184-
let min_len = before.len() as u64 + after.len() as u64;
1185-
if slice.is_none() {
1186-
// ...and since there is no variable-length pattern,
1187-
// we require an exact match between the number of elements
1188-
// in the array pattern and as provided by the matched type.
1189-
if min_len != size {
1190-
self.error_scrutinee_inconsistent_length(span, min_len, size)
1191-
}
1192-
tcx.types.err
1193-
} else if let Some(rest) = size.checked_sub(min_len) {
1194-
// The variable-length pattern was there,
1195-
// so it has an array type with the remaining elements left as its size...
1196-
tcx.mk_array(inner_ty, rest)
1197-
} else {
1198-
// ...however, in this case, there were no remaining elements.
1199-
// That is, the slice pattern requires more than the array type offers.
1200-
self.error_scrutinee_with_rest_inconsistent_length(span, min_len, size);
1201-
tcx.types.err
1202-
}
1203-
} else {
1204-
// No idea what the length is, which happens if we have e.g.,
1205-
// `let [a, b] = arr` where `arr: [T; N]` where `const N: usize`.
1206-
self.error_scrutinee_unfixed_length(span);
1207-
tcx.types.err
1208-
};
1209-
(inner_ty, slice_ty)
1181+
ty::Array(inner_ty, len) => {
1182+
let min = before.len() as u64 + after.len() as u64;
1183+
let slice_ty = self.check_array_pat_len(span, slice, len, min)
1184+
.map_or(err, |len| self.tcx.mk_array(inner_ty, len));
1185+
(inner_ty, slice_ty, expected)
12101186
}
1211-
ty::Slice(inner_ty) => (inner_ty, expected_ty),
1187+
ty::Slice(inner_ty) => (inner_ty, expected, expected),
12121188
// The expected type must be an array or slice, but was neither, so error.
12131189
_ => {
1214-
if !expected_ty.references_error() {
1215-
self.error_expected_array_or_slice(span, expected_ty);
1190+
if !expected.references_error() {
1191+
self.error_expected_array_or_slice(span, expected);
12161192
}
1217-
(tcx.types.err, tcx.types.err)
1193+
(err, err, err)
12181194
}
12191195
};
12201196

@@ -1230,7 +1206,44 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12301206
for elt in after {
12311207
self.check_pat(&elt, inner_ty, def_bm, discrim_span);
12321208
}
1233-
expected_ty
1209+
expected
1210+
}
1211+
1212+
/// Type check the length of an array pattern.
1213+
///
1214+
/// Return the length of the variable length pattern,
1215+
/// if it exists and there are no errors.
1216+
fn check_array_pat_len(
1217+
&self,
1218+
span: Span,
1219+
slice: Option<&'tcx Pat>,
1220+
len: &ty::Const<'tcx>,
1221+
min_len: u64,
1222+
) -> Option<u64> {
1223+
if let Some(len) = len.try_eval_usize(self.tcx, self.param_env) {
1224+
// Now we know the length...
1225+
if slice.is_none() {
1226+
// ...and since there is no variable-length pattern,
1227+
// we require an exact match between the number of elements
1228+
// in the array pattern and as provided by the matched type.
1229+
if min_len != len {
1230+
self.error_scrutinee_inconsistent_length(span, min_len, len);
1231+
}
1232+
} else if let r @ Some(_) = len.checked_sub(min_len) {
1233+
// The variable-length pattern was there,
1234+
// so it has an array type with the remaining elements left as its size...
1235+
return r;
1236+
} else {
1237+
// ...however, in this case, there were no remaining elements.
1238+
// That is, the slice pattern requires more than the array type offers.
1239+
self.error_scrutinee_with_rest_inconsistent_length(span, min_len, len);
1240+
}
1241+
} else {
1242+
// No idea what the length is, which happens if we have e.g.,
1243+
// `let [a, b] = arr` where `arr: [T; N]` where `const N: usize`.
1244+
self.error_scrutinee_unfixed_length(span);
1245+
}
1246+
None
12341247
}
12351248

12361249
fn error_scrutinee_inconsistent_length(&self, span: Span, min_len: u64, size: u64) {

0 commit comments

Comments
 (0)