@@ -18,7 +18,7 @@ use std::mem;
18
18
use creader:: { CrateLoader , Macros } ;
19
19
20
20
use rustc:: hir:: def_id:: DefIndex ;
21
- use rustc:: middle:: cstore:: LoadedMacro ;
21
+ use rustc:: middle:: cstore:: { LoadedMacro , LoadedMacroKind } ;
22
22
use rustc:: session:: Session ;
23
23
use rustc:: util:: nodemap:: FnvHashMap ;
24
24
use rustc_back:: dynamic_lib:: DynamicLibrary ;
@@ -28,14 +28,19 @@ use syntax::ast;
28
28
use syntax:: attr;
29
29
use syntax:: parse:: token;
30
30
use syntax_ext:: deriving:: custom:: CustomDerive ;
31
- use syntax_pos:: Span ;
31
+ use syntax_pos:: { Span , DUMMY_SP } ;
32
32
33
33
pub fn call_bad_macro_reexport ( a : & Session , b : Span ) {
34
34
span_err ! ( a, b, E0467 , "bad macro reexport" ) ;
35
35
}
36
36
37
37
pub type MacroSelection = FnvHashMap < token:: InternedString , Span > ;
38
38
39
+ enum ImportSelection {
40
+ All ( Span ) ,
41
+ Some ( MacroSelection ) ,
42
+ }
43
+
39
44
pub fn load_macros ( loader : & mut CrateLoader , extern_crate : & ast:: Item , allows_macros : bool )
40
45
-> Vec < LoadedMacro > {
41
46
loader. load_crate ( extern_crate, allows_macros)
@@ -46,7 +51,7 @@ impl<'a> CrateLoader<'a> {
46
51
extern_crate : & ast:: Item ,
47
52
allows_macros : bool ) -> Vec < LoadedMacro > {
48
53
// Parse the attributes relating to macros.
49
- let mut import = Some ( FnvHashMap ( ) ) ; // None => load all
54
+ let mut import = ImportSelection :: Some ( FnvHashMap ( ) ) ;
50
55
let mut reexport = FnvHashMap ( ) ;
51
56
52
57
for attr in & extern_crate. attrs {
@@ -55,11 +60,9 @@ impl<'a> CrateLoader<'a> {
55
60
"macro_use" => {
56
61
let names = attr. meta_item_list ( ) ;
57
62
if names. is_none ( ) {
58
- // no names => load all
59
- import = None ;
60
- }
61
- if let ( Some ( sel) , Some ( names) ) = ( import. as_mut ( ) , names) {
62
- for attr in names {
63
+ import = ImportSelection :: All ( attr. span ) ;
64
+ } else if let ImportSelection :: Some ( ref mut sel) = import {
65
+ for attr in names. unwrap ( ) {
63
66
if let Some ( word) = attr. word ( ) {
64
67
sel. insert ( word. name ( ) . clone ( ) , attr. span ( ) ) ;
65
68
} else {
@@ -98,10 +101,10 @@ impl<'a> CrateLoader<'a> {
98
101
fn load_macros < ' b > ( & mut self ,
99
102
vi : & ast:: Item ,
100
103
allows_macros : bool ,
101
- import : Option < MacroSelection > ,
104
+ import : ImportSelection ,
102
105
reexport : MacroSelection )
103
106
-> Vec < LoadedMacro > {
104
- if let Some ( sel) = import. as_ref ( ) {
107
+ if let ImportSelection :: Some ( ref sel) = import {
105
108
if sel. is_empty ( ) && reexport. is_empty ( ) {
106
109
return Vec :: new ( ) ;
107
110
}
@@ -120,15 +123,19 @@ impl<'a> CrateLoader<'a> {
120
123
for mut def in macros. macro_rules . drain ( ..) {
121
124
let name = def. ident . name . as_str ( ) ;
122
125
123
- def . use_locally = match import. as_ref ( ) {
124
- None => true ,
125
- Some ( sel) => sel. contains_key ( & name) ,
126
+ let import_site = match import {
127
+ ImportSelection :: All ( span ) => Some ( span ) ,
128
+ ImportSelection :: Some ( ref sel) => sel. get ( & name) . cloned ( )
126
129
} ;
130
+ def. use_locally = import_site. is_some ( ) ;
127
131
def. export = reexport. contains_key ( & name) ;
128
132
def. allow_internal_unstable = attr:: contains_name ( & def. attrs ,
129
133
"allow_internal_unstable" ) ;
130
134
debug ! ( "load_macros: loaded: {:?}" , def) ;
131
- ret. push ( LoadedMacro :: Def ( def) ) ;
135
+ ret. push ( LoadedMacro {
136
+ kind : LoadedMacroKind :: Def ( def) ,
137
+ import_site : import_site. unwrap_or ( DUMMY_SP ) ,
138
+ } ) ;
132
139
seen. insert ( name) ;
133
140
}
134
141
@@ -137,7 +144,7 @@ impl<'a> CrateLoader<'a> {
137
144
// exported macros, enforced elsewhere
138
145
assert_eq ! ( ret. len( ) , 0 ) ;
139
146
140
- if import . is_some ( ) {
147
+ if let ImportSelection :: Some ( .. ) = import {
141
148
self . sess . span_err ( vi. span , "`rustc-macro` crates cannot be \
142
149
selectively imported from, must \
143
150
use `#[macro_use]`") ;
@@ -151,10 +158,10 @@ impl<'a> CrateLoader<'a> {
151
158
self . load_derive_macros ( vi. span , & macros, index, & mut ret) ;
152
159
}
153
160
154
- if let Some ( sel) = import. as_ref ( ) {
161
+ if let ImportSelection :: Some ( sel) = import {
155
162
for ( name, span) in sel {
156
163
if !seen. contains ( & name) {
157
- span_err ! ( self . sess, * span, E0469 ,
164
+ span_err ! ( self . sess, span, E0469 ,
158
165
"imported macro not found" ) ;
159
166
}
160
167
}
@@ -199,18 +206,21 @@ impl<'a> CrateLoader<'a> {
199
206
mem:: transmute :: < * mut u8 , fn ( & mut Registry ) > ( sym)
200
207
} ;
201
208
202
- struct MyRegistrar < ' a > ( & ' a mut Vec < LoadedMacro > ) ;
209
+ struct MyRegistrar < ' a > ( & ' a mut Vec < LoadedMacro > , Span ) ;
203
210
204
211
impl < ' a > Registry for MyRegistrar < ' a > {
205
212
fn register_custom_derive ( & mut self ,
206
213
trait_name : & str ,
207
214
expand : fn ( TokenStream ) -> TokenStream ) {
208
215
let derive = Rc :: new ( CustomDerive :: new ( expand) ) ;
209
- self . 0 . push ( LoadedMacro :: CustomDerive ( trait_name. to_string ( ) , derive) ) ;
216
+ self . 0 . push ( LoadedMacro {
217
+ kind : LoadedMacroKind :: CustomDerive ( trait_name. to_string ( ) , derive) ,
218
+ import_site : self . 1 ,
219
+ } ) ;
210
220
}
211
221
}
212
222
213
- registrar ( & mut MyRegistrar ( ret) ) ;
223
+ registrar ( & mut MyRegistrar ( ret, span ) ) ;
214
224
215
225
// Intentionally leak the dynamic library. We can't ever unload it
216
226
// since the library can make things that will live arbitrarily long.
0 commit comments