feat(charka): PERFORM VARYING — el bucle con variable de control

El bucle más usado de COBOL, que antes el parser degradaba a un
PERFORM vacío (un hueco de corrección real). Ahora atraviesa el
pipeline entero como una rebanada vertical.

- IR: PerformControl::Varying { var, from, by, until }.
- Parser: reconoce PERFORM VARYING var FROM x BY y UNTIL cond en
  línea (END-PERFORM) y fuera de línea (PERFORM párrafo VARYING ...).
- Codegen: emite var = from; while !(until) { cuerpo; var += by; }.
- Shadow: el intérprete inicializa la variable, evalúa la condición
  antes de cada vuelta e incrementa al final.
- Corpus: programa nuevo 08-varying (suma 1..10). Verificado: el
  intérprete sombra y el crate compilado por scaffold dan ambos
  SUMA 1 A 10 = 00055 — las dos rutas concuerdan.

Tests: charka-ir 18, charka-codegen 15, charka-shadow 13. fmt +
clippy limpios.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
sergio
2026-05-21 21:33:14 +00:00
parent b052c41e3c
commit 321e6f8e27
12 changed files with 177 additions and 7 deletions
@@ -334,6 +334,22 @@ mod tests {
assert!(out.contains("for _ in 0..3usize {"));
}
#[test]
fn perform_varying_emits_init_loop_and_increment() {
let out = gen("DATA DIVISION.\n\
WORKING-STORAGE SECTION.\n\
01 WS-I PIC 9(2).\n\
01 WS-N PIC 9(3).\n\
PROCEDURE DIVISION.\n\
MAIN.\n\
PERFORM VARYING WS-I FROM 1 BY 1 UNTIL WS-I > 5\n\
ADD 1 TO WS-N\n\
END-PERFORM.\n");
assert!(out.contains("self.ws_i.store(dec(\"1\"));"));
assert!(out.contains("while !((self.ws_i.value()) > (dec(\"5\"))) {"));
assert!(out.contains("self.ws_i.store(self.ws_i.value().add(&(dec(\"1\"))));"));
}
#[test]
fn empty_program_still_compiles_shape() {
let out = gen("");