@@ -339,22 +339,25 @@ impl<'tcx> LateLintPass<'tcx> for ExistingDocKeyword {
339
339
}
340
340
341
341
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.
344
344
///
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).
346
347
pub rustc:: UNTRANSLATABLE_DIAGNOSTIC ,
347
348
Deny ,
348
349
"prevent creation of diagnostics which cannot be translated" ,
349
350
report_in_external_macro: true
350
351
}
351
352
352
353
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.
356
358
///
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).
358
361
pub rustc:: DIAGNOSTIC_OUTSIDE_OF_IMPL ,
359
362
Deny ,
360
363
"prevent creation of diagnostics outside of `IntoDiagnostic`/`AddToDiagnostic` impls" ,
@@ -369,10 +372,15 @@ declare_tool_lint! {
369
372
report_in_external_macro: true
370
373
}
371
374
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
+ ] ) ;
373
380
374
381
impl LateLintPass < ' _ > for Diagnostics {
375
382
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]`.
376
384
let Some ( ( span, def_id, args) ) = typeck_results_of_method_fn ( cx, expr) else { return } ;
377
385
debug ! ( ?span, ?def_id, ?args) ;
378
386
let has_attr = ty:: Instance :: resolve ( cx. tcx , cx. param_env , def_id, args)
@@ -383,12 +391,19 @@ impl LateLintPass<'_> for Diagnostics {
383
391
return ;
384
392
}
385
393
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.
386
399
let mut found_parent_with_attr = false ;
387
400
let mut found_impl = false ;
388
401
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 ;
392
407
}
393
408
394
409
debug ! ( ?parent) ;
@@ -407,6 +422,10 @@ impl LateLintPass<'_> for Diagnostics {
407
422
cx. emit_span_lint ( DIAGNOSTIC_OUTSIDE_OF_IMPL , span, DiagOutOfImpl ) ;
408
423
}
409
424
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.
410
429
let mut found_diagnostic_message = false ;
411
430
for ty in args. types ( ) {
412
431
debug ! ( ?ty) ;
@@ -506,7 +525,7 @@ declare_tool_lint! {
506
525
report_in_external_macro: true
507
526
}
508
527
509
- declare_lint_pass ! ( BadOptAccess => [ BAD_OPT_ACCESS ] ) ;
528
+ declare_lint_pass ! ( BadOptAccess => [ BAD_OPT_ACCESS ] ) ;
510
529
511
530
impl LateLintPass < ' _ > for BadOptAccess {
512
531
fn check_expr ( & mut self , cx : & LateContext < ' _ > , expr : & Expr < ' _ > ) {
0 commit comments