Skip to content

Commit a7557e7

Browse files
authored
Auto merge of #37361 - jseyfried:fix_crate_var_regressions, r=nrc
Fix `$crate`-related regressions Fixes #37345, fixes #37357, fixes #37352, and improves the `unused_extern_crates` lint. r? @nrc
2 parents aef18be + 0d30325 commit a7557e7

File tree

6 files changed

+74
-12
lines changed

6 files changed

+74
-12
lines changed

src/librustc_resolve/build_reduced_graph.rs

+17-3
Original file line numberDiff line numberDiff line change
@@ -130,14 +130,27 @@ impl<'b> Resolver<'b> {
130130

131131
match view_path.node {
132132
ViewPathSimple(binding, ref full_path) => {
133-
let source_name = full_path.segments.last().unwrap().identifier.name;
134-
if source_name.as_str() == "mod" || source_name.as_str() == "self" {
133+
let mut source = full_path.segments.last().unwrap().identifier;
134+
let source_name = source.name.as_str();
135+
if source_name == "mod" || source_name == "self" {
135136
resolve_error(self,
136137
view_path.span,
137138
ResolutionError::SelfImportsOnlyAllowedWithin);
139+
} else if source_name == "$crate" && full_path.segments.len() == 1 {
140+
let crate_root = self.resolve_crate_var(source.ctxt);
141+
let crate_name = match crate_root.kind {
142+
ModuleKind::Def(_, name) => name,
143+
ModuleKind::Block(..) => unreachable!(),
144+
};
145+
source.name = crate_name;
146+
147+
self.session.struct_span_warn(item.span, "`$crate` may not be imported")
148+
.note("`use $crate;` was erroneously allowed and \
149+
will become a hard error in a future release")
150+
.emit();
138151
}
139152

140-
let subclass = ImportDirectiveSubclass::single(binding.name, source_name);
153+
let subclass = ImportDirectiveSubclass::single(binding.name, source.name);
141154
let span = view_path.span;
142155
self.add_import_directive(module_path, subclass, span, item.id, vis);
143156
}
@@ -500,6 +513,7 @@ impl<'b> Resolver<'b> {
500513
legacy_imports: LegacyMacroImports,
501514
allow_shadowing: bool) {
502515
let import_macro = |this: &mut Self, name, ext: Rc<_>, span| {
516+
this.used_crates.insert(module.def_id().unwrap().krate);
503517
if let SyntaxExtension::NormalTT(..) = *ext {
504518
this.macro_names.insert(name);
505519
}

src/librustc_resolve/lib.rs

+15-9
Original file line numberDiff line numberDiff line change
@@ -845,6 +845,10 @@ impl<'a> ModuleS<'a> {
845845
_ => false,
846846
}
847847
}
848+
849+
fn is_local(&self) -> bool {
850+
self.normal_ancestor_id.is_some()
851+
}
848852
}
849853

850854
impl<'a> fmt::Debug for ModuleS<'a> {
@@ -1580,14 +1584,7 @@ impl<'a> Resolver<'a> {
15801584
fn resolve_module_prefix(&mut self, module_path: &[Ident], span: Option<Span>)
15811585
-> ResolveResult<ModulePrefixResult<'a>> {
15821586
if &*module_path[0].name.as_str() == "$crate" {
1583-
let mut ctxt = module_path[0].ctxt;
1584-
while ctxt.source().0 != SyntaxContext::empty() {
1585-
ctxt = ctxt.source().0;
1586-
}
1587-
let module = self.invocations[&ctxt.source().1].module.get();
1588-
let crate_root =
1589-
if module.def_id().unwrap().is_local() { self.graph_root } else { module };
1590-
return Success(PrefixFound(crate_root, 1))
1587+
return Success(PrefixFound(self.resolve_crate_var(module_path[0].ctxt), 1));
15911588
}
15921589

15931590
// Start at the current module if we see `self` or `super`, or at the
@@ -1620,6 +1617,14 @@ impl<'a> Resolver<'a> {
16201617
return Success(PrefixFound(containing_module, i));
16211618
}
16221619

1620+
fn resolve_crate_var(&mut self, mut crate_var_ctxt: SyntaxContext) -> Module<'a> {
1621+
while crate_var_ctxt.source().0 != SyntaxContext::empty() {
1622+
crate_var_ctxt = crate_var_ctxt.source().0;
1623+
}
1624+
let module = self.invocations[&crate_var_ctxt.source().1].module.get();
1625+
if module.is_local() { self.graph_root } else { module }
1626+
}
1627+
16231628
// AST resolution
16241629
//
16251630
// We maintain a list of value ribs and type ribs.
@@ -2569,7 +2574,8 @@ impl<'a> Resolver<'a> {
25692574
let unqualified_def = resolve_identifier_with_fallback(self, None);
25702575
let qualified_binding = self.resolve_module_relative_path(span, segments, namespace);
25712576
match (qualified_binding, unqualified_def) {
2572-
(Ok(binding), Some(ref ud)) if binding.def() == ud.def => {
2577+
(Ok(binding), Some(ref ud)) if binding.def() == ud.def &&
2578+
segments[0].identifier.name.as_str() != "$crate" => {
25732579
self.session
25742580
.add_lint(lint::builtin::UNUSED_QUALIFICATIONS,
25752581
id,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#[macro_export]
12+
macro_rules! m { () => { use $crate; } }
+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// aux-build:import_crate_var.rs
12+
// error-pattern: `$crate` may not be imported
13+
// error-pattern: `use $crate;` was erroneously allowed and will become a hard error
14+
15+
#![feature(rustc_attrs)]
16+
17+
#[macro_use] extern crate import_crate_var;
18+
m!();
19+
20+
#[rustc_error]
21+
fn main() {}

src/test/compile-fail/lint-qualification.rs

+7
Original file line numberDiff line numberDiff line change
@@ -18,4 +18,11 @@ fn main() {
1818
use foo::bar;
1919
foo::bar(); //~ ERROR: unnecessary qualification
2020
bar();
21+
22+
let _ = || -> Result<(), ()> { try!(Ok(())); Ok(()) }; // issue #37345
23+
24+
macro_rules! m {
25+
() => { $crate::foo::bar(); }
26+
}
27+
m!(); // issue #37357
2128
}

src/test/compile-fail/lint-unused-extern-crate.rs

+2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,8 @@ extern crate rand; // no error, the use marks it as used
2626

2727
extern crate lint_unused_extern_crate as other; // no error, the use * marks it as used
2828

29+
#[macro_use] extern crate core; // no error, the `#[macro_use]` marks it as used
30+
2931
#[allow(unused_imports)]
3032
use rand::isaac::IsaacRng;
3133

0 commit comments

Comments
 (0)