Mugrate mismatched_static_lifetime.rs
This commit is contained in:
parent
af3343ae29
commit
e0e9b21c78
4 changed files with 303 additions and 20 deletions
176
compiler/rustc_infer/src/errors/note_and_explain.rs
Normal file
176
compiler/rustc_infer/src/errors/note_and_explain.rs
Normal file
|
@ -0,0 +1,176 @@
|
|||
use crate::infer::error_reporting::nice_region_error::find_anon_type;
|
||||
use rustc_errors::{self, fluent, AddSubdiagnostic};
|
||||
use rustc_middle::ty::{self, TyCtxt};
|
||||
use rustc_span::{symbol::kw, Span};
|
||||
|
||||
#[derive(Default)]
|
||||
struct DescriptionCtx<'a> {
|
||||
span: Option<Span>,
|
||||
kind: &'a str,
|
||||
arg: String,
|
||||
num_arg: u32,
|
||||
}
|
||||
|
||||
impl<'a> DescriptionCtx<'a> {
|
||||
fn new<'tcx>(
|
||||
tcx: TyCtxt<'tcx>,
|
||||
region: ty::Region<'tcx>,
|
||||
alt_span: Option<Span>,
|
||||
) -> Option<Self> {
|
||||
let mut me = DescriptionCtx::default();
|
||||
me.span = alt_span;
|
||||
match *region {
|
||||
ty::ReEarlyBound(_) | ty::ReFree(_) => {
|
||||
return Self::from_early_bound_and_free_regions(tcx, region);
|
||||
}
|
||||
ty::ReStatic => {
|
||||
me.kind = "restatic";
|
||||
}
|
||||
|
||||
ty::ReEmpty(ty::UniverseIndex::ROOT) => me.kind = "reempty",
|
||||
|
||||
// uh oh, hope no user ever sees THIS
|
||||
ty::ReEmpty(ui) => {
|
||||
me.kind = "reemptyuni";
|
||||
me.arg = format!("{:?}", ui);
|
||||
}
|
||||
|
||||
ty::RePlaceholder(_) => return None,
|
||||
|
||||
// FIXME(#13998) RePlaceholder should probably print like
|
||||
// ReFree rather than dumping Debug output on the user.
|
||||
//
|
||||
// We shouldn't really be having unification failures with ReVar
|
||||
// and ReLateBound though.
|
||||
ty::ReVar(_) | ty::ReLateBound(..) | ty::ReErased => {
|
||||
me.kind = "revar";
|
||||
me.arg = format!("{:?}", region);
|
||||
}
|
||||
};
|
||||
Some(me)
|
||||
}
|
||||
|
||||
fn from_early_bound_and_free_regions<'tcx>(
|
||||
tcx: TyCtxt<'tcx>,
|
||||
region: ty::Region<'tcx>,
|
||||
) -> Option<Self> {
|
||||
let mut me = DescriptionCtx::default();
|
||||
let scope = region.free_region_binding_scope(tcx).expect_local();
|
||||
match *region {
|
||||
ty::ReEarlyBound(ref br) => {
|
||||
let mut sp = tcx.def_span(scope);
|
||||
if let Some(param) =
|
||||
tcx.hir().get_generics(scope).and_then(|generics| generics.get_named(br.name))
|
||||
{
|
||||
sp = param.span;
|
||||
}
|
||||
if br.has_name() {
|
||||
me.kind = "as_defined";
|
||||
me.arg = br.name.to_string();
|
||||
} else {
|
||||
me.kind = "as_defined_anon";
|
||||
};
|
||||
me.span = Some(sp)
|
||||
}
|
||||
ty::ReFree(ref fr) => {
|
||||
if !fr.bound_region.is_named()
|
||||
&& let Some((ty, _)) = find_anon_type(tcx, region, &fr.bound_region)
|
||||
{
|
||||
me.kind = "defined_here";
|
||||
me.span = Some(ty.span);
|
||||
} else {
|
||||
match fr.bound_region {
|
||||
ty::BoundRegionKind::BrNamed(_, name) => {
|
||||
let mut sp = tcx.def_span(scope);
|
||||
if let Some(param) =
|
||||
tcx.hir().get_generics(scope).and_then(|generics| generics.get_named(name))
|
||||
{
|
||||
sp = param.span;
|
||||
}
|
||||
if name == kw::UnderscoreLifetime {
|
||||
me.kind = "as_defined_anon";
|
||||
} else {
|
||||
me.kind = "as_defined";
|
||||
me.arg = name.to_string();
|
||||
};
|
||||
me.span = Some(sp);
|
||||
}
|
||||
ty::BrAnon(idx) => {
|
||||
me.kind = "anon_num_here";
|
||||
me.num_arg = idx+1;
|
||||
me.span = Some(tcx.def_span(scope));
|
||||
},
|
||||
_ => {
|
||||
me.kind = "defined_here_reg";
|
||||
me.arg = region.to_string();
|
||||
me.span = Some(tcx.def_span(scope));
|
||||
},
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => bug!(),
|
||||
}
|
||||
Some(me)
|
||||
}
|
||||
|
||||
fn add_to(self, diag: &mut rustc_errors::Diagnostic) {
|
||||
diag.set_arg("desc_kind", self.kind);
|
||||
diag.set_arg("desc_arg", self.arg);
|
||||
diag.set_arg("desc_num_arg", self.num_arg);
|
||||
}
|
||||
}
|
||||
|
||||
pub enum PrefixKind {
|
||||
Empty,
|
||||
}
|
||||
|
||||
pub enum SuffixKind {
|
||||
Continues,
|
||||
}
|
||||
|
||||
impl PrefixKind {
|
||||
fn add_to(self, diag: &mut rustc_errors::Diagnostic) {
|
||||
match self {
|
||||
Self::Empty => diag.set_arg("pref_kind", "empty"),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
impl SuffixKind {
|
||||
fn add_to(self, diag: &mut rustc_errors::Diagnostic) {
|
||||
match self {
|
||||
Self::Continues => diag.set_arg("suff_kind", "continues"),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
pub struct RegionExplanation<'a> {
|
||||
desc: DescriptionCtx<'a>,
|
||||
prefix: PrefixKind,
|
||||
suffix: SuffixKind,
|
||||
}
|
||||
|
||||
impl RegionExplanation<'_> {
|
||||
pub fn new<'tcx>(
|
||||
tcx: TyCtxt<'tcx>,
|
||||
region: ty::Region<'tcx>,
|
||||
alt_span: Option<Span>,
|
||||
prefix: PrefixKind,
|
||||
suffix: SuffixKind,
|
||||
) -> Option<Self> {
|
||||
Some(Self { desc: DescriptionCtx::new(tcx, region, alt_span)?, prefix, suffix })
|
||||
}
|
||||
}
|
||||
|
||||
impl AddSubdiagnostic for RegionExplanation<'_> {
|
||||
fn add_to_diagnostic(self, diag: &mut rustc_errors::Diagnostic) {
|
||||
if let Some(span) = self.desc.span {
|
||||
diag.span_note(span, fluent::infer::region_explanation);
|
||||
} else {
|
||||
diag.note(fluent::infer::region_explanation);
|
||||
}
|
||||
self.desc.add_to(diag);
|
||||
self.prefix.add_to(diag);
|
||||
self.suffix.add_to(diag);
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue