diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2017-01-21 21:13:33 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2017-01-21 21:13:33 -0800 |
commit | b64bcbb6e7bd7f331f660518e97e2bfa898ae340 (patch) | |
tree | 8b80eb794c7f4777346947fccea8b0faafeb4185 /match.c | |
parent | 9cac065f8374ed008bf8274909b3f431dd9e8da6 (diff) | |
download | txr-b64bcbb6e7bd7f331f660518e97e2bfa898ae340.tar.gz txr-b64bcbb6e7bd7f331f660518e97e2bfa898ae340.tar.bz2 txr-b64bcbb6e7bd7f331f660518e97e2bfa898ae340.zip |
bugfix: expand macros in a number of directives.
This is the last round of changes on this topic, bringing
proper macro expansion to the arguments to @(skip),
@(fuzz), @(next), @(call), @(cat), @(load) and @(close).
* match.c (match_expand_keyword_args): Only process the
keyword arguments if they are followed by an argument.
Process @(next) arguments here too: :list and :string
take a Lisp expression, but :tlist and :var take an
argument which is not a Lisp expression and must be
handled properly. Also, expand any non-keyword expression.
This handles the <source> argument of @(next).
(match_expand_elem): New function.
* match.h (match_expand_elem): Declared.
* parser.h (expand_meta): Declared.
* parser.y (expand_meta): Static function becomes external.
(elem): Expand elem other than require or do using
match_expand_elem. We don't fold require and do into this
because match_expand_elem has a backward compat switch
in it that doesn't apply to these.
Diffstat (limited to 'match.c')
-rw-r--r-- | match.c | 80 |
1 files changed, 72 insertions, 8 deletions
@@ -2894,27 +2894,91 @@ val match_expand_keyword_args(val args) while (consp(args)) { val sym = car(args); - if (sym == maxgap_k || sym == mingap_k || sym == gap_k || sym == times_k || - sym == mintimes_k || sym == maxtimes_k || sym == lines_k || - sym == counter_k || sym == vars_k) + val next = cdr(args); + val more = consp(next); + + if (more && + (sym == maxgap_k || sym == mingap_k || sym == gap_k || + sym == times_k || sym == mintimes_k || sym == maxtimes_k || + sym == lines_k || sym == counter_k || sym == vars_k || + sym == list_k || sym == string_k)) { - val d = cdr(args); - val form = car(d); + val form = car(next); val form_ex = if3(sym == vars_k, match_expand_vars(form), expand(form, nil)); ptail = list_collect(ptail, sym); ptail = list_collect(ptail, form_ex); - args = cdr(d); - } else { + args = cdr(next); + } else if (more && + (sym == tlist_k)) { + ptail = list_collect(ptail, sym); + ptail = list_collect(ptail, expand_meta(car(next), nil)); + args = cdr(next); + } else if (more && + (sym == var_k)) { ptail = list_collect(ptail, sym); - args = cdr(args); + ptail = list_collect(ptail, car(next)); + args = cdr(next); + } else { + ptail = list_collect(ptail, expand(sym, nil)); + args = next; } } return out; } +val match_expand_elem(val elem) +{ + if (atom(elem)) { + return elem; + } else { + val sym = car(elem); + val args = cdr(elem); + + if (opt_compat && opt_compat < 166) { + goto out; + } else if (sym == skip_s || sym == fuzz_s || sym == load_s || + sym == close_s) + { + val args_ex = expand_forms(args, nil); + if (args == args_ex) + return elem; + return rlcp(cons(sym, args_ex), elem); + } else if (sym == call_s) { + if (atom(args)) { + return elem; + } else { + val arg1 = car(args); + val arg1_ex = expand(arg1, nil); + if (arg1 == arg1_ex) + return elem; + return rlcp(cons(sym, cons(arg1_ex, cdr(args))), elem); + } + } else if (sym == cat_s) { + if (atom(args) || atom(cdr(args))) { + return elem; + } else { + val arg1 = car(args); + val arg2 = cadr(args); + val arg2_ex = expand(arg2, nil); + if (arg2 == arg2_ex) + return elem; + return rlcp(cons(sym, cons(arg1, cons(arg2_ex, cddr(args)))), elem); + } + } else if (sym == next_s) { + val args_ex = match_expand_keyword_args(args); + if (args == args_ex) + return elem; + return rlcp(cons(sym, args_ex), elem); + } else { +out: + return rlcp(cons(sym, expand_meta(args, nil)), elem); + } + } +} + static val v_collect(match_files_ctx *c) { spec_bind (specline, first_spec, c->spec); |