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,
                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,
                                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            safety: Safety::Default,
190            default: None,
191        }]),
192        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 {
193            attrs: Default::default(),
194            data: ast::VariantData::Struct {
195                fields: Default::default(),
196                recovered: ast::Recovered::No
197            },
198            disr_expr: None,
199            id,
200            ident,
201            span,
202            vis,
203            is_placeholder: true,
204        }]),
205        AstFragmentKind::WherePredicates => {
206            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 {
207                attrs: Default::default(),
208                id,
209                span,
210                kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
211                    bound_generic_params: Default::default(),
212                    bounded_ty: ty(),
213                    bounds: Default::default(),
214                }),
215                is_placeholder: true,
216            }])
217        }
218    }
219}
220
221pub(crate) struct PlaceholderExpander {
222    expanded_fragments: FxHashMap<ast::NodeId, AstFragment>,
223}
224
225impl PlaceholderExpander {
226    pub(crate) fn with_capacity(capacity: usize) -> Self {
227        PlaceholderExpander {
228            expanded_fragments: FxHashMap::with_capacity_and_hasher(capacity, Default::default()),
229        }
230    }
231
232    pub(crate) fn add(&mut self, id: ast::NodeId, mut fragment: AstFragment) {
233        fragment.mut_visit_with(self);
234        self.expanded_fragments.insert(id, fragment);
235    }
236
237    fn remove(&mut self, id: ast::NodeId) -> AstFragment {
238        self.expanded_fragments.remove(&id).unwrap()
239    }
240}
241
242impl MutVisitor for PlaceholderExpander {
243    fn flat_map_arm(&mut self, arm: ast::Arm) -> SmallVec<[ast::Arm; 1]> {
244        if arm.is_placeholder {
245            self.remove(arm.id).make_arms()
246        } else {
247            walk_flat_map_arm(self, arm)
248        }
249    }
250
251    fn flat_map_expr_field(&mut self, field: ast::ExprField) -> SmallVec<[ast::ExprField; 1]> {
252        if field.is_placeholder {
253            self.remove(field.id).make_expr_fields()
254        } else {
255            walk_flat_map_expr_field(self, field)
256        }
257    }
258
259    fn flat_map_pat_field(&mut self, fp: ast::PatField) -> SmallVec<[ast::PatField; 1]> {
260        if fp.is_placeholder {
261            self.remove(fp.id).make_pat_fields()
262        } else {
263            walk_flat_map_pat_field(self, fp)
264        }
265    }
266
267    fn flat_map_generic_param(
268        &mut self,
269        param: ast::GenericParam,
270    ) -> SmallVec<[ast::GenericParam; 1]> {
271        if param.is_placeholder {
272            self.remove(param.id).make_generic_params()
273        } else {
274            walk_flat_map_generic_param(self, param)
275        }
276    }
277
278    fn flat_map_param(&mut self, p: ast::Param) -> SmallVec<[ast::Param; 1]> {
279        if p.is_placeholder {
280            self.remove(p.id).make_params()
281        } else {
282            walk_flat_map_param(self, p)
283        }
284    }
285
286    fn flat_map_field_def(&mut self, sf: ast::FieldDef) -> SmallVec<[ast::FieldDef; 1]> {
287        if sf.is_placeholder {
288            self.remove(sf.id).make_field_defs()
289        } else {
290            walk_flat_map_field_def(self, sf)
291        }
292    }
293
294    fn flat_map_variant(&mut self, variant: ast::Variant) -> SmallVec<[ast::Variant; 1]> {
295        if variant.is_placeholder {
296            self.remove(variant.id).make_variants()
297        } else {
298            walk_flat_map_variant(self, variant)
299        }
300    }
301
302    fn flat_map_where_predicate(
303        &mut self,
304        predicate: ast::WherePredicate,
305    ) -> SmallVec<[ast::WherePredicate; 1]> {
306        if predicate.is_placeholder {
307            self.remove(predicate.id).make_where_predicates()
308        } else {
309            walk_flat_map_where_predicate(self, predicate)
310        }
311    }
312
313    fn flat_map_item(&mut self, item: Box<ast::Item>) -> SmallVec<[Box<ast::Item>; 1]> {
314        match item.kind {
315            ast::ItemKind::MacCall(_) => self.remove(item.id).make_items(),
316            _ => walk_flat_map_item(self, item),
317        }
318    }
319
320    fn flat_map_assoc_item(
321        &mut self,
322        item: Box<ast::AssocItem>,
323        ctxt: AssocCtxt,
324    ) -> SmallVec<[Box<ast::AssocItem>; 1]> {
325        match item.kind {
326            ast::AssocItemKind::MacCall(_) => {
327                let it = self.remove(item.id);
328                match ctxt {
329                    AssocCtxt::Trait => it.make_trait_items(),
330                    AssocCtxt::Impl { of_trait: false } => it.make_impl_items(),
331                    AssocCtxt::Impl { of_trait: true } => it.make_trait_impl_items(),
332                }
333            }
334            _ => walk_flat_map_assoc_item(self, item, ctxt),
335        }
336    }
337
338    fn flat_map_foreign_item(
339        &mut self,
340        item: Box<ast::ForeignItem>,
341    ) -> SmallVec<[Box<ast::ForeignItem>; 1]> {
342        match item.kind {
343            ast::ForeignItemKind::MacCall(_) => self.remove(item.id).make_foreign_items(),
344            _ => walk_flat_map_foreign_item(self, item),
345        }
346    }
347
348    fn visit_expr(&mut self, expr: &mut ast::Expr) {
349        match expr.kind {
350            ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_expr(),
351            _ => walk_expr(self, expr),
352        }
353    }
354
355    fn visit_method_receiver_expr(&mut self, expr: &mut ast::Expr) {
356        match expr.kind {
357            ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_method_receiver_expr(),
358            _ => walk_expr(self, expr),
359        }
360    }
361
362    fn filter_map_expr(&mut self, expr: Box<ast::Expr>) -> Option<Box<ast::Expr>> {
363        match expr.kind {
364            ast::ExprKind::MacCall(_) => self.remove(expr.id).make_opt_expr(),
365            _ => walk_filter_map_expr(self, expr),
366        }
367    }
368
369    fn flat_map_stmt(&mut self, stmt: ast::Stmt) -> SmallVec<[ast::Stmt; 1]> {
370        let (style, mut stmts) = match stmt.kind {
371            ast::StmtKind::MacCall(mac) => (mac.style, self.remove(stmt.id).make_stmts()),
372            _ => return walk_flat_map_stmt(self, stmt),
373        };
374
375        if style == ast::MacStmtStyle::Semicolon {
376            // Implement the proposal described in
377            // https://github.com/rust-lang/rust/issues/61733#issuecomment-509626449
378            //
379            // The macro invocation expands to the list of statements. If the
380            // list of statements is empty, then 'parse' the trailing semicolon
381            // on the original invocation as an empty statement. That is:
382            //
383            // `empty();` is parsed as a single `StmtKind::Empty`
384            //
385            // If the list of statements is non-empty, see if the final
386            // statement already has a trailing semicolon.
387            //
388            // If it doesn't have a semicolon, then 'parse' the trailing
389            // semicolon from the invocation as part of the final statement,
390            // using `stmt.add_trailing_semicolon()`
391            //
392            // If it does have a semicolon, then 'parse' the trailing semicolon
393            // from the invocation as a new StmtKind::Empty
394
395            // FIXME: We will need to preserve the original semicolon token and
396            // span as part of #15701
397            let empty_stmt =
398                ast::Stmt { id: ast::DUMMY_NODE_ID, kind: ast::StmtKind::Empty, span: DUMMY_SP };
399
400            if let Some(stmt) = stmts.pop() {
401                if stmt.has_trailing_semicolon() {
402                    stmts.push(stmt);
403                    stmts.push(empty_stmt);
404                } else {
405                    stmts.push(stmt.add_trailing_semicolon());
406                }
407            } else {
408                stmts.push(empty_stmt);
409            }
410        }
411
412        stmts
413    }
414
415    fn visit_pat(&mut self, pat: &mut ast::Pat) {
416        match pat.kind {
417            ast::PatKind::MacCall(_) => *pat = *self.remove(pat.id).make_pat(),
418            _ => walk_pat(self, pat),
419        }
420    }
421
422    fn visit_ty(&mut self, ty: &mut ast::Ty) {
423        match ty.kind {
424            ast::TyKind::MacCall(_) => *ty = *self.remove(ty.id).make_ty(),
425            _ => walk_ty(self, ty),
426        }
427    }
428
429    fn visit_crate(&mut self, krate: &mut ast::Crate) {
430        if krate.is_placeholder {
431            *krate = self.remove(krate.id).make_crate();
432        } else {
433            walk_crate(self, krate)
434        }
435    }
436}