Skip to main content

rustc_expand/
placeholders.rs

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            // Implement the proposal described in
382            // https://github.com/rust-lang/rust/issues/61733#issuecomment-509626449
383            //
384            // The macro invocation expands to the list of statements. If the
385            // list of statements is empty, then 'parse' the trailing semicolon
386            // on the original invocation as an empty statement. That is:
387            //
388            // `empty();` is parsed as a single `StmtKind::Empty`
389            //
390            // If the list of statements is non-empty, see if the final
391            // statement already has a trailing semicolon.
392            //
393            // If it doesn't have a semicolon, then 'parse' the trailing
394            // semicolon from the invocation as part of the final statement,
395            // using `stmt.add_trailing_semicolon()`
396            //
397            // If it does have a semicolon, then 'parse' the trailing semicolon
398            // from the invocation as a new StmtKind::Empty
399
400            // FIXME: We will need to preserve the original semicolon token and
401            // span as part of #15701
402            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}