summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-15 07:47:29 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-15 07:47:29 -0700
commit5a584a973b632f28edcc0312d92e4a38ee567d8c (patch)
treebe339f33937f3a883172939d7e82b35233421d43 /eval.c
parent561428042eeb26eb0d5c05c3985bd4c8ecdcec1d (diff)
downloadtxr-5a584a973b632f28edcc0312d92e4a38ee567d8c.tar.gz
txr-5a584a973b632f28edcc0312d92e4a38ee567d8c.tar.bz2
txr-5a584a973b632f28edcc0312d92e4a38ee567d8c.zip
defsymacro: regression: don't expand replacement.
This is a regression that was introduced in 191. The change in 191 was trying to prevent defsymacro from being expanded immediately by the expander except in 190 compatibility. Unfortunately, this caused the whole defsymacro block not to be entered unless in 190 compatibility, otherwise taking the common exit which returns form_ex, containing the expanded replacement form. * eval.c (do_expand): Split up implementation of defvarl and defsymacro. In the defsymacro block, do not do any expanding on entry. Absent of compatibility mode, we just do some sanity checks and pass the entire form through. In 262 compatibility, we do the expansion to obtain form_ex. Then all the previous compat logic is wrapped in that block. * tests/011/macros-3.tl: Add a test case which confirms that symbol macros are lazily expanded. Weakness in the test suite is how these regressions creep in. * txr.1: Improve defsymacro documentation, spelling out clearly that the unexpanded replacement form is associated with the symbol. Eliminate obsolescent text suggesting that defsymacro is evaluated at macro time.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c41
1 files changed, 29 insertions, 12 deletions
diff --git a/eval.c b/eval.c
index 2ad82dec..50d20fad 100644
--- a/eval.c
+++ b/eval.c
@@ -4749,32 +4749,49 @@ again:
if (pairs == pairs_ex)
return form;
return rlcp(cons(cond_s, pairs_ex), form);
- } else if (sym == defvarl_s || sym == defsymacro_s) {
+ } else if (sym == defvarl_s) {
val name = second(form);
val init = third(form);
val init_ex = expand(init, menv);
val form_ex = form;
- if (sym == defsymacro_s && length(form) != three)
- eval_error(form, lit("~s: two arguments expected"), sym, nao);
-
if (!bindable(name))
not_bindable_error(form, name);
- if (sym == defvarl_s)
- uw_register_tentative_def(cons(var_s, name));
+ uw_register_tentative_def(cons(var_s, name));
if (init != init_ex)
form_ex = rlcp(cons(sym, cons(name, cons(init_ex, nil))), form);
- if (opt_compat && opt_compat <= 190 && sym == defsymacro_s) {
- val result = eval(if3(opt_compat && opt_compat <= 137,
- form_ex, form),
- make_env(nil, nil, nil), form);
- return cons(quote_s, cons(result, nil));
+ return form_ex;
+ } else if (sym == defsymacro_s) {
+ val name = second(form);
+ val init = third(form);
+
+ if (length(form) != three)
+ eval_error(form, lit("~s: two arguments expected"), sym, nao);
+
+ if (!bindable(name))
+ not_bindable_error(form, name);
+
+ if (opt_compat && opt_compat <= 262) {
+ val init_ex = expand(init, menv);
+ val form_ex = form;
+
+ if (init != init_ex)
+ form_ex = rlcp(cons(sym, cons(name, cons(init_ex, nil))), form);
+
+ if (opt_compat <= 190 && sym == defsymacro_s) {
+ val result = eval(if3(opt_compat && opt_compat <= 137,
+ form_ex, form),
+ make_env(nil, nil, nil), form);
+ return cons(quote_s, cons(result, nil));
+ }
+
+ return form_ex;
}
- return form_ex;
+ return form;
} else if (sym == lambda_s) {
if (!cdr(form))
eval_error(form, lit("~s: missing argument list"), sym, nao);