Skip to main content

php_ast/
fold.rs

1//! Arena-to-arena AST transformation via the [`Fold`] trait.
2//!
3//! # Overview
4//!
5//! [`Fold`] is the transformation counterpart of [`crate::visitor::Visitor`].
6//! Where `Visitor` walks a tree read-only, `Fold` rebuilds it — reading from an
7//! input node (in any arena `'_`) and writing into a caller-supplied output arena
8//! `'new`.  This is the only sound design for arena-allocated ASTs: in-place
9//! mutation (`VisitorMut`) would let you write a `&'new Expr` into a slot that
10//! requires a `&'old Expr`, silently breaking the arena lifetime invariant.
11//!
12//! # Lifetimes
13//!
14//! * `'src` — the source-text lifetime, bound on the `Fold` trait.  `&'src str`
15//!   slices that point directly into the original PHP source are passed through
16//!   unchanged across the fold; they are never re-allocated.
17//! * `'new` — the output arena lifetime; a generic on each method so one
18//!   `impl Fold<'src>` can fold into different arenas over its lifetime.
19//! * `'_` — the input arena lifetime, intentionally erased.  The fold never
20//!   reads back through output-arena pointers, so the input and output arenas
21//!   are fully independent.
22//!
23//! # Arena-allocated strings
24//!
25//! Several AST nodes store `&'arena str` values that were *not* borrowed from
26//! the source buffer — for example string literals, [`StmtKind::Label`], and
27//! [`ExprKind::Nowdoc`] values.  The identity fold re-allocates these via
28//! `arena.alloc_str(s)` so the output nodes are self-contained within the
29//! new arena.  Source-borrowed [`NameStr`] values (`NameStr::__src`) are
30//! preserved as-is without copying.
31//!
32//! # Usage
33//!
34//! ```
35//! use bumpalo::Bump;
36//! use php_ast::fold::{Fold, fold_program};
37//! use php_ast::ast::*;
38//!
39//! /// An identity fold — rebuilds the AST without changes.
40//! struct Identity;
41//! impl<'src> Fold<'src> for Identity {}
42//!
43//! // Fold `program` (in `src_arena`) into `out_arena`:
44//! // let folded = Identity.fold_program(&out_arena, &program);
45//! ```
46//!
47//! To transform specific nodes, override the corresponding method and call the
48//! free `fold_*` function for the default recursion:
49//!
50//! ```
51//! use bumpalo::Bump;
52//! use php_ast::fold::{Fold, fold_expr};
53//! use php_ast::ast::*;
54//!
55//! struct NegateInts;
56//!
57//! impl<'src> Fold<'src> for NegateInts {
58//!     fn fold_expr<'new>(&mut self, arena: &'new Bump, expr: &Expr<'_, 'src>) -> Expr<'new, 'src> {
59//!         if let ExprKind::Int(n) = expr.kind {
60//!             return Expr { kind: ExprKind::Int(-n), span: expr.span };
61//!         }
62//!         fold_expr(self, arena, expr)
63//!     }
64//! }
65//! ```
66
67use bumpalo::Bump;
68
69use crate::ast::*;
70
71// =============================================================================
72// Fold trait
73// =============================================================================
74
75/// Trait for arena-to-arena PHP AST transformation.
76///
77/// All methods have identity-fold default implementations that call the
78/// corresponding free `fold_*` function.  Override only the node types you want
79/// to change; the rest recurse automatically.
80///
81/// See the [module documentation](self) for design rationale and lifetime notes.
82pub trait Fold<'src> {
83    fn fold_program<'new>(
84        &mut self,
85        arena: &'new Bump,
86        program: &Program<'_, 'src>,
87    ) -> Program<'new, 'src> {
88        fold_program(self, arena, program)
89    }
90
91    fn fold_stmt<'new>(&mut self, arena: &'new Bump, stmt: &Stmt<'_, 'src>) -> Stmt<'new, 'src> {
92        fold_stmt(self, arena, stmt)
93    }
94
95    fn fold_block<'new>(
96        &mut self,
97        arena: &'new Bump,
98        block: &Block<'_, 'src>,
99    ) -> Block<'new, 'src> {
100        fold_block(self, arena, block)
101    }
102
103    fn fold_expr<'new>(&mut self, arena: &'new Bump, expr: &Expr<'_, 'src>) -> Expr<'new, 'src> {
104        fold_expr(self, arena, expr)
105    }
106
107    fn fold_param<'new>(
108        &mut self,
109        arena: &'new Bump,
110        param: &Param<'_, 'src>,
111    ) -> Param<'new, 'src> {
112        fold_param(self, arena, param)
113    }
114
115    fn fold_arg<'new>(&mut self, arena: &'new Bump, arg: &Arg<'_, 'src>) -> Arg<'new, 'src> {
116        fold_arg(self, arena, arg)
117    }
118
119    fn fold_class_member<'new>(
120        &mut self,
121        arena: &'new Bump,
122        member: &ClassMember<'_, 'src>,
123    ) -> ClassMember<'new, 'src> {
124        fold_class_member(self, arena, member)
125    }
126
127    fn fold_enum_member<'new>(
128        &mut self,
129        arena: &'new Bump,
130        member: &EnumMember<'_, 'src>,
131    ) -> EnumMember<'new, 'src> {
132        fold_enum_member(self, arena, member)
133    }
134
135    fn fold_property_hook<'new>(
136        &mut self,
137        arena: &'new Bump,
138        hook: &PropertyHook<'_, 'src>,
139    ) -> PropertyHook<'new, 'src> {
140        fold_property_hook(self, arena, hook)
141    }
142
143    fn fold_type_hint<'new>(
144        &mut self,
145        arena: &'new Bump,
146        type_hint: &TypeHint<'_, 'src>,
147    ) -> TypeHint<'new, 'src> {
148        fold_type_hint(self, arena, type_hint)
149    }
150
151    fn fold_attribute<'new>(
152        &mut self,
153        arena: &'new Bump,
154        attribute: &Attribute<'_, 'src>,
155    ) -> Attribute<'new, 'src> {
156        fold_attribute(self, arena, attribute)
157    }
158
159    fn fold_catch_clause<'new>(
160        &mut self,
161        arena: &'new Bump,
162        catch: &CatchClause<'_, 'src>,
163    ) -> CatchClause<'new, 'src> {
164        fold_catch_clause(self, arena, catch)
165    }
166
167    fn fold_match_arm<'new>(
168        &mut self,
169        arena: &'new Bump,
170        arm: &MatchArm<'_, 'src>,
171    ) -> MatchArm<'new, 'src> {
172        fold_match_arm(self, arena, arm)
173    }
174
175    fn fold_closure_use_var(&mut self, var: &ClosureUseVar<'src>) -> ClosureUseVar<'src> {
176        var.clone()
177    }
178
179    fn fold_trait_use<'new>(
180        &mut self,
181        arena: &'new Bump,
182        trait_use: &TraitUseDecl<'_, 'src>,
183    ) -> TraitUseDecl<'new, 'src> {
184        fold_trait_use(self, arena, trait_use)
185    }
186
187    fn fold_trait_adaptation<'new>(
188        &mut self,
189        arena: &'new Bump,
190        adaptation: &TraitAdaptation<'_, 'src>,
191    ) -> TraitAdaptation<'new, 'src> {
192        fold_trait_adaptation(self, arena, adaptation)
193    }
194
195    fn fold_name<'new>(&mut self, arena: &'new Bump, name: &Name<'_, 'src>) -> Name<'new, 'src> {
196        fold_name(self, arena, name)
197    }
198}
199
200// =============================================================================
201// Public free functions — default recursion for each trait method
202// =============================================================================
203
204pub fn fold_program<'new, 'src, F: Fold<'src> + ?Sized>(
205    folder: &mut F,
206    arena: &'new Bump,
207    program: &Program<'_, 'src>,
208) -> Program<'new, 'src> {
209    Program {
210        stmts: fold_stmts(folder, arena, &program.stmts),
211        span: program.span,
212    }
213}
214
215pub fn fold_stmt<'new, 'src, F: Fold<'src> + ?Sized>(
216    folder: &mut F,
217    arena: &'new Bump,
218    stmt: &Stmt<'_, 'src>,
219) -> Stmt<'new, 'src> {
220    let kind = match &stmt.kind {
221        StmtKind::Expression(expr) => {
222            StmtKind::Expression(arena.alloc(folder.fold_expr(arena, expr)))
223        }
224        StmtKind::Echo(exprs) => StmtKind::Echo(fold_exprs(folder, arena, exprs)),
225        StmtKind::Return(expr) => {
226            StmtKind::Return(expr.map(|e| &*arena.alloc(folder.fold_expr(arena, e))))
227        }
228        StmtKind::Block(block) => StmtKind::Block(arena.alloc(folder.fold_block(arena, block))),
229        StmtKind::If(if_stmt) => {
230            let mut elseif_branches =
231                ArenaVec::with_capacity_in(if_stmt.elseif_branches.len(), arena);
232            for branch in if_stmt.elseif_branches.iter() {
233                elseif_branches.push(ElseIfBranch {
234                    condition: folder.fold_expr(arena, &branch.condition),
235                    body: folder.fold_stmt(arena, &branch.body),
236                    span: branch.span,
237                });
238            }
239            let new_if = arena.alloc(IfStmt {
240                condition: folder.fold_expr(arena, &if_stmt.condition),
241                then_branch: arena.alloc(folder.fold_stmt(arena, if_stmt.then_branch)),
242                elseif_branches,
243                else_branch: if_stmt
244                    .else_branch
245                    .map(|b| &*arena.alloc(folder.fold_stmt(arena, b))),
246                else_kw_start: if_stmt.else_kw_start,
247                uses_alternative: if_stmt.uses_alternative,
248            });
249            StmtKind::If(new_if)
250        }
251        StmtKind::While(w) => {
252            let new_w = arena.alloc(WhileStmt {
253                condition: folder.fold_expr(arena, &w.condition),
254                body: arena.alloc(folder.fold_stmt(arena, w.body)),
255                uses_alternative: w.uses_alternative,
256            });
257            StmtKind::While(new_w)
258        }
259        StmtKind::For(f) => {
260            let new_f = arena.alloc(ForStmt {
261                init: fold_exprs(folder, arena, &f.init),
262                condition: fold_exprs(folder, arena, &f.condition),
263                update: fold_exprs(folder, arena, &f.update),
264                body: arena.alloc(folder.fold_stmt(arena, f.body)),
265                uses_alternative: f.uses_alternative,
266            });
267            StmtKind::For(new_f)
268        }
269        StmtKind::Foreach(fe) => {
270            let new_fe = arena.alloc(ForeachStmt {
271                expr: folder.fold_expr(arena, &fe.expr),
272                key: fe.key.as_ref().map(|k| folder.fold_expr(arena, k)),
273                value: folder.fold_expr(arena, &fe.value),
274                body: arena.alloc(folder.fold_stmt(arena, fe.body)),
275                uses_alternative: fe.uses_alternative,
276            });
277            StmtKind::Foreach(new_fe)
278        }
279        StmtKind::DoWhile(dw) => {
280            let new_dw = arena.alloc(DoWhileStmt {
281                body: arena.alloc(folder.fold_stmt(arena, dw.body)),
282                condition: folder.fold_expr(arena, &dw.condition),
283            });
284            StmtKind::DoWhile(new_dw)
285        }
286        StmtKind::Function(func) => {
287            StmtKind::Function(arena.alloc(fold_function_decl(folder, arena, func)))
288        }
289        StmtKind::Break(expr) => {
290            StmtKind::Break(expr.map(|e| &*arena.alloc(folder.fold_expr(arena, e))))
291        }
292        StmtKind::Continue(expr) => {
293            StmtKind::Continue(expr.map(|e| &*arena.alloc(folder.fold_expr(arena, e))))
294        }
295        StmtKind::Switch(sw) => {
296            let mut cases = ArenaVec::with_capacity_in(sw.body.cases.len(), arena);
297            for case in sw.body.cases.iter() {
298                cases.push(SwitchCase {
299                    value: case.value.as_ref().map(|v| folder.fold_expr(arena, v)),
300                    body: fold_stmts(folder, arena, &case.body),
301                    span: case.span,
302                });
303            }
304            let new_sw = arena.alloc(SwitchStmt {
305                expr: folder.fold_expr(arena, &sw.expr),
306                body: SwitchBody {
307                    cases,
308                    span: sw.body.span,
309                },
310                uses_alternative: sw.uses_alternative,
311            });
312            StmtKind::Switch(new_sw)
313        }
314        StmtKind::Goto(ident) => StmtKind::Goto(*ident),
315        StmtKind::Label(s) => StmtKind::Label(arena.alloc_str(s)),
316        StmtKind::Declare(decl) => {
317            let mut directives = ArenaVec::with_capacity_in(decl.directives.len(), arena);
318            for (name, expr) in decl.directives.iter() {
319                directives.push((*name, folder.fold_expr(arena, expr)));
320            }
321            let new_decl = arena.alloc(DeclareStmt {
322                directives,
323                body: decl.body.map(|b| &*arena.alloc(folder.fold_stmt(arena, b))),
324                uses_alternative: decl.uses_alternative,
325            });
326            StmtKind::Declare(new_decl)
327        }
328        StmtKind::Unset(exprs) => StmtKind::Unset(fold_exprs(folder, arena, exprs)),
329        StmtKind::Throw(expr) => StmtKind::Throw(arena.alloc(folder.fold_expr(arena, expr))),
330        StmtKind::TryCatch(tc) => {
331            let mut catches = ArenaVec::with_capacity_in(tc.catches.len(), arena);
332            for catch in tc.catches.iter() {
333                catches.push(folder.fold_catch_clause(arena, catch));
334            }
335            let new_tc = arena.alloc(TryCatchStmt {
336                body: arena.alloc(folder.fold_block(arena, tc.body)),
337                catches,
338                finally: tc
339                    .finally
340                    .map(|f| &*arena.alloc(folder.fold_block(arena, f))),
341                finally_kw_start: tc.finally_kw_start,
342            });
343            StmtKind::TryCatch(new_tc)
344        }
345        StmtKind::Global(exprs) => StmtKind::Global(fold_exprs(folder, arena, exprs)),
346        StmtKind::Class(class) => {
347            StmtKind::Class(arena.alloc(fold_class_decl(folder, arena, class)))
348        }
349        StmtKind::Interface(iface) => {
350            StmtKind::Interface(arena.alloc(fold_interface_decl(folder, arena, iface)))
351        }
352        StmtKind::Trait(t) => StmtKind::Trait(arena.alloc(fold_trait_decl(folder, arena, t))),
353        StmtKind::Enum(e) => StmtKind::Enum(arena.alloc(fold_enum_decl(folder, arena, e))),
354        StmtKind::Namespace(ns) => {
355            let new_ns = arena.alloc(NamespaceDecl {
356                name: ns.name.as_ref().map(|n| folder.fold_name(arena, n)),
357                body: match &ns.body {
358                    NamespaceBody::Braced(block) => {
359                        NamespaceBody::Braced(arena.alloc(folder.fold_block(arena, block)))
360                    }
361                    NamespaceBody::Simple => NamespaceBody::Simple,
362                },
363            });
364            StmtKind::Namespace(new_ns)
365        }
366        StmtKind::Use(use_decl) => {
367            let mut uses = ArenaVec::with_capacity_in(use_decl.uses.len(), arena);
368            for item in use_decl.uses.iter() {
369                uses.push(UseItem {
370                    name: folder.fold_name(arena, &item.name),
371                    alias: item.alias,
372                    kind: item.kind,
373                    span: item.span,
374                });
375            }
376            let new_use = arena.alloc(UseDecl {
377                kind: use_decl.kind,
378                uses,
379            });
380            StmtKind::Use(new_use)
381        }
382        StmtKind::Const(items) => {
383            let mut new_items = ArenaVec::with_capacity_in(items.len(), arena);
384            for item in items.iter() {
385                new_items.push(ConstItem {
386                    name: item.name,
387                    value: folder.fold_expr(arena, &item.value),
388                    attributes: fold_attrs(folder, arena, &item.attributes),
389                    span: item.span,
390                    doc_comment: item.doc_comment.as_ref().map(fold_comment),
391                });
392            }
393            StmtKind::Const(new_items)
394        }
395        StmtKind::StaticVar(vars) => {
396            let mut new_vars = ArenaVec::with_capacity_in(vars.len(), arena);
397            for var in vars.iter() {
398                new_vars.push(StaticVar {
399                    name: var.name,
400                    default: var.default.as_ref().map(|d| folder.fold_expr(arena, d)),
401                    span: var.span,
402                });
403            }
404            StmtKind::StaticVar(new_vars)
405        }
406        StmtKind::HaltCompiler(s) => StmtKind::HaltCompiler(s),
407        StmtKind::Nop => StmtKind::Nop,
408        StmtKind::InlineHtml(s) => StmtKind::InlineHtml(s),
409        StmtKind::Error => StmtKind::Error,
410    };
411    Stmt {
412        kind,
413        span: stmt.span,
414    }
415}
416
417pub fn fold_expr<'new, 'src, F: Fold<'src> + ?Sized>(
418    folder: &mut F,
419    arena: &'new Bump,
420    expr: &Expr<'_, 'src>,
421) -> Expr<'new, 'src> {
422    let kind = match &expr.kind {
423        ExprKind::Int(n) => ExprKind::Int(*n),
424        ExprKind::Float(f) => ExprKind::Float(*f),
425        ExprKind::String(s) => ExprKind::String(arena.alloc_str(s)),
426        ExprKind::InterpolatedString(parts) => {
427            ExprKind::InterpolatedString(fold_string_parts(folder, arena, parts))
428        }
429        ExprKind::Heredoc { label, parts } => ExprKind::Heredoc {
430            label,
431            parts: fold_string_parts(folder, arena, parts),
432        },
433        ExprKind::Nowdoc { label, value } => ExprKind::Nowdoc {
434            label,
435            value: arena.alloc_str(value),
436        },
437        ExprKind::ShellExec(parts) => ExprKind::ShellExec(fold_string_parts(folder, arena, parts)),
438        ExprKind::Bool(b) => ExprKind::Bool(*b),
439        ExprKind::Null => ExprKind::Null,
440        ExprKind::Variable(name) => ExprKind::Variable(fold_name_str(*name, arena)),
441        ExprKind::VariableVariable(inner) => {
442            ExprKind::VariableVariable(arena.alloc(folder.fold_expr(arena, inner)))
443        }
444        ExprKind::Identifier(name) => ExprKind::Identifier(fold_name_str(*name, arena)),
445        ExprKind::Assign(assign) => ExprKind::Assign(AssignExpr {
446            target: arena.alloc(folder.fold_expr(arena, assign.target)),
447            op: assign.op,
448            value: arena.alloc(folder.fold_expr(arena, assign.value)),
449            by_ref: assign.by_ref,
450        }),
451        ExprKind::Binary(binary) => ExprKind::Binary(BinaryExpr {
452            left: arena.alloc(folder.fold_expr(arena, binary.left)),
453            op: binary.op,
454            right: arena.alloc(folder.fold_expr(arena, binary.right)),
455        }),
456        ExprKind::UnaryPrefix(u) => ExprKind::UnaryPrefix(UnaryPrefixExpr {
457            op: u.op,
458            operand: arena.alloc(folder.fold_expr(arena, u.operand)),
459        }),
460        ExprKind::UnaryPostfix(u) => ExprKind::UnaryPostfix(UnaryPostfixExpr {
461            operand: arena.alloc(folder.fold_expr(arena, u.operand)),
462            op: u.op,
463        }),
464        ExprKind::Ternary(t) => ExprKind::Ternary(TernaryExpr {
465            condition: arena.alloc(folder.fold_expr(arena, t.condition)),
466            then_expr: t
467                .then_expr
468                .map(|e| &*arena.alloc(folder.fold_expr(arena, e))),
469            else_expr: arena.alloc(folder.fold_expr(arena, t.else_expr)),
470        }),
471        ExprKind::NullCoalesce(nc) => ExprKind::NullCoalesce(NullCoalesceExpr {
472            left: arena.alloc(folder.fold_expr(arena, nc.left)),
473            right: arena.alloc(folder.fold_expr(arena, nc.right)),
474        }),
475        ExprKind::FunctionCall(call) => ExprKind::FunctionCall(FunctionCallExpr {
476            name: arena.alloc(folder.fold_expr(arena, call.name)),
477            args: fold_args(folder, arena, &call.args),
478        }),
479        ExprKind::Array(elements) => {
480            let mut new_elements = ArenaVec::with_capacity_in(elements.len(), arena);
481            for elem in elements.iter() {
482                new_elements.push(ArrayElement {
483                    key: elem.key.as_ref().map(|k| folder.fold_expr(arena, k)),
484                    value: folder.fold_expr(arena, &elem.value),
485                    unpack: elem.unpack,
486                    by_ref: elem.by_ref,
487                    span: elem.span,
488                });
489            }
490            ExprKind::Array(new_elements)
491        }
492        ExprKind::ArrayAccess(access) => ExprKind::ArrayAccess(ArrayAccessExpr {
493            array: arena.alloc(folder.fold_expr(arena, access.array)),
494            index: access
495                .index
496                .map(|i| &*arena.alloc(folder.fold_expr(arena, i))),
497        }),
498        ExprKind::Print(e) => ExprKind::Print(arena.alloc(folder.fold_expr(arena, e))),
499        ExprKind::Parenthesized(e) => {
500            ExprKind::Parenthesized(arena.alloc(folder.fold_expr(arena, e)))
501        }
502        ExprKind::Cast(kind, e) => ExprKind::Cast(*kind, arena.alloc(folder.fold_expr(arena, e))),
503        ExprKind::ErrorSuppress(e) => {
504            ExprKind::ErrorSuppress(arena.alloc(folder.fold_expr(arena, e)))
505        }
506        ExprKind::Isset(exprs) => ExprKind::Isset(fold_exprs(folder, arena, exprs)),
507        ExprKind::Empty(e) => ExprKind::Empty(arena.alloc(folder.fold_expr(arena, e))),
508        ExprKind::Include(kind, e) => {
509            ExprKind::Include(*kind, arena.alloc(folder.fold_expr(arena, e)))
510        }
511        ExprKind::Eval(e) => ExprKind::Eval(arena.alloc(folder.fold_expr(arena, e))),
512        ExprKind::Exit(e) => ExprKind::Exit(e.map(|e| &*arena.alloc(folder.fold_expr(arena, e)))),
513        ExprKind::MagicConst(k) => ExprKind::MagicConst(*k),
514        ExprKind::Clone(e) => ExprKind::Clone(arena.alloc(folder.fold_expr(arena, e))),
515        ExprKind::CloneWith(obj, overrides) => ExprKind::CloneWith(
516            arena.alloc(folder.fold_expr(arena, obj)),
517            arena.alloc(folder.fold_expr(arena, overrides)),
518        ),
519        ExprKind::New(new_expr) => ExprKind::New(NewExpr {
520            class: arena.alloc(folder.fold_expr(arena, new_expr.class)),
521            args: fold_args(folder, arena, &new_expr.args),
522        }),
523        ExprKind::PropertyAccess(access) => ExprKind::PropertyAccess(PropertyAccessExpr {
524            object: arena.alloc(folder.fold_expr(arena, access.object)),
525            property: arena.alloc(folder.fold_expr(arena, access.property)),
526        }),
527        ExprKind::NullsafePropertyAccess(access) => {
528            ExprKind::NullsafePropertyAccess(PropertyAccessExpr {
529                object: arena.alloc(folder.fold_expr(arena, access.object)),
530                property: arena.alloc(folder.fold_expr(arena, access.property)),
531            })
532        }
533        ExprKind::MethodCall(call) => ExprKind::MethodCall(arena.alloc(MethodCallExpr {
534            object: arena.alloc(folder.fold_expr(arena, call.object)),
535            method: arena.alloc(folder.fold_expr(arena, call.method)),
536            args: fold_args(folder, arena, &call.args),
537        })),
538        ExprKind::NullsafeMethodCall(call) => {
539            ExprKind::NullsafeMethodCall(arena.alloc(MethodCallExpr {
540                object: arena.alloc(folder.fold_expr(arena, call.object)),
541                method: arena.alloc(folder.fold_expr(arena, call.method)),
542                args: fold_args(folder, arena, &call.args),
543            }))
544        }
545        ExprKind::StaticPropertyAccess(access) => {
546            ExprKind::StaticPropertyAccess(StaticAccessExpr {
547                class: arena.alloc(folder.fold_expr(arena, access.class)),
548                member: arena.alloc(folder.fold_expr(arena, access.member)),
549            })
550        }
551        ExprKind::StaticMethodCall(call) => {
552            ExprKind::StaticMethodCall(arena.alloc(StaticMethodCallExpr {
553                class: arena.alloc(folder.fold_expr(arena, call.class)),
554                method: arena.alloc(folder.fold_expr(arena, call.method)),
555                args: fold_args(folder, arena, &call.args),
556            }))
557        }
558        ExprKind::StaticDynMethodCall(call) => {
559            ExprKind::StaticDynMethodCall(arena.alloc(StaticDynMethodCallExpr {
560                class: arena.alloc(folder.fold_expr(arena, call.class)),
561                method: arena.alloc(folder.fold_expr(arena, call.method)),
562                args: fold_args(folder, arena, &call.args),
563            }))
564        }
565        ExprKind::ClassConstAccess(access) => ExprKind::ClassConstAccess(StaticAccessExpr {
566            class: arena.alloc(folder.fold_expr(arena, access.class)),
567            member: arena.alloc(folder.fold_expr(arena, access.member)),
568        }),
569        ExprKind::ClassConstAccessDynamic { class, member } => ExprKind::ClassConstAccessDynamic {
570            class: arena.alloc(folder.fold_expr(arena, class)),
571            member: arena.alloc(folder.fold_expr(arena, member)),
572        },
573        ExprKind::StaticPropertyAccessDynamic { class, member } => {
574            ExprKind::StaticPropertyAccessDynamic {
575                class: arena.alloc(folder.fold_expr(arena, class)),
576                member: arena.alloc(folder.fold_expr(arena, member)),
577            }
578        }
579        ExprKind::Closure(closure) => {
580            let mut use_vars = ArenaVec::with_capacity_in(closure.use_vars.len(), arena);
581            for var in closure.use_vars.iter() {
582                use_vars.push(folder.fold_closure_use_var(var));
583            }
584            let new_closure = arena.alloc(ClosureExpr {
585                is_static: closure.is_static,
586                by_ref: closure.by_ref,
587                params: fold_params(folder, arena, &closure.params),
588                use_vars,
589                return_type: closure
590                    .return_type
591                    .as_ref()
592                    .map(|t| folder.fold_type_hint(arena, t)),
593                body: arena.alloc(folder.fold_block(arena, closure.body)),
594                attributes: fold_attrs(folder, arena, &closure.attributes),
595            });
596            ExprKind::Closure(new_closure)
597        }
598        ExprKind::ArrowFunction(arrow) => {
599            let new_arrow = arena.alloc(ArrowFunctionExpr {
600                is_static: arrow.is_static,
601                by_ref: arrow.by_ref,
602                params: fold_params(folder, arena, &arrow.params),
603                return_type: arrow
604                    .return_type
605                    .as_ref()
606                    .map(|t| folder.fold_type_hint(arena, t)),
607                body: arena.alloc(folder.fold_expr(arena, arrow.body)),
608                attributes: fold_attrs(folder, arena, &arrow.attributes),
609            });
610            ExprKind::ArrowFunction(new_arrow)
611        }
612        ExprKind::Match(match_expr) => ExprKind::Match(MatchExpr {
613            subject: arena.alloc(folder.fold_expr(arena, match_expr.subject)),
614            arms: {
615                let mut arms = ArenaVec::with_capacity_in(match_expr.arms.len(), arena);
616                for arm in match_expr.arms.iter() {
617                    arms.push(folder.fold_match_arm(arena, arm));
618                }
619                arms
620            },
621            brace_start: match_expr.brace_start,
622        }),
623        ExprKind::ThrowExpr(e) => ExprKind::ThrowExpr(arena.alloc(folder.fold_expr(arena, e))),
624        ExprKind::Yield(y) => ExprKind::Yield(YieldExpr {
625            key: y.key.map(|k| &*arena.alloc(folder.fold_expr(arena, k))),
626            value: y.value.map(|v| &*arena.alloc(folder.fold_expr(arena, v))),
627            is_from: y.is_from,
628        }),
629        ExprKind::AnonymousClass(class) => {
630            ExprKind::AnonymousClass(arena.alloc(fold_class_decl(folder, arena, class)))
631        }
632        ExprKind::CallableCreate(cc) => {
633            let kind = match &cc.kind {
634                CallableCreateKind::Function(name) => {
635                    CallableCreateKind::Function(arena.alloc(folder.fold_expr(arena, name)))
636                }
637                CallableCreateKind::Method { object, method } => CallableCreateKind::Method {
638                    object: arena.alloc(folder.fold_expr(arena, object)),
639                    method: arena.alloc(folder.fold_expr(arena, method)),
640                },
641                CallableCreateKind::NullsafeMethod { object, method } => {
642                    CallableCreateKind::NullsafeMethod {
643                        object: arena.alloc(folder.fold_expr(arena, object)),
644                        method: arena.alloc(folder.fold_expr(arena, method)),
645                    }
646                }
647                CallableCreateKind::StaticMethod { class, method } => {
648                    CallableCreateKind::StaticMethod {
649                        class: arena.alloc(folder.fold_expr(arena, class)),
650                        method: arena.alloc(folder.fold_expr(arena, method)),
651                    }
652                }
653            };
654            ExprKind::CallableCreate(CallableCreateExpr { kind })
655        }
656        ExprKind::Omit => ExprKind::Omit,
657        ExprKind::Error => ExprKind::Error,
658    };
659    Expr {
660        kind,
661        span: expr.span,
662    }
663}
664
665pub fn fold_param<'new, 'src, F: Fold<'src> + ?Sized>(
666    folder: &mut F,
667    arena: &'new Bump,
668    param: &Param<'_, 'src>,
669) -> Param<'new, 'src> {
670    Param {
671        name: param.name,
672        type_hint: param
673            .type_hint
674            .as_ref()
675            .map(|t| folder.fold_type_hint(arena, t)),
676        default: param.default.as_ref().map(|d| folder.fold_expr(arena, d)),
677        by_ref: param.by_ref,
678        variadic: param.variadic,
679        is_readonly: param.is_readonly,
680        is_final: param.is_final,
681        visibility: param.visibility,
682        set_visibility: param.set_visibility,
683        attributes: fold_attrs(folder, arena, &param.attributes),
684        hooks: fold_hooks(folder, arena, &param.hooks),
685        span: param.span,
686    }
687}
688
689pub fn fold_arg<'new, 'src, F: Fold<'src> + ?Sized>(
690    folder: &mut F,
691    arena: &'new Bump,
692    arg: &Arg<'_, 'src>,
693) -> Arg<'new, 'src> {
694    Arg {
695        name: arg.name.as_ref().map(|n| folder.fold_name(arena, n)),
696        value: folder.fold_expr(arena, &arg.value),
697        unpack: arg.unpack,
698        by_ref: arg.by_ref,
699        span: arg.span,
700    }
701}
702
703pub fn fold_class_member<'new, 'src, F: Fold<'src> + ?Sized>(
704    folder: &mut F,
705    arena: &'new Bump,
706    member: &ClassMember<'_, 'src>,
707) -> ClassMember<'new, 'src> {
708    let kind = match &member.kind {
709        ClassMemberKind::Property(prop) => {
710            ClassMemberKind::Property(fold_property_decl(folder, arena, prop))
711        }
712        ClassMemberKind::Method(method) => {
713            ClassMemberKind::Method(fold_method_decl(folder, arena, method))
714        }
715        ClassMemberKind::ClassConst(cc) => {
716            ClassMemberKind::ClassConst(fold_class_const_decl(folder, arena, cc))
717        }
718        ClassMemberKind::TraitUse(tu) => {
719            ClassMemberKind::TraitUse(folder.fold_trait_use(arena, tu))
720        }
721    };
722    ClassMember {
723        kind,
724        span: member.span,
725    }
726}
727
728pub fn fold_enum_member<'new, 'src, F: Fold<'src> + ?Sized>(
729    folder: &mut F,
730    arena: &'new Bump,
731    member: &EnumMember<'_, 'src>,
732) -> EnumMember<'new, 'src> {
733    let kind = match &member.kind {
734        EnumMemberKind::Case(case) => EnumMemberKind::Case(EnumCase {
735            name: case.name,
736            value: case.value.as_ref().map(|v| folder.fold_expr(arena, v)),
737            attributes: fold_attrs(folder, arena, &case.attributes),
738            doc_comment: case.doc_comment.as_ref().map(fold_comment),
739        }),
740        EnumMemberKind::Method(method) => {
741            EnumMemberKind::Method(fold_method_decl(folder, arena, method))
742        }
743        EnumMemberKind::ClassConst(cc) => {
744            EnumMemberKind::ClassConst(fold_class_const_decl(folder, arena, cc))
745        }
746        EnumMemberKind::TraitUse(tu) => EnumMemberKind::TraitUse(folder.fold_trait_use(arena, tu)),
747    };
748    EnumMember {
749        kind,
750        span: member.span,
751    }
752}
753
754pub fn fold_property_hook<'new, 'src, F: Fold<'src> + ?Sized>(
755    folder: &mut F,
756    arena: &'new Bump,
757    hook: &PropertyHook<'_, 'src>,
758) -> PropertyHook<'new, 'src> {
759    let body = match &hook.body {
760        PropertyHookBody::Block(block) => {
761            PropertyHookBody::Block(arena.alloc(fold_block(folder, arena, block)))
762        }
763        PropertyHookBody::Expression(expr) => {
764            PropertyHookBody::Expression(folder.fold_expr(arena, expr))
765        }
766        PropertyHookBody::Abstract => PropertyHookBody::Abstract,
767    };
768    PropertyHook {
769        kind: hook.kind,
770        body,
771        is_final: hook.is_final,
772        by_ref: hook.by_ref,
773        params: fold_params(folder, arena, &hook.params),
774        attributes: fold_attrs(folder, arena, &hook.attributes),
775        span: hook.span,
776    }
777}
778
779pub fn fold_type_hint<'new, 'src, F: Fold<'src> + ?Sized>(
780    folder: &mut F,
781    arena: &'new Bump,
782    type_hint: &TypeHint<'_, 'src>,
783) -> TypeHint<'new, 'src> {
784    let kind = match &type_hint.kind {
785        TypeHintKind::Named(name) => TypeHintKind::Named(folder.fold_name(arena, name)),
786        TypeHintKind::Keyword(builtin, span) => TypeHintKind::Keyword(*builtin, *span),
787        TypeHintKind::Nullable(inner) => {
788            TypeHintKind::Nullable(arena.alloc(folder.fold_type_hint(arena, inner)))
789        }
790        TypeHintKind::Union(types) => {
791            let mut new_types = ArenaVec::with_capacity_in(types.len(), arena);
792            for t in types.iter() {
793                new_types.push(folder.fold_type_hint(arena, t));
794            }
795            TypeHintKind::Union(new_types)
796        }
797        TypeHintKind::Intersection(types) => {
798            let mut new_types = ArenaVec::with_capacity_in(types.len(), arena);
799            for t in types.iter() {
800                new_types.push(folder.fold_type_hint(arena, t));
801            }
802            TypeHintKind::Intersection(new_types)
803        }
804    };
805    TypeHint {
806        kind,
807        span: type_hint.span,
808    }
809}
810
811pub fn fold_attribute<'new, 'src, F: Fold<'src> + ?Sized>(
812    folder: &mut F,
813    arena: &'new Bump,
814    attribute: &Attribute<'_, 'src>,
815) -> Attribute<'new, 'src> {
816    Attribute {
817        name: folder.fold_name(arena, &attribute.name),
818        args: fold_args(folder, arena, &attribute.args),
819        span: attribute.span,
820    }
821}
822
823pub fn fold_catch_clause<'new, 'src, F: Fold<'src> + ?Sized>(
824    folder: &mut F,
825    arena: &'new Bump,
826    catch: &CatchClause<'_, 'src>,
827) -> CatchClause<'new, 'src> {
828    let mut types = ArenaVec::with_capacity_in(catch.types.len(), arena);
829    for ty in catch.types.iter() {
830        types.push(folder.fold_name(arena, ty));
831    }
832    CatchClause {
833        types,
834        var: catch.var,
835        body: arena.alloc(folder.fold_block(arena, catch.body)),
836        span: catch.span,
837    }
838}
839
840pub fn fold_match_arm<'new, 'src, F: Fold<'src> + ?Sized>(
841    folder: &mut F,
842    arena: &'new Bump,
843    arm: &MatchArm<'_, 'src>,
844) -> MatchArm<'new, 'src> {
845    let conditions = arm.conditions.as_ref().map(|conds| {
846        let mut new_conds = ArenaVec::with_capacity_in(conds.len(), arena);
847        for c in conds.iter() {
848            new_conds.push(folder.fold_expr(arena, c));
849        }
850        new_conds
851    });
852    MatchArm {
853        conditions,
854        body: folder.fold_expr(arena, &arm.body),
855        span: arm.span,
856    }
857}
858
859pub fn fold_trait_use<'new, 'src, F: Fold<'src> + ?Sized>(
860    folder: &mut F,
861    arena: &'new Bump,
862    trait_use: &TraitUseDecl<'_, 'src>,
863) -> TraitUseDecl<'new, 'src> {
864    let mut traits = ArenaVec::with_capacity_in(trait_use.traits.len(), arena);
865    for t in trait_use.traits.iter() {
866        traits.push(folder.fold_name(arena, t));
867    }
868    let mut adaptations = ArenaVec::with_capacity_in(trait_use.adaptations.len(), arena);
869    for a in trait_use.adaptations.iter() {
870        adaptations.push(folder.fold_trait_adaptation(arena, a));
871    }
872    TraitUseDecl {
873        traits,
874        adaptations,
875        adaptations_brace_start: trait_use.adaptations_brace_start,
876    }
877}
878
879pub fn fold_trait_adaptation<'new, 'src, F: Fold<'src> + ?Sized>(
880    folder: &mut F,
881    arena: &'new Bump,
882    adaptation: &TraitAdaptation<'_, 'src>,
883) -> TraitAdaptation<'new, 'src> {
884    let kind = match &adaptation.kind {
885        TraitAdaptationKind::Precedence {
886            trait_name,
887            method,
888            insteadof,
889        } => {
890            let mut new_insteadof = ArenaVec::with_capacity_in(insteadof.len(), arena);
891            for n in insteadof.iter() {
892                new_insteadof.push(folder.fold_name(arena, n));
893            }
894            TraitAdaptationKind::Precedence {
895                trait_name: folder.fold_name(arena, trait_name),
896                method: folder.fold_name(arena, method),
897                insteadof: new_insteadof,
898            }
899        }
900        TraitAdaptationKind::Alias {
901            trait_name,
902            method,
903            new_modifier,
904            new_name,
905        } => TraitAdaptationKind::Alias {
906            trait_name: trait_name.as_ref().map(|n| folder.fold_name(arena, n)),
907            method: folder.fold_name(arena, method),
908            new_modifier: *new_modifier,
909            new_name: new_name.as_ref().map(|n| folder.fold_name(arena, n)),
910        },
911    };
912    TraitAdaptation {
913        kind,
914        span: adaptation.span,
915    }
916}
917
918pub fn fold_name<'new, 'src, F: Fold<'src> + ?Sized>(
919    _folder: &mut F,
920    arena: &'new Bump,
921    name: &Name<'_, 'src>,
922) -> Name<'new, 'src> {
923    match name {
924        Name::Simple { value, span } => Name::Simple { value, span: *span },
925        Name::Complex { parts, kind, span } => {
926            let mut new_parts = ArenaVec::with_capacity_in(parts.len(), arena);
927            for &part in parts.iter() {
928                new_parts.push(part);
929            }
930            Name::Complex {
931                parts: new_parts,
932                kind: *kind,
933                span: *span,
934            }
935        }
936        Name::Error { span } => Name::Error { span: *span },
937    }
938}
939
940// =============================================================================
941// Private helpers — complex declaration types
942// =============================================================================
943
944fn fold_function_decl<'new, 'src, F: Fold<'src> + ?Sized>(
945    folder: &mut F,
946    arena: &'new Bump,
947    func: &FunctionDecl<'_, 'src>,
948) -> FunctionDecl<'new, 'src> {
949    FunctionDecl {
950        name: func.name,
951        params: fold_params(folder, arena, &func.params),
952        body: arena.alloc(folder.fold_block(arena, func.body)),
953        return_type: func
954            .return_type
955            .as_ref()
956            .map(|t| folder.fold_type_hint(arena, t)),
957        by_ref: func.by_ref,
958        attributes: fold_attrs(folder, arena, &func.attributes),
959        doc_comment: func.doc_comment.as_ref().map(fold_comment),
960    }
961}
962
963fn fold_method_decl<'new, 'src, F: Fold<'src> + ?Sized>(
964    folder: &mut F,
965    arena: &'new Bump,
966    method: &MethodDecl<'_, 'src>,
967) -> MethodDecl<'new, 'src> {
968    MethodDecl {
969        name: method.name,
970        visibility: method.visibility,
971        is_static: method.is_static,
972        is_abstract: method.is_abstract,
973        is_final: method.is_final,
974        by_ref: method.by_ref,
975        params: fold_params(folder, arena, &method.params),
976        return_type: method
977            .return_type
978            .as_ref()
979            .map(|t| folder.fold_type_hint(arena, t)),
980        body: method
981            .body
982            .map(|b| &*arena.alloc(folder.fold_block(arena, b))),
983        attributes: fold_attrs(folder, arena, &method.attributes),
984        doc_comment: method.doc_comment.as_ref().map(fold_comment),
985    }
986}
987
988fn fold_property_decl<'new, 'src, F: Fold<'src> + ?Sized>(
989    folder: &mut F,
990    arena: &'new Bump,
991    prop: &PropertyDecl<'_, 'src>,
992) -> PropertyDecl<'new, 'src> {
993    PropertyDecl {
994        name: prop.name,
995        visibility: prop.visibility,
996        set_visibility: prop.set_visibility,
997        is_static: prop.is_static,
998        is_readonly: prop.is_readonly,
999        type_hint: prop
1000            .type_hint
1001            .as_ref()
1002            .map(|t| folder.fold_type_hint(arena, t)),
1003        default: prop.default.as_ref().map(|d| folder.fold_expr(arena, d)),
1004        attributes: fold_attrs(folder, arena, &prop.attributes),
1005        hooks: fold_hooks(folder, arena, &prop.hooks),
1006        doc_comment: prop.doc_comment.as_ref().map(fold_comment),
1007    }
1008}
1009
1010fn fold_class_const_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1011    folder: &mut F,
1012    arena: &'new Bump,
1013    cc: &ClassConstDecl<'_, 'src>,
1014) -> ClassConstDecl<'new, 'src> {
1015    ClassConstDecl {
1016        name: cc.name,
1017        visibility: cc.visibility,
1018        is_final: cc.is_final,
1019        type_hint: cc
1020            .type_hint
1021            .map(|t| &*arena.alloc(folder.fold_type_hint(arena, t))),
1022        value: folder.fold_expr(arena, &cc.value),
1023        attributes: fold_attrs(folder, arena, &cc.attributes),
1024        doc_comment: cc.doc_comment.as_ref().map(fold_comment),
1025    }
1026}
1027
1028fn fold_class_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1029    folder: &mut F,
1030    arena: &'new Bump,
1031    class: &ClassDecl<'_, 'src>,
1032) -> ClassDecl<'new, 'src> {
1033    let mut members = ArenaVec::with_capacity_in(class.body.members.len(), arena);
1034    for member in class.body.members.iter() {
1035        members.push(folder.fold_class_member(arena, member));
1036    }
1037    ClassDecl {
1038        name: class.name,
1039        modifiers: class.modifiers.clone(),
1040        extends: class.extends.as_ref().map(|n| folder.fold_name(arena, n)),
1041        implements: {
1042            let mut v = ArenaVec::with_capacity_in(class.implements.len(), arena);
1043            for n in class.implements.iter() {
1044                v.push(folder.fold_name(arena, n));
1045            }
1046            v
1047        },
1048        body: ClassBody {
1049            members,
1050            span: class.body.span,
1051        },
1052        attributes: fold_attrs(folder, arena, &class.attributes),
1053        doc_comment: class.doc_comment.as_ref().map(fold_comment),
1054    }
1055}
1056
1057fn fold_interface_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1058    folder: &mut F,
1059    arena: &'new Bump,
1060    iface: &InterfaceDecl<'_, 'src>,
1061) -> InterfaceDecl<'new, 'src> {
1062    let mut extends = ArenaVec::with_capacity_in(iface.extends.len(), arena);
1063    for n in iface.extends.iter() {
1064        extends.push(folder.fold_name(arena, n));
1065    }
1066    let mut members = ArenaVec::with_capacity_in(iface.body.members.len(), arena);
1067    for member in iface.body.members.iter() {
1068        members.push(folder.fold_class_member(arena, member));
1069    }
1070    InterfaceDecl {
1071        name: iface.name,
1072        extends,
1073        body: ClassBody {
1074            members,
1075            span: iface.body.span,
1076        },
1077        attributes: fold_attrs(folder, arena, &iface.attributes),
1078        doc_comment: iface.doc_comment.as_ref().map(fold_comment),
1079    }
1080}
1081
1082fn fold_trait_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1083    folder: &mut F,
1084    arena: &'new Bump,
1085    t: &TraitDecl<'_, 'src>,
1086) -> TraitDecl<'new, 'src> {
1087    let mut members = ArenaVec::with_capacity_in(t.body.members.len(), arena);
1088    for member in t.body.members.iter() {
1089        members.push(folder.fold_class_member(arena, member));
1090    }
1091    TraitDecl {
1092        name: t.name,
1093        body: ClassBody {
1094            members,
1095            span: t.body.span,
1096        },
1097        attributes: fold_attrs(folder, arena, &t.attributes),
1098        doc_comment: t.doc_comment.as_ref().map(fold_comment),
1099    }
1100}
1101
1102fn fold_enum_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1103    folder: &mut F,
1104    arena: &'new Bump,
1105    e: &EnumDecl<'_, 'src>,
1106) -> EnumDecl<'new, 'src> {
1107    let mut members = ArenaVec::with_capacity_in(e.body.members.len(), arena);
1108    for member in e.body.members.iter() {
1109        members.push(folder.fold_enum_member(arena, member));
1110    }
1111    EnumDecl {
1112        name: e.name,
1113        scalar_type: e.scalar_type.as_ref().map(|n| folder.fold_name(arena, n)),
1114        implements: {
1115            let mut v = ArenaVec::with_capacity_in(e.implements.len(), arena);
1116            for n in e.implements.iter() {
1117                v.push(folder.fold_name(arena, n));
1118            }
1119            v
1120        },
1121        body: EnumBody {
1122            members,
1123            span: e.body.span,
1124        },
1125        attributes: fold_attrs(folder, arena, &e.attributes),
1126        doc_comment: e.doc_comment.as_ref().map(fold_comment),
1127    }
1128}
1129
1130// =============================================================================
1131// Private helpers — collection folding
1132// =============================================================================
1133
1134fn fold_stmts<'new, 'src, F: Fold<'src> + ?Sized>(
1135    folder: &mut F,
1136    arena: &'new Bump,
1137    stmts: &[Stmt<'_, 'src>],
1138) -> ArenaVec<'new, Stmt<'new, 'src>> {
1139    let mut vec = ArenaVec::with_capacity_in(stmts.len(), arena);
1140    for stmt in stmts {
1141        vec.push(folder.fold_stmt(arena, stmt));
1142    }
1143    vec
1144}
1145
1146pub fn fold_block<'new, 'src, F: Fold<'src> + ?Sized>(
1147    folder: &mut F,
1148    arena: &'new Bump,
1149    block: &Block<'_, 'src>,
1150) -> Block<'new, 'src> {
1151    Block {
1152        stmts: fold_stmts(folder, arena, &block.stmts),
1153        span: block.span,
1154    }
1155}
1156
1157fn fold_exprs<'new, 'src, F: Fold<'src> + ?Sized>(
1158    folder: &mut F,
1159    arena: &'new Bump,
1160    exprs: &[Expr<'_, 'src>],
1161) -> ArenaVec<'new, Expr<'new, 'src>> {
1162    let mut vec = ArenaVec::with_capacity_in(exprs.len(), arena);
1163    for expr in exprs {
1164        vec.push(folder.fold_expr(arena, expr));
1165    }
1166    vec
1167}
1168
1169fn fold_args<'new, 'src, F: Fold<'src> + ?Sized>(
1170    folder: &mut F,
1171    arena: &'new Bump,
1172    args: &[Arg<'_, 'src>],
1173) -> ArenaVec<'new, Arg<'new, 'src>> {
1174    let mut vec = ArenaVec::with_capacity_in(args.len(), arena);
1175    for arg in args {
1176        vec.push(folder.fold_arg(arena, arg));
1177    }
1178    vec
1179}
1180
1181fn fold_params<'new, 'src, F: Fold<'src> + ?Sized>(
1182    folder: &mut F,
1183    arena: &'new Bump,
1184    params: &[Param<'_, 'src>],
1185) -> ArenaVec<'new, Param<'new, 'src>> {
1186    let mut vec = ArenaVec::with_capacity_in(params.len(), arena);
1187    for param in params {
1188        vec.push(folder.fold_param(arena, param));
1189    }
1190    vec
1191}
1192
1193fn fold_attrs<'new, 'src, F: Fold<'src> + ?Sized>(
1194    folder: &mut F,
1195    arena: &'new Bump,
1196    attrs: &[Attribute<'_, 'src>],
1197) -> ArenaVec<'new, Attribute<'new, 'src>> {
1198    let mut vec = ArenaVec::with_capacity_in(attrs.len(), arena);
1199    for attr in attrs {
1200        vec.push(folder.fold_attribute(arena, attr));
1201    }
1202    vec
1203}
1204
1205fn fold_hooks<'new, 'src, F: Fold<'src> + ?Sized>(
1206    folder: &mut F,
1207    arena: &'new Bump,
1208    hooks: &[PropertyHook<'_, 'src>],
1209) -> ArenaVec<'new, PropertyHook<'new, 'src>> {
1210    let mut vec = ArenaVec::with_capacity_in(hooks.len(), arena);
1211    for hook in hooks {
1212        vec.push(folder.fold_property_hook(arena, hook));
1213    }
1214    vec
1215}
1216
1217fn fold_string_parts<'new, 'src, F: Fold<'src> + ?Sized>(
1218    folder: &mut F,
1219    arena: &'new Bump,
1220    parts: &[StringPart<'_, 'src>],
1221) -> ArenaVec<'new, StringPart<'new, 'src>> {
1222    let mut vec = ArenaVec::with_capacity_in(parts.len(), arena);
1223    for part in parts {
1224        vec.push(match part {
1225            StringPart::Literal(s) => StringPart::Literal(arena.alloc_str(s)),
1226            StringPart::Expr(e) => StringPart::Expr(folder.fold_expr(arena, e)),
1227        });
1228    }
1229    vec
1230}
1231
1232// =============================================================================
1233// Private helpers — leaf types
1234// =============================================================================
1235
1236pub fn fold_name_str<'new, 'src>(
1237    name: NameStr<'_, 'src>,
1238    arena: &'new Bump,
1239) -> NameStr<'new, 'src> {
1240    if let Some(s) = name.__into_src_str() {
1241        NameStr::__src(s)
1242    } else {
1243        NameStr::__arena(arena.alloc_str(name.as_str()))
1244    }
1245}
1246
1247fn fold_comment<'src>(comment: &Comment<'src>) -> Comment<'src> {
1248    Comment {
1249        kind: comment.kind,
1250        text: comment.text,
1251        span: comment.span,
1252    }
1253}