Files
hakorune/docs/reference/language/variables-and-scope.md
tomoaki ab76e39036 feat(parser): Phase 285A1.4 & A1.5 - Weak field sugar + Parser hang fix
A1.4: Add sugar syntax `public weak parent` ≡ `public { weak parent }`
A1.5: Fix parser hang on unsupported `param: Type` syntax

Key changes:
- A1.4: Extend visibility parser to handle weak modifier (fields.rs)
- A1.5: Shared helper `parse_param_name_list()` with progress-zero detection
- A1.5: Fix 6 vulnerable parameter parsing loops (methods, constructors, functions)
- Tests: Sugar syntax (OK/NG), parser hang (timeout-based)
- Docs: lifecycle.md, EBNF.md, phase-285a1-boxification.md

Additional changes:
- weak() builtin implementation (handlers/weak.rs)
- Leak tracking improvements (leak_tracker.rs)
- Documentation updates (lifecycle, types, memory-finalization, etc.)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2025-12-24 07:44:50 +09:00

3.4 KiB
Raw Blame History

Variables and Scope (Local/Block Semantics)

Status: Stable (Stage3 surface for local), default strong references.

This document defines the variable model used by Hakorune/Nyash and clarifies how locals interact with blocks, memory, and references across VMs (Rust VM, Hakorune VM, LLVM harness).

For the lifecycle/finalization SSOT, see: docs/reference/language/lifecycle.md.

Local Variables

  • Syntax: local name = expr
  • Scope: Blockscoped. The variable is visible from its declaration to the end of the lexical block.
  • Redeclaration: Writing local name = ... inside a nested block creates a new shadowing binding. Writing name = ... without local updates the nearest existing binding in an enclosing scope.
  • Mutability: Locals are mutable unless future keywords specify otherwise (e.g., const).
  • Lifetime: The variable binding is dropped at block end (}); object lifetime/finalization is defined separately in docs/reference/language/lifecycle.md.

Notes:

  • Stage3 gate: Parsing local requires Stage3 to be enabled (NYASH_PARSER_STAGE3=1 or equivalent runner profile).

Assignment Resolution (Enclosing Scope Update)

Assignment to an identifier resolves as follows:

  1. If a local declaration with the same name exists in the current block, update that binding.
  2. Otherwise, search outward through enclosing blocks and update the first found binding.
  3. If no binding exists in any enclosing scope, it is an error (undeclared variable). Declare it with local.

This matches intuitive blockscoped semantics (Lualike), and differs from Python where inner blocks do not create a new scope (function scope), and assignment would create a local unless nonlocal/global is used.

Reference Semantics (Strong/Weak)

  • Default: Locals hold strong references to boxes/collections.
  • Weak references: Use weak(x) (and fields that store WeakRef) to hold a nonowning reference. Weak refs do not keep the object alive; they can be upgraded at use sites (see SSOT: docs/reference/language/lifecycle.md).
  • Typical guidance:
    • Locals and return values: strong references.
    • Object fields that create cycles (child→parent): weak references.

Example (nested block retains object via outer local):

local a = null
{
  local b = new Box(a)
  a = b  // outer binding updated; a and b point to the same object
}
// leaving the block drops `b` (strongcount 1), but `a` still keeps the object alive

Shadowing vs. Updating

  • Shadowing: local x = ... inside a block hides an outer x for the remainder of the inner block. The outer x remains unchanged.
  • Updating: x = ... without local updates the nearest enclosing x binding.

Prefer clarity: avoid accidental shadowing. If you intentionally shadow, consider naming or comments to clarify intent.

Const/Immutability (Future)

  • A separate keyword (e.g., const) can introduce an immutable local. Semantics: same scoping as local, but reassignment is a compile error. This does not affect reference ownership (still strong by default).

CrossVM Consistency

The above semantics are enforced consistently across:

  • Rust VM (MIR interpreter): scope updates propagate to enclosing locals.
  • Hakorune VM/runner: same resolution rules.
  • LLVM harness/EXE: parity tests validate identical exit codes/behavior.

See also: quick/integration smokes scope_assign_vm.sh, vm_llvm_scope_assign.sh.