Skip to content

Commit d39d609

Browse files
committedMar 7, 2022
Handle #[expect(unfulfilled_lint_expectations)] with a lint message
1 parent 165b558 commit d39d609

File tree

6 files changed

+119
-14
lines changed

6 files changed

+119
-14
lines changed
 

‎compiler/rustc_errors/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -970,6 +970,7 @@ impl Handler {
970970

971971
/// This methods steals all [`LintExpectationId`]s that are stored inside
972972
/// [`HandlerInner`] and indicate that the linked expectation has been fulfilled.
973+
#[must_use]
973974
pub fn steal_fulfilled_expectation_ids(&self) -> FxHashSet<LintExpectationId> {
974975
assert!(
975976
self.inner.borrow().unstable_expect_diagnostics.is_empty(),

‎compiler/rustc_lint/src/expect.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -30,10 +30,6 @@ fn emit_unfulfilled_expectation_lint(
3030
hir_id: HirId,
3131
expectation: &LintExpectation,
3232
) {
33-
// FIXME: The current implementation doesn't cover cases where the
34-
// `unfulfilled_lint_expectations` is actually expected by another lint
35-
// expectation. This can be added here by checking the lint level and
36-
// retrieving the `LintExpectationId` if it was expected.
3733
tcx.struct_span_lint_hir(
3834
builtin::UNFULFILLED_LINT_EXPECTATIONS,
3935
hir_id,
@@ -43,6 +39,11 @@ fn emit_unfulfilled_expectation_lint(
4339
if let Some(rationale) = expectation.reason {
4440
diag.note(&rationale.as_str());
4541
}
42+
43+
if expectation.is_unfulfilled_lint_expectations {
44+
diag.note("the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message");
45+
}
46+
4647
diag.emit();
4748
},
4849
);

‎compiler/rustc_lint/src/levels.rs

+26-8
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use rustc_middle::lint::{
1414
use rustc_middle::ty::query::Providers;
1515
use rustc_middle::ty::{RegisteredTools, TyCtxt};
1616
use rustc_session::lint::{
17-
builtin::{self, FORBIDDEN_LINT_GROUPS},
17+
builtin::{self, FORBIDDEN_LINT_GROUPS, UNFULFILLED_LINT_EXPECTATIONS},
1818
Level, Lint, LintExpectationId, LintId,
1919
};
2020
use rustc_session::parse::feature_err;
@@ -212,6 +212,14 @@ impl<'s> LintLevelsBuilder<'s> {
212212
}
213213
}
214214
}
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+
215223
if let Level::ForceWarn = old_level {
216224
specs.insert(id, (old_level, old_src));
217225
} else {
@@ -344,6 +352,20 @@ impl<'s> LintLevelsBuilder<'s> {
344352
self.store.check_lint_name(&name, tool_name, self.registered_tools);
345353
match &lint_result {
346354
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+
}
347369
let src = LintLevelSource::Node(
348370
meta_item.path.segments.last().expect("empty lint name").ident.name,
349371
sp,
@@ -353,10 +375,6 @@ impl<'s> LintLevelsBuilder<'s> {
353375
self.check_gated_lint(id, attr.span);
354376
self.insert_spec(&mut specs, id, (level, src));
355377
}
356-
if let Level::Expect(expect_id) = level {
357-
self.lint_expectations
358-
.push((expect_id, LintExpectation::new(reason, sp)));
359-
}
360378
}
361379

362380
CheckLintNameResult::Tool(result) => {
@@ -374,7 +392,7 @@ impl<'s> LintLevelsBuilder<'s> {
374392
}
375393
if let Level::Expect(expect_id) = level {
376394
self.lint_expectations
377-
.push((expect_id, LintExpectation::new(reason, sp)));
395+
.push((expect_id, LintExpectation::new(reason, sp, false)));
378396
}
379397
}
380398
Err((Some(ids), ref new_lint_name)) => {
@@ -414,7 +432,7 @@ impl<'s> LintLevelsBuilder<'s> {
414432
}
415433
if let Level::Expect(expect_id) = level {
416434
self.lint_expectations
417-
.push((expect_id, LintExpectation::new(reason, sp)));
435+
.push((expect_id, LintExpectation::new(reason, sp, false)));
418436
}
419437
}
420438
Err((None, _)) => {
@@ -511,7 +529,7 @@ impl<'s> LintLevelsBuilder<'s> {
511529
}
512530
if let Level::Expect(expect_id) = level {
513531
self.lint_expectations
514-
.push((expect_id, LintExpectation::new(reason, sp)));
532+
.push((expect_id, LintExpectation::new(reason, sp, false)));
515533
}
516534
} else {
517535
panic!("renamed lint does not exist: {}", new_name);

‎compiler/rustc_middle/src/lint.rs

+10-2
Original file line numberDiff line numberDiff line change
@@ -204,11 +204,19 @@ pub struct LintExpectation {
204204
pub reason: Option<Symbol>,
205205
/// The [`Span`] of the attribute that this expectation originated from.
206206
pub emission_span: Span,
207+
/// Lint messages for the `unfulfilled_lint_expectations` lint will be
208+
/// adjusted to include an additional note. Therefore, we have to track if
209+
/// the expectation is for the lint.
210+
pub is_unfulfilled_lint_expectations: bool,
207211
}
208212

209213
impl LintExpectation {
210-
pub fn new(reason: Option<Symbol>, attr_span: Span) -> Self {
211-
Self { reason, emission_span: attr_span }
214+
pub fn new(
215+
reason: Option<Symbol>,
216+
emission_span: Span,
217+
is_unfulfilled_lint_expectations: bool,
218+
) -> Self {
219+
Self { reason, emission_span, is_unfulfilled_lint_expectations }
212220
}
213221
}
214222

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
// check-pass
2+
// ignore-tidy-linelength
3+
4+
#![feature(lint_reasons)]
5+
#![warn(unused_mut)]
6+
7+
#![expect(unfulfilled_lint_expectations, reason = "idk why you would expect this")]
8+
//~^ WARNING this lint expectation is unfulfilled
9+
//~| NOTE `#[warn(unfulfilled_lint_expectations)]` on by default
10+
//~| NOTE idk why you would expect this
11+
//~| NOTE the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
12+
13+
#[expect(unfulfilled_lint_expectations, reason = "a local: idk why you would expect this")]
14+
//~^ WARNING this lint expectation is unfulfilled
15+
//~| NOTE a local: idk why you would expect this
16+
//~| NOTE the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
17+
pub fn normal_test_fn() {
18+
#[expect(unused_mut, reason = "this expectation will create a diagnostic with the default lint level")]
19+
//~^ WARNING this lint expectation is unfulfilled
20+
//~| NOTE this expectation will create a diagnostic with the default lint level
21+
let mut v = vec![1, 1, 2, 3, 5];
22+
v.sort();
23+
24+
// Check that lint lists including `unfulfilled_lint_expectations` are also handled correctly
25+
#[expect(unused, unfulfilled_lint_expectations, reason = "the expectation for `unused` should be fulfilled")]
26+
//~^ WARNING this lint expectation is unfulfilled
27+
//~| NOTE the expectation for `unused` should be fulfilled
28+
//~| NOTE the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
29+
let value = "I'm unused";
30+
}
31+
32+
#[expect(warnings, reason = "this suppresses all warnings and also suppresses itself. No warning will be issued")]
33+
pub fn expect_warnings() {
34+
// This lint trigger will be suppressed
35+
#[warn(unused_mut)]
36+
let mut v = vec![1, 1, 2, 3, 5];
37+
}
38+
39+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
warning: this lint expectation is unfulfilled
2+
--> $DIR/expect_unfulfilled_expectation.rs:7:11
3+
|
4+
LL | #![expect(unfulfilled_lint_expectations, reason = "idk why you would expect this")]
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6+
|
7+
= note: `#[warn(unfulfilled_lint_expectations)]` on by default
8+
= note: idk why you would expect this
9+
= note: the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
10+
11+
warning: this lint expectation is unfulfilled
12+
--> $DIR/expect_unfulfilled_expectation.rs:13:10
13+
|
14+
LL | #[expect(unfulfilled_lint_expectations, reason = "a local: idk why you would expect this")]
15+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
16+
|
17+
= note: a local: idk why you would expect this
18+
= note: the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
19+
20+
warning: this lint expectation is unfulfilled
21+
--> $DIR/expect_unfulfilled_expectation.rs:18:14
22+
|
23+
LL | #[expect(unused_mut, reason = "this expectation will create a diagnostic with the default lint level")]
24+
| ^^^^^^^^^^
25+
|
26+
= note: this expectation will create a diagnostic with the default lint level
27+
28+
warning: this lint expectation is unfulfilled
29+
--> $DIR/expect_unfulfilled_expectation.rs:25:22
30+
|
31+
LL | #[expect(unused, unfulfilled_lint_expectations, reason = "the expectation for `unused` should be fulfilled")]
32+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
33+
|
34+
= note: the expectation for `unused` should be fulfilled
35+
= note: the `unfulfilled_lint_expectations` lint can't be expected and will always produce this message
36+
37+
warning: 4 warnings emitted
38+

0 commit comments

Comments
 (0)