From 94fc09c68f36da4b35f870c9399743102bea393f Mon Sep 17 00:00:00 2001 From: est31 Date: Sat, 15 Jul 2017 05:53:58 +0200 Subject: [PATCH] Tidy: allow common lang+lib features This allows changes to the Rust language that have both library and language components share one feature gate. The feature gates need to be "about the same change", so that both library and language components must either be both unstable, or both stable, and share the tracking issue. Removes the ugly "proc_macro" exception. Closes #43089 --- src/tools/tidy/src/features.rs | 57 ++++++++++++++++++++++++---------- 1 file changed, 41 insertions(+), 16 deletions(-) diff --git a/src/tools/tidy/src/features.rs b/src/tools/tidy/src/features.rs index 4c94ade98d96..e72f24989691 100644 --- a/src/tools/tidy/src/features.rs +++ b/src/tools/tidy/src/features.rs @@ -50,6 +50,34 @@ pub struct Feature { pub tracking_issue: Option, } +impl Feature { + fn check_match(&self, other: &Feature)-> Result<(), Vec<&'static str>> { + let mut mismatches = Vec::new(); + if self.level != other.level { + mismatches.push("stability level"); + } + if self.level == Status::Stable || other.level == Status::Stable { + // As long as a feature is unstable, the since field tracks + // when the given part of the feature has been implemented. + // Mismatches are tolerable as features evolve and functionality + // gets added. + // Once a feature is stable, the since field tracks the first version + // it was part of the stable distribution, and mismatches are disallowed. + if self.since != other.since { + mismatches.push("since"); + } + } + if self.tracking_issue != other.tracking_issue { + mismatches.push("tracking issue"); + } + if mismatches.is_empty() { + Ok(()) + } else { + Err(mismatches) + } + } +} + pub type Features = HashMap; pub fn check(path: &Path, bad: &mut bool, quiet: bool) { @@ -242,23 +270,20 @@ fn get_and_check_lib_features(base_src_path: &Path, &mut |res, file, line| { match res { Ok((name, f)) => { - let mut err = |msg: &str| { - tidy_error!(bad, "{}:{}: {}", file.display(), line, msg); - }; - if lang_features.contains_key(name) && name != "proc_macro" { - err("duplicating a lang feature"); - } - if let Some(ref s) = lib_features.get(name) { - if s.level != f.level { - err("different stability level than before"); - } - if s.since != f.since { - err("different `since` than before"); + let mut check_features = |f: &Feature, list: &Features, display: &str| { + if let Some(ref s) = list.get(name) { + if let Err(m) = (&f).check_match(s) { + tidy_error!(bad, + "{}:{}: mismatches to {} in: {:?}", + file.display(), + line, + display, + &m); + } } - if s.tracking_issue != f.tracking_issue { - err("different `tracking_issue` than before"); - } - } + }; + check_features(&f, &lang_features, "corresponding lang feature"); + check_features(&f, &lib_features, "previous"); lib_features.insert(name.to_owned(), f); }, Err(msg) => {