Skip to content

Commit 6766970

Browse files
committed
Auto merge of #122213 - estebank:issue-50195, r=oli-obk,estebank
Provide suggestion to dereference closure tail if appropriate When encoutnering a case like ```rust use std::collections::HashMap; fn main() { let vs = vec![0, 0, 1, 1, 3, 4, 5, 6, 3, 3, 3]; let mut counts = HashMap::new(); for num in vs { let count = counts.entry(num).or_insert(0); *count += 1; } let _ = counts.iter().max_by_key(|(_, v)| v); ``` produce the following suggestion ``` error: lifetime may not live long enough --> $DIR/return-value-lifetime-error.rs:13:47 | LL | let _ = counts.iter().max_by_key(|(_, v)| v); | ------- ^ returning this value requires that `'1` must outlive `'2` | | | | | return type of closure is &'2 &i32 | has type `&'1 (&i32, &i32)` | help: dereference the return value | LL | let _ = counts.iter().max_by_key(|(_, v)| **v); | ++ ``` Fix #50195.
2 parents aa067fb + f2465f8 commit 6766970

File tree

8 files changed

+241
-5
lines changed

8 files changed

+241
-5
lines changed

compiler/rustc_borrowck/src/diagnostics/conflict_errors.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -1469,27 +1469,31 @@ impl<'cx, 'tcx> MirBorrowckCtxt<'cx, 'tcx> {
14691469
let hir = tcx.hir();
14701470
let Some(body_id) = tcx.hir_node(self.mir_hir_id()).body_id() else { return };
14711471
struct FindUselessClone<'hir> {
1472+
tcx: TyCtxt<'hir>,
1473+
def_id: DefId,
14721474
pub clones: Vec<&'hir hir::Expr<'hir>>,
14731475
}
14741476
impl<'hir> FindUselessClone<'hir> {
1475-
pub fn new() -> Self {
1476-
Self { clones: vec![] }
1477+
pub fn new(tcx: TyCtxt<'hir>, def_id: DefId) -> Self {
1478+
Self { tcx, def_id, clones: vec![] }
14771479
}
14781480
}
14791481

14801482
impl<'v> Visitor<'v> for FindUselessClone<'v> {
14811483
fn visit_expr(&mut self, ex: &'v hir::Expr<'v>) {
1482-
// FIXME: use `lookup_method_for_diagnostic`?
14831484
if let hir::ExprKind::MethodCall(segment, _rcvr, args, _span) = ex.kind
14841485
&& segment.ident.name == sym::clone
14851486
&& args.len() == 0
1487+
&& let Some(def_id) = self.def_id.as_local()
1488+
&& let Some(method) = self.tcx.lookup_method_for_diagnostic((def_id, ex.hir_id))
1489+
&& Some(self.tcx.parent(method)) == self.tcx.lang_items().clone_trait()
14861490
{
14871491
self.clones.push(ex);
14881492
}
14891493
hir::intravisit::walk_expr(self, ex);
14901494
}
14911495
}
1492-
let mut expr_finder = FindUselessClone::new();
1496+
let mut expr_finder = FindUselessClone::new(tcx, self.mir_def_id().into());
14931497

14941498
let body = hir.body(body_id).value;
14951499
expr_finder.visit_expr(body);

compiler/rustc_borrowck/src/diagnostics/region_errors.rs

+145
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,9 @@ use rustc_middle::ty::{self, RegionVid, Ty};
2626
use rustc_middle::ty::{Region, TyCtxt};
2727
use rustc_span::symbol::{kw, Ident};
2828
use rustc_span::Span;
29+
use rustc_trait_selection::infer::type_variable::{TypeVariableOrigin, TypeVariableOriginKind};
30+
use rustc_trait_selection::infer::InferCtxtExt;
31+
use rustc_trait_selection::traits::{Obligation, ObligationCtxt};
2932

3033
use crate::borrowck_errors;
3134
use crate::session_diagnostics::{
@@ -810,6 +813,7 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
810813
self.add_static_impl_trait_suggestion(&mut diag, *fr, fr_name, *outlived_fr);
811814
self.suggest_adding_lifetime_params(&mut diag, *fr, *outlived_fr);
812815
self.suggest_move_on_borrowing_closure(&mut diag);
816+
self.suggest_deref_closure_value(&mut diag);
813817

814818
diag
815819
}
@@ -1039,6 +1043,147 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
10391043
suggest_adding_lifetime_params(self.infcx.tcx, sub, ty_sup, ty_sub, diag);
10401044
}
10411045

1046+
#[allow(rustc::diagnostic_outside_of_impl)]
1047+
#[allow(rustc::untranslatable_diagnostic)] // FIXME: make this translatable
1048+
/// When encountering a lifetime error caused by the return type of a closure, check the
1049+
/// corresponding trait bound and see if dereferencing the closure return value would satisfy
1050+
/// them. If so, we produce a structured suggestion.
1051+
fn suggest_deref_closure_value(&self, diag: &mut Diag<'_>) {
1052+
let tcx = self.infcx.tcx;
1053+
let map = tcx.hir();
1054+
1055+
// Get the closure return value and type.
1056+
let body_id = map.body_owned_by(self.mir_def_id());
1057+
let body = &map.body(body_id);
1058+
let value = &body.value.peel_blocks();
1059+
let hir::Node::Expr(closure_expr) = tcx.hir_node_by_def_id(self.mir_def_id()) else {
1060+
return;
1061+
};
1062+
let fn_call_id = tcx.parent_hir_id(self.mir_hir_id());
1063+
let hir::Node::Expr(expr) = tcx.hir_node(fn_call_id) else { return };
1064+
let def_id = map.enclosing_body_owner(fn_call_id);
1065+
let tables = tcx.typeck(def_id);
1066+
let Some(return_value_ty) = tables.node_type_opt(value.hir_id) else { return };
1067+
let return_value_ty = self.infcx.resolve_vars_if_possible(return_value_ty);
1068+
1069+
// We don't use `ty.peel_refs()` to get the number of `*`s needed to get the root type.
1070+
let mut ty = return_value_ty;
1071+
let mut count = 0;
1072+
while let ty::Ref(_, t, _) = ty.kind() {
1073+
ty = *t;
1074+
count += 1;
1075+
}
1076+
if !self.infcx.type_is_copy_modulo_regions(self.param_env, ty) {
1077+
return;
1078+
}
1079+
1080+
// Build a new closure where the return type is an owned value, instead of a ref.
1081+
let Some(ty::Closure(did, args)) =
1082+
tables.node_type_opt(closure_expr.hir_id).as_ref().map(|ty| ty.kind())
1083+
else {
1084+
return;
1085+
};
1086+
let sig = args.as_closure().sig();
1087+
let closure_sig_as_fn_ptr_ty = Ty::new_fn_ptr(
1088+
tcx,
1089+
sig.map_bound(|s| {
1090+
let unsafety = hir::Unsafety::Normal;
1091+
use rustc_target::spec::abi;
1092+
tcx.mk_fn_sig(
1093+
[s.inputs()[0]],
1094+
s.output().peel_refs(),
1095+
s.c_variadic,
1096+
unsafety,
1097+
abi::Abi::Rust,
1098+
)
1099+
}),
1100+
);
1101+
let parent_args = GenericArgs::identity_for_item(
1102+
tcx,
1103+
tcx.typeck_root_def_id(self.mir_def_id().to_def_id()),
1104+
);
1105+
let closure_kind = args.as_closure().kind();
1106+
let closure_kind_ty = Ty::from_closure_kind(tcx, closure_kind);
1107+
let tupled_upvars_ty = self.infcx.next_ty_var(TypeVariableOrigin {
1108+
kind: TypeVariableOriginKind::ClosureSynthetic,
1109+
span: closure_expr.span,
1110+
});
1111+
let closure_args = ty::ClosureArgs::new(
1112+
tcx,
1113+
ty::ClosureArgsParts {
1114+
parent_args,
1115+
closure_kind_ty,
1116+
closure_sig_as_fn_ptr_ty,
1117+
tupled_upvars_ty,
1118+
},
1119+
);
1120+
let closure_ty = Ty::new_closure(tcx, *did, closure_args.args);
1121+
let closure_ty = tcx.erase_regions(closure_ty);
1122+
1123+
let hir::ExprKind::MethodCall(_, rcvr, args, _) = expr.kind else { return };
1124+
let Some(pos) = args
1125+
.iter()
1126+
.enumerate()
1127+
.find(|(_, arg)| arg.hir_id == closure_expr.hir_id)
1128+
.map(|(i, _)| i)
1129+
else {
1130+
return;
1131+
};
1132+
// The found `Self` type of the method call.
1133+
let Some(possible_rcvr_ty) = tables.node_type_opt(rcvr.hir_id) else { return };
1134+
1135+
// The `MethodCall` expression is `Res::Err`, so we search for the method on the `rcvr_ty`.
1136+
let Some(method) = tcx.lookup_method_for_diagnostic((self.mir_def_id(), expr.hir_id))
1137+
else {
1138+
return;
1139+
};
1140+
1141+
// Get the type for the parameter corresponding to the argument the closure with the
1142+
// lifetime error we had.
1143+
let Some(input) = tcx
1144+
.fn_sig(method)
1145+
.instantiate_identity()
1146+
.inputs()
1147+
.skip_binder()
1148+
// Methods have a `self` arg, so `pos` is actually `+ 1` to match the method call arg.
1149+
.get(pos + 1)
1150+
else {
1151+
return;
1152+
};
1153+
1154+
trace!(?input);
1155+
1156+
let ty::Param(closure_param) = input.kind() else { return };
1157+
1158+
// Get the arguments for the found method, only specifying that `Self` is the receiver type.
1159+
let args = GenericArgs::for_item(tcx, method, |param, _| {
1160+
if param.index == 0 {
1161+
possible_rcvr_ty.into()
1162+
} else if param.index == closure_param.index {
1163+
closure_ty.into()
1164+
} else {
1165+
self.infcx.var_for_def(expr.span, param)
1166+
}
1167+
});
1168+
1169+
let preds = tcx.predicates_of(method).instantiate(tcx, args);
1170+
1171+
let ocx = ObligationCtxt::new(&self.infcx);
1172+
ocx.register_obligations(preds.iter().map(|(pred, span)| {
1173+
trace!(?pred);
1174+
Obligation::misc(tcx, span, self.mir_def_id(), self.param_env, pred)
1175+
}));
1176+
1177+
if ocx.select_all_or_error().is_empty() {
1178+
diag.span_suggestion_verbose(
1179+
value.span.shrink_to_lo(),
1180+
"dereference the return value",
1181+
"*".repeat(count),
1182+
Applicability::MachineApplicable,
1183+
);
1184+
}
1185+
}
1186+
10421187
#[allow(rustc::diagnostic_outside_of_impl)]
10431188
#[allow(rustc::untranslatable_diagnostic)] // FIXME: make this translatable
10441189
fn suggest_move_on_borrowing_closure(&self, diag: &mut Diag<'_>) {

compiler/rustc_hir_typeck/src/lib.rs

+24-1
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@ use rustc_data_structures::unord::UnordSet;
5656
use rustc_errors::{codes::*, struct_span_code_err, ErrorGuaranteed};
5757
use rustc_hir as hir;
5858
use rustc_hir::def::{DefKind, Res};
59-
use rustc_hir::intravisit::Visitor;
59+
use rustc_hir::intravisit::{Map, Visitor};
6060
use rustc_hir::{HirIdMap, Node};
6161
use rustc_hir_analysis::check::check_abi;
6262
use rustc_hir_analysis::hir_ty_lowering::HirTyLowerer;
@@ -436,13 +436,36 @@ fn fatally_break_rust(tcx: TyCtxt<'_>, span: Span) -> ! {
436436
diag.emit()
437437
}
438438

439+
pub fn lookup_method_for_diagnostic<'tcx>(
440+
tcx: TyCtxt<'tcx>,
441+
(def_id, hir_id): (LocalDefId, hir::HirId),
442+
) -> Option<DefId> {
443+
let root_ctxt = TypeckRootCtxt::new(tcx, def_id);
444+
let param_env = tcx.param_env(def_id);
445+
let fn_ctxt = FnCtxt::new(&root_ctxt, param_env, def_id);
446+
let hir::Node::Expr(expr) = tcx.hir().hir_node(hir_id) else {
447+
return None;
448+
};
449+
let hir::ExprKind::MethodCall(segment, rcvr, _, _) = expr.kind else {
450+
return None;
451+
};
452+
let tables = tcx.typeck(def_id);
453+
// The found `Self` type of the method call.
454+
let possible_rcvr_ty = tables.node_type_opt(rcvr.hir_id)?;
455+
fn_ctxt
456+
.lookup_method_for_diagnostic(possible_rcvr_ty, segment, expr.span, expr, rcvr)
457+
.ok()
458+
.map(|method| method.def_id)
459+
}
460+
439461
pub fn provide(providers: &mut Providers) {
440462
method::provide(providers);
441463
*providers = Providers {
442464
typeck,
443465
diagnostic_only_typeck,
444466
has_typeck_results,
445467
used_trait_imports,
468+
lookup_method_for_diagnostic: lookup_method_for_diagnostic,
446469
..*providers
447470
};
448471
}

compiler/rustc_middle/src/query/keys.rs

+13
Original file line numberDiff line numberDiff line change
@@ -555,6 +555,19 @@ impl Key for HirId {
555555
}
556556
}
557557

558+
impl Key for (LocalDefId, HirId) {
559+
type Cache<V> = DefaultCache<Self, V>;
560+
561+
fn default_span(&self, tcx: TyCtxt<'_>) -> Span {
562+
tcx.hir().span(self.1)
563+
}
564+
565+
#[inline(always)]
566+
fn key_as_def_id(&self) -> Option<DefId> {
567+
Some(self.0.into())
568+
}
569+
}
570+
558571
impl<'tcx> Key for (ValidityRequirement, ty::ParamEnvAnd<'tcx, Ty<'tcx>>) {
559572
type Cache<V> = DefaultCache<Self, V>;
560573

compiler/rustc_middle/src/query/mod.rs

+3
Original file line numberDiff line numberDiff line change
@@ -983,6 +983,9 @@ rustc_queries! {
983983
query diagnostic_only_typeck(key: LocalDefId) -> &'tcx ty::TypeckResults<'tcx> {
984984
desc { |tcx| "type-checking `{}`", tcx.def_path_str(key) }
985985
}
986+
query lookup_method_for_diagnostic((def_id, hir_id): (LocalDefId, hir::HirId)) -> Option<DefId> {
987+
desc { |tcx| "lookup_method_for_diagnostics `{}`", tcx.def_path_str(def_id) }
988+
}
986989

987990
query used_trait_imports(key: LocalDefId) -> &'tcx UnordSet<LocalDefId> {
988991
desc { |tcx| "finding used_trait_imports `{}`", tcx.def_path_str(key) }
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
//@ run-rustfix
2+
use std::collections::HashMap;
3+
4+
fn main() {
5+
let vs = vec![0, 0, 1, 1, 3, 4, 5, 6, 3, 3, 3];
6+
7+
let mut counts = HashMap::new();
8+
for num in vs {
9+
let count = counts.entry(num).or_insert(0);
10+
*count += 1;
11+
}
12+
13+
let _ = counts.iter().max_by_key(|(_, v)| **v);
14+
//~^ ERROR lifetime may not live long enough
15+
//~| HELP dereference the return value
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
//@ run-rustfix
2+
use std::collections::HashMap;
3+
4+
fn main() {
5+
let vs = vec![0, 0, 1, 1, 3, 4, 5, 6, 3, 3, 3];
6+
7+
let mut counts = HashMap::new();
8+
for num in vs {
9+
let count = counts.entry(num).or_insert(0);
10+
*count += 1;
11+
}
12+
13+
let _ = counts.iter().max_by_key(|(_, v)| v);
14+
//~^ ERROR lifetime may not live long enough
15+
//~| HELP dereference the return value
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
error: lifetime may not live long enough
2+
--> $DIR/return-value-lifetime-error.rs:13:47
3+
|
4+
LL | let _ = counts.iter().max_by_key(|(_, v)| v);
5+
| ------- ^ returning this value requires that `'1` must outlive `'2`
6+
| | |
7+
| | return type of closure is &'2 &i32
8+
| has type `&'1 (&i32, &i32)`
9+
|
10+
help: dereference the return value
11+
|
12+
LL | let _ = counts.iter().max_by_key(|(_, v)| **v);
13+
| ++
14+
15+
error: aborting due to 1 previous error
16+

0 commit comments

Comments
 (0)