@@ -27,10 +27,10 @@ use syntax::ext::base::{NormalTT, Resolver as SyntaxResolver, SyntaxExtension};
27
27
use syntax:: ext:: expand:: { Expansion , mark_tts} ;
28
28
use syntax:: ext:: hygiene:: Mark ;
29
29
use syntax:: ext:: tt:: macro_rules;
30
- use syntax:: feature_gate:: { emit_feature_err , GateIssue , is_builtin_attr } ;
30
+ use syntax:: feature_gate:: { self , emit_feature_err , GateIssue } ;
31
31
use syntax:: fold:: { self , Folder } ;
32
32
use syntax:: ptr:: P ;
33
- use syntax:: symbol:: keywords;
33
+ use syntax:: symbol:: { Symbol , keywords} ;
34
34
use syntax:: util:: lev_distance:: find_best_match_for_name;
35
35
use syntax:: visit:: Visitor ;
36
36
use syntax_pos:: { Span , DUMMY_SP } ;
@@ -130,12 +130,16 @@ impl<'a> base::Resolver for Resolver<'a> {
130
130
self . whitelisted_legacy_custom_derives . contains ( & name)
131
131
}
132
132
133
- fn visit_expansion ( & mut self , mark : Mark , expansion : & Expansion ) {
133
+ fn visit_expansion ( & mut self , mark : Mark , expansion : & Expansion , derives : & [ Mark ] ) {
134
134
let invocation = self . invocations [ & mark] ;
135
135
self . collect_def_ids ( invocation, expansion) ;
136
136
137
137
self . current_module = invocation. module . get ( ) ;
138
138
self . current_module . unresolved_invocations . borrow_mut ( ) . remove ( & mark) ;
139
+ self . current_module . unresolved_invocations . borrow_mut ( ) . extend ( derives) ;
140
+ for & derive in derives {
141
+ self . invocations . insert ( derive, invocation) ;
142
+ }
139
143
let mut visitor = BuildReducedGraphVisitor {
140
144
resolver : self ,
141
145
legacy_scope : LegacyScope :: Invocation ( invocation) ,
@@ -172,7 +176,9 @@ impl<'a> base::Resolver for Resolver<'a> {
172
176
ImportResolver { resolver : self } . resolve_imports ( )
173
177
}
174
178
175
- fn find_attr_invoc ( & mut self , attrs : & mut Vec < ast:: Attribute > ) -> Option < ast:: Attribute > {
179
+ // Resolves attribute and derive legacy macros from `#![plugin(..)]`.
180
+ fn find_legacy_attr_invoc ( & mut self , attrs : & mut Vec < ast:: Attribute > )
181
+ -> Option < ast:: Attribute > {
176
182
for i in 0 ..attrs. len ( ) {
177
183
match self . builtin_macros . get ( & attrs[ i] . name ( ) ) . cloned ( ) {
178
184
Some ( binding) => match * binding. get_macro ( self ) {
@@ -183,11 +189,50 @@ impl<'a> base::Resolver for Resolver<'a> {
183
189
} ,
184
190
None => { }
185
191
}
192
+ }
186
193
187
- if self . proc_macro_enabled && !is_builtin_attr ( & attrs[ i] ) {
188
- return Some ( attrs. remove ( i) ) ;
194
+ // Check for legacy derives
195
+ for i in 0 ..attrs. len ( ) {
196
+ if attrs[ i] . name ( ) == "derive" {
197
+ let mut traits = match attrs[ i] . meta_item_list ( ) {
198
+ Some ( traits) if !traits. is_empty ( ) => traits. to_owned ( ) ,
199
+ _ => continue ,
200
+ } ;
201
+
202
+ for j in 0 ..traits. len ( ) {
203
+ let legacy_name = Symbol :: intern ( & match traits[ j] . word ( ) {
204
+ Some ( ..) => format ! ( "derive_{}" , traits[ j] . name( ) . unwrap( ) ) ,
205
+ None => continue ,
206
+ } ) ;
207
+ if !self . builtin_macros . contains_key ( & legacy_name) {
208
+ continue
209
+ }
210
+ let span = traits. remove ( j) . span ;
211
+ self . gate_legacy_custom_derive ( legacy_name, span) ;
212
+ if traits. is_empty ( ) {
213
+ attrs. remove ( i) ;
214
+ } else {
215
+ attrs[ i] . value = ast:: MetaItem {
216
+ name : attrs[ i] . name ( ) ,
217
+ span : attrs[ i] . span ,
218
+ node : ast:: MetaItemKind :: List ( traits) ,
219
+ } ;
220
+ }
221
+ return Some ( ast:: Attribute {
222
+ value : ast:: MetaItem {
223
+ name : legacy_name,
224
+ span : span,
225
+ node : ast:: MetaItemKind :: Word ,
226
+ } ,
227
+ id : attr:: mk_attr_id ( ) ,
228
+ style : ast:: AttrStyle :: Outer ,
229
+ is_sugared_doc : false ,
230
+ span : span,
231
+ } ) ;
232
+ }
189
233
}
190
234
}
235
+
191
236
None
192
237
}
193
238
@@ -236,7 +281,7 @@ impl<'a> base::Resolver for Resolver<'a> {
236
281
Ok ( binding) => Ok ( binding. get_macro ( self ) ) ,
237
282
Err ( Determinacy :: Undetermined ) if !force => return Err ( Determinacy :: Undetermined ) ,
238
283
_ => {
239
- let msg = format ! ( "macro undefined: '{}!' " , name) ;
284
+ let msg = format ! ( "macro undefined: `{}` " , name) ;
240
285
let mut err = self . session . struct_span_err ( span, & msg) ;
241
286
self . suggest_macro_name ( & name. as_str ( ) , & mut err) ;
242
287
err. emit ( ) ;
@@ -251,13 +296,6 @@ impl<'a> base::Resolver for Resolver<'a> {
251
296
result
252
297
}
253
298
254
- fn resolve_builtin_macro ( & mut self , tname : Name ) -> Result < Rc < SyntaxExtension > , Determinacy > {
255
- match self . builtin_macros . get ( & tname) . cloned ( ) {
256
- Some ( binding) => Ok ( binding. get_macro ( self ) ) ,
257
- None => Err ( Determinacy :: Undetermined ) ,
258
- }
259
- }
260
-
261
299
fn resolve_derive_macro ( & mut self , scope : Mark , path : & ast:: Path , force : bool )
262
300
-> Result < Rc < SyntaxExtension > , Determinacy > {
263
301
let ast:: Path { span, .. } = * path;
@@ -540,4 +578,14 @@ impl<'a> Resolver<'a> {
540
578
`use {}::{};`", crate_name, name) )
541
579
. emit ( ) ;
542
580
}
581
+
582
+ fn gate_legacy_custom_derive ( & mut self , name : Symbol , span : Span ) {
583
+ if !self . session . features . borrow ( ) . custom_derive {
584
+ let sess = & self . session . parse_sess ;
585
+ let explain = feature_gate:: EXPLAIN_CUSTOM_DERIVE ;
586
+ emit_feature_err ( sess, "custom_derive" , span, GateIssue :: Language , explain) ;
587
+ } else if !self . is_whitelisted_legacy_custom_derive ( name) {
588
+ self . session . span_warn ( span, feature_gate:: EXPLAIN_DEPR_CUSTOM_DERIVE ) ;
589
+ }
590
+ }
543
591
}
0 commit comments