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 | |
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.
-rw-r--r-- | match.c | 33 | ||||
-rw-r--r-- | match.h | 2 | ||||
-rw-r--r-- | txr.1 | 18 |
3 files changed, 47 insertions, 6 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); @@ -27,7 +27,7 @@ extern val text_s, choose_s, gather_s, do_s, mdo_s, require_s, in_package_s; extern val close_s, load_s, include_s, mod_s, modlast_s, line_s; -extern val counter_k, vars_k, env_k, var_k, into_k, named_k; +extern val counter_k, vars_k, lists_k, env_k, var_k, into_k, named_k; val match_expand_keyword_args(val elem); val match_expand_elem(val elem); val match_filter(val name, val arg, val other_args); @@ -5526,6 +5526,24 @@ The behavior of the keyword is specified in the following section, "Specifying variables in .codn collect \(dq. +.meIP :lists <> ( variable *) +The +.code :lists +keyword indicates a list of variables. After the +.code collect +terminates, each +.meta variable +in the list which does not have a binding is bound to the empty +list symbol +.codn nil . +Unlike +.code :vars +the +.code :lists +mechanism doesn't assert that only the listed variables may emanate +from the collect. It also doesn't assert that each iteration of the +collect must bind each of those variables. + .meIP :counter >> { variable | >> ( variable << starting-value )} The .code :counter |