diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2016-04-11 20:26:56 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2016-04-11 20:26:56 -0700 |
commit | 9f36af8a86b7c377f2bfab2cb9fafefa5ed173d4 (patch) | |
tree | 9b25a804307d954b2fa150b317172e5062f229d4 /eval.c | |
parent | 2a441b7edd2c92ac9371ccafccc6e2820eab22a7 (diff) | |
download | txr-9f36af8a86b7c377f2bfab2cb9fafefa5ed173d4.tar.gz txr-9f36af8a86b7c377f2bfab2cb9fafefa5ed173d4.tar.bz2 txr-9f36af8a86b7c377f2bfab2cb9fafefa5ed173d4.zip |
Better handling of dot position function calls.
The expander now actually produces apply forms for dot
position function call and dwim forms. This allows symbol
macros to work naturally.
* eval.c (sys_apply_s): New symbol variable.
(imp_list_to_list, dot_to_apply): New static functions.
(expand_forms, expand_forms_lisp1): We now throw an error if a
non-nil atom terminates a form, Except in compatibility mode
with TXR 137 or less, whereby we emulate the old behavior of
not expanding this atom.
(do_expand): Perform the dot_to_apply transformation
on the arguments of the dwim form.
Perform the dot_to_apply transformation on an
entire function call form.
(eval_init): Initialize sys_apply_s and register
sys:apply function (using the same function object that
is registered under apply).
* txr.1: Documented that both DWIM forms and regular
function call forms work as if by a transformation to apply form,
removing verbiage which separately described the DWIM
handling. Documented that symbol macros work properly in
dot position.
Diffstat (limited to 'eval.c')
-rw-r--r-- | eval.c | 62 |
1 files changed, 50 insertions, 12 deletions
@@ -87,7 +87,7 @@ val for_s, for_star_s, each_s, each_star_s, collect_each_s, collect_each_star_s; val append_each_s, append_each_star_s, while_s, while_star_s, until_star_s; val dohash_s; val uw_protect_s, return_s, return_from_s, sys_abscond_from_s, block_star_s; -val list_s, append_s, apply_s, iapply_s; +val list_s, append_s, apply_s, sys_apply_s, iapply_s; val gen_s, gun_s, generate_s, rest_s, plus_s; val promise_s, promise_forced_s, promise_inprogress_s, force_s; val op_s, ap_s, identity_s, apf_s, ipf_s; @@ -2456,14 +2456,38 @@ static val me_quasilist(val form, val menv) return cons(list_s, cdr(form)); } +static val imp_list_to_list(val list) +{ + list_collect_decl (out, ptail); + + for (; consp(list); list = cdr(list)) + ptail = list_collect(ptail, car(list)); + + list_collect(ptail, list); + return out; +} + +static val dot_to_apply(val form, val lisp1_p) +{ + if ((opt_compat && opt_compat <= 137) || proper_listp(form)) { + return form; + } else { + val sym = car(form); + val args = imp_list_to_list(cdr(form)); + return cons(sys_apply_s, cons(if3(lisp1_p, + sym, + list(fun_s, sym, nao)), + args)); + } +} + val expand_forms(val form, val menv) { if (atom(form)) { - val ex_f = expand(form, menv); - if (consp(ex_f)) - uw_throwf(error_s, lit("symbol macro ~s in dot position must produce " - "atom form, not compound"), form, nao); - return ex_f; + if (!form || (opt_compat && opt_compat <= 137)) + return form; + uw_throwf(error_s, lit("dotted argument ~!~s " + "was not converted to apply form"), form, nao); } else { val f = car(form); val r = cdr(form); @@ -2527,7 +2551,10 @@ tail: static val expand_forms_lisp1(val form, val menv) { if (atom(form)) { - return form; + if (!form || (opt_compat && opt_compat <= 137)) + return form; + uw_throwf(error_s, lit("dotted function call ~!~s " + "was not converted to apply form"), form, nao); } else { val f = car(form); val r = cdr(form); @@ -3513,7 +3540,7 @@ tail: return expand_symacrolet(form, menv); } else if (sym == dwim_s) { val args = rest(form); - val args_ex = expand_forms_lisp1(args, menv); + val args_ex = expand_forms_lisp1(dot_to_apply(args, t), menv); if (args == args_ex) return form; @@ -3547,12 +3574,18 @@ tail: also handles: prog1, call, if, and, or, unwind-protect, return, dwim, set, inc, dec, push, pop, flip, and with-saved-vars. */ - val args = rest(form); + val form_ex = dot_to_apply(form, nil); + val sym_ex = first(form_ex); + val args = rest(form_ex); val args_ex = expand_forms(args, menv); - if (args == args_ex) + if (form_ex == form && args_ex == args) return form; - return rlcp(cons(sym, args_ex), form); + + if (args_ex == args) + return form_ex; + + return rlcp(cons(sym_ex, args_ex), form); } abort(); } @@ -4551,6 +4584,7 @@ void eval_init(void) list_s = intern(lit("list"), user_package); append_s = intern(lit("append"), user_package); apply_s = intern(lit("apply"), user_package); + sys_apply_s = intern(lit("apply"), system_package); iapply_s = intern(lit("iapply"), user_package); gen_s = intern(lit("gen"), user_package); gun_s = intern(lit("gun"), user_package); @@ -4735,7 +4769,11 @@ void eval_init(void) reg_fun(intern(lit("mapdo"), user_package), func_n1v(mapdov)); reg_fun(intern(lit("window-map"), user_package), func_n4(window_map)); reg_fun(intern(lit("window-mappend"), user_package), func_n4(window_mappend)); - reg_fun(apply_s, func_n1v(applyv)); + { + val apply_f = func_n1v(applyv); + reg_fun(apply_s, apply_f); + reg_fun(sys_apply_s, apply_f); + } reg_fun(iapply_s, func_n1v(iapply)); reg_fun(call_s, call_f); reg_fun(intern(lit("reduce-left"), user_package), func_n4o(reduce_left, 2)); |