Auto merge of #113470 - compiler-errors:new-solver-structurally-resolve-pat, r=lcnr

Structurally resolve in pattern matching when peeling refs in new solver

Let me know if you want me to commit the minimized test:
```rust
fn test() {}

fn test2() {}

fn main() {
    let tests: &[(_, fn())] = &[
        ("test", test),
        ("test2", test2),
    ];

    for (a, b) in tests {
        todo!();
    }
}
```

In that test above, the match scrutinee is `<std::vec::Iter<(&'static str, fn())> as Iterator>::Item`, which we cannot peel the refs from.

We also need to structurally resolve in the loop, since structural resolve is inherently shallow. I haven't come up with a test where this matters, but I can if you care.

Also,  I removed two other calls to `resolve_vars_with_obligations` in diagnostics code that I'm pretty convinced are not useful.

r? `@lcnr`
This commit is contained in:
bors 2023-07-11 15:29:47 +00:00
commit 0a2681cc49
3 changed files with 11 additions and 5 deletions

View File

@ -335,8 +335,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
expected: Ty<'tcx>, expected: Ty<'tcx>,
mut def_bm: BindingMode, mut def_bm: BindingMode,
) -> (Ty<'tcx>, BindingMode) { ) -> (Ty<'tcx>, BindingMode) {
let mut expected = self.resolve_vars_with_obligations(expected); let mut expected = self.try_structurally_resolve_type(pat.span, expected);
// Peel off as many `&` or `&mut` from the scrutinee type as possible. For example, // Peel off as many `&` or `&mut` from the scrutinee type as possible. For example,
// for `match &&&mut Some(5)` the loop runs three times, aborting when it reaches // for `match &&&mut Some(5)` the loop runs three times, aborting when it reaches
// the `Some(5)` which is not of type Ref. // the `Some(5)` which is not of type Ref.
@ -353,7 +352,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
// Preserve the reference type. We'll need it later during THIR lowering. // Preserve the reference type. We'll need it later during THIR lowering.
pat_adjustments.push(expected); pat_adjustments.push(expected);
expected = inner_ty; expected = self.try_structurally_resolve_type(pat.span, inner_ty);
def_bm = ty::BindByReference(match def_bm { def_bm = ty::BindByReference(match def_bm {
// If default binding mode is by value, make it `ref` or `ref mut` // If default binding mode is by value, make it `ref` or `ref mut`
// (depending on whether we observe `&` or `&mut`). // (depending on whether we observe `&` or `&mut`).
@ -627,6 +626,9 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
local_ty local_ty
} }
/// When a variable is bound several times in a `PatKind::Or`, it'll resolve all of the
/// subsequent bindings of the same name to the first usage. Verify that all of these
/// bindings have the same type by comparing them all against the type of that first pat.
fn check_binding_alt_eq_ty( fn check_binding_alt_eq_ty(
&self, &self,
ba: hir::BindingAnnotation, ba: hir::BindingAnnotation,
@ -638,7 +640,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
let var_ty = self.local_ty(span, var_id); let var_ty = self.local_ty(span, var_id);
if let Some(mut err) = self.demand_eqtype_pat_diag(span, var_ty, ty, ti) { if let Some(mut err) = self.demand_eqtype_pat_diag(span, var_ty, ty, ti) {
let hir = self.tcx.hir(); let hir = self.tcx.hir();
let var_ty = self.resolve_vars_with_obligations(var_ty); let var_ty = self.resolve_vars_if_possible(var_ty);
let msg = format!("first introduced with type `{var_ty}` here"); let msg = format!("first introduced with type `{var_ty}` here");
err.span_label(hir.span(var_id), msg); err.span_label(hir.span(var_id), msg);
let in_match = hir.parent_iter(var_id).any(|(_, n)| { let in_match = hir.parent_iter(var_id).any(|(_, n)| {
@ -656,7 +658,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
&mut err, &mut err,
span, span,
var_ty, var_ty,
self.resolve_vars_with_obligations(ty), self.resolve_vars_if_possible(ty),
ba, ba,
); );
err.emit(); err.emit();

View File

@ -1640,6 +1640,8 @@ impl<'tcx> InferCtxtPrivExt<'tcx> for TypeErrCtxt<'_, 'tcx> {
) )
.into(), .into(),
}; };
// FIXME(-Ztrait-solver=next): For diagnostic purposes, it would be nice
// to deeply normalize this type.
let normalized_term = let normalized_term =
ocx.normalize(&obligation.cause, obligation.param_env, unnormalized_term); ocx.normalize(&obligation.cause, obligation.param_env, unnormalized_term);

View File

@ -1,4 +1,6 @@
// run-pass // run-pass
// revisions: current next
//[next] compile-flags: -Ztrait-solver=next
#![allow(unused_must_use)] #![allow(unused_must_use)]
#![feature(c_unwind)] #![feature(c_unwind)]