typeck: resolve type vars before calling try_index_step
`try_index_step` does not resolve type variables by itself and would fail otherwise. Also harden the failure path in `confirm` to cause less confusing errors.
This commit is contained in:
parent
54ef80043a
commit
a260df27e0
@ -451,7 +451,10 @@ fn pat_ty(&self, pat: &hir::Pat) -> McResult<Ty<'tcx>> {
|
||||
// So peel off one-level, turning the &T into T.
|
||||
match base_ty.builtin_deref(false, ty::NoPreference) {
|
||||
Some(t) => t.ty,
|
||||
None => { return Err(()); }
|
||||
None => {
|
||||
debug!("By-ref binding of non-derefable type {:?}", base_ty);
|
||||
return Err(());
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => base_ty,
|
||||
@ -1039,6 +1042,7 @@ pub fn cat_index<N:ast_node>(&self,
|
||||
match base_cmt.ty.builtin_index() {
|
||||
Some(ty) => (ty, ElementKind::VecElement),
|
||||
None => {
|
||||
debug!("Explicit index of non-indexable type {:?}", base_cmt);
|
||||
return Err(());
|
||||
}
|
||||
}
|
||||
@ -1154,7 +1158,10 @@ fn cat_pattern_<F>(&self, cmt: cmt<'tcx>, pat: &hir::Pat, op: &mut F) -> McResul
|
||||
PatKind::TupleStruct(hir::QPath::Resolved(_, ref path), ..) |
|
||||
PatKind::Struct(hir::QPath::Resolved(_, ref path), ..) => {
|
||||
match path.def {
|
||||
Def::Err => return Err(()),
|
||||
Def::Err => {
|
||||
debug!("access to unresolvable pattern {:?}", pat);
|
||||
return Err(())
|
||||
}
|
||||
Def::Variant(variant_did) |
|
||||
Def::VariantCtor(variant_did, ..) => {
|
||||
// univariant enums do not need downcasts
|
||||
|
@ -375,6 +375,9 @@ pub fn span_bug<S: Into<MultiSpan>>(&self, sp: S, msg: &str) -> ! {
|
||||
panic!(ExplicitBug);
|
||||
}
|
||||
pub fn delay_span_bug<S: Into<MultiSpan>>(&self, sp: S, msg: &str) {
|
||||
if self.treat_err_as_bug {
|
||||
self.span_bug(sp, msg);
|
||||
}
|
||||
let mut delayed = self.delayed_span_bug.borrow_mut();
|
||||
*delayed = Some((sp.into(), msg.to_string()));
|
||||
}
|
||||
|
@ -516,6 +516,8 @@ fn convert_lvalue_derefs_to_mutable(&self) {
|
||||
};
|
||||
|
||||
let index_expr_ty = self.node_ty(index_expr.id);
|
||||
let adjusted_base_ty = self.resolve_type_vars_if_possible(&adjusted_base_ty);
|
||||
let index_expr_ty = self.resolve_type_vars_if_possible(&index_expr_ty);
|
||||
|
||||
let result = self.try_index_step(ty::MethodCall::expr(expr.id),
|
||||
expr,
|
||||
@ -531,6 +533,15 @@ fn convert_lvalue_derefs_to_mutable(&self) {
|
||||
|
||||
let expr_ty = self.node_ty(expr.id);
|
||||
self.demand_suptype(expr.span, expr_ty, return_ty);
|
||||
} else {
|
||||
// We could not perform a mutable index. Re-apply the
|
||||
// immutable index adjustments - borrowck will detect
|
||||
// this as an error.
|
||||
if let Some(adjustment) = adjustment {
|
||||
self.apply_adjustment(expr.id, adjustment);
|
||||
}
|
||||
self.tcx.sess.delay_span_bug(
|
||||
expr.span, "convert_lvalue_derefs_to_mutable failed");
|
||||
}
|
||||
}
|
||||
hir::ExprUnary(hir::UnDeref, ref base_expr) => {
|
||||
|
26
src/test/run-pass/issue-41498.rs
Normal file
26
src/test/run-pass/issue-41498.rs
Normal file
@ -0,0 +1,26 @@
|
||||
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
// regression test for issue #41498.
|
||||
|
||||
struct S;
|
||||
impl S {
|
||||
fn mutate(&mut self) {}
|
||||
}
|
||||
|
||||
fn call_and_ref<T, F: FnOnce() -> T>(x: &mut Option<T>, f: F) -> &mut T {
|
||||
*x = Some(f());
|
||||
x.as_mut().unwrap()
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let mut n = None;
|
||||
call_and_ref(&mut n, || [S])[0].mutate();
|
||||
}
|
Loading…
Reference in New Issue
Block a user