Skip to content

Commit 737b461

Browse files
committed
Auto merge of rust-lang#113086 - lcnr:rust8, r=compiler-errors
implement deep normalization via the new solver together with rust-lang#112869 this should remove all uses of the old solver with `-Ztrait-solver=next`. see https://hackmd.io/V0qsUB_fTxexfQO_pcOcrg for a description of this PR. Will move that doc to the `rustc-dev-guide` after merging this. r? `@compiler-errors`
2 parents 1623634 + be6a344 commit 737b461

File tree

63 files changed

+834
-296
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

63 files changed

+834
-296
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -4136,6 +4136,7 @@ dependencies = [
41364136
name = "rustc_ty_utils"
41374137
version = "0.0.0"
41384138
dependencies = [
4139+
"itertools",
41394140
"rustc_data_structures",
41404141
"rustc_errors",
41414142
"rustc_fluent_macro",

compiler/rustc_hir_analysis/src/check/check.rs

+10-7
Original file line numberDiff line numberDiff line change
@@ -224,7 +224,8 @@ fn check_opaque(tcx: TyCtxt<'_>, id: hir::ItemId) {
224224
if check_opaque_for_cycles(tcx, item.owner_id.def_id, substs, span, &origin).is_err() {
225225
return;
226226
}
227-
check_opaque_meets_bounds(tcx, item.owner_id.def_id, span, &origin);
227+
228+
let _ = check_opaque_meets_bounds(tcx, item.owner_id.def_id, span, &origin);
228229
}
229230

230231
/// Checks that an opaque type does not use `Self` or `T::Foo` projections that would result
@@ -395,7 +396,7 @@ fn check_opaque_meets_bounds<'tcx>(
395396
def_id: LocalDefId,
396397
span: Span,
397398
origin: &hir::OpaqueTyOrigin,
398-
) {
399+
) -> Result<(), ErrorGuaranteed> {
399400
let defining_use_anchor = match *origin {
400401
hir::OpaqueTyOrigin::FnReturn(did) | hir::OpaqueTyOrigin::AsyncFn(did) => did,
401402
hir::OpaqueTyOrigin::TyAlias { .. } => tcx.impl_trait_parent(def_id),
@@ -429,10 +430,10 @@ fn check_opaque_meets_bounds<'tcx>(
429430
Ok(()) => {}
430431
Err(ty_err) => {
431432
let ty_err = ty_err.to_string(tcx);
432-
tcx.sess.delay_span_bug(
433+
return Err(tcx.sess.delay_span_bug(
433434
span,
434435
format!("could not unify `{hidden_ty}` with revealed type:\n{ty_err}"),
435-
);
436+
));
436437
}
437438
}
438439

@@ -447,7 +448,8 @@ fn check_opaque_meets_bounds<'tcx>(
447448
// version.
448449
let errors = ocx.select_all_or_error();
449450
if !errors.is_empty() {
450-
infcx.err_ctxt().report_fulfillment_errors(&errors);
451+
let guar = infcx.err_ctxt().report_fulfillment_errors(&errors);
452+
return Err(guar);
451453
}
452454
match origin {
453455
// Checked when type checking the function containing them.
@@ -461,14 +463,15 @@ fn check_opaque_meets_bounds<'tcx>(
461463
if tcx.def_kind(tcx.parent(def_id.to_def_id())) == DefKind::OpaqueTy => {}
462464
// Can have different predicates to their defining use
463465
hir::OpaqueTyOrigin::TyAlias { .. } => {
464-
let wf_tys = ocx.assumed_wf_types(param_env, span, def_id);
466+
let wf_tys = ocx.assumed_wf_types_and_report_errors(param_env, def_id)?;
465467
let implied_bounds = infcx.implied_bounds_tys(param_env, def_id, wf_tys);
466468
let outlives_env = OutlivesEnvironment::with_bounds(param_env, implied_bounds);
467-
let _ = ocx.resolve_regions_and_report_errors(defining_use_anchor, &outlives_env);
469+
ocx.resolve_regions_and_report_errors(defining_use_anchor, &outlives_env)?;
468470
}
469471
}
470472
// Clean up after ourselves
471473
let _ = infcx.take_opaque_types();
474+
Ok(())
472475
}
473476

474477
fn is_enum_of_nonnullable_ptr<'tcx>(

compiler/rustc_hir_analysis/src/check/compare_impl_item.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2121,7 +2121,7 @@ pub(super) fn check_type_bounds<'tcx>(
21212121
_ => bug!(),
21222122
}
21232123
};
2124-
let assumed_wf_types = ocx.assumed_wf_types(param_env, impl_ty_span, impl_ty_def_id);
2124+
let assumed_wf_types = ocx.assumed_wf_types_and_report_errors(param_env, impl_ty_def_id)?;
21252125

21262126
let normalize_cause = ObligationCause::new(
21272127
impl_ty_span,

compiler/rustc_hir_analysis/src/check/wfcheck.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,12 @@ pub(super) fn enter_wf_checking_ctxt<'tcx, F>(
105105
}
106106
f(&mut wfcx);
107107

108-
let assumed_wf_types = wfcx.ocx.assumed_wf_types(param_env, span, body_def_id);
108+
let assumed_wf_types = match wfcx.ocx.assumed_wf_types_and_report_errors(param_env, body_def_id)
109+
{
110+
Ok(wf_types) => wf_types,
111+
Err(_guar) => return,
112+
};
113+
109114
let implied_bounds = infcx.implied_bounds_tys(param_env, body_def_id, assumed_wf_types);
110115

111116
let errors = wfcx.select_all_or_error();

compiler/rustc_hir_analysis/src/impl_wf_check/min_specialization.rs

+10-12
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ use rustc_infer::traits::specialization_graph::Node;
7777
use rustc_middle::ty::subst::{GenericArg, InternalSubsts, SubstsRef};
7878
use rustc_middle::ty::trait_def::TraitSpecializationKind;
7979
use rustc_middle::ty::{self, TyCtxt, TypeVisitableExt};
80-
use rustc_span::Span;
80+
use rustc_span::{ErrorGuaranteed, Span};
8181
use rustc_trait_selection::traits::error_reporting::TypeErrCtxtExt;
8282
use rustc_trait_selection::traits::outlives_bounds::InferCtxtExt as _;
8383
use rustc_trait_selection::traits::{self, translate_substs_with_cause, wf, ObligationCtxt};
@@ -113,7 +113,7 @@ fn check_always_applicable(tcx: TyCtxt<'_>, impl1_def_id: LocalDefId, impl2_node
113113
let span = tcx.def_span(impl1_def_id);
114114
check_has_items(tcx, impl1_def_id, impl2_node, span);
115115

116-
if let Some((impl1_substs, impl2_substs)) = get_impl_substs(tcx, impl1_def_id, impl2_node) {
116+
if let Ok((impl1_substs, impl2_substs)) = get_impl_substs(tcx, impl1_def_id, impl2_node) {
117117
let impl2_def_id = impl2_node.def_id();
118118
debug!(?impl2_def_id, ?impl2_substs);
119119

@@ -171,16 +171,14 @@ fn get_impl_substs(
171171
tcx: TyCtxt<'_>,
172172
impl1_def_id: LocalDefId,
173173
impl2_node: Node,
174-
) -> Option<(SubstsRef<'_>, SubstsRef<'_>)> {
174+
) -> Result<(SubstsRef<'_>, SubstsRef<'_>), ErrorGuaranteed> {
175175
let infcx = &tcx.infer_ctxt().build();
176176
let ocx = ObligationCtxt::new(infcx);
177177
let param_env = tcx.param_env(impl1_def_id);
178-
179-
let assumed_wf_types =
180-
ocx.assumed_wf_types(param_env, tcx.def_span(impl1_def_id), impl1_def_id);
178+
let impl1_span = tcx.def_span(impl1_def_id);
179+
let assumed_wf_types = ocx.assumed_wf_types_and_report_errors(param_env, impl1_def_id)?;
181180

182181
let impl1_substs = InternalSubsts::identity_for_item(tcx, impl1_def_id);
183-
let impl1_span = tcx.def_span(impl1_def_id);
184182
let impl2_substs = translate_substs_with_cause(
185183
infcx,
186184
param_env,
@@ -198,19 +196,19 @@ fn get_impl_substs(
198196

199197
let errors = ocx.select_all_or_error();
200198
if !errors.is_empty() {
201-
ocx.infcx.err_ctxt().report_fulfillment_errors(&errors);
202-
return None;
199+
let guar = ocx.infcx.err_ctxt().report_fulfillment_errors(&errors);
200+
return Err(guar);
203201
}
204202

205203
let implied_bounds = infcx.implied_bounds_tys(param_env, impl1_def_id, assumed_wf_types);
206204
let outlives_env = OutlivesEnvironment::with_bounds(param_env, implied_bounds);
207205
let _ = ocx.resolve_regions_and_report_errors(impl1_def_id, &outlives_env);
208206
let Ok(impl2_substs) = infcx.fully_resolve(impl2_substs) else {
209207
let span = tcx.def_span(impl1_def_id);
210-
tcx.sess.emit_err(SubstsOnOverriddenImpl { span });
211-
return None;
208+
let guar = tcx.sess.emit_err(SubstsOnOverriddenImpl { span });
209+
return Err(guar);
212210
};
213-
Some((impl1_substs, impl2_substs))
211+
Ok((impl1_substs, impl2_substs))
214212
}
215213

216214
/// Returns a list of all of the unconstrained subst of the given impl.

compiler/rustc_hir_typeck/src/callee.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -89,7 +89,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
8989
_ => self.check_expr(callee_expr),
9090
};
9191

92-
let expr_ty = self.structurally_resolved_type(call_expr.span, original_callee_ty);
92+
let expr_ty = self.structurally_resolve_type(call_expr.span, original_callee_ty);
9393

9494
let mut autoderef = self.autoderef(callee_expr.span, expr_ty);
9595
let mut result = None;
@@ -138,7 +138,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
138138
autoderef: &Autoderef<'a, 'tcx>,
139139
) -> Option<CallStep<'tcx>> {
140140
let adjusted_ty =
141-
self.structurally_resolved_type(autoderef.span(), autoderef.final_ty(false));
141+
self.structurally_resolve_type(autoderef.span(), autoderef.final_ty(false));
142142

143143
// If the callee is a bare function or a closure, then we're all set.
144144
match *adjusted_ty.kind() {

compiler/rustc_hir_typeck/src/cast.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -717,8 +717,8 @@ impl<'a, 'tcx> CastCheck<'tcx> {
717717

718718
#[instrument(skip(fcx), level = "debug")]
719719
pub fn check(mut self, fcx: &FnCtxt<'a, 'tcx>) {
720-
self.expr_ty = fcx.structurally_resolved_type(self.expr_span, self.expr_ty);
721-
self.cast_ty = fcx.structurally_resolved_type(self.cast_span, self.cast_ty);
720+
self.expr_ty = fcx.structurally_resolve_type(self.expr_span, self.expr_ty);
721+
self.cast_ty = fcx.structurally_resolve_type(self.cast_span, self.cast_ty);
722722

723723
debug!("check_cast({}, {:?} as {:?})", self.expr.hir_id, self.expr_ty, self.cast_ty);
724724

compiler/rustc_hir_typeck/src/coercion.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1005,7 +1005,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
10051005
allow_two_phase: AllowTwoPhase,
10061006
cause: Option<ObligationCause<'tcx>>,
10071007
) -> RelateResult<'tcx, Ty<'tcx>> {
1008-
let source = self.resolve_vars_with_obligations(expr_ty);
1008+
let source = self.try_structurally_resolve_type(expr.span, expr_ty);
10091009
debug!("coercion::try({:?}: {:?} -> {:?})", expr, source, target);
10101010

10111011
let cause =

compiler/rustc_hir_typeck/src/expr.rs

+8-8
Original file line numberDiff line numberDiff line change
@@ -380,7 +380,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
380380
let mut oprnd_t = self.check_expr_with_expectation(&oprnd, expected_inner);
381381

382382
if !oprnd_t.references_error() {
383-
oprnd_t = self.structurally_resolved_type(expr.span, oprnd_t);
383+
oprnd_t = self.structurally_resolve_type(expr.span, oprnd_t);
384384
match unop {
385385
hir::UnOp::Deref => {
386386
if let Some(ty) = self.lookup_derefing(expr, oprnd, oprnd_t) {
@@ -1266,13 +1266,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12661266
) -> Ty<'tcx> {
12671267
let rcvr_t = self.check_expr(&rcvr);
12681268
// no need to check for bot/err -- callee does that
1269-
let rcvr_t = self.structurally_resolved_type(rcvr.span, rcvr_t);
1269+
let rcvr_t = self.structurally_resolve_type(rcvr.span, rcvr_t);
12701270
let span = segment.ident.span;
12711271

12721272
let method = match self.lookup_method(rcvr_t, segment, span, expr, rcvr, args) {
12731273
Ok(method) => {
12741274
// We could add a "consider `foo::<params>`" suggestion here, but I wasn't able to
1275-
// trigger this codepath causing `structurally_resolved_type` to emit an error.
1275+
// trigger this codepath causing `structurally_resolve_type` to emit an error.
12761276

12771277
self.write_method_call(expr.hir_id, method);
12781278
Ok(method)
@@ -2252,7 +2252,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
22522252
) -> Ty<'tcx> {
22532253
debug!("check_field(expr: {:?}, base: {:?}, field: {:?})", expr, base, field);
22542254
let base_ty = self.check_expr(base);
2255-
let base_ty = self.structurally_resolved_type(base.span, base_ty);
2255+
let base_ty = self.structurally_resolve_type(base.span, base_ty);
22562256
let mut private_candidate = None;
22572257
let mut autoderef = self.autoderef(expr.span, base_ty);
22582258
while let Some((deref_base_ty, _)) = autoderef.next() {
@@ -2300,7 +2300,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
23002300
_ => {}
23012301
}
23022302
}
2303-
self.structurally_resolved_type(autoderef.span(), autoderef.final_ty(false));
2303+
self.structurally_resolve_type(autoderef.span(), autoderef.final_ty(false));
23042304

23052305
if let Some((adjustments, did)) = private_candidate {
23062306
// (#90483) apply adjustments to avoid ExprUseVisitor from
@@ -2857,7 +2857,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
28572857
} else if idx_t.references_error() {
28582858
idx_t
28592859
} else {
2860-
let base_t = self.structurally_resolved_type(base.span, base_t);
2860+
let base_t = self.structurally_resolve_type(base.span, base_t);
28612861
match self.lookup_indexing(expr, base, base_t, idx, idx_t) {
28622862
Some((index_ty, element_ty)) => {
28632863
// two-phase not needed because index_ty is never mutable
@@ -3084,7 +3084,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
30843084
// allows them to be inferred based on how they are used later in the
30853085
// function.
30863086
if is_input {
3087-
let ty = self.structurally_resolved_type(expr.span, ty);
3087+
let ty = self.structurally_resolve_type(expr.span, ty);
30883088
match *ty.kind() {
30893089
ty::FnDef(..) => {
30903090
let fnptr_ty = self.tcx.mk_fn_ptr(ty.fn_sig(self.tcx));
@@ -3142,7 +3142,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
31423142
let mut current_container = container;
31433143

31443144
for &field in fields {
3145-
let container = self.structurally_resolved_type(expr.span, current_container);
3145+
let container = self.structurally_resolve_type(expr.span, current_container);
31463146

31473147
match container.kind() {
31483148
ty::Adt(container_def, substs) if !container_def.is_enum() => {

compiler/rustc_hir_typeck/src/fn_ctxt/_impl.rs

+24-13
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,8 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
8383
/// version (resolve_vars_if_possible), this version will
8484
/// also select obligations if it seems useful, in an effort
8585
/// to get more type information.
86+
// FIXME(-Ztrait-solver=next): A lot of the calls to this method should
87+
// probably be `try_structurally_resolve_type` or `structurally_resolve_type` instead.
8688
pub(in super::super) fn resolve_vars_with_obligations(&self, ty: Ty<'tcx>) -> Ty<'tcx> {
8789
self.resolve_vars_with_obligations_and_mutate_fulfillment(ty, |_| {})
8890
}
@@ -1465,16 +1467,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
14651467
}
14661468
}
14671469

1468-
/// Resolves `typ` by a single level if `typ` is a type variable.
1470+
/// Try to resolve `ty` to a structural type, normalizing aliases.
14691471
///
1470-
/// When the new solver is enabled, this will also attempt to normalize
1471-
/// the type if it's a projection (note that it will not deeply normalize
1472-
/// projections within the type, just the outermost layer of the type).
1473-
///
1474-
/// If no resolution is possible, then an error is reported.
1475-
/// Numeric inference variables may be left unresolved.
1476-
pub fn structurally_resolved_type(&self, sp: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
1477-
let mut ty = self.resolve_vars_with_obligations(ty);
1472+
/// In case there is still ambiguity, the returned type may be an inference
1473+
/// variable. This is different from `structurally_resolve_type` which errors
1474+
/// in this case.
1475+
pub fn try_structurally_resolve_type(&self, sp: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
1476+
let ty = self.resolve_vars_with_obligations(ty);
14781477

14791478
if self.next_trait_solver()
14801479
&& let ty::Alias(ty::Projection, _) = ty.kind()
@@ -1483,15 +1482,27 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
14831482
.at(&self.misc(sp), self.param_env)
14841483
.structurally_normalize(ty, &mut **self.fulfillment_cx.borrow_mut())
14851484
{
1486-
Ok(normalized_ty) => {
1487-
ty = normalized_ty;
1488-
},
1485+
Ok(normalized_ty) => normalized_ty,
14891486
Err(errors) => {
14901487
let guar = self.err_ctxt().report_fulfillment_errors(&errors);
14911488
return self.tcx.ty_error(guar);
14921489
}
14931490
}
1494-
}
1491+
} else {
1492+
ty
1493+
}
1494+
}
1495+
1496+
/// Resolves `ty` by a single level if `ty` is a type variable.
1497+
///
1498+
/// When the new solver is enabled, this will also attempt to normalize
1499+
/// the type if it's a projection (note that it will not deeply normalize
1500+
/// projections within the type, just the outermost layer of the type).
1501+
///
1502+
/// If no resolution is possible, then an error is reported.
1503+
/// Numeric inference variables may be left unresolved.
1504+
pub fn structurally_resolve_type(&self, sp: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
1505+
let ty = self.try_structurally_resolve_type(sp, ty);
14951506

14961507
if !ty.is_ty_var() {
14971508
ty

compiler/rustc_hir_typeck/src/fn_ctxt/checks.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -184,7 +184,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
184184

185185
// If the arguments should be wrapped in a tuple (ex: closures), unwrap them here
186186
let (formal_input_tys, expected_input_tys) = if tuple_arguments == TupleArguments {
187-
let tuple_type = self.structurally_resolved_type(call_span, formal_input_tys[0]);
187+
let tuple_type = self.structurally_resolve_type(call_span, formal_input_tys[0]);
188188
match tuple_type.kind() {
189189
// We expected a tuple and got a tuple
190190
ty::Tuple(arg_types) => {
@@ -412,7 +412,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
412412

413413
// There are a few types which get autopromoted when passed via varargs
414414
// in C but we just error out instead and require explicit casts.
415-
let arg_ty = self.structurally_resolved_type(arg.span, arg_ty);
415+
let arg_ty = self.structurally_resolve_type(arg.span, arg_ty);
416416
match arg_ty.kind() {
417417
ty::Float(ty::FloatTy::F32) => {
418418
variadic_error(tcx.sess, arg.span, arg_ty, "c_double");

compiler/rustc_hir_typeck/src/method/confirm.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -179,7 +179,7 @@ impl<'a, 'tcx> ConfirmContext<'a, 'tcx> {
179179
assert_eq!(n, pick.autoderefs);
180180

181181
let mut adjustments = self.adjust_steps(&autoderef);
182-
let mut target = self.structurally_resolved_type(autoderef.span(), ty);
182+
let mut target = self.structurally_resolve_type(autoderef.span(), ty);
183183

184184
match pick.autoref_or_ptr_adjustment {
185185
Some(probe::AutorefOrPtrAdjustment::Autoref { mutbl, unsize }) => {

compiler/rustc_hir_typeck/src/method/probe.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -451,7 +451,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
451451
} else {
452452
// Ended up encountering a type variable when doing autoderef,
453453
// but it may not be a type variable after processing obligations
454-
// in our local `FnCtxt`, so don't call `structurally_resolved_type`.
454+
// in our local `FnCtxt`, so don't call `structurally_resolve_type`.
455455
let ty = &bad_ty.ty;
456456
let ty = self
457457
.probe_instantiate_query_response(span, &orig_values, ty)

compiler/rustc_hir_typeck/src/pat.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -393,7 +393,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
393393
// They can denote both statically and dynamically-sized byte arrays.
394394
let mut pat_ty = ty;
395395
if let hir::ExprKind::Lit(Spanned { node: ast::LitKind::ByteStr(..), .. }) = lt.kind {
396-
let expected = self.structurally_resolved_type(span, expected);
396+
let expected = self.structurally_resolve_type(span, expected);
397397
if let ty::Ref(_, inner_ty, _) = expected.kind()
398398
&& matches!(inner_ty.kind(), ty::Slice(_))
399399
{
@@ -501,7 +501,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
501501
// This check is needed if both sides are inference variables.
502502
// We require types to be resolved here so that we emit inference failure
503503
// rather than "_ is not a char or numeric".
504-
let ty = self.structurally_resolved_type(span, expected);
504+
let ty = self.structurally_resolve_type(span, expected);
505505
if !(ty.is_numeric() || ty.is_char() || ty.references_error()) {
506506
if let Some((ref mut fail, _, _)) = lhs {
507507
*fail = true;
@@ -1289,7 +1289,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12891289
let mut expected_len = elements.len();
12901290
if ddpos.as_opt_usize().is_some() {
12911291
// Require known type only when `..` is present.
1292-
if let ty::Tuple(tys) = self.structurally_resolved_type(span, expected).kind() {
1292+
if let ty::Tuple(tys) = self.structurally_resolve_type(span, expected).kind() {
12931293
expected_len = tys.len();
12941294
}
12951295
}
@@ -2042,7 +2042,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
20422042
def_bm: BindingMode,
20432043
ti: TopInfo<'tcx>,
20442044
) -> Ty<'tcx> {
2045-
let expected = self.structurally_resolved_type(span, expected);
2045+
let expected = self.structurally_resolve_type(span, expected);
20462046
let (element_ty, opt_slice_ty, inferred) = match *expected.kind() {
20472047
// An array, so we might have something like `let [a, b, c] = [0, 1, 2];`.
20482048
ty::Array(element_ty, len) => {

0 commit comments

Comments
 (0)