Skip to content

Commit 238fd72

Browse files
committed
Auto merge of rust-lang#86572 - rylev:force-warnings-always, r=nikomatsakis
Force warnings even when can_emit_warnings == false Fixes an issue mentioned in rust-lang#85512 with --cap-lints overriding --force-warnings. Fixes rust-lang#86751 r? `@ehuss`
2 parents b09dad3 + 5af5a6d commit 238fd72

10 files changed

+116
-12
lines changed

compiler/rustc_errors/src/annotate_snippet_emitter_writer.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -145,8 +145,9 @@ impl AnnotateSnippetEmitterWriter {
145145
title: Some(Annotation {
146146
label: Some(&message),
147147
id: code.as_ref().map(|c| match c {
148-
DiagnosticId::Error(val)
149-
| DiagnosticId::Lint { name: val, has_future_breakage: _ } => val.as_str(),
148+
DiagnosticId::Error(val) | DiagnosticId::Lint { name: val, .. } => {
149+
val.as_str()
150+
}
150151
}),
151152
annotation_type: annotation_type_for_level(*level),
152153
}),

compiler/rustc_errors/src/diagnostic.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ pub struct Diagnostic {
2929
#[derive(Clone, Debug, PartialEq, Eq, Hash, Encodable, Decodable)]
3030
pub enum DiagnosticId {
3131
Error(String),
32-
Lint { name: String, has_future_breakage: bool },
32+
Lint { name: String, has_future_breakage: bool, is_force_warn: bool },
3333
}
3434

3535
/// A "sub"-diagnostic attached to a parent diagnostic.
@@ -109,6 +109,13 @@ impl Diagnostic {
109109
}
110110
}
111111

112+
pub fn is_force_warn(&self) -> bool {
113+
match self.code {
114+
Some(DiagnosticId::Lint { is_force_warn, .. }) => is_force_warn,
115+
_ => false,
116+
}
117+
}
118+
112119
/// Cancel the diagnostic (a structured diagnostic must either be emitted or
113120
/// canceled or it will panic when dropped).
114121
pub fn cancel(&mut self) {

compiler/rustc_errors/src/json.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -559,7 +559,7 @@ impl DiagnosticCode {
559559
s.map(|s| {
560560
let s = match s {
561561
DiagnosticId::Error(s) => s,
562-
DiagnosticId::Lint { name, has_future_breakage: _ } => name,
562+
DiagnosticId::Lint { name, .. } => name,
563563
};
564564
let je_result =
565565
je.registry.as_ref().map(|registry| registry.try_find_description(&s)).unwrap();

compiler/rustc_errors/src/lib.rs

+31-2
Original file line numberDiff line numberDiff line change
@@ -520,12 +520,28 @@ impl Handler {
520520
}
521521

522522
/// Construct a builder at the `Warning` level at the given `span` and with the `msg`.
523+
///
524+
/// The builder will be canceled if warnings cannot be emitted.
523525
pub fn struct_span_warn(&self, span: impl Into<MultiSpan>, msg: &str) -> DiagnosticBuilder<'_> {
524526
let mut result = self.struct_warn(msg);
525527
result.set_span(span);
526528
result
527529
}
528530

531+
/// Construct a builder at the `Warning` level at the given `span` and with the `msg`.
532+
///
533+
/// This will "force" the warning meaning it will not be canceled even
534+
/// if warnings cannot be emitted.
535+
pub fn struct_span_force_warn(
536+
&self,
537+
span: impl Into<MultiSpan>,
538+
msg: &str,
539+
) -> DiagnosticBuilder<'_> {
540+
let mut result = self.struct_force_warn(msg);
541+
result.set_span(span);
542+
result
543+
}
544+
529545
/// Construct a builder at the `Allow` level at the given `span` and with the `msg`.
530546
pub fn struct_span_allow(
531547
&self,
@@ -551,6 +567,8 @@ impl Handler {
551567
}
552568

553569
/// Construct a builder at the `Warning` level with the `msg`.
570+
///
571+
/// The builder will be canceled if warnings cannot be emitted.
554572
pub fn struct_warn(&self, msg: &str) -> DiagnosticBuilder<'_> {
555573
let mut result = DiagnosticBuilder::new(self, Level::Warning, msg);
556574
if !self.flags.can_emit_warnings {
@@ -559,6 +577,14 @@ impl Handler {
559577
result
560578
}
561579

580+
/// Construct a builder at the `Warning` level with the `msg`.
581+
///
582+
/// This will "force" a warning meaning it will not be canceled even
583+
/// if warnings cannot be emitted.
584+
pub fn struct_force_warn(&self, msg: &str) -> DiagnosticBuilder<'_> {
585+
DiagnosticBuilder::new(self, Level::Warning, msg)
586+
}
587+
562588
/// Construct a builder at the `Allow` level with the `msg`.
563589
pub fn struct_allow(&self, msg: &str) -> DiagnosticBuilder<'_> {
564590
DiagnosticBuilder::new(self, Level::Allow, msg)
@@ -801,7 +827,10 @@ impl HandlerInner {
801827
self.future_breakage_diagnostics.push(diagnostic.clone());
802828
}
803829

804-
if diagnostic.level == Warning && !self.flags.can_emit_warnings {
830+
if diagnostic.level == Warning
831+
&& !self.flags.can_emit_warnings
832+
&& !diagnostic.is_force_warn()
833+
{
805834
if diagnostic.has_future_breakage() {
806835
(*TRACK_DIAGNOSTICS)(diagnostic);
807836
}
@@ -873,7 +902,7 @@ impl HandlerInner {
873902

874903
match (errors.len(), warnings.len()) {
875904
(0, 0) => return,
876-
(0, _) => self.emit_diagnostic(&Diagnostic::new(Level::Warning, &warnings)),
905+
(0, _) => self.emitter.emit_diagnostic(&Diagnostic::new(Level::Warning, &warnings)),
877906
(_, 0) => {
878907
let _ = self.fatal(&errors);
879908
}

compiler/rustc_middle/src/lint.rs

+13-5
Original file line numberDiff line numberDiff line change
@@ -111,8 +111,13 @@ impl LintLevelSets {
111111
}
112112
}
113113

114-
// Ensure that we never exceed the `--cap-lints` argument.
115-
level = cmp::min(level, self.lint_cap);
114+
// Ensure that we never exceed the `--cap-lints` argument
115+
// unless the source is a --force-warn
116+
level = if let LintLevelSource::CommandLine(_, Level::ForceWarn) = src {
117+
level
118+
} else {
119+
cmp::min(level, self.lint_cap)
120+
};
116121

117122
if let Some(driver_level) = sess.driver_lint_caps.get(&LintId::of(lint)) {
118123
// Ensure that we never exceed driver level.
@@ -233,8 +238,10 @@ pub fn struct_lint_level<'s, 'd>(
233238
return;
234239
}
235240
}
236-
(Level::Warn | Level::ForceWarn, Some(span)) => sess.struct_span_warn(span, ""),
237-
(Level::Warn | Level::ForceWarn, None) => sess.struct_warn(""),
241+
(Level::Warn, Some(span)) => sess.struct_span_warn(span, ""),
242+
(Level::Warn, None) => sess.struct_warn(""),
243+
(Level::ForceWarn, Some(span)) => sess.struct_span_force_warn(span, ""),
244+
(Level::ForceWarn, None) => sess.struct_force_warn(""),
238245
(Level::Deny | Level::Forbid, Some(span)) => sess.struct_span_err(span, ""),
239246
(Level::Deny | Level::Forbid, None) => sess.struct_err(""),
240247
};
@@ -324,7 +331,8 @@ pub fn struct_lint_level<'s, 'd>(
324331
}
325332
}
326333

327-
err.code(DiagnosticId::Lint { name, has_future_breakage });
334+
let is_force_warn = matches!(level, Level::ForceWarn);
335+
err.code(DiagnosticId::Lint { name, has_future_breakage, is_force_warn });
328336

329337
if let Some(future_incompatible) = future_incompatible {
330338
let explanation = if lint_id == LintId::of(builtin::UNSTABLE_NAME_COLLISIONS) {

compiler/rustc_session/src/session.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -324,7 +324,7 @@ impl Session {
324324
.into_iter()
325325
.map(|diag| {
326326
let lint_name = match &diag.code {
327-
Some(DiagnosticId::Lint { name, has_future_breakage: true }) => name,
327+
Some(DiagnosticId::Lint { name, has_future_breakage: true, .. }) => name,
328328
_ => panic!("Unexpected code in diagnostic {:?}", diag),
329329
};
330330
let lint = lint_store.name_to_lint(&lint_name);
@@ -351,6 +351,13 @@ impl Session {
351351
pub fn struct_span_warn<S: Into<MultiSpan>>(&self, sp: S, msg: &str) -> DiagnosticBuilder<'_> {
352352
self.diagnostic().struct_span_warn(sp, msg)
353353
}
354+
pub fn struct_span_force_warn<S: Into<MultiSpan>>(
355+
&self,
356+
sp: S,
357+
msg: &str,
358+
) -> DiagnosticBuilder<'_> {
359+
self.diagnostic().struct_span_force_warn(sp, msg)
360+
}
354361
pub fn struct_span_warn_with_code<S: Into<MultiSpan>>(
355362
&self,
356363
sp: S,
@@ -362,6 +369,9 @@ impl Session {
362369
pub fn struct_warn(&self, msg: &str) -> DiagnosticBuilder<'_> {
363370
self.diagnostic().struct_warn(msg)
364371
}
372+
pub fn struct_force_warn(&self, msg: &str) -> DiagnosticBuilder<'_> {
373+
self.diagnostic().struct_force_warn(msg)
374+
}
365375
pub fn struct_span_allow<S: Into<MultiSpan>>(&self, sp: S, msg: &str) -> DiagnosticBuilder<'_> {
366376
self.diagnostic().struct_span_allow(sp, msg)
367377
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// compile-flags: --cap-lints warn --force-warns rust-2021-compatibility -Zunstable-options
2+
// check-pass
3+
#![allow(ellipsis_inclusive_range_patterns)]
4+
5+
pub fn f() -> bool {
6+
let x = 123;
7+
match x {
8+
0...100 => true,
9+
//~^ WARN range patterns are deprecated
10+
//~| WARN this is accepted in the current edition
11+
_ => false,
12+
}
13+
}
14+
15+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
warning: `...` range patterns are deprecated
2+
--> $DIR/force-warn-cap-lints-warn.rs:8:10
3+
|
4+
LL | 0...100 => true,
5+
| ^^^ help: use `..=` for an inclusive range
6+
|
7+
= note: `--force-warns ellipsis-inclusive-range-patterns` implied by `--force-warns rust-2021-compatibility`
8+
= warning: this is accepted in the current edition (Rust 2015) but is a hard error in Rust 2021!
9+
= note: for more information, see issue #80165 <https://github.com/rust-lang/rust/issues/80165>
10+
11+
warning: 1 warning emitted
12+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
// compile-flags: --cap-lints allow --force-warns bare_trait_objects -Zunstable-options
2+
// check-pass
3+
4+
pub trait SomeTrait {}
5+
6+
pub fn function(_x: Box<SomeTrait>) {}
7+
//~^ WARN trait objects without an explicit `dyn` are deprecated
8+
//~| WARN this is accepted in the current edition
9+
10+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
warning: trait objects without an explicit `dyn` are deprecated
2+
--> $DIR/force-warns-cap-lints-allow.rs:6:25
3+
|
4+
LL | pub fn function(_x: Box<SomeTrait>) {}
5+
| ^^^^^^^^^ help: use `dyn`: `dyn SomeTrait`
6+
|
7+
= note: requested on the command line with `--force-warns bare-trait-objects`
8+
= warning: this is accepted in the current edition (Rust 2015) but is a hard error in Rust 2021!
9+
= note: for more information, see issue #80165 <https://github.com/rust-lang/rust/issues/80165>
10+
11+
warning: 1 warning emitted
12+

0 commit comments

Comments
 (0)