forked from rust-lang/rust
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Rollup merge of rust-lang#100843 - IntQuant:issue-100717-infer, r=com…
…piler-errors Migrate part of rustc_infer to session diagnostic
- Loading branch information
Showing
10 changed files
with
649 additions
and
212 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
infer_opaque_hidden_type = | ||
opaque type's hidden type cannot be another opaque type from the same scope | ||
.label = one of the two opaque types used here has to be outside its defining scope | ||
.opaque_type = opaque type whose hidden type is being assigned | ||
.hidden_type = opaque type being used as hidden type | ||
infer_type_annotations_needed = {$source_kind -> | ||
[closure] type annotations needed for the closure `{$source_name}` | ||
[normal] type annotations needed for `{$source_name}` | ||
*[other] type annotations needed | ||
} | ||
.label = type must be known at this point | ||
infer_label_bad = {$bad_kind -> | ||
*[other] cannot infer type | ||
[more_info] cannot infer {$prefix_kind -> | ||
*[type] type for {$prefix} | ||
[const_with_param] the value of const parameter | ||
[const] the value of the constant | ||
} `{$name}`{$has_parent -> | ||
[true] {" "}declared on the {$parent_prefix} `{$parent_name}` | ||
*[false] {""} | ||
} | ||
} | ||
infer_source_kind_subdiag_let = {$kind -> | ||
[with_pattern] consider giving `{$name}` an explicit type | ||
[closure] consider giving this closure parameter an explicit type | ||
*[other] consider giving this pattern a type | ||
}{$x_kind -> | ||
[has_name] , where the {$prefix_kind -> | ||
*[type] type for {$prefix} | ||
[const_with_param] the value of const parameter | ||
[const] the value of the constant | ||
} `{$arg_name}` is specified | ||
[underscore] , where the placeholders `_` are specified | ||
*[empty] {""} | ||
} | ||
infer_source_kind_subdiag_generic_label = | ||
cannot infer {$is_type -> | ||
[true] type | ||
*[false] the value | ||
} of the {$is_type -> | ||
[true] type | ||
*[false] const | ||
} {$parent_exists -> | ||
[true] parameter `{$param_name}` declared on the {$parent_prefix} `{$parent_name}` | ||
*[false] parameter {$param_name} | ||
} | ||
infer_source_kind_subdiag_generic_suggestion = | ||
consider specifying the generic {$arg_count -> | ||
[one] argument | ||
*[other] arguments | ||
} | ||
infer_source_kind_fully_qualified = | ||
try using a fully qualified path to specify the expected types | ||
infer_source_kind_closure_return = | ||
try giving this closure an explicit return type | ||
# generator_kind may need to be translated | ||
infer_need_type_info_in_generator = | ||
type inside {$generator_kind -> | ||
[async_block] `async` block | ||
[async_closure] `async` closure | ||
[async_fn] `async fn` body | ||
*[generator] generator | ||
} must be known in this context | ||
infer_subtype = ...so that the {$requirement -> | ||
[method_compat] method type is compatible with trait | ||
[type_compat] associated type is compatible with trait | ||
[const_compat] const is compatible with trait | ||
[expr_assignable] expression is assignable | ||
[if_else_different] `if` and `else` have incompatible types | ||
[no_else] `if` missing an `else` returns `()` | ||
[fn_main_correct_type] `main` function has the correct type | ||
[fn_start_correct_type] #[start]` function has the correct type | ||
[intristic_correct_type] intrinsic has the correct type | ||
[method_correct_type] method receiver has the correct type | ||
*[other] types are compatible | ||
} | ||
infer_subtype_2 = ...so that {$requirement -> | ||
[method_compat] method type is compatible with trait | ||
[type_compat] associated type is compatible with trait | ||
[const_compat] const is compatible with trait | ||
[expr_assignable] expression is assignable | ||
[if_else_different] `if` and `else` have incompatible types | ||
[no_else] `if` missing an `else` returns `()` | ||
[fn_main_correct_type] `main` function has the correct type | ||
[fn_start_correct_type] #[start]` function has the correct type | ||
[intristic_correct_type] intrinsic has the correct type | ||
[method_correct_type] method receiver has the correct type | ||
*[other] types are compatible | ||
} | ||
infer_reborrow = ...so that reference does not outlive borrowed content | ||
infer_reborrow_upvar = ...so that closure can access `{$name}` | ||
infer_relate_object_bound = ...so that it can be closed over into an object | ||
infer_data_borrowed = ...so that the type `{$name}` is not borrowed for too long | ||
infer_reference_outlives_referent = ...so that the reference type `{$name}` does not outlive the data it points at | ||
infer_relate_param_bound = ...so that the type `{$name}` will meet its required lifetime bounds{$continues -> | ||
[true] ... | ||
*[false] {""} | ||
} | ||
infer_relate_param_bound_2 = ...that is required by this bound | ||
infer_relate_region_param_bound = ...so that the declared lifetime parameter bounds are satisfied | ||
infer_compare_impl_item_obligation = ...so that the definition in impl matches the definition from the trait |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,254 @@ | ||
use rustc_errors::{fluent, AddSubdiagnostic, DiagnosticMessage, DiagnosticStyledString}; | ||
use rustc_hir::FnRetTy; | ||
use rustc_macros::SessionDiagnostic; | ||
use rustc_span::{BytePos, Span}; | ||
|
||
use crate::infer::error_reporting::{ | ||
need_type_info::{GeneratorKindAsDiagArg, UnderspecifiedArgKind}, | ||
ObligationCauseAsDiagArg, | ||
}; | ||
|
||
#[derive(SessionDiagnostic)] | ||
#[diag(infer::opaque_hidden_type)] | ||
pub struct OpaqueHiddenTypeDiag { | ||
#[primary_span] | ||
#[label] | ||
pub span: Span, | ||
#[note(infer::opaque_type)] | ||
pub opaque_type: Span, | ||
#[note(infer::hidden_type)] | ||
pub hidden_type: Span, | ||
} | ||
|
||
#[derive(SessionDiagnostic)] | ||
#[diag(infer::type_annotations_needed, code = "E0282")] | ||
pub struct AnnotationRequired<'a> { | ||
#[primary_span] | ||
pub span: Span, | ||
pub source_kind: &'static str, | ||
pub source_name: &'a str, | ||
#[label] | ||
pub failure_span: Option<Span>, | ||
#[subdiagnostic] | ||
pub bad_label: Option<InferenceBadError<'a>>, | ||
#[subdiagnostic] | ||
pub infer_subdiags: Vec<SourceKindSubdiag<'a>>, | ||
#[subdiagnostic] | ||
pub multi_suggestions: Vec<SourceKindMultiSuggestion<'a>>, | ||
} | ||
|
||
// Copy of `AnnotationRequired` for E0283 | ||
#[derive(SessionDiagnostic)] | ||
#[diag(infer::type_annotations_needed, code = "E0283")] | ||
pub struct AmbigousImpl<'a> { | ||
#[primary_span] | ||
pub span: Span, | ||
pub source_kind: &'static str, | ||
pub source_name: &'a str, | ||
#[label] | ||
pub failure_span: Option<Span>, | ||
#[subdiagnostic] | ||
pub bad_label: Option<InferenceBadError<'a>>, | ||
#[subdiagnostic] | ||
pub infer_subdiags: Vec<SourceKindSubdiag<'a>>, | ||
#[subdiagnostic] | ||
pub multi_suggestions: Vec<SourceKindMultiSuggestion<'a>>, | ||
} | ||
|
||
// Copy of `AnnotationRequired` for E0284 | ||
#[derive(SessionDiagnostic)] | ||
#[diag(infer::type_annotations_needed, code = "E0284")] | ||
pub struct AmbigousReturn<'a> { | ||
#[primary_span] | ||
pub span: Span, | ||
pub source_kind: &'static str, | ||
pub source_name: &'a str, | ||
#[label] | ||
pub failure_span: Option<Span>, | ||
#[subdiagnostic] | ||
pub bad_label: Option<InferenceBadError<'a>>, | ||
#[subdiagnostic] | ||
pub infer_subdiags: Vec<SourceKindSubdiag<'a>>, | ||
#[subdiagnostic] | ||
pub multi_suggestions: Vec<SourceKindMultiSuggestion<'a>>, | ||
} | ||
|
||
#[derive(SessionDiagnostic)] | ||
#[diag(infer::need_type_info_in_generator, code = "E0698")] | ||
pub struct NeedTypeInfoInGenerator<'a> { | ||
#[primary_span] | ||
pub span: Span, | ||
pub generator_kind: GeneratorKindAsDiagArg, | ||
#[subdiagnostic] | ||
pub bad_label: InferenceBadError<'a>, | ||
} | ||
|
||
// Used when a better one isn't available | ||
#[derive(SessionSubdiagnostic)] | ||
#[label(infer::label_bad)] | ||
pub struct InferenceBadError<'a> { | ||
#[primary_span] | ||
pub span: Span, | ||
pub bad_kind: &'static str, | ||
pub prefix_kind: UnderspecifiedArgKind, | ||
pub has_parent: bool, | ||
pub prefix: &'a str, | ||
pub parent_prefix: &'a str, | ||
pub parent_name: String, | ||
pub name: String, | ||
} | ||
|
||
#[derive(SessionSubdiagnostic)] | ||
pub enum SourceKindSubdiag<'a> { | ||
#[suggestion_verbose( | ||
infer::source_kind_subdiag_let, | ||
code = ": {type_name}", | ||
applicability = "has-placeholders" | ||
)] | ||
LetLike { | ||
#[primary_span] | ||
span: Span, | ||
name: String, | ||
type_name: String, | ||
kind: &'static str, | ||
x_kind: &'static str, | ||
prefix_kind: UnderspecifiedArgKind, | ||
prefix: &'a str, | ||
arg_name: String, | ||
}, | ||
#[label(infer::source_kind_subdiag_generic_label)] | ||
GenericLabel { | ||
#[primary_span] | ||
span: Span, | ||
is_type: bool, | ||
param_name: String, | ||
parent_exists: bool, | ||
parent_prefix: String, | ||
parent_name: String, | ||
}, | ||
#[suggestion_verbose( | ||
infer::source_kind_subdiag_generic_suggestion, | ||
code = "::<{args}>", | ||
applicability = "has-placeholders" | ||
)] | ||
GenericSuggestion { | ||
#[primary_span] | ||
span: Span, | ||
arg_count: usize, | ||
args: String, | ||
}, | ||
} | ||
|
||
// Has to be implemented manually because multipart suggestions are not supported by the derive macro. | ||
// Would be a part of `SourceKindSubdiag` otherwise. | ||
pub enum SourceKindMultiSuggestion<'a> { | ||
FullyQualified { | ||
span: Span, | ||
def_path: String, | ||
adjustment: &'a str, | ||
successor: (&'a str, BytePos), | ||
}, | ||
ClosureReturn { | ||
ty_info: String, | ||
data: &'a FnRetTy<'a>, | ||
should_wrap_expr: Option<Span>, | ||
}, | ||
} | ||
|
||
impl AddSubdiagnostic for SourceKindMultiSuggestion<'_> { | ||
fn add_to_diagnostic(self, diag: &mut rustc_errors::Diagnostic) { | ||
match self { | ||
Self::FullyQualified { span, def_path, adjustment, successor } => { | ||
let suggestion = vec![ | ||
(span.shrink_to_lo(), format!("{def_path}({adjustment}")), | ||
(span.shrink_to_hi().with_hi(successor.1), successor.0.to_string()), | ||
]; | ||
diag.multipart_suggestion_verbose( | ||
fluent::infer::source_kind_fully_qualified, | ||
suggestion, | ||
rustc_errors::Applicability::HasPlaceholders, | ||
); | ||
} | ||
Self::ClosureReturn { ty_info, data, should_wrap_expr } => { | ||
let (arrow, post) = match data { | ||
FnRetTy::DefaultReturn(_) => ("-> ", " "), | ||
_ => ("", ""), | ||
}; | ||
let suggestion = match should_wrap_expr { | ||
Some(end_span) => vec![ | ||
(data.span(), format!("{}{}{}{{ ", arrow, ty_info, post)), | ||
(end_span, " }".to_string()), | ||
], | ||
None => vec![(data.span(), format!("{}{}{}", arrow, ty_info, post))], | ||
}; | ||
diag.multipart_suggestion_verbose( | ||
fluent::infer::source_kind_closure_return, | ||
suggestion, | ||
rustc_errors::Applicability::HasPlaceholders, | ||
); | ||
} | ||
} | ||
} | ||
} | ||
|
||
pub enum RegionOriginNote<'a> { | ||
Plain { | ||
span: Span, | ||
msg: DiagnosticMessage, | ||
}, | ||
WithName { | ||
span: Span, | ||
msg: DiagnosticMessage, | ||
name: &'a str, | ||
continues: bool, | ||
}, | ||
WithRequirement { | ||
span: Span, | ||
requirement: ObligationCauseAsDiagArg<'a>, | ||
expected_found: Option<(DiagnosticStyledString, DiagnosticStyledString)>, | ||
}, | ||
} | ||
|
||
impl AddSubdiagnostic for RegionOriginNote<'_> { | ||
fn add_to_diagnostic(self, diag: &mut rustc_errors::Diagnostic) { | ||
let mut label_or_note = |span, msg: DiagnosticMessage| { | ||
let sub_count = diag.children.iter().filter(|d| d.span.is_dummy()).count(); | ||
let expanded_sub_count = diag.children.iter().filter(|d| !d.span.is_dummy()).count(); | ||
let span_is_primary = diag.span.primary_spans().iter().all(|&sp| sp == span); | ||
if span_is_primary && sub_count == 0 && expanded_sub_count == 0 { | ||
diag.span_label(span, msg); | ||
} else if span_is_primary && expanded_sub_count == 0 { | ||
diag.note(msg); | ||
} else { | ||
diag.span_note(span, msg); | ||
} | ||
}; | ||
match self { | ||
RegionOriginNote::Plain { span, msg } => { | ||
label_or_note(span, msg); | ||
} | ||
RegionOriginNote::WithName { span, msg, name, continues } => { | ||
label_or_note(span, msg); | ||
diag.set_arg("name", name); | ||
diag.set_arg("continues", continues); | ||
} | ||
RegionOriginNote::WithRequirement { | ||
span, | ||
requirement, | ||
expected_found: Some((expected, found)), | ||
} => { | ||
label_or_note(span, fluent::infer::subtype); | ||
diag.set_arg("requirement", requirement); | ||
|
||
diag.note_expected_found(&"", expected, &"", found); | ||
} | ||
RegionOriginNote::WithRequirement { span, requirement, expected_found: None } => { | ||
// FIXME: this really should be handled at some earlier stage. Our | ||
// handling of region checking when type errors are present is | ||
// *terrible*. | ||
label_or_note(span, fluent::infer::subtype_2); | ||
diag.set_arg("requirement", requirement); | ||
} | ||
}; | ||
} | ||
} |
Oops, something went wrong.