Skip to content

Commit 4c50f8b

Browse files
committed
Replace enum LintId with an extensible alternative
1 parent d79e624 commit 4c50f8b

File tree

16 files changed

+751
-776
lines changed

16 files changed

+751
-776
lines changed

src/librustc/driver/config.rs

+13-25
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,8 @@ pub struct Options {
7070
pub gc: bool,
7171
pub optimize: OptLevel,
7272
pub debuginfo: DebugInfoLevel,
73-
pub lint_opts: Vec<(lint::LintId, lint::Level)> ,
73+
pub lint_opts: Vec<(String, lint::Level)>,
74+
pub describe_lints: bool,
7475
pub output_types: Vec<back::link::OutputType> ,
7576
// This was mutable for rustpkg, which updates search paths based on the
7677
// parsed code. It remains mutable in case its replacements wants to use
@@ -104,6 +105,7 @@ pub fn basic_options() -> Options {
104105
optimize: No,
105106
debuginfo: NoDebugInfo,
106107
lint_opts: Vec::new(),
108+
describe_lints: false,
107109
output_types: Vec::new(),
108110
addl_lib_search_paths: RefCell::new(HashSet::new()),
109111
maybe_sysroot: None,
@@ -588,30 +590,15 @@ pub fn build_session_options(matches: &getopts::Matches) -> Options {
588590
let no_trans = matches.opt_present("no-trans");
589591
let no_analysis = matches.opt_present("no-analysis");
590592

591-
let lint_levels = [lint::Allow, lint::Warn,
592-
lint::Deny, lint::Forbid];
593-
let mut lint_opts = Vec::new();
594-
let lint_dict = lint::get_lint_dict();
595-
for level in lint_levels.iter() {
596-
let level_name = lint::level_to_str(*level);
597-
598-
let level_short = level_name.slice_chars(0, 1);
599-
let level_short = level_short.to_ascii().to_upper().into_str();
600-
let flags = matches.opt_strs(level_short.as_slice())
601-
.move_iter()
602-
.collect::<Vec<_>>()
603-
.append(matches.opt_strs(level_name).as_slice());
604-
for lint_name in flags.iter() {
605-
let lint_name = lint_name.replace("-", "_").into_string();
606-
match lint_dict.find_equiv(&lint_name) {
607-
None => {
608-
early_error(format!("unknown {} flag: {}",
609-
level_name,
610-
lint_name).as_slice());
611-
}
612-
Some(lint) => {
613-
lint_opts.push((lint.lint, *level));
614-
}
593+
let mut lint_opts = vec!();
594+
let mut describe_lints = false;
595+
596+
for &level in [lint::Allow, lint::Warn, lint::Deny, lint::Forbid].iter() {
597+
for lint_name in matches.opt_strs(level.as_str()).move_iter() {
598+
if lint_name.as_slice() == "help" {
599+
describe_lints = true;
600+
} else {
601+
lint_opts.push((lint_name.replace("-", "_").into_string(), level));
615602
}
616603
}
617604
}
@@ -755,6 +742,7 @@ pub fn build_session_options(matches: &getopts::Matches) -> Options {
755742
optimize: opt_level,
756743
debuginfo: debuginfo,
757744
lint_opts: lint_opts,
745+
describe_lints: describe_lints,
758746
output_types: output_types,
759747
addl_lib_search_paths: RefCell::new(addl_lib_search_paths),
760748
maybe_sysroot: sysroot_opt,

src/librustc/driver/driver.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -749,15 +749,15 @@ pub fn collect_crate_types(session: &Session,
749749
}
750750
Some(ref n) if n.equiv(&("bin")) => Some(config::CrateTypeExecutable),
751751
Some(_) => {
752-
session.add_lint(lint::UnknownCrateType,
752+
session.add_lint(lint::builtin::unknown_crate_type,
753753
ast::CRATE_NODE_ID,
754754
a.span,
755755
"invalid `crate_type` \
756756
value".to_string());
757757
None
758758
}
759759
_ => {
760-
session.add_lint(lint::UnknownCrateType,
760+
session.add_lint(lint::builtin::unknown_crate_type,
761761
ast::CRATE_NODE_ID,
762762
a.span,
763763
"`crate_type` requires a \

src/librustc/driver/mod.rs

+31-30
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@ use lint;
1717
use metadata;
1818

1919
use std::any::AnyRefExt;
20-
use std::cmp;
2120
use std::io;
2221
use std::os;
2322
use std::str;
@@ -50,6 +49,12 @@ fn run_compiler(args: &[String]) {
5049
None => return
5150
};
5251

52+
let sopts = config::build_session_options(&matches);
53+
if sopts.describe_lints {
54+
describe_lints();
55+
return;
56+
}
57+
5358
let (input, input_file_path) = match matches.free.len() {
5459
0u => early_error("no input filename given"),
5560
1u => {
@@ -66,7 +71,6 @@ fn run_compiler(args: &[String]) {
6671
_ => early_error("multiple input filenames provided")
6772
};
6873

69-
let sopts = config::build_session_options(&matches);
7074
let sess = build_session(sopts, input_file_path);
7175
let cfg = config::build_configuration(&sess);
7276
let odir = matches.opt_str("out-dir").map(|o| Path::new(o));
@@ -124,7 +128,7 @@ Additional help:
124128
config::optgroups().as_slice()));
125129
}
126130

127-
fn describe_warnings() {
131+
fn describe_lints() {
128132
println!("
129133
Available lint options:
130134
-W <foo> Warn about <foo>
@@ -133,30 +137,32 @@ Available lint options:
133137
-F <foo> Forbid <foo> (deny, and deny all overrides)
134138
");
135139

136-
let lint_dict = lint::get_lint_dict();
137-
let mut lint_dict = lint_dict.move_iter()
138-
.map(|(k, v)| (v, k))
139-
.collect::<Vec<(lint::LintSpec, &'static str)> >();
140-
lint_dict.as_mut_slice().sort();
140+
let mut builtin_specs = lint::builtin_lint_specs();
141+
builtin_specs.sort_by(|x, y| {
142+
match x.default_level.cmp(&y.default_level) {
143+
Equal => x.name.cmp(&y.name),
144+
r => r,
145+
}
146+
});
147+
148+
// FIXME: What if someone uses combining characters or East Asian fullwidth
149+
// characters in a lint name?!?!?
150+
let max_name_len = builtin_specs.iter()
151+
.map(|&s| s.name.char_len())
152+
.max().unwrap_or(0);
153+
let padded = |x: &str| {
154+
format!("{}{}", " ".repeat(max_name_len - x.char_len()), x)
155+
};
141156

142-
let mut max_key = 0;
143-
for &(_, name) in lint_dict.iter() {
144-
max_key = cmp::max(name.len(), max_key);
145-
}
146-
fn padded(max: uint, s: &str) -> String {
147-
format!("{}{}", " ".repeat(max - s.len()), s)
148-
}
149157
println!("\nAvailable lint checks:\n");
150-
println!(" {} {:7.7s} {}",
151-
padded(max_key, "name"), "default", "meaning");
152-
println!(" {} {:7.7s} {}\n",
153-
padded(max_key, "----"), "-------", "-------");
154-
for (spec, name) in lint_dict.move_iter() {
155-
let name = name.replace("_", "-");
158+
println!(" {} {:7.7s} {}", padded("name"), "default", "meaning");
159+
println!(" {} {:7.7s} {}", padded("----"), "-------", "-------");
160+
println!("");
161+
162+
for spec in builtin_specs.move_iter() {
163+
let name = spec.name.replace("_", "-");
156164
println!(" {} {:7.7s} {}",
157-
padded(max_key, name.as_slice()),
158-
lint::level_to_str(spec.default),
159-
spec.desc);
165+
padded(name.as_slice()), spec.default_level.as_str(), spec.desc);
160166
}
161167
println!("");
162168
}
@@ -214,12 +220,7 @@ pub fn handle_options(mut args: Vec<String>) -> Option<getopts::Matches> {
214220
return None;
215221
}
216222

217-
let lint_flags = matches.opt_strs("W").move_iter().collect::<Vec<_>>().append(
218-
matches.opt_strs("warn").as_slice());
219-
if lint_flags.iter().any(|x| x.as_slice() == "help") {
220-
describe_warnings();
221-
return None;
222-
}
223+
// Don't handle -W help here, because we might first load plugins.
223224

224225
let r = matches.opt_strs("Z");
225226
if r.iter().any(|x| x.as_slice() == "help") {

src/librustc/driver/session.rs

+4-3
Original file line numberDiff line numberDiff line change
@@ -105,16 +105,17 @@ impl Session {
105105
self.diagnostic().handler().unimpl(msg)
106106
}
107107
pub fn add_lint(&self,
108-
lint: lint::LintId,
108+
lint: &'static lint::Lint,
109109
id: ast::NodeId,
110110
sp: Span,
111111
msg: String) {
112+
let lint_id = lint::LintId::of(lint);
112113
let mut lints = self.lints.borrow_mut();
113114
match lints.find_mut(&id) {
114-
Some(arr) => { arr.push((lint, sp, msg)); return; }
115+
Some(arr) => { arr.push((lint_id, sp, msg)); return; }
115116
None => {}
116117
}
117-
lints.insert(id, vec!((lint, sp, msg)));
118+
lints.insert(id, vec!((lint_id, sp, msg)));
118119
}
119120
pub fn next_node_id(&self) -> ast::NodeId {
120121
self.reserve_node_ids(1)

src/librustc/front/feature_gate.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -366,7 +366,7 @@ pub fn check_crate(sess: &Session, krate: &ast::Crate) {
366366
directive not necessary");
367367
}
368368
None => {
369-
sess.add_lint(lint::UnknownFeatures,
369+
sess.add_lint(lint::builtin::unknown_features,
370370
ast::CRATE_NODE_ID,
371371
mi.span,
372372
"unknown feature".to_string());

src/librustc/lib.rs

+9
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,15 @@ pub mod lib {
130130
pub mod llvmdeps;
131131
}
132132

133+
// A private module so that macro-expanded idents like
134+
// `::rustc::lint::Lint` will also work in `rustc` itself.
135+
//
136+
// `libstd` uses the same trick.
137+
#[doc(hidden)]
138+
mod rustc {
139+
pub use lint;
140+
}
141+
133142
pub fn main() {
134143
let args = std::os::args().iter()
135144
.map(|x| x.to_string())

0 commit comments

Comments
 (0)