2025-09-16 01:54:00 +09:00
|
|
|
use crate::ast::ASTNode;
|
|
|
|
|
use std::collections::HashSet;
|
|
|
|
|
|
|
|
|
|
/// Collect free variables used in `node` into `used`, excluding names present in `locals`.
|
|
|
|
|
/// `locals` is updated as new local declarations are encountered.
|
2025-09-19 15:11:57 +09:00
|
|
|
#[allow(dead_code)]
|
2025-09-17 07:43:07 +09:00
|
|
|
pub(super) fn collect_free_vars(
|
|
|
|
|
node: &ASTNode,
|
|
|
|
|
used: &mut HashSet<String>,
|
|
|
|
|
locals: &mut HashSet<String>,
|
|
|
|
|
) {
|
2025-09-16 01:54:00 +09:00
|
|
|
match node {
|
|
|
|
|
ASTNode::Variable { name, .. } => {
|
|
|
|
|
if name != "me" && name != "this" && !locals.contains(name) {
|
|
|
|
|
used.insert(name.clone());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ASTNode::Local { variables, .. } => {
|
2025-09-17 07:43:07 +09:00
|
|
|
for v in variables {
|
|
|
|
|
locals.insert(v.clone());
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
|
|
|
|
ASTNode::Assignment { target, value, .. } => {
|
|
|
|
|
collect_free_vars(target, used, locals);
|
|
|
|
|
collect_free_vars(value, used, locals);
|
|
|
|
|
}
|
2025-12-04 13:32:58 +09:00
|
|
|
// Phase 152-A: Grouped assignment expression
|
|
|
|
|
ASTNode::GroupedAssignmentExpr { rhs, .. } => {
|
|
|
|
|
collect_free_vars(rhs, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
ASTNode::BinaryOp { left, right, .. } => {
|
|
|
|
|
collect_free_vars(left, used, locals);
|
|
|
|
|
collect_free_vars(right, used, locals);
|
|
|
|
|
}
|
2025-09-17 07:43:07 +09:00
|
|
|
ASTNode::UnaryOp { operand, .. } => {
|
|
|
|
|
collect_free_vars(operand, used, locals);
|
|
|
|
|
}
|
|
|
|
|
ASTNode::MethodCall {
|
|
|
|
|
object, arguments, ..
|
|
|
|
|
} => {
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(object, used, locals);
|
2025-09-17 07:43:07 +09:00
|
|
|
for a in arguments {
|
|
|
|
|
collect_free_vars(a, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
|
|
|
|
ASTNode::FunctionCall { arguments, .. } => {
|
2025-09-17 07:43:07 +09:00
|
|
|
for a in arguments {
|
|
|
|
|
collect_free_vars(a, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
2025-09-17 07:43:07 +09:00
|
|
|
ASTNode::Call {
|
|
|
|
|
callee, arguments, ..
|
|
|
|
|
} => {
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(callee, used, locals);
|
2025-09-17 07:43:07 +09:00
|
|
|
for a in arguments {
|
|
|
|
|
collect_free_vars(a, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ASTNode::FieldAccess { object, .. } => {
|
|
|
|
|
collect_free_vars(object, used, locals);
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
2025-10-31 20:18:39 +09:00
|
|
|
ASTNode::Index { target, index, .. } => {
|
|
|
|
|
collect_free_vars(target, used, locals);
|
|
|
|
|
collect_free_vars(index, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
ASTNode::New { arguments, .. } => {
|
2025-09-17 07:43:07 +09:00
|
|
|
for a in arguments {
|
|
|
|
|
collect_free_vars(a, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
2025-09-17 07:43:07 +09:00
|
|
|
ASTNode::If {
|
|
|
|
|
condition,
|
|
|
|
|
then_body,
|
|
|
|
|
else_body,
|
|
|
|
|
..
|
|
|
|
|
} => {
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(condition, used, locals);
|
2025-09-17 07:43:07 +09:00
|
|
|
for st in then_body {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
if let Some(eb) = else_body {
|
|
|
|
|
for st in eb {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
2025-09-17 07:43:07 +09:00
|
|
|
ASTNode::Loop {
|
|
|
|
|
condition, body, ..
|
|
|
|
|
} => {
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(condition, used, locals);
|
2025-09-17 07:43:07 +09:00
|
|
|
for st in body {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ASTNode::TryCatch {
|
|
|
|
|
try_body,
|
|
|
|
|
catch_clauses,
|
|
|
|
|
finally_body,
|
|
|
|
|
..
|
|
|
|
|
} => {
|
|
|
|
|
for st in try_body {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
for c in catch_clauses {
|
|
|
|
|
for st in &c.body {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if let Some(fb) = finally_body {
|
|
|
|
|
for st in fb {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ASTNode::Throw { expression, .. } => {
|
|
|
|
|
collect_free_vars(expression, used, locals);
|
|
|
|
|
}
|
|
|
|
|
ASTNode::Print { expression, .. } => {
|
|
|
|
|
collect_free_vars(expression, used, locals);
|
|
|
|
|
}
|
|
|
|
|
ASTNode::Return { value, .. } => {
|
|
|
|
|
if let Some(v) = value {
|
|
|
|
|
collect_free_vars(v, used, locals);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ASTNode::AwaitExpression { expression, .. } => {
|
|
|
|
|
collect_free_vars(expression, used, locals);
|
|
|
|
|
}
|
2025-09-23 09:00:07 +09:00
|
|
|
ASTNode::MatchExpr {
|
2025-09-17 07:43:07 +09:00
|
|
|
scrutinee,
|
|
|
|
|
arms,
|
|
|
|
|
else_expr,
|
|
|
|
|
..
|
|
|
|
|
} => {
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(scrutinee, used, locals);
|
2025-09-17 07:43:07 +09:00
|
|
|
for (_, e) in arms {
|
|
|
|
|
collect_free_vars(e, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
collect_free_vars(else_expr, used, locals);
|
|
|
|
|
}
|
|
|
|
|
ASTNode::Program { statements, .. } => {
|
2025-09-17 07:43:07 +09:00
|
|
|
for st in statements {
|
|
|
|
|
collect_free_vars(st, used, locals);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
|
|
|
|
ASTNode::FunctionDeclaration { params, body, .. } => {
|
|
|
|
|
let mut inner = locals.clone();
|
2025-09-17 07:43:07 +09:00
|
|
|
for p in params {
|
|
|
|
|
inner.insert(p.clone());
|
|
|
|
|
}
|
|
|
|
|
for st in body {
|
|
|
|
|
collect_free_vars(st, used, &mut inner);
|
|
|
|
|
}
|
2025-09-16 01:54:00 +09:00
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
}
|
|
|
|
|
}
|