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 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, ¶m.attributes),
685 hooks: fold_hooks(folder, arena, ¶m.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
941fn 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
1131fn 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
1233pub 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}