Skip to content

Commit 5a66d65

Browse files
committed
Some rustc_lint_diagnostic cleanups.
This commit: - Expands and corrects some comments. - Does some minor formatting improvements. - Adds missing `DecorateLint` cases to `tests/ui-fulldeps/internal-lints/diagnostics.rs`.
1 parent bb59453 commit 5a66d65

File tree

4 files changed

+70
-19
lines changed

4 files changed

+70
-19
lines changed

compiler/rustc_feature/src/builtin_attrs.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -625,7 +625,8 @@ pub const BUILTIN_ATTRIBUTES: &[BuiltinAttribute] = &[
625625
// might not be stable during incremental compilation.
626626
rustc_attr!(rustc_lint_query_instability, Normal, template!(Word), WarnFollowing, INTERNAL_UNSTABLE),
627627
// Used by the `rustc::untranslatable_diagnostic` and `rustc::diagnostic_outside_of_impl` lints
628-
// to assist in changes to diagnostic APIs.
628+
// to assist in changes to diagnostic APIs. Any function with this attribute will be checked by
629+
// those lints.
629630
rustc_attr!(rustc_lint_diagnostics, Normal, template!(Word), WarnFollowing, INTERNAL_UNSTABLE),
630631
// Used by the `rustc::bad_opt_access` lint to identify `DebuggingOptions` and `CodegenOptions`
631632
// types (as well as any others in future).

compiler/rustc_lint/src/internal.rs

+31-12
Original file line numberDiff line numberDiff line change
@@ -339,22 +339,25 @@ impl<'tcx> LateLintPass<'tcx> for ExistingDocKeyword {
339339
}
340340

341341
declare_tool_lint! {
342-
/// The `untranslatable_diagnostic` lint detects diagnostics created
343-
/// without using translatable Fluent strings.
342+
/// The `untranslatable_diagnostic` lint detects messages passed to functions annotated with
343+
/// `#[rustc_lint_diagnostics]` without using translatable Fluent strings.
344344
///
345-
/// More details on translatable diagnostics can be found [here](https://rustc-dev-guide.rust-lang.org/diagnostics/translation.html).
345+
/// More details on translatable diagnostics can be found
346+
/// [here](https://rustc-dev-guide.rust-lang.org/diagnostics/translation.html).
346347
pub rustc::UNTRANSLATABLE_DIAGNOSTIC,
347348
Deny,
348349
"prevent creation of diagnostics which cannot be translated",
349350
report_in_external_macro: true
350351
}
351352

352353
declare_tool_lint! {
353-
/// The `diagnostic_outside_of_impl` lint detects diagnostics created manually,
354-
/// and inside an `IntoDiagnostic`/`AddToDiagnostic` implementation,
355-
/// or a `#[derive(Diagnostic)]`/`#[derive(Subdiagnostic)]` expansion.
354+
/// The `diagnostic_outside_of_impl` lint detects calls to functions annotated with
355+
/// `#[rustc_lint_diagnostics]` that are outside a `IntoDiagnostic`, `AddToDiagnostic`, or
356+
/// `DecorateLint` impl, or a `#[derive(Diagnostic)]`, `#[derive(Subdiagnostic)]`,
357+
/// `#[derive(DecorateLint)]` expansion.
356358
///
357-
/// More details on diagnostics implementations can be found [here](https://rustc-dev-guide.rust-lang.org/diagnostics/diagnostic-structs.html).
359+
/// More details on diagnostics implementations can be found
360+
/// [here](https://rustc-dev-guide.rust-lang.org/diagnostics/diagnostic-structs.html).
358361
pub rustc::DIAGNOSTIC_OUTSIDE_OF_IMPL,
359362
Deny,
360363
"prevent creation of diagnostics outside of `IntoDiagnostic`/`AddToDiagnostic` impls",
@@ -369,10 +372,15 @@ declare_tool_lint! {
369372
report_in_external_macro: true
370373
}
371374

372-
declare_lint_pass!(Diagnostics => [ UNTRANSLATABLE_DIAGNOSTIC, DIAGNOSTIC_OUTSIDE_OF_IMPL, UNTRANSLATABLE_DIAGNOSTIC_TRIVIAL ]);
375+
declare_lint_pass!(Diagnostics => [
376+
UNTRANSLATABLE_DIAGNOSTIC,
377+
DIAGNOSTIC_OUTSIDE_OF_IMPL,
378+
UNTRANSLATABLE_DIAGNOSTIC_TRIVIAL
379+
]);
373380

374381
impl LateLintPass<'_> for Diagnostics {
375382
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
383+
// Return early if it's not a call to a function marked with `#[rustc_lint_diagnostics]`.
376384
let Some((span, def_id, args)) = typeck_results_of_method_fn(cx, expr) else { return };
377385
debug!(?span, ?def_id, ?args);
378386
let has_attr = ty::Instance::resolve(cx.tcx, cx.param_env, def_id, args)
@@ -383,12 +391,19 @@ impl LateLintPass<'_> for Diagnostics {
383391
return;
384392
}
385393

394+
// Calls to `#[rustc_lint_diagnostics]`-marked functions should only occur:
395+
// - inside an impl of `IntoDiagnostic`, `AddToDiagnostic`, or `DecorateLint`, or
396+
// - inside a function that is itself marked with `#[rustc_lint_diagnostics]`.
397+
//
398+
// Otherwise, emit a `DIAGNOSTIC_OUTSIDE_OF_IMPL` lint.
386399
let mut found_parent_with_attr = false;
387400
let mut found_impl = false;
388401
for (hir_id, parent) in cx.tcx.hir().parent_iter(expr.hir_id) {
389-
if let Some(owner_did) = hir_id.as_owner() {
390-
found_parent_with_attr = found_parent_with_attr
391-
|| cx.tcx.has_attr(owner_did, sym::rustc_lint_diagnostics);
402+
if let Some(owner_did) = hir_id.as_owner()
403+
&& cx.tcx.has_attr(owner_did, sym::rustc_lint_diagnostics)
404+
{
405+
found_parent_with_attr = true;
406+
break;
392407
}
393408

394409
debug!(?parent);
@@ -407,6 +422,10 @@ impl LateLintPass<'_> for Diagnostics {
407422
cx.emit_span_lint(DIAGNOSTIC_OUTSIDE_OF_IMPL, span, DiagOutOfImpl);
408423
}
409424

425+
// Calls to `#[rustc_lint_diagnostics]`-marked functions should only be passed
426+
// `DiagnosticMessage` or `SubdiagnosticMessage` arguments, unless they are inside a
427+
// function that is itself marked with `#[rustc_lint_diagnostics]`. Otherwise, emit an
428+
// `UNTRANSLATABLE_DIAGNOSTIC` lint.
410429
let mut found_diagnostic_message = false;
411430
for ty in args.types() {
412431
debug!(?ty);
@@ -506,7 +525,7 @@ declare_tool_lint! {
506525
report_in_external_macro: true
507526
}
508527

509-
declare_lint_pass!(BadOptAccess => [ BAD_OPT_ACCESS ]);
528+
declare_lint_pass!(BadOptAccess => [BAD_OPT_ACCESS]);
510529

511530
impl LateLintPass<'_> for BadOptAccess {
512531
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {

tests/ui-fulldeps/internal-lints/diagnostics.rs

+27-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,8 @@ extern crate rustc_session;
1313
extern crate rustc_span;
1414

1515
use rustc_errors::{
16-
AddToDiagnostic, Diagnostic, DiagnosticBuilder, DiagnosticMessage, EmissionGuarantee, DiagCtxt,
17-
IntoDiagnostic, Level, SubdiagnosticMessageOp,
16+
AddToDiagnostic, DecorateLint, Diagnostic, DiagnosticBuilder, DiagnosticMessage,
17+
EmissionGuarantee, DiagCtxt, IntoDiagnostic, Level, SubdiagnosticMessageOp,
1818
};
1919
use rustc_macros::{Diagnostic, Subdiagnostic};
2020
use rustc_span::Span;
@@ -77,6 +77,31 @@ impl AddToDiagnostic for TranslatableInAddToDiagnostic {
7777
}
7878
}
7979

80+
pub struct UntranslatableInDecorateLint;
81+
82+
impl<'a> DecorateLint<'a, ()> for UntranslatableInDecorateLint {
83+
fn decorate_lint<'b, >(self, diag: &'b mut DiagnosticBuilder<'a, ()>) {
84+
diag.note("untranslatable diagnostic");
85+
//~^ ERROR diagnostics should be created using translatable messages
86+
}
87+
88+
fn msg(&self) -> DiagnosticMessage {
89+
unreachable!();
90+
}
91+
}
92+
93+
pub struct TranslatableInDecorateLint;
94+
95+
impl<'a> DecorateLint<'a, ()> for TranslatableInDecorateLint {
96+
fn decorate_lint<'b>(self, diag: &'b mut DiagnosticBuilder<'a, ()>) {
97+
diag.note(crate::fluent_generated::no_crate_note);
98+
}
99+
100+
fn msg(&self) -> DiagnosticMessage {
101+
unreachable!();
102+
}
103+
}
104+
80105
pub fn make_diagnostics<'a>(dcx: &'a DiagCtxt) {
81106
let _diag = dcx.struct_err(crate::fluent_generated::no_crate_example);
82107
//~^ ERROR diagnostics should only be created in `IntoDiagnostic`/`AddToDiagnostic` impls

tests/ui-fulldeps/internal-lints/diagnostics.stderr

+10-4
Original file line numberDiff line numberDiff line change
@@ -16,8 +16,14 @@ error: diagnostics should be created using translatable messages
1616
LL | diag.note("untranslatable diagnostic");
1717
| ^^^^
1818

19+
error: diagnostics should be created using translatable messages
20+
--> $DIR/diagnostics.rs:84:14
21+
|
22+
LL | diag.note("untranslatable diagnostic");
23+
| ^^^^
24+
1925
error: diagnostics should only be created in `IntoDiagnostic`/`AddToDiagnostic` impls
20-
--> $DIR/diagnostics.rs:81:21
26+
--> $DIR/diagnostics.rs:106:21
2127
|
2228
LL | let _diag = dcx.struct_err(crate::fluent_generated::no_crate_example);
2329
| ^^^^^^^^^^
@@ -29,16 +35,16 @@ LL | #![deny(rustc::diagnostic_outside_of_impl)]
2935
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
3036

3137
error: diagnostics should only be created in `IntoDiagnostic`/`AddToDiagnostic` impls
32-
--> $DIR/diagnostics.rs:84:21
38+
--> $DIR/diagnostics.rs:109:21
3339
|
3440
LL | let _diag = dcx.struct_err("untranslatable diagnostic");
3541
| ^^^^^^^^^^
3642

3743
error: diagnostics should be created using translatable messages
38-
--> $DIR/diagnostics.rs:84:21
44+
--> $DIR/diagnostics.rs:109:21
3945
|
4046
LL | let _diag = dcx.struct_err("untranslatable diagnostic");
4147
| ^^^^^^^^^^
4248

43-
error: aborting due to 5 previous errors
49+
error: aborting due to 6 previous errors
4450

0 commit comments

Comments
 (0)