mirror of
https://github.com/rust-lang/rust.git
synced 2026-04-27 18:57:42 +03:00
Rollup merge of #154674 - Enselic:unused-outlive, r=jdonszelmann
borrowck: Don't mention unused vars in closure outlive errors When there is a closure arg lifetime error, the code does its best to find variables that correspond to the lifetimes involved. This commit stops mentioning arguments that are unused in closures, since they are not relevant. This removes the "red herring" of rust-lang/rust#113121. The `user_arg_index` nomenclature can be applied more broadly, but then the diff becomes annoyingly big, so I chose not to do that. Review each commit individually for a nicer experience.
This commit is contained in:
@@ -1,5 +1,6 @@
|
||||
use rustc_index::IndexSlice;
|
||||
use rustc_middle::mir::{Body, Local};
|
||||
use rustc_middle::mir::visit::{PlaceContext, VisitPlacesWith, Visitor};
|
||||
use rustc_middle::mir::{Body, Local, Place};
|
||||
use rustc_middle::ty::{self, RegionVid, TyCtxt};
|
||||
use rustc_span::{Span, Symbol};
|
||||
use tracing::debug;
|
||||
@@ -7,6 +8,8 @@
|
||||
use crate::region_infer::RegionInferenceContext;
|
||||
|
||||
impl<'tcx> RegionInferenceContext<'tcx> {
|
||||
/// Find the the name and span of the variable corresponding to the given region.
|
||||
/// The returned var will also be ensured to actually be used in `body`.
|
||||
pub(crate) fn get_var_name_and_span_for_region(
|
||||
&self,
|
||||
tcx: TyCtxt<'tcx>,
|
||||
@@ -22,13 +25,20 @@ pub(crate) fn get_var_name_and_span_for_region(
|
||||
self.get_upvar_index_for_region(tcx, fr)
|
||||
.map(|index| {
|
||||
// FIXME(project-rfc-2229#8): Use place span for diagnostics
|
||||
// We know our upvars are used thanks to `fn compute_min_captures()` in `upvar.rs`.
|
||||
let (name, span) = self.get_upvar_name_and_span_for_region(tcx, upvars, index);
|
||||
(Some(name), span)
|
||||
})
|
||||
.or_else(|| {
|
||||
debug!("get_var_name_and_span_for_region: attempting argument");
|
||||
self.get_argument_index_for_region(tcx, fr).map(|index| {
|
||||
self.get_argument_name_and_span_for_region(body, local_names, index)
|
||||
self.get_argument_index_for_region(tcx, fr).and_then(|index| {
|
||||
let local = self.user_arg_index_to_local(body, index);
|
||||
if body_uses_local(body, local) {
|
||||
Some(self.get_argument_name_and_span_for_region(body, local_names, index))
|
||||
} else {
|
||||
debug!("get_var_name_and_span_for_region: skipping unused local {local:?}");
|
||||
None
|
||||
}
|
||||
})
|
||||
})
|
||||
}
|
||||
@@ -105,6 +115,13 @@ pub(crate) fn get_argument_index_for_region(
|
||||
Some(argument_index)
|
||||
}
|
||||
|
||||
/// Given the index of an argument as seen from the user (i.e. excluding
|
||||
/// implicit inputs), returns the corresponding MIR local.
|
||||
fn user_arg_index_to_local(&self, body: &Body<'tcx>, user_arg_index: usize) -> Local {
|
||||
let implicit_inputs = self.universal_regions().defining_ty.implicit_inputs();
|
||||
body.args_iter().nth(implicit_inputs + user_arg_index).unwrap()
|
||||
}
|
||||
|
||||
/// Given the index of an argument, finds its name (if any) and the span from where it was
|
||||
/// declared.
|
||||
pub(crate) fn get_argument_name_and_span_for_region(
|
||||
@@ -113,8 +130,7 @@ pub(crate) fn get_argument_name_and_span_for_region(
|
||||
local_names: &IndexSlice<Local, Option<Symbol>>,
|
||||
argument_index: usize,
|
||||
) -> (Option<Symbol>, Span) {
|
||||
let implicit_inputs = self.universal_regions().defining_ty.implicit_inputs();
|
||||
let argument_local = Local::from_usize(implicit_inputs + argument_index + 1);
|
||||
let argument_local = self.user_arg_index_to_local(body, argument_index);
|
||||
debug!("get_argument_name_and_span_for_region: argument_local={argument_local:?}");
|
||||
|
||||
let argument_name = local_names[argument_local];
|
||||
@@ -126,3 +142,14 @@ pub(crate) fn get_argument_name_and_span_for_region(
|
||||
(argument_name, argument_span)
|
||||
}
|
||||
}
|
||||
|
||||
fn body_uses_local<'tcx>(body: &Body<'tcx>, target: Local) -> bool {
|
||||
let mut used = false;
|
||||
VisitPlacesWith(|place: Place<'_>, context: PlaceContext| {
|
||||
if !matches!(context, PlaceContext::NonUse(_)) && place.local == target {
|
||||
used = true;
|
||||
}
|
||||
})
|
||||
.visit_body(body);
|
||||
used
|
||||
}
|
||||
|
||||
@@ -0,0 +1,17 @@
|
||||
//! Regression test for <https://github.com/rust-lang/rust/issues/113121>.
|
||||
|
||||
#![allow(unused_variables)]
|
||||
|
||||
fn consume<T: 'static>(_: T) {}
|
||||
|
||||
fn foo<'a>(
|
||||
used_arg: &'a u8,
|
||||
unused_arg: &'static u16, // Unused in closure. Must not appear in error.
|
||||
) {
|
||||
let unused_var: &'static u32 = &42; // Unused in closure. Must not appear in error.
|
||||
|
||||
let c = move || used_arg;
|
||||
consume(c); //~ ERROR: borrowed data escapes outside of function
|
||||
}
|
||||
|
||||
fn main() {}
|
||||
@@ -0,0 +1,17 @@
|
||||
error[E0521]: borrowed data escapes outside of function
|
||||
--> $DIR/var-matching-lifetime-but-unused-not-mentioned.rs:14:5
|
||||
|
|
||||
LL | fn foo<'a>(
|
||||
| -- lifetime `'a` defined here
|
||||
LL | used_arg: &'a u8,
|
||||
| -------- `used_arg` is a reference that is only valid in the function body
|
||||
...
|
||||
LL | consume(c);
|
||||
| ^^^^^^^^^^
|
||||
| |
|
||||
| `used_arg` escapes the function body here
|
||||
| argument requires that `'a` must outlive `'static`
|
||||
|
||||
error: aborting due to 1 previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0521`.
|
||||
Reference in New Issue
Block a user