diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2017-07-11 22:45:48 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2017-07-11 22:45:48 -0700 |
commit | 974532041c1eaeb55c9fcb4c395e4da5f19a4773 (patch) | |
tree | 6b71ae2d02f9b6fc43b6e34fd301df1d3d30e0c8 /match.c | |
parent | afe3787a368ec6753ed9a69867b943a3238d5796 (diff) | |
download | txr-974532041c1eaeb55c9fcb4c395e4da5f19a4773.tar.gz txr-974532041c1eaeb55c9fcb4c395e4da5f19a4773.tar.bz2 txr-974532041c1eaeb55c9fcb4c395e4da5f19a4773.zip |
New :lists feature in @(collect)/@(coll).
This is for those situations when multiple lists are being
collected, and must all be bound even if empty. Yet, the
lists are independent rather than parallel, so the discipline
of :vars is too rigid.
* match.c (lists_k): New keyword symbol variable.
(h_coll, v_collect): Extract :lists argument as local variable
lists, validate that :lists is not used on @(repeat)/@(rep)
and implement logic.
(match_expand_keyword_args): Treat expansion of :lists
the same way as :vars.
(syms_init): Initialize lists_k.
* match.c (lists_k): Declared.
* txr.1: Documented.
Diffstat (limited to 'match.c')
-rw-r--r-- | match.c | 33 |
1 files changed, 28 insertions, 5 deletions
@@ -62,7 +62,7 @@ val text_s, choose_s, gather_s, do_s, mdo_s, mod_s, modlast_s; val line_s, data_s, fuzz_s, load_s; val include_s, close_s, require_s, in_package_s; val longest_k, shortest_k, greedy_k; -val vars_k, resolve_k; +val vars_k, lists_k, resolve_k; val append_k, into_k, var_k, list_k, tlist_k, string_k, env_k, counter_k; val named_k, continue_k, finish_k, mandatory_k; @@ -935,8 +935,9 @@ static val h_coll(match_line_ctx *c) c->bindings), zero); val counter_binding = if2(counter, cons(counter, nil)); val bindings_with_counter = if2(counter, cons(counter_binding, nil)); - val have_vars; + val have_vars, have_lists; val vars = getplist_f(args, vars_k, mkcloc(have_vars)); + val lists = getplist_f(args, lists_k, mkcloc(have_lists)); cnum cmax = if3(gap, c_num(gap), if3(max, c_num(max), 0)); cnum cmin = if3(gap, c_num(gap), if3(min, c_num(min), 0)); cnum mincounter = cmin, maxcounter = 0; @@ -950,6 +951,9 @@ static val h_coll(match_line_ctx *c) if (have_vars) sem_error(elem, lit("~s: coll takes :vars, rep does not"), op_sym, nao); + if (have_lists) + sem_error(elem, lit("~s: coll takes :lists, rep does not"), + op_sym, nao); have_vars = t; } @@ -1113,6 +1117,13 @@ next_coll: } } + for (iter = lists; iter; iter = cdr(iter)) { + val sym = car(iter); + val exists = tx_lookup_var(sym, c->bindings); + if (!exists) + c->bindings = acons(sym, nil, c->bindings); + } + return next_spec_k; } @@ -3114,11 +3125,11 @@ val match_expand_keyword_args(val args) if (more && (sym == maxgap_k || sym == mingap_k || sym == gap_k || sym == times_k || sym == mintimes_k || sym == maxtimes_k || - sym == lines_k || sym == vars_k || + sym == lines_k || sym == vars_k || sym == lists_k || sym == list_k || sym == string_k)) { val form = car(next); - val form_ex = if3(sym == vars_k, + val form_ex = if3(sym == vars_k || sym == lists_k, match_expand_vars(form), expand(form, nil)); ptail = list_collect(ptail, sym); @@ -3234,8 +3245,9 @@ static val v_collect(match_files_ctx *c) c->bindings), zero); val counter_binding = if2(counter, cons(counter, nil)); val bindings_with_counter = if2(counter, cons(counter_binding, nil)); - val have_vars; + val have_vars, have_lists; volatile val vars = getplist_f(args, vars_k, mkcloc(have_vars)); + val lists = getplist_f(args, lists_k, mkcloc(have_lists)); cnum cmax = if3(gap, c_num(gap), if3(max, c_num(max), 0)); cnum cmin = if3(gap, c_num(gap), if3(min, c_num(min), 0)); cnum mincounter = cmin, maxcounter = 0; @@ -3256,6 +3268,9 @@ static val v_collect(match_files_ctx *c) if (have_vars) sem_error(specline, lit("~s: collect takes :vars, repeat does not"), op_sym, nao); + if (have_lists) + sem_error(specline, lit("~s: collect takes :lists, repeat does not"), + op_sym, nao); have_vars = t; } @@ -3449,6 +3464,13 @@ next_collect: } } + for (iter = lists; iter; iter = cdr(iter)) { + val sym = car(iter); + val exists = tx_lookup_var(sym, c->bindings); + if (!exists) + c->bindings = acons(sym, nil, c->bindings); + } + return next_spec_k; } @@ -4738,6 +4760,7 @@ static void syms_init(void) shortest_k = intern(lit("shortest"), keyword_package); greedy_k = intern(lit("greedy"), keyword_package); vars_k = intern(lit("vars"), keyword_package); + lists_k = intern(lit("lists"), keyword_package); resolve_k = intern(lit("resolve"), keyword_package); append_k = intern(lit("append"), keyword_package); into_k = intern(lit("into"), keyword_package); |