@@ -14,7 +14,7 @@ use rustc_middle::lint::{
14
14
use rustc_middle:: ty:: query:: Providers ;
15
15
use rustc_middle:: ty:: { RegisteredTools , TyCtxt } ;
16
16
use rustc_session:: lint:: {
17
- builtin:: { self , FORBIDDEN_LINT_GROUPS } ,
17
+ builtin:: { self , FORBIDDEN_LINT_GROUPS , UNFULFILLED_LINT_EXPECTATIONS } ,
18
18
Level , Lint , LintExpectationId , LintId ,
19
19
} ;
20
20
use rustc_session:: parse:: feature_err;
@@ -212,6 +212,14 @@ impl<'s> LintLevelsBuilder<'s> {
212
212
}
213
213
}
214
214
}
215
+
216
+ // The lint `unfulfilled_lint_expectations` can't be expected, as it would suppress itself.
217
+ // Handling expectations of this lint would add additional complexity with little to no
218
+ // benefit. The expect level for this lint will therefore be ignored.
219
+ if let Level :: Expect ( _) = level && id == LintId :: of ( UNFULFILLED_LINT_EXPECTATIONS ) {
220
+ return ;
221
+ }
222
+
215
223
if let Level :: ForceWarn = old_level {
216
224
specs. insert ( id, ( old_level, old_src) ) ;
217
225
} else {
@@ -344,6 +352,20 @@ impl<'s> LintLevelsBuilder<'s> {
344
352
self . store . check_lint_name ( & name, tool_name, self . registered_tools ) ;
345
353
match & lint_result {
346
354
CheckLintNameResult :: Ok ( ids) => {
355
+ // This checks for instances where the user writes `#[expect(unfulfilled_lint_expectations)]`
356
+ // in that case we want to avoid overriding the lint level but instead add an expectation that
357
+ // can't be fulfilled. The lint message will include an explanation, that the
358
+ // `unfulfilled_lint_expectations` lint can't be expected.
359
+ if let Level :: Expect ( expect_id) = level {
360
+ let is_unfulfilled_lint_expectations = match ids {
361
+ [ lint] => * lint == LintId :: of ( UNFULFILLED_LINT_EXPECTATIONS ) ,
362
+ _ => false ,
363
+ } ;
364
+ self . lint_expectations . push ( (
365
+ expect_id,
366
+ LintExpectation :: new ( reason, sp, is_unfulfilled_lint_expectations) ,
367
+ ) ) ;
368
+ }
347
369
let src = LintLevelSource :: Node (
348
370
meta_item. path . segments . last ( ) . expect ( "empty lint name" ) . ident . name ,
349
371
sp,
@@ -353,10 +375,6 @@ impl<'s> LintLevelsBuilder<'s> {
353
375
self . check_gated_lint ( id, attr. span ) ;
354
376
self . insert_spec ( & mut specs, id, ( level, src) ) ;
355
377
}
356
- if let Level :: Expect ( expect_id) = level {
357
- self . lint_expectations
358
- . push ( ( expect_id, LintExpectation :: new ( reason, sp) ) ) ;
359
- }
360
378
}
361
379
362
380
CheckLintNameResult :: Tool ( result) => {
@@ -374,7 +392,7 @@ impl<'s> LintLevelsBuilder<'s> {
374
392
}
375
393
if let Level :: Expect ( expect_id) = level {
376
394
self . lint_expectations
377
- . push ( ( expect_id, LintExpectation :: new ( reason, sp) ) ) ;
395
+ . push ( ( expect_id, LintExpectation :: new ( reason, sp, false ) ) ) ;
378
396
}
379
397
}
380
398
Err ( ( Some ( ids) , ref new_lint_name) ) => {
@@ -414,7 +432,7 @@ impl<'s> LintLevelsBuilder<'s> {
414
432
}
415
433
if let Level :: Expect ( expect_id) = level {
416
434
self . lint_expectations
417
- . push ( ( expect_id, LintExpectation :: new ( reason, sp) ) ) ;
435
+ . push ( ( expect_id, LintExpectation :: new ( reason, sp, false ) ) ) ;
418
436
}
419
437
}
420
438
Err ( ( None , _) ) => {
@@ -511,7 +529,7 @@ impl<'s> LintLevelsBuilder<'s> {
511
529
}
512
530
if let Level :: Expect ( expect_id) = level {
513
531
self . lint_expectations
514
- . push ( ( expect_id, LintExpectation :: new ( reason, sp) ) ) ;
532
+ . push ( ( expect_id, LintExpectation :: new ( reason, sp, false ) ) ) ;
515
533
}
516
534
} else {
517
535
panic ! ( "renamed lint does not exist: {}" , new_name) ;
0 commit comments