1use bumpalo::Bump;
68
69use crate::ast::*;
70
71pub 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
200pub 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, ¶m.attributes),
684 hooks: fold_hooks(folder, arena, ¶m.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
940fn 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
1130fn 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
1232pub 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}