1use rustc_ast::mut_visit::*;
2use rustc_ast::token::Delimiter;
3use rustc_ast::visit::AssocCtxt;
4use rustc_ast::{self as ast, Safety};
5use rustc_data_structures::fx::FxHashMap;
6use rustc_span::{DUMMY_SP, Ident};
7use smallvec::{SmallVec, smallvec};
8use thin_vec::ThinVec;
9
10use crate::expand::{AstFragment, AstFragmentKind};
11
12pub(crate) fn placeholder(
13 kind: AstFragmentKind,
14 id: ast::NodeId,
15 vis: Option<ast::Visibility>,
16) -> AstFragment {
17 fn mac_placeholder() -> Box<ast::MacCall> {
18 Box::new(ast::MacCall {
19 path: ast::Path { span: DUMMY_SP, segments: ThinVec::new(), tokens: None },
20 args: Box::new(ast::DelimArgs {
21 dspan: ast::tokenstream::DelimSpan::dummy(),
22 delim: Delimiter::Parenthesis,
23 tokens: ast::tokenstream::TokenStream::new(Vec::new()),
24 }),
25 })
26 }
27
28 let ident = Ident::dummy();
29 let attrs = ast::AttrVec::new();
30 let vis = vis.unwrap_or(ast::Visibility {
31 span: DUMMY_SP,
32 kind: ast::VisibilityKind::Inherited,
33 tokens: None,
34 });
35 let span = DUMMY_SP;
36 let expr_placeholder = || {
37 Box::new(ast::Expr {
38 id,
39 span,
40 attrs: ast::AttrVec::new(),
41 kind: ast::ExprKind::MacCall(mac_placeholder()),
42 tokens: None,
43 })
44 };
45 let ty = || {
46 Box::new(ast::Ty { id, kind: ast::TyKind::MacCall(mac_placeholder()), span, tokens: None })
47 };
48 let pat = || {
49 Box::new(ast::Pat {
50 id,
51 kind: ast::PatKind::MacCall(mac_placeholder()),
52 span,
53 tokens: None,
54 })
55 };
56
57 match kind {
58 AstFragmentKind::Crate => AstFragment::Crate(ast::Crate {
59 attrs: Default::default(),
60 items: Default::default(),
61 spans: ast::ModSpans { inner_span: span, ..Default::default() },
62 id,
63 is_placeholder: true,
64 }),
65 AstFragmentKind::Expr => AstFragment::Expr(expr_placeholder()),
66 AstFragmentKind::OptExpr => AstFragment::OptExpr(Some(expr_placeholder())),
67 AstFragmentKind::MethodReceiverExpr => AstFragment::MethodReceiverExpr(expr_placeholder()),
68 AstFragmentKind::Items => AstFragment::Items({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::Item {
id,
span,
vis,
attrs,
kind: ast::ItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::Item {
id,
span,
vis,
attrs,
kind: ast::ItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::Item {
69 id,
70 span,
71 vis,
72 attrs,
73 kind: ast::ItemKind::MacCall(mac_placeholder()),
74 tokens: None,
75 })]),
76 AstFragmentKind::TraitItems => {
77 AstFragment::TraitItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
78 id,
79 span,
80 vis,
81 attrs,
82 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
83 tokens: None,
84 })])
85 }
86 AstFragmentKind::ImplItems => AstFragment::ImplItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
87 id,
88 span,
89 vis,
90 attrs,
91 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
92 tokens: None,
93 })]),
94 AstFragmentKind::TraitImplItems => {
95 AstFragment::TraitImplItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
96 id,
97 span,
98 vis,
99 attrs,
100 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
101 tokens: None,
102 })])
103 }
104 AstFragmentKind::ForeignItems => {
105 AstFragment::ForeignItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::ForeignItem {
id,
span,
vis,
attrs,
kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::ForeignItem {
id,
span,
vis,
attrs,
kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::ForeignItem {
106 id,
107 span,
108 vis,
109 attrs,
110 kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
111 tokens: None,
112 })])
113 }
114 AstFragmentKind::Pat => AstFragment::Pat(Box::new(ast::Pat {
115 id,
116 span,
117 kind: ast::PatKind::MacCall(mac_placeholder()),
118 tokens: None,
119 })),
120 AstFragmentKind::Ty => AstFragment::Ty(Box::new(ast::Ty {
121 id,
122 span,
123 kind: ast::TyKind::MacCall(mac_placeholder()),
124 tokens: None,
125 })),
126 AstFragmentKind::Stmts => AstFragment::Stmts({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push({
let mac =
Box::new(ast::MacCallStmt {
mac: mac_placeholder(),
style: ast::MacStmtStyle::Braces,
attrs: ast::AttrVec::new(),
tokens: None,
});
ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[{
let mac =
Box::new(ast::MacCallStmt {
mac: mac_placeholder(),
style: ast::MacStmtStyle::Braces,
attrs: ast::AttrVec::new(),
tokens: None,
});
ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
}])))
}
}smallvec![{
127 let mac = Box::new(ast::MacCallStmt {
128 mac: mac_placeholder(),
129 style: ast::MacStmtStyle::Braces,
130 attrs: ast::AttrVec::new(),
131 tokens: None,
132 });
133 ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
134 }]),
135 AstFragmentKind::Arms => AstFragment::Arms({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Arm {
attrs: Default::default(),
body: Some(expr_placeholder()),
guard: None,
id,
pat: pat(),
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Arm {
attrs: Default::default(),
body: Some(expr_placeholder()),
guard: None,
id,
pat: pat(),
span,
is_placeholder: true,
}])))
}
}smallvec![ast::Arm {
136 attrs: Default::default(),
137 body: Some(expr_placeholder()),
138 guard: None,
139 id,
140 pat: pat(),
141 span,
142 is_placeholder: true,
143 }]),
144 AstFragmentKind::ExprFields => AstFragment::ExprFields({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::ExprField {
attrs: Default::default(),
expr: expr_placeholder(),
id,
ident,
is_shorthand: false,
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::ExprField {
attrs: Default::default(),
expr: expr_placeholder(),
id,
ident,
is_shorthand: false,
span,
is_placeholder: true,
}])))
}
}smallvec![ast::ExprField {
145 attrs: Default::default(),
146 expr: expr_placeholder(),
147 id,
148 ident,
149 is_shorthand: false,
150 span,
151 is_placeholder: true,
152 }]),
153 AstFragmentKind::PatFields => AstFragment::PatFields({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::PatField {
attrs: Default::default(),
id,
ident,
is_shorthand: false,
pat: pat(),
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::PatField {
attrs: Default::default(),
id,
ident,
is_shorthand: false,
pat: pat(),
span,
is_placeholder: true,
}])))
}
}smallvec![ast::PatField {
154 attrs: Default::default(),
155 id,
156 ident,
157 is_shorthand: false,
158 pat: pat(),
159 span,
160 is_placeholder: true,
161 }]),
162 AstFragmentKind::GenericParams => AstFragment::GenericParams({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push({
ast::GenericParam {
attrs: Default::default(),
bounds: Default::default(),
id,
ident,
is_placeholder: true,
kind: ast::GenericParamKind::Lifetime,
colon_span: None,
}
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[{
ast::GenericParam {
attrs: Default::default(),
bounds: Default::default(),
id,
ident,
is_placeholder: true,
kind: ast::GenericParamKind::Lifetime,
colon_span: None,
}
}])))
}
}smallvec![{
163 ast::GenericParam {
164 attrs: Default::default(),
165 bounds: Default::default(),
166 id,
167 ident,
168 is_placeholder: true,
169 kind: ast::GenericParamKind::Lifetime,
170 colon_span: None,
171 }
172 }]),
173 AstFragmentKind::Params => AstFragment::Params({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Param {
attrs: Default::default(),
id,
pat: pat(),
span,
ty: ty(),
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Param {
attrs: Default::default(),
id,
pat: pat(),
span,
ty: ty(),
is_placeholder: true,
}])))
}
}smallvec![ast::Param {
174 attrs: Default::default(),
175 id,
176 pat: pat(),
177 span,
178 ty: ty(),
179 is_placeholder: true,
180 }]),
181 AstFragmentKind::FieldDefs => AstFragment::FieldDefs({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::FieldDef {
attrs: Default::default(),
id,
ident: None,
span,
ty: ty(),
vis,
is_placeholder: true,
mut_restriction: ast::MutRestriction {
kind: ast::RestrictionKind::Unrestricted,
span: DUMMY_SP,
tokens: None,
},
safety: Safety::Default,
default: None,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::FieldDef {
attrs: Default::default(),
id,
ident: None,
span,
ty: ty(),
vis,
is_placeholder: true,
mut_restriction: ast::MutRestriction {
kind: ast::RestrictionKind::Unrestricted,
span: DUMMY_SP,
tokens: None,
},
safety: Safety::Default,
default: None,
}])))
}
}smallvec![ast::FieldDef {
182 attrs: Default::default(),
183 id,
184 ident: None,
185 span,
186 ty: ty(),
187 vis,
188 is_placeholder: true,
189 mut_restriction: ast::MutRestriction {
190 kind: ast::RestrictionKind::Unrestricted,
191 span: DUMMY_SP,
192 tokens: None,
193 },
194 safety: Safety::Default,
195 default: None,
196 }]),
197 AstFragmentKind::Variants => AstFragment::Variants({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Variant {
attrs: Default::default(),
data: ast::VariantData::Struct {
fields: Default::default(),
recovered: ast::Recovered::No,
},
disr_expr: None,
id,
ident,
span,
vis,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Variant {
attrs: Default::default(),
data: ast::VariantData::Struct {
fields: Default::default(),
recovered: ast::Recovered::No,
},
disr_expr: None,
id,
ident,
span,
vis,
is_placeholder: true,
}])))
}
}smallvec![ast::Variant {
198 attrs: Default::default(),
199 data: ast::VariantData::Struct {
200 fields: Default::default(),
201 recovered: ast::Recovered::No
202 },
203 disr_expr: None,
204 id,
205 ident,
206 span,
207 vis,
208 is_placeholder: true,
209 }]),
210 AstFragmentKind::WherePredicates => {
211 AstFragment::WherePredicates({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::WherePredicate {
attrs: Default::default(),
id,
span,
kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
bound_generic_params: Default::default(),
bounded_ty: ty(),
bounds: Default::default(),
}),
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::WherePredicate {
attrs: Default::default(),
id,
span,
kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
bound_generic_params: Default::default(),
bounded_ty: ty(),
bounds: Default::default(),
}),
is_placeholder: true,
}])))
}
}smallvec![ast::WherePredicate {
212 attrs: Default::default(),
213 id,
214 span,
215 kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
216 bound_generic_params: Default::default(),
217 bounded_ty: ty(),
218 bounds: Default::default(),
219 }),
220 is_placeholder: true,
221 }])
222 }
223 }
224}
225
226pub(crate) struct PlaceholderExpander {
227 expanded_fragments: FxHashMap<ast::NodeId, AstFragment>,
228}
229
230impl PlaceholderExpander {
231 pub(crate) fn with_capacity(capacity: usize) -> Self {
232 PlaceholderExpander {
233 expanded_fragments: FxHashMap::with_capacity_and_hasher(capacity, Default::default()),
234 }
235 }
236
237 pub(crate) fn add(&mut self, id: ast::NodeId, mut fragment: AstFragment) {
238 fragment.mut_visit_with(self);
239 self.expanded_fragments.insert(id, fragment);
240 }
241
242 fn remove(&mut self, id: ast::NodeId) -> AstFragment {
243 self.expanded_fragments.remove(&id).unwrap()
244 }
245}
246
247impl MutVisitor for PlaceholderExpander {
248 fn flat_map_arm(&mut self, arm: ast::Arm) -> SmallVec<[ast::Arm; 1]> {
249 if arm.is_placeholder {
250 self.remove(arm.id).make_arms()
251 } else {
252 walk_flat_map_arm(self, arm)
253 }
254 }
255
256 fn flat_map_expr_field(&mut self, field: ast::ExprField) -> SmallVec<[ast::ExprField; 1]> {
257 if field.is_placeholder {
258 self.remove(field.id).make_expr_fields()
259 } else {
260 walk_flat_map_expr_field(self, field)
261 }
262 }
263
264 fn flat_map_pat_field(&mut self, fp: ast::PatField) -> SmallVec<[ast::PatField; 1]> {
265 if fp.is_placeholder {
266 self.remove(fp.id).make_pat_fields()
267 } else {
268 walk_flat_map_pat_field(self, fp)
269 }
270 }
271
272 fn flat_map_generic_param(
273 &mut self,
274 param: ast::GenericParam,
275 ) -> SmallVec<[ast::GenericParam; 1]> {
276 if param.is_placeholder {
277 self.remove(param.id).make_generic_params()
278 } else {
279 walk_flat_map_generic_param(self, param)
280 }
281 }
282
283 fn flat_map_param(&mut self, p: ast::Param) -> SmallVec<[ast::Param; 1]> {
284 if p.is_placeholder {
285 self.remove(p.id).make_params()
286 } else {
287 walk_flat_map_param(self, p)
288 }
289 }
290
291 fn flat_map_field_def(&mut self, sf: ast::FieldDef) -> SmallVec<[ast::FieldDef; 1]> {
292 if sf.is_placeholder {
293 self.remove(sf.id).make_field_defs()
294 } else {
295 walk_flat_map_field_def(self, sf)
296 }
297 }
298
299 fn flat_map_variant(&mut self, variant: ast::Variant) -> SmallVec<[ast::Variant; 1]> {
300 if variant.is_placeholder {
301 self.remove(variant.id).make_variants()
302 } else {
303 walk_flat_map_variant(self, variant)
304 }
305 }
306
307 fn flat_map_where_predicate(
308 &mut self,
309 predicate: ast::WherePredicate,
310 ) -> SmallVec<[ast::WherePredicate; 1]> {
311 if predicate.is_placeholder {
312 self.remove(predicate.id).make_where_predicates()
313 } else {
314 walk_flat_map_where_predicate(self, predicate)
315 }
316 }
317
318 fn flat_map_item(&mut self, item: Box<ast::Item>) -> SmallVec<[Box<ast::Item>; 1]> {
319 match item.kind {
320 ast::ItemKind::MacCall(_) => self.remove(item.id).make_items(),
321 _ => walk_flat_map_item(self, item),
322 }
323 }
324
325 fn flat_map_assoc_item(
326 &mut self,
327 item: Box<ast::AssocItem>,
328 ctxt: AssocCtxt,
329 ) -> SmallVec<[Box<ast::AssocItem>; 1]> {
330 match item.kind {
331 ast::AssocItemKind::MacCall(_) => {
332 let it = self.remove(item.id);
333 match ctxt {
334 AssocCtxt::Trait => it.make_trait_items(),
335 AssocCtxt::Impl { of_trait: false } => it.make_impl_items(),
336 AssocCtxt::Impl { of_trait: true } => it.make_trait_impl_items(),
337 }
338 }
339 _ => walk_flat_map_assoc_item(self, item, ctxt),
340 }
341 }
342
343 fn flat_map_foreign_item(
344 &mut self,
345 item: Box<ast::ForeignItem>,
346 ) -> SmallVec<[Box<ast::ForeignItem>; 1]> {
347 match item.kind {
348 ast::ForeignItemKind::MacCall(_) => self.remove(item.id).make_foreign_items(),
349 _ => walk_flat_map_foreign_item(self, item),
350 }
351 }
352
353 fn visit_expr(&mut self, expr: &mut ast::Expr) {
354 match expr.kind {
355 ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_expr(),
356 _ => walk_expr(self, expr),
357 }
358 }
359
360 fn visit_method_receiver_expr(&mut self, expr: &mut ast::Expr) {
361 match expr.kind {
362 ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_method_receiver_expr(),
363 _ => walk_expr(self, expr),
364 }
365 }
366
367 fn filter_map_expr(&mut self, expr: Box<ast::Expr>) -> Option<Box<ast::Expr>> {
368 match expr.kind {
369 ast::ExprKind::MacCall(_) => self.remove(expr.id).make_opt_expr(),
370 _ => walk_filter_map_expr(self, expr),
371 }
372 }
373
374 fn flat_map_stmt(&mut self, stmt: ast::Stmt) -> SmallVec<[ast::Stmt; 1]> {
375 let (style, mut stmts) = match stmt.kind {
376 ast::StmtKind::MacCall(mac) => (mac.style, self.remove(stmt.id).make_stmts()),
377 _ => return walk_flat_map_stmt(self, stmt),
378 };
379
380 if style == ast::MacStmtStyle::Semicolon {
381 let empty_stmt =
403 ast::Stmt { id: ast::DUMMY_NODE_ID, kind: ast::StmtKind::Empty, span: DUMMY_SP };
404
405 if let Some(stmt) = stmts.pop() {
406 if stmt.has_trailing_semicolon() {
407 stmts.push(stmt);
408 stmts.push(empty_stmt);
409 } else {
410 stmts.push(stmt.add_trailing_semicolon());
411 }
412 } else {
413 stmts.push(empty_stmt);
414 }
415 }
416
417 stmts
418 }
419
420 fn visit_pat(&mut self, pat: &mut ast::Pat) {
421 match pat.kind {
422 ast::PatKind::MacCall(_) => *pat = *self.remove(pat.id).make_pat(),
423 _ => walk_pat(self, pat),
424 }
425 }
426
427 fn visit_ty(&mut self, ty: &mut ast::Ty) {
428 match ty.kind {
429 ast::TyKind::MacCall(_) => *ty = *self.remove(ty.id).make_ty(),
430 _ => walk_ty(self, ty),
431 }
432 }
433
434 fn visit_crate(&mut self, krate: &mut ast::Crate) {
435 if krate.is_placeholder {
436 *krate = self.remove(krate.id).make_crate();
437 } else {
438 walk_crate(self, krate)
439 }
440 }
441}