@@ -27,21 +27,25 @@ use syntax_pos::{Span, DUMMY_SP};
27
27
28
28
use deriving;
29
29
30
+ const PROC_MACRO_KINDS : [ & ' static str ; 3 ] =
31
+ [ "proc_macro_derive" , "proc_macro_attribute" , "proc_macro" ] ;
32
+
30
33
struct ProcMacroDerive {
31
34
trait_name : ast:: Name ,
32
35
function_name : Ident ,
33
36
span : Span ,
34
37
attrs : Vec < ast:: Name > ,
35
38
}
36
39
37
- struct AttrProcMacro {
40
+ struct ProcMacroDef {
38
41
function_name : Ident ,
39
42
span : Span ,
40
43
}
41
44
42
45
struct CollectProcMacros < ' a > {
43
46
derives : Vec < ProcMacroDerive > ,
44
- attr_macros : Vec < AttrProcMacro > ,
47
+ attr_macros : Vec < ProcMacroDef > ,
48
+ bang_macros : Vec < ProcMacroDef > ,
45
49
in_root : bool ,
46
50
handler : & ' a errors:: Handler ,
47
51
is_proc_macro_crate : bool ,
@@ -58,17 +62,18 @@ pub fn modify(sess: &ParseSess,
58
62
let ecfg = ExpansionConfig :: default ( "proc_macro" . to_string ( ) ) ;
59
63
let mut cx = ExtCtxt :: new ( sess, ecfg, resolver) ;
60
64
61
- let ( derives, attr_macros) = {
65
+ let ( derives, attr_macros, bang_macros ) = {
62
66
let mut collect = CollectProcMacros {
63
67
derives : Vec :: new ( ) ,
64
68
attr_macros : Vec :: new ( ) ,
69
+ bang_macros : Vec :: new ( ) ,
65
70
in_root : true ,
66
71
handler : handler,
67
72
is_proc_macro_crate : is_proc_macro_crate,
68
73
is_test_crate : is_test_crate,
69
74
} ;
70
75
visit:: walk_crate ( & mut collect, & krate) ;
71
- ( collect. derives , collect. attr_macros )
76
+ ( collect. derives , collect. attr_macros , collect . bang_macros )
72
77
} ;
73
78
74
79
if !is_proc_macro_crate {
@@ -83,7 +88,7 @@ pub fn modify(sess: &ParseSess,
83
88
return krate;
84
89
}
85
90
86
- krate. module . items . push ( mk_registrar ( & mut cx, & derives, & attr_macros) ) ;
91
+ krate. module . items . push ( mk_registrar ( & mut cx, & derives, & attr_macros, & bang_macros ) ) ;
87
92
88
93
if krate. exported_macros . len ( ) > 0 {
89
94
handler. err ( "cannot export macro_rules! macros from a `proc-macro` \
@@ -93,6 +98,10 @@ pub fn modify(sess: &ParseSess,
93
98
return krate
94
99
}
95
100
101
+ fn is_proc_macro_attr ( attr : & ast:: Attribute ) -> bool {
102
+ PROC_MACRO_KINDS . iter ( ) . any ( |kind| attr. check_name ( kind) )
103
+ }
104
+
96
105
impl < ' a > CollectProcMacros < ' a > {
97
106
fn check_not_pub_in_root ( & self , vis : & ast:: Visibility , sp : Span ) {
98
107
if self . is_proc_macro_crate &&
@@ -196,12 +205,12 @@ impl<'a> CollectProcMacros<'a> {
196
205
fn collect_attr_proc_macro ( & mut self , item : & ' a ast:: Item , attr : & ' a ast:: Attribute ) {
197
206
if let Some ( _) = attr. meta_item_list ( ) {
198
207
self . handler . span_err ( attr. span , "`#[proc_macro_attribute]` attribute
199
- cannot contain any meta items " ) ;
208
+ does not take any arguments " ) ;
200
209
return ;
201
210
}
202
211
203
212
if self . in_root && item. vis == ast:: Visibility :: Public {
204
- self . attr_macros . push ( AttrProcMacro {
213
+ self . attr_macros . push ( ProcMacroDef {
205
214
span : item. span ,
206
215
function_name : item. ident ,
207
216
} ) ;
@@ -215,6 +224,29 @@ impl<'a> CollectProcMacros<'a> {
215
224
self . handler . span_err ( item. span , msg) ;
216
225
}
217
226
}
227
+
228
+ fn collect_bang_proc_macro ( & mut self , item : & ' a ast:: Item , attr : & ' a ast:: Attribute ) {
229
+ if let Some ( _) = attr. meta_item_list ( ) {
230
+ self . handler . span_err ( attr. span , "`#[proc_macro]` attribute
231
+ does not take any arguments" ) ;
232
+ return ;
233
+ }
234
+
235
+ if self . in_root && item. vis == ast:: Visibility :: Public {
236
+ self . bang_macros . push ( ProcMacroDef {
237
+ span : item. span ,
238
+ function_name : item. ident ,
239
+ } ) ;
240
+ } else {
241
+ let msg = if !self . in_root {
242
+ "functions tagged with `#[proc_macro]` must \
243
+ currently reside in the root of the crate"
244
+ } else {
245
+ "functions tagged with `#[proc_macro]` must be `pub`"
246
+ } ;
247
+ self . handler . span_err ( item. span , msg) ;
248
+ }
249
+ }
218
250
}
219
251
220
252
impl < ' a > Visitor < ' a > for CollectProcMacros < ' a > {
@@ -232,7 +264,7 @@ impl<'a> Visitor<'a> for CollectProcMacros<'a> {
232
264
let mut found_attr: Option < & ' a ast:: Attribute > = None ;
233
265
234
266
for attr in & item. attrs {
235
- if attr . check_name ( "proc_macro_derive" ) || attr. check_name ( "proc_macro_attribute" ) {
267
+ if is_proc_macro_attr ( & attr) {
236
268
if let Some ( prev_attr) = found_attr {
237
269
let msg = if attr. name ( ) == prev_attr. name ( ) {
238
270
format ! ( "Only one `#[{}]` attribute is allowed on any given function" ,
@@ -285,6 +317,8 @@ impl<'a> Visitor<'a> for CollectProcMacros<'a> {
285
317
self . collect_custom_derive ( item, attr) ;
286
318
} else if attr. check_name ( "proc_macro_attribute" ) {
287
319
self . collect_attr_proc_macro ( item, attr) ;
320
+ } else if attr. check_name ( "proc_macro" ) {
321
+ self . collect_bang_proc_macro ( item, attr) ;
288
322
} ;
289
323
290
324
visit:: walk_item ( self , item) ;
@@ -320,7 +354,8 @@ impl<'a> Visitor<'a> for CollectProcMacros<'a> {
320
354
// }
321
355
fn mk_registrar ( cx : & mut ExtCtxt ,
322
356
custom_derives : & [ ProcMacroDerive ] ,
323
- custom_attrs : & [ AttrProcMacro ] ) -> P < ast:: Item > {
357
+ custom_attrs : & [ ProcMacroDef ] ,
358
+ custom_macros : & [ ProcMacroDef ] ) -> P < ast:: Item > {
324
359
let eid = cx. codemap ( ) . record_expansion ( ExpnInfo {
325
360
call_site : DUMMY_SP ,
326
361
callee : NameAndSpan {
@@ -342,6 +377,7 @@ fn mk_registrar(cx: &mut ExtCtxt,
342
377
let registrar = Ident :: from_str ( "registrar" ) ;
343
378
let register_custom_derive = Ident :: from_str ( "register_custom_derive" ) ;
344
379
let register_attr_proc_macro = Ident :: from_str ( "register_attr_proc_macro" ) ;
380
+ let register_bang_proc_macro = Ident :: from_str ( "register_bang_proc_macro" ) ;
345
381
346
382
let mut stmts = custom_derives. iter ( ) . map ( |cd| {
347
383
let path = cx. path_global ( cd. span , vec ! [ cd. function_name] ) ;
@@ -371,6 +407,18 @@ fn mk_registrar(cx: &mut ExtCtxt,
371
407
vec ! [ registrar, name, cx. expr_path( path) ] ) )
372
408
} ) ) ;
373
409
410
+ stmts. extend ( custom_macros. iter ( ) . map ( |cm| {
411
+ let name = cx. expr_str ( cm. span , cm. function_name . name ) ;
412
+ let path = cx. path_global ( cm. span , vec ! [ cm. function_name] ) ;
413
+ let registrar = cx. expr_ident ( cm. span , registrar) ;
414
+
415
+ let ufcs_path = cx. path ( span,
416
+ vec ! [ proc_macro, __internal, registry, register_bang_proc_macro] ) ;
417
+
418
+ cx. stmt_expr ( cx. expr_call ( span, cx. expr_path ( ufcs_path) ,
419
+ vec ! [ registrar, name, cx. expr_path( path) ] ) )
420
+ } ) ) ;
421
+
374
422
let path = cx. path ( span, vec ! [ proc_macro, __internal, registry] ) ;
375
423
let registrar_path = cx. ty_path ( path) ;
376
424
let arg_ty = cx. ty_rptr ( span, registrar_path, None , ast:: Mutability :: Mutable ) ;
0 commit comments