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        doc_comment: stmt.doc_comment.map(|c| &*arena.alloc(fold_comment(c))),
415    }
416}
417
418pub fn fold_expr<'new, 'src, F: Fold<'src> + ?Sized>(
419    folder: &mut F,
420    arena: &'new Bump,
421    expr: &Expr<'_, 'src>,
422) -> Expr<'new, 'src> {
423    let kind = match &expr.kind {
424        ExprKind::Int(n) => ExprKind::Int(*n),
425        ExprKind::Float(f) => ExprKind::Float(*f),
426        ExprKind::String(s) => ExprKind::String(arena.alloc_str(s)),
427        ExprKind::InterpolatedString(parts) => {
428            ExprKind::InterpolatedString(fold_string_parts(folder, arena, parts))
429        }
430        ExprKind::Heredoc { label, parts } => ExprKind::Heredoc {
431            label,
432            parts: fold_string_parts(folder, arena, parts),
433        },
434        ExprKind::Nowdoc { label, value } => ExprKind::Nowdoc {
435            label,
436            value: arena.alloc_str(value),
437        },
438        ExprKind::ShellExec(parts) => ExprKind::ShellExec(fold_string_parts(folder, arena, parts)),
439        ExprKind::Bool(b) => ExprKind::Bool(*b),
440        ExprKind::Null => ExprKind::Null,
441        ExprKind::Variable(name) => ExprKind::Variable(fold_name_str(*name, arena)),
442        ExprKind::VariableVariable(inner) => {
443            ExprKind::VariableVariable(arena.alloc(folder.fold_expr(arena, inner)))
444        }
445        ExprKind::Identifier(name) => ExprKind::Identifier(fold_name_str(*name, arena)),
446        ExprKind::Assign(assign) => ExprKind::Assign(AssignExpr {
447            target: arena.alloc(folder.fold_expr(arena, assign.target)),
448            op: assign.op,
449            value: arena.alloc(folder.fold_expr(arena, assign.value)),
450            by_ref: assign.by_ref,
451        }),
452        ExprKind::Binary(binary) => ExprKind::Binary(BinaryExpr {
453            left: arena.alloc(folder.fold_expr(arena, binary.left)),
454            op: binary.op,
455            right: arena.alloc(folder.fold_expr(arena, binary.right)),
456        }),
457        ExprKind::UnaryPrefix(u) => ExprKind::UnaryPrefix(UnaryPrefixExpr {
458            op: u.op,
459            operand: arena.alloc(folder.fold_expr(arena, u.operand)),
460        }),
461        ExprKind::UnaryPostfix(u) => ExprKind::UnaryPostfix(UnaryPostfixExpr {
462            operand: arena.alloc(folder.fold_expr(arena, u.operand)),
463            op: u.op,
464        }),
465        ExprKind::Ternary(t) => ExprKind::Ternary(TernaryExpr {
466            condition: arena.alloc(folder.fold_expr(arena, t.condition)),
467            then_expr: t
468                .then_expr
469                .map(|e| &*arena.alloc(folder.fold_expr(arena, e))),
470            else_expr: arena.alloc(folder.fold_expr(arena, t.else_expr)),
471        }),
472        ExprKind::NullCoalesce(nc) => ExprKind::NullCoalesce(NullCoalesceExpr {
473            left: arena.alloc(folder.fold_expr(arena, nc.left)),
474            right: arena.alloc(folder.fold_expr(arena, nc.right)),
475        }),
476        ExprKind::FunctionCall(call) => ExprKind::FunctionCall(FunctionCallExpr {
477            name: arena.alloc(folder.fold_expr(arena, call.name)),
478            args: fold_args(folder, arena, &call.args),
479        }),
480        ExprKind::Array(elements) => {
481            let mut new_elements = ArenaVec::with_capacity_in(elements.len(), arena);
482            for elem in elements.iter() {
483                new_elements.push(ArrayElement {
484                    key: elem.key.as_ref().map(|k| folder.fold_expr(arena, k)),
485                    value: folder.fold_expr(arena, &elem.value),
486                    unpack: elem.unpack,
487                    by_ref: elem.by_ref,
488                    span: elem.span,
489                });
490            }
491            ExprKind::Array(new_elements)
492        }
493        ExprKind::ArrayAccess(access) => ExprKind::ArrayAccess(ArrayAccessExpr {
494            array: arena.alloc(folder.fold_expr(arena, access.array)),
495            index: access
496                .index
497                .map(|i| &*arena.alloc(folder.fold_expr(arena, i))),
498        }),
499        ExprKind::Print(e) => ExprKind::Print(arena.alloc(folder.fold_expr(arena, e))),
500        ExprKind::Parenthesized(e) => {
501            ExprKind::Parenthesized(arena.alloc(folder.fold_expr(arena, e)))
502        }
503        ExprKind::Cast(kind, e) => ExprKind::Cast(*kind, arena.alloc(folder.fold_expr(arena, e))),
504        ExprKind::ErrorSuppress(e) => {
505            ExprKind::ErrorSuppress(arena.alloc(folder.fold_expr(arena, e)))
506        }
507        ExprKind::Isset(exprs) => ExprKind::Isset(fold_exprs(folder, arena, exprs)),
508        ExprKind::Empty(e) => ExprKind::Empty(arena.alloc(folder.fold_expr(arena, e))),
509        ExprKind::Include(kind, e) => {
510            ExprKind::Include(*kind, arena.alloc(folder.fold_expr(arena, e)))
511        }
512        ExprKind::Eval(e) => ExprKind::Eval(arena.alloc(folder.fold_expr(arena, e))),
513        ExprKind::Exit(e) => ExprKind::Exit(e.map(|e| &*arena.alloc(folder.fold_expr(arena, e)))),
514        ExprKind::MagicConst(k) => ExprKind::MagicConst(*k),
515        ExprKind::Clone(e) => ExprKind::Clone(arena.alloc(folder.fold_expr(arena, e))),
516        ExprKind::CloneWith(obj, overrides) => ExprKind::CloneWith(
517            arena.alloc(folder.fold_expr(arena, obj)),
518            arena.alloc(folder.fold_expr(arena, overrides)),
519        ),
520        ExprKind::New(new_expr) => ExprKind::New(NewExpr {
521            class: arena.alloc(folder.fold_expr(arena, new_expr.class)),
522            args: fold_args(folder, arena, &new_expr.args),
523        }),
524        ExprKind::PropertyAccess(access) => ExprKind::PropertyAccess(PropertyAccessExpr {
525            object: arena.alloc(folder.fold_expr(arena, access.object)),
526            property: arena.alloc(folder.fold_expr(arena, access.property)),
527        }),
528        ExprKind::NullsafePropertyAccess(access) => {
529            ExprKind::NullsafePropertyAccess(PropertyAccessExpr {
530                object: arena.alloc(folder.fold_expr(arena, access.object)),
531                property: arena.alloc(folder.fold_expr(arena, access.property)),
532            })
533        }
534        ExprKind::MethodCall(call) => ExprKind::MethodCall(arena.alloc(MethodCallExpr {
535            object: arena.alloc(folder.fold_expr(arena, call.object)),
536            method: arena.alloc(folder.fold_expr(arena, call.method)),
537            args: fold_args(folder, arena, &call.args),
538        })),
539        ExprKind::NullsafeMethodCall(call) => {
540            ExprKind::NullsafeMethodCall(arena.alloc(MethodCallExpr {
541                object: arena.alloc(folder.fold_expr(arena, call.object)),
542                method: arena.alloc(folder.fold_expr(arena, call.method)),
543                args: fold_args(folder, arena, &call.args),
544            }))
545        }
546        ExprKind::StaticPropertyAccess(access) => {
547            ExprKind::StaticPropertyAccess(StaticAccessExpr {
548                class: arena.alloc(folder.fold_expr(arena, access.class)),
549                member: arena.alloc(folder.fold_expr(arena, access.member)),
550            })
551        }
552        ExprKind::StaticMethodCall(call) => {
553            ExprKind::StaticMethodCall(arena.alloc(StaticMethodCallExpr {
554                class: arena.alloc(folder.fold_expr(arena, call.class)),
555                method: arena.alloc(folder.fold_expr(arena, call.method)),
556                args: fold_args(folder, arena, &call.args),
557            }))
558        }
559        ExprKind::StaticDynMethodCall(call) => {
560            ExprKind::StaticDynMethodCall(arena.alloc(StaticDynMethodCallExpr {
561                class: arena.alloc(folder.fold_expr(arena, call.class)),
562                method: arena.alloc(folder.fold_expr(arena, call.method)),
563                args: fold_args(folder, arena, &call.args),
564            }))
565        }
566        ExprKind::ClassConstAccess(access) => ExprKind::ClassConstAccess(StaticAccessExpr {
567            class: arena.alloc(folder.fold_expr(arena, access.class)),
568            member: arena.alloc(folder.fold_expr(arena, access.member)),
569        }),
570        ExprKind::ClassConstAccessDynamic { class, member } => ExprKind::ClassConstAccessDynamic {
571            class: arena.alloc(folder.fold_expr(arena, class)),
572            member: arena.alloc(folder.fold_expr(arena, member)),
573        },
574        ExprKind::StaticPropertyAccessDynamic { class, member } => {
575            ExprKind::StaticPropertyAccessDynamic {
576                class: arena.alloc(folder.fold_expr(arena, class)),
577                member: arena.alloc(folder.fold_expr(arena, member)),
578            }
579        }
580        ExprKind::Closure(closure) => {
581            let mut use_vars = ArenaVec::with_capacity_in(closure.use_vars.len(), arena);
582            for var in closure.use_vars.iter() {
583                use_vars.push(folder.fold_closure_use_var(var));
584            }
585            let new_closure = arena.alloc(ClosureExpr {
586                is_static: closure.is_static,
587                by_ref: closure.by_ref,
588                params: fold_params(folder, arena, &closure.params),
589                use_vars,
590                return_type: closure
591                    .return_type
592                    .as_ref()
593                    .map(|t| folder.fold_type_hint(arena, t)),
594                body: arena.alloc(folder.fold_block(arena, closure.body)),
595                attributes: fold_attrs(folder, arena, &closure.attributes),
596            });
597            ExprKind::Closure(new_closure)
598        }
599        ExprKind::ArrowFunction(arrow) => {
600            let new_arrow = arena.alloc(ArrowFunctionExpr {
601                is_static: arrow.is_static,
602                by_ref: arrow.by_ref,
603                params: fold_params(folder, arena, &arrow.params),
604                return_type: arrow
605                    .return_type
606                    .as_ref()
607                    .map(|t| folder.fold_type_hint(arena, t)),
608                body: arena.alloc(folder.fold_expr(arena, arrow.body)),
609                attributes: fold_attrs(folder, arena, &arrow.attributes),
610            });
611            ExprKind::ArrowFunction(new_arrow)
612        }
613        ExprKind::Match(match_expr) => ExprKind::Match(MatchExpr {
614            subject: arena.alloc(folder.fold_expr(arena, match_expr.subject)),
615            arms: {
616                let mut arms = ArenaVec::with_capacity_in(match_expr.arms.len(), arena);
617                for arm in match_expr.arms.iter() {
618                    arms.push(folder.fold_match_arm(arena, arm));
619                }
620                arms
621            },
622            brace_start: match_expr.brace_start,
623        }),
624        ExprKind::ThrowExpr(e) => ExprKind::ThrowExpr(arena.alloc(folder.fold_expr(arena, e))),
625        ExprKind::Yield(y) => ExprKind::Yield(YieldExpr {
626            key: y.key.map(|k| &*arena.alloc(folder.fold_expr(arena, k))),
627            value: y.value.map(|v| &*arena.alloc(folder.fold_expr(arena, v))),
628            is_from: y.is_from,
629        }),
630        ExprKind::AnonymousClass(class) => {
631            ExprKind::AnonymousClass(arena.alloc(fold_class_decl(folder, arena, class)))
632        }
633        ExprKind::CallableCreate(cc) => {
634            let kind = match &cc.kind {
635                CallableCreateKind::Function(name) => {
636                    CallableCreateKind::Function(arena.alloc(folder.fold_expr(arena, name)))
637                }
638                CallableCreateKind::Method { object, method } => CallableCreateKind::Method {
639                    object: arena.alloc(folder.fold_expr(arena, object)),
640                    method: arena.alloc(folder.fold_expr(arena, method)),
641                },
642                CallableCreateKind::NullsafeMethod { object, method } => {
643                    CallableCreateKind::NullsafeMethod {
644                        object: arena.alloc(folder.fold_expr(arena, object)),
645                        method: arena.alloc(folder.fold_expr(arena, method)),
646                    }
647                }
648                CallableCreateKind::StaticMethod { class, method } => {
649                    CallableCreateKind::StaticMethod {
650                        class: arena.alloc(folder.fold_expr(arena, class)),
651                        method: arena.alloc(folder.fold_expr(arena, method)),
652                    }
653                }
654            };
655            ExprKind::CallableCreate(CallableCreateExpr { kind })
656        }
657        ExprKind::Omit => ExprKind::Omit,
658        ExprKind::Error => ExprKind::Error,
659    };
660    Expr {
661        kind,
662        span: expr.span,
663    }
664}
665
666pub fn fold_param<'new, 'src, F: Fold<'src> + ?Sized>(
667    folder: &mut F,
668    arena: &'new Bump,
669    param: &Param<'_, 'src>,
670) -> Param<'new, 'src> {
671    Param {
672        name: param.name,
673        type_hint: param
674            .type_hint
675            .as_ref()
676            .map(|t| folder.fold_type_hint(arena, t)),
677        default: param.default.as_ref().map(|d| folder.fold_expr(arena, d)),
678        by_ref: param.by_ref,
679        variadic: param.variadic,
680        is_readonly: param.is_readonly,
681        is_final: param.is_final,
682        visibility: param.visibility,
683        set_visibility: param.set_visibility,
684        attributes: fold_attrs(folder, arena, &param.attributes),
685        hooks: fold_hooks(folder, arena, &param.hooks),
686        span: param.span,
687    }
688}
689
690pub fn fold_arg<'new, 'src, F: Fold<'src> + ?Sized>(
691    folder: &mut F,
692    arena: &'new Bump,
693    arg: &Arg<'_, 'src>,
694) -> Arg<'new, 'src> {
695    Arg {
696        name: arg.name.as_ref().map(|n| folder.fold_name(arena, n)),
697        value: folder.fold_expr(arena, &arg.value),
698        unpack: arg.unpack,
699        by_ref: arg.by_ref,
700        span: arg.span,
701    }
702}
703
704pub fn fold_class_member<'new, 'src, F: Fold<'src> + ?Sized>(
705    folder: &mut F,
706    arena: &'new Bump,
707    member: &ClassMember<'_, 'src>,
708) -> ClassMember<'new, 'src> {
709    let kind = match &member.kind {
710        ClassMemberKind::Property(prop) => {
711            ClassMemberKind::Property(fold_property_decl(folder, arena, prop))
712        }
713        ClassMemberKind::Method(method) => {
714            ClassMemberKind::Method(fold_method_decl(folder, arena, method))
715        }
716        ClassMemberKind::ClassConst(cc) => {
717            ClassMemberKind::ClassConst(fold_class_const_decl(folder, arena, cc))
718        }
719        ClassMemberKind::TraitUse(tu) => {
720            ClassMemberKind::TraitUse(folder.fold_trait_use(arena, tu))
721        }
722    };
723    ClassMember {
724        kind,
725        span: member.span,
726    }
727}
728
729pub fn fold_enum_member<'new, 'src, F: Fold<'src> + ?Sized>(
730    folder: &mut F,
731    arena: &'new Bump,
732    member: &EnumMember<'_, 'src>,
733) -> EnumMember<'new, 'src> {
734    let kind = match &member.kind {
735        EnumMemberKind::Case(case) => EnumMemberKind::Case(EnumCase {
736            name: case.name,
737            value: case.value.as_ref().map(|v| folder.fold_expr(arena, v)),
738            attributes: fold_attrs(folder, arena, &case.attributes),
739            doc_comment: case.doc_comment.as_ref().map(fold_comment),
740        }),
741        EnumMemberKind::Method(method) => {
742            EnumMemberKind::Method(fold_method_decl(folder, arena, method))
743        }
744        EnumMemberKind::ClassConst(cc) => {
745            EnumMemberKind::ClassConst(fold_class_const_decl(folder, arena, cc))
746        }
747        EnumMemberKind::TraitUse(tu) => EnumMemberKind::TraitUse(folder.fold_trait_use(arena, tu)),
748    };
749    EnumMember {
750        kind,
751        span: member.span,
752    }
753}
754
755pub fn fold_property_hook<'new, 'src, F: Fold<'src> + ?Sized>(
756    folder: &mut F,
757    arena: &'new Bump,
758    hook: &PropertyHook<'_, 'src>,
759) -> PropertyHook<'new, 'src> {
760    let body = match &hook.body {
761        PropertyHookBody::Block(block) => {
762            PropertyHookBody::Block(arena.alloc(fold_block(folder, arena, block)))
763        }
764        PropertyHookBody::Expression(expr) => {
765            PropertyHookBody::Expression(folder.fold_expr(arena, expr))
766        }
767        PropertyHookBody::Abstract => PropertyHookBody::Abstract,
768    };
769    PropertyHook {
770        kind: hook.kind,
771        body,
772        is_final: hook.is_final,
773        by_ref: hook.by_ref,
774        params: fold_params(folder, arena, &hook.params),
775        attributes: fold_attrs(folder, arena, &hook.attributes),
776        span: hook.span,
777    }
778}
779
780pub fn fold_type_hint<'new, 'src, F: Fold<'src> + ?Sized>(
781    folder: &mut F,
782    arena: &'new Bump,
783    type_hint: &TypeHint<'_, 'src>,
784) -> TypeHint<'new, 'src> {
785    let kind = match &type_hint.kind {
786        TypeHintKind::Named(name) => TypeHintKind::Named(folder.fold_name(arena, name)),
787        TypeHintKind::Keyword(builtin, span) => TypeHintKind::Keyword(*builtin, *span),
788        TypeHintKind::Nullable(inner) => {
789            TypeHintKind::Nullable(arena.alloc(folder.fold_type_hint(arena, inner)))
790        }
791        TypeHintKind::Union(types) => {
792            let mut new_types = ArenaVec::with_capacity_in(types.len(), arena);
793            for t in types.iter() {
794                new_types.push(folder.fold_type_hint(arena, t));
795            }
796            TypeHintKind::Union(new_types)
797        }
798        TypeHintKind::Intersection(types) => {
799            let mut new_types = ArenaVec::with_capacity_in(types.len(), arena);
800            for t in types.iter() {
801                new_types.push(folder.fold_type_hint(arena, t));
802            }
803            TypeHintKind::Intersection(new_types)
804        }
805    };
806    TypeHint {
807        kind,
808        span: type_hint.span,
809    }
810}
811
812pub fn fold_attribute<'new, 'src, F: Fold<'src> + ?Sized>(
813    folder: &mut F,
814    arena: &'new Bump,
815    attribute: &Attribute<'_, 'src>,
816) -> Attribute<'new, 'src> {
817    Attribute {
818        name: folder.fold_name(arena, &attribute.name),
819        args: fold_args(folder, arena, &attribute.args),
820        span: attribute.span,
821    }
822}
823
824pub fn fold_catch_clause<'new, 'src, F: Fold<'src> + ?Sized>(
825    folder: &mut F,
826    arena: &'new Bump,
827    catch: &CatchClause<'_, 'src>,
828) -> CatchClause<'new, 'src> {
829    let mut types = ArenaVec::with_capacity_in(catch.types.len(), arena);
830    for ty in catch.types.iter() {
831        types.push(folder.fold_name(arena, ty));
832    }
833    CatchClause {
834        types,
835        var: catch.var,
836        body: arena.alloc(folder.fold_block(arena, catch.body)),
837        span: catch.span,
838    }
839}
840
841pub fn fold_match_arm<'new, 'src, F: Fold<'src> + ?Sized>(
842    folder: &mut F,
843    arena: &'new Bump,
844    arm: &MatchArm<'_, 'src>,
845) -> MatchArm<'new, 'src> {
846    let conditions = arm.conditions.as_ref().map(|conds| {
847        let mut new_conds = ArenaVec::with_capacity_in(conds.len(), arena);
848        for c in conds.iter() {
849            new_conds.push(folder.fold_expr(arena, c));
850        }
851        new_conds
852    });
853    MatchArm {
854        conditions,
855        body: folder.fold_expr(arena, &arm.body),
856        span: arm.span,
857    }
858}
859
860pub fn fold_trait_use<'new, 'src, F: Fold<'src> + ?Sized>(
861    folder: &mut F,
862    arena: &'new Bump,
863    trait_use: &TraitUseDecl<'_, 'src>,
864) -> TraitUseDecl<'new, 'src> {
865    let mut traits = ArenaVec::with_capacity_in(trait_use.traits.len(), arena);
866    for t in trait_use.traits.iter() {
867        traits.push(folder.fold_name(arena, t));
868    }
869    let mut adaptations = ArenaVec::with_capacity_in(trait_use.adaptations.len(), arena);
870    for a in trait_use.adaptations.iter() {
871        adaptations.push(folder.fold_trait_adaptation(arena, a));
872    }
873    TraitUseDecl {
874        traits,
875        adaptations,
876        adaptations_brace_start: trait_use.adaptations_brace_start,
877    }
878}
879
880pub fn fold_trait_adaptation<'new, 'src, F: Fold<'src> + ?Sized>(
881    folder: &mut F,
882    arena: &'new Bump,
883    adaptation: &TraitAdaptation<'_, 'src>,
884) -> TraitAdaptation<'new, 'src> {
885    let kind = match &adaptation.kind {
886        TraitAdaptationKind::Precedence {
887            trait_name,
888            method,
889            insteadof,
890        } => {
891            let mut new_insteadof = ArenaVec::with_capacity_in(insteadof.len(), arena);
892            for n in insteadof.iter() {
893                new_insteadof.push(folder.fold_name(arena, n));
894            }
895            TraitAdaptationKind::Precedence {
896                trait_name: folder.fold_name(arena, trait_name),
897                method: folder.fold_name(arena, method),
898                insteadof: new_insteadof,
899            }
900        }
901        TraitAdaptationKind::Alias {
902            trait_name,
903            method,
904            new_modifier,
905            new_name,
906        } => TraitAdaptationKind::Alias {
907            trait_name: trait_name.as_ref().map(|n| folder.fold_name(arena, n)),
908            method: folder.fold_name(arena, method),
909            new_modifier: *new_modifier,
910            new_name: new_name.as_ref().map(|n| folder.fold_name(arena, n)),
911        },
912    };
913    TraitAdaptation {
914        kind,
915        span: adaptation.span,
916    }
917}
918
919pub fn fold_name<'new, 'src, F: Fold<'src> + ?Sized>(
920    _folder: &mut F,
921    arena: &'new Bump,
922    name: &Name<'_, 'src>,
923) -> Name<'new, 'src> {
924    match name {
925        Name::Simple { value, span } => Name::Simple { value, span: *span },
926        Name::Complex { parts, kind, span } => {
927            let mut new_parts = ArenaVec::with_capacity_in(parts.len(), arena);
928            for &part in parts.iter() {
929                new_parts.push(part);
930            }
931            Name::Complex {
932                parts: new_parts,
933                kind: *kind,
934                span: *span,
935            }
936        }
937        Name::Error { span } => Name::Error { span: *span },
938    }
939}
940
941// =============================================================================
942// Private helpers — complex declaration types
943// =============================================================================
944
945fn fold_function_decl<'new, 'src, F: Fold<'src> + ?Sized>(
946    folder: &mut F,
947    arena: &'new Bump,
948    func: &FunctionDecl<'_, 'src>,
949) -> FunctionDecl<'new, 'src> {
950    FunctionDecl {
951        name: func.name,
952        params: fold_params(folder, arena, &func.params),
953        body: arena.alloc(folder.fold_block(arena, func.body)),
954        return_type: func
955            .return_type
956            .as_ref()
957            .map(|t| folder.fold_type_hint(arena, t)),
958        by_ref: func.by_ref,
959        attributes: fold_attrs(folder, arena, &func.attributes),
960        doc_comment: func.doc_comment.as_ref().map(fold_comment),
961    }
962}
963
964fn fold_method_decl<'new, 'src, F: Fold<'src> + ?Sized>(
965    folder: &mut F,
966    arena: &'new Bump,
967    method: &MethodDecl<'_, 'src>,
968) -> MethodDecl<'new, 'src> {
969    MethodDecl {
970        name: method.name,
971        visibility: method.visibility,
972        is_static: method.is_static,
973        is_abstract: method.is_abstract,
974        is_final: method.is_final,
975        by_ref: method.by_ref,
976        params: fold_params(folder, arena, &method.params),
977        return_type: method
978            .return_type
979            .as_ref()
980            .map(|t| folder.fold_type_hint(arena, t)),
981        body: method
982            .body
983            .map(|b| &*arena.alloc(folder.fold_block(arena, b))),
984        attributes: fold_attrs(folder, arena, &method.attributes),
985        doc_comment: method.doc_comment.as_ref().map(fold_comment),
986    }
987}
988
989fn fold_property_decl<'new, 'src, F: Fold<'src> + ?Sized>(
990    folder: &mut F,
991    arena: &'new Bump,
992    prop: &PropertyDecl<'_, 'src>,
993) -> PropertyDecl<'new, 'src> {
994    PropertyDecl {
995        name: prop.name,
996        visibility: prop.visibility,
997        set_visibility: prop.set_visibility,
998        is_static: prop.is_static,
999        is_readonly: prop.is_readonly,
1000        type_hint: prop
1001            .type_hint
1002            .as_ref()
1003            .map(|t| folder.fold_type_hint(arena, t)),
1004        default: prop.default.as_ref().map(|d| folder.fold_expr(arena, d)),
1005        attributes: fold_attrs(folder, arena, &prop.attributes),
1006        hooks: fold_hooks(folder, arena, &prop.hooks),
1007        doc_comment: prop.doc_comment.as_ref().map(fold_comment),
1008    }
1009}
1010
1011fn fold_class_const_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1012    folder: &mut F,
1013    arena: &'new Bump,
1014    cc: &ClassConstDecl<'_, 'src>,
1015) -> ClassConstDecl<'new, 'src> {
1016    ClassConstDecl {
1017        name: cc.name,
1018        visibility: cc.visibility,
1019        is_final: cc.is_final,
1020        type_hint: cc
1021            .type_hint
1022            .map(|t| &*arena.alloc(folder.fold_type_hint(arena, t))),
1023        value: folder.fold_expr(arena, &cc.value),
1024        attributes: fold_attrs(folder, arena, &cc.attributes),
1025        doc_comment: cc.doc_comment.as_ref().map(fold_comment),
1026    }
1027}
1028
1029fn fold_class_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1030    folder: &mut F,
1031    arena: &'new Bump,
1032    class: &ClassDecl<'_, 'src>,
1033) -> ClassDecl<'new, 'src> {
1034    let mut members = ArenaVec::with_capacity_in(class.body.members.len(), arena);
1035    for member in class.body.members.iter() {
1036        members.push(folder.fold_class_member(arena, member));
1037    }
1038    ClassDecl {
1039        name: class.name,
1040        modifiers: class.modifiers.clone(),
1041        extends: class.extends.as_ref().map(|n| folder.fold_name(arena, n)),
1042        implements: {
1043            let mut v = ArenaVec::with_capacity_in(class.implements.len(), arena);
1044            for n in class.implements.iter() {
1045                v.push(folder.fold_name(arena, n));
1046            }
1047            v
1048        },
1049        body: ClassBody {
1050            members,
1051            span: class.body.span,
1052        },
1053        attributes: fold_attrs(folder, arena, &class.attributes),
1054        doc_comment: class.doc_comment.as_ref().map(fold_comment),
1055    }
1056}
1057
1058fn fold_interface_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1059    folder: &mut F,
1060    arena: &'new Bump,
1061    iface: &InterfaceDecl<'_, 'src>,
1062) -> InterfaceDecl<'new, 'src> {
1063    let mut extends = ArenaVec::with_capacity_in(iface.extends.len(), arena);
1064    for n in iface.extends.iter() {
1065        extends.push(folder.fold_name(arena, n));
1066    }
1067    let mut members = ArenaVec::with_capacity_in(iface.body.members.len(), arena);
1068    for member in iface.body.members.iter() {
1069        members.push(folder.fold_class_member(arena, member));
1070    }
1071    InterfaceDecl {
1072        name: iface.name,
1073        extends,
1074        body: ClassBody {
1075            members,
1076            span: iface.body.span,
1077        },
1078        attributes: fold_attrs(folder, arena, &iface.attributes),
1079        doc_comment: iface.doc_comment.as_ref().map(fold_comment),
1080    }
1081}
1082
1083fn fold_trait_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1084    folder: &mut F,
1085    arena: &'new Bump,
1086    t: &TraitDecl<'_, 'src>,
1087) -> TraitDecl<'new, 'src> {
1088    let mut members = ArenaVec::with_capacity_in(t.body.members.len(), arena);
1089    for member in t.body.members.iter() {
1090        members.push(folder.fold_class_member(arena, member));
1091    }
1092    TraitDecl {
1093        name: t.name,
1094        body: ClassBody {
1095            members,
1096            span: t.body.span,
1097        },
1098        attributes: fold_attrs(folder, arena, &t.attributes),
1099        doc_comment: t.doc_comment.as_ref().map(fold_comment),
1100    }
1101}
1102
1103fn fold_enum_decl<'new, 'src, F: Fold<'src> + ?Sized>(
1104    folder: &mut F,
1105    arena: &'new Bump,
1106    e: &EnumDecl<'_, 'src>,
1107) -> EnumDecl<'new, 'src> {
1108    let mut members = ArenaVec::with_capacity_in(e.body.members.len(), arena);
1109    for member in e.body.members.iter() {
1110        members.push(folder.fold_enum_member(arena, member));
1111    }
1112    EnumDecl {
1113        name: e.name,
1114        scalar_type: e.scalar_type.as_ref().map(|n| folder.fold_name(arena, n)),
1115        implements: {
1116            let mut v = ArenaVec::with_capacity_in(e.implements.len(), arena);
1117            for n in e.implements.iter() {
1118                v.push(folder.fold_name(arena, n));
1119            }
1120            v
1121        },
1122        body: EnumBody {
1123            members,
1124            span: e.body.span,
1125        },
1126        attributes: fold_attrs(folder, arena, &e.attributes),
1127        doc_comment: e.doc_comment.as_ref().map(fold_comment),
1128    }
1129}
1130
1131// =============================================================================
1132// Private helpers — collection folding
1133// =============================================================================
1134
1135fn fold_stmts<'new, 'src, F: Fold<'src> + ?Sized>(
1136    folder: &mut F,
1137    arena: &'new Bump,
1138    stmts: &[Stmt<'_, 'src>],
1139) -> ArenaVec<'new, Stmt<'new, 'src>> {
1140    let mut vec = ArenaVec::with_capacity_in(stmts.len(), arena);
1141    for stmt in stmts {
1142        vec.push(folder.fold_stmt(arena, stmt));
1143    }
1144    vec
1145}
1146
1147pub fn fold_block<'new, 'src, F: Fold<'src> + ?Sized>(
1148    folder: &mut F,
1149    arena: &'new Bump,
1150    block: &Block<'_, 'src>,
1151) -> Block<'new, 'src> {
1152    Block {
1153        stmts: fold_stmts(folder, arena, &block.stmts),
1154        span: block.span,
1155    }
1156}
1157
1158fn fold_exprs<'new, 'src, F: Fold<'src> + ?Sized>(
1159    folder: &mut F,
1160    arena: &'new Bump,
1161    exprs: &[Expr<'_, 'src>],
1162) -> ArenaVec<'new, Expr<'new, 'src>> {
1163    let mut vec = ArenaVec::with_capacity_in(exprs.len(), arena);
1164    for expr in exprs {
1165        vec.push(folder.fold_expr(arena, expr));
1166    }
1167    vec
1168}
1169
1170fn fold_args<'new, 'src, F: Fold<'src> + ?Sized>(
1171    folder: &mut F,
1172    arena: &'new Bump,
1173    args: &[Arg<'_, 'src>],
1174) -> ArenaVec<'new, Arg<'new, 'src>> {
1175    let mut vec = ArenaVec::with_capacity_in(args.len(), arena);
1176    for arg in args {
1177        vec.push(folder.fold_arg(arena, arg));
1178    }
1179    vec
1180}
1181
1182fn fold_params<'new, 'src, F: Fold<'src> + ?Sized>(
1183    folder: &mut F,
1184    arena: &'new Bump,
1185    params: &[Param<'_, 'src>],
1186) -> ArenaVec<'new, Param<'new, 'src>> {
1187    let mut vec = ArenaVec::with_capacity_in(params.len(), arena);
1188    for param in params {
1189        vec.push(folder.fold_param(arena, param));
1190    }
1191    vec
1192}
1193
1194fn fold_attrs<'new, 'src, F: Fold<'src> + ?Sized>(
1195    folder: &mut F,
1196    arena: &'new Bump,
1197    attrs: &[Attribute<'_, 'src>],
1198) -> ArenaVec<'new, Attribute<'new, 'src>> {
1199    let mut vec = ArenaVec::with_capacity_in(attrs.len(), arena);
1200    for attr in attrs {
1201        vec.push(folder.fold_attribute(arena, attr));
1202    }
1203    vec
1204}
1205
1206fn fold_hooks<'new, 'src, F: Fold<'src> + ?Sized>(
1207    folder: &mut F,
1208    arena: &'new Bump,
1209    hooks: &[PropertyHook<'_, 'src>],
1210) -> ArenaVec<'new, PropertyHook<'new, 'src>> {
1211    let mut vec = ArenaVec::with_capacity_in(hooks.len(), arena);
1212    for hook in hooks {
1213        vec.push(folder.fold_property_hook(arena, hook));
1214    }
1215    vec
1216}
1217
1218fn fold_string_parts<'new, 'src, F: Fold<'src> + ?Sized>(
1219    folder: &mut F,
1220    arena: &'new Bump,
1221    parts: &[StringPart<'_, 'src>],
1222) -> ArenaVec<'new, StringPart<'new, 'src>> {
1223    let mut vec = ArenaVec::with_capacity_in(parts.len(), arena);
1224    for part in parts {
1225        vec.push(match part {
1226            StringPart::Literal(s) => StringPart::Literal(arena.alloc_str(s)),
1227            StringPart::Expr(e) => StringPart::Expr(folder.fold_expr(arena, e)),
1228        });
1229    }
1230    vec
1231}
1232
1233// =============================================================================
1234// Private helpers — leaf types
1235// =============================================================================
1236
1237pub fn fold_name_str<'new, 'src>(
1238    name: NameStr<'_, 'src>,
1239    arena: &'new Bump,
1240) -> NameStr<'new, 'src> {
1241    if let Some(s) = name.__into_src_str() {
1242        NameStr::__src(s)
1243    } else {
1244        NameStr::__arena(arena.alloc_str(name.as_str()))
1245    }
1246}
1247
1248fn fold_comment<'src>(comment: &Comment<'src>) -> Comment<'src> {
1249    Comment {
1250        kind: comment.kind,
1251        text: comment.text,
1252        span: comment.span,
1253    }
1254}