feat(charka): EVALUATE — el case de COBOL

EVALUATE atraviesa el pipeline entero — antes el parser lo guardaba
crudo como Stmt::Unknown.

- IR: Stmt::Evaluate { subject, whens, other } con
  WhenBranch { values, body }. Varios WHEN apilados comparten cuerpo;
  WHEN OTHER es el caso por defecto.
- Parser: EVALUATE subject WHEN v1 WHEN v2 ... [WHEN OTHER ...]
  END-EVALUATE.
- Codegen: lo baja a una cadena if / else if / else — una rama se
  elige si el sujeto es igual a alguno de sus valores, sin caída.
- Shadow: el intérprete evalúa el sujeto y ejecuta la primera rama
  cuyos valores casen, o el WHEN OTHER.
- Corpus: programa nuevo 09-evaluar (EVALUATE por valor anidado en un
  PERFORM VARYING, con WHEN apilados y WHEN OTHER). Verificado: el
  intérprete sombra y el crate compilado por scaffold dan la misma
  salida.

Alcance v1: EVALUATE por igualdad de valor; no la forma EVALUATE TRUE
con condiciones ni los rangos THRU.

Tests: charka-ir 19, charka-codegen 16, charka-shadow 14. fmt +
clippy limpios.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
sergio
2026-05-21 21:37:28 +00:00
parent 321e6f8e27
commit 4df7478b71
13 changed files with 259 additions and 8 deletions
@@ -155,6 +155,15 @@ pub enum Stmt {
then_branch: Vec<Stmt>,
else_branch: Vec<Stmt>,
},
/// `EVALUATE subject WHEN ... [WHEN OTHER ...] END-EVALUATE` — el
/// `case` de COBOL. Una rama se elige si `subject` es igual a
/// alguno de sus valores; sin caída entre ramas.
Evaluate {
subject: Operand,
whens: Vec<WhenBranch>,
/// El cuerpo de `WHEN OTHER` (vacío si no hay).
other: Vec<Stmt>,
},
/// `PERFORM ...` — ver [`Perform`].
Perform(Perform),
/// `GO TO target`
@@ -172,6 +181,14 @@ pub enum Stmt {
Unknown { verb: String, tokens: Vec<Token> },
}
/// Una rama `WHEN` de un `EVALUATE`: los valores que la disparan
/// (varios `WHEN` apilados comparten cuerpo) y el cuerpo a ejecutar.
#[derive(Debug, Clone, PartialEq)]
pub struct WhenBranch {
pub values: Vec<Operand>,
pub body: Vec<Stmt>,
}
/// Un statement `PERFORM`: a quién ejecuta y cuántas veces.
#[derive(Debug, Clone, PartialEq)]
pub struct Perform {