diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2017-02-15 06:57:59 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2017-02-15 06:57:59 -0800 |
commit | e7831ed8f0b67b463be1e01e6c0254fb98d13848 (patch) | |
tree | 8cc88f1bccb454b4f191b6197830dbb0ffea80ab | |
parent | 28020fb827625d41c1252c9e34690114607506ec (diff) | |
download | txr-e7831ed8f0b67b463be1e01e6c0254fb98d13848.tar.gz txr-e7831ed8f0b67b463be1e01e6c0254fb98d13848.tar.bz2 txr-e7831ed8f0b67b463be1e01e6c0254fb98d13848.zip |
Support horizontal @(block), phase 1.
Unresolved issue: horizontal @(accept) terminating in a
vertical @(block) or horizontal @(block) in a different line,
or vertical @(accept) caught in horizontal context.
* match.c (h_block, h_accept_fail): New functions.
(dir_tables_init): Register horizontal @(block),
@(accept) and @(fail).
* parser.y (elem): Support BLOCK syntax.
-rw-r--r-- | match.c | 45 | ||||
-rw-r--r-- | parser.y | 5 |
2 files changed, 50 insertions, 0 deletions
@@ -810,6 +810,48 @@ static val h_skip(match_line_ctx *c) return nil; } +static val h_block(match_line_ctx *c) +{ + val elem = first(c->specline); + val name = second(elem); + val specs = car(third(elem)); + + { + uw_block_begin(name, result); + result = match_line(ml_specline(*c, specs)); + uw_block_end; + if (result) { + cons_bind (bindings, new_pos, result); + c->bindings = bindings; + c->pos = minus(new_pos, c->base); + return next_spec_k; + } + return nil; + } +} + +static val h_accept_fail(match_line_ctx *c) +{ + val elem = first(c->specline); + val sym = first(elem); + val target = second(elem); + + uw_block_return_proto(target, + if2(sym == accept_s, + cons(c->bindings, c->pos)), + sym); + + /* TODO: uw_block_return could just throw this */ + if (target) + sem_error(elem, lit("~a: no block named ~a in scope"), + sym, target, nao); + else + sem_error(elem, lit("~a: no anonymous block in scope"), + sym, nao); + return nil; +} + + static val h_coll(match_line_ctx *c) { val elem = first(c->specline); @@ -4561,6 +4603,9 @@ static void dir_tables_init(void) sethash(h_directive_table, text_s, cptr(coerce(mem_t *, h_text))); sethash(h_directive_table, var_s, cptr(coerce(mem_t *, h_var))); sethash(h_directive_table, skip_s, cptr(coerce(mem_t *, h_skip))); + sethash(h_directive_table, block_s, cptr(coerce(mem_t *, h_block))); + sethash(h_directive_table, accept_s, cptr(coerce(mem_t *, h_accept_fail))); + sethash(h_directive_table, fail_s, cptr(coerce(mem_t *, h_accept_fail))); sethash(h_directive_table, coll_s, cptr(coerce(mem_t *, h_coll))); sethash(h_directive_table, rep_s, cptr(coerce(mem_t *, h_coll))); sethash(h_directive_table, flatten_s, cptr(coerce(mem_t *, hv_trampoline))); @@ -486,10 +486,15 @@ elem : texts { $$ = rlcp(cons(text_s, $1), $1); args, nao); rl($$, num($1)); rl($6, car($5)); } + | BLOCK exprs_opt ')' elems END { $$ = list(block_s, car($2), + cons($4, nil), nao); + rl($$, num($1)); } | COLL error { $$ = nil; yybadtok(yychar, lit("coll clause")); } | REP error { $$ = nil; yybadtok(yychar, lit("rep clause")); } + | BLOCK error { $$ = nil; + yybadtok(yychar, lit("rep clause")); } | ALL clause_parts_h { $$ = rl(list(all_s, t, $2, nao), num($1)); } | SOME exprs_opt ')' clause_parts_h { $$ = rl(list(some_s, t, $4, $2, nao), num($1)); } |