diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-02-03 10:46:26 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-02-03 10:46:26 -0800 |
commit | b9efd7c864fdfb82f9141fbacfbeb0d963750bef (patch) | |
tree | 41c073d24d1e8633b266c0b2a8ccf18b6befc50e | |
parent | 57f259e3fafc0304aaac74b50581dcd5f4d4e144 (diff) | |
download | txr-b9efd7c864fdfb82f9141fbacfbeb0d963750bef.tar.gz txr-b9efd7c864fdfb82f9141fbacfbeb0d963750bef.tar.bz2 txr-b9efd7c864fdfb82f9141fbacfbeb0d963750bef.zip |
Symbol macros shadowed by functions under dwim.
* eval.c (lookup_symac_lisp1, expand_lisp1,
expand_forms_lisp1): New static functions.
(expand): Handle dwim's arguments via
expand_forms_lisp1, rather than as a generic
compound form.
* txr.1: Revised text in several places to clarify
this expansion rule.
-rw-r--r-- | ChangeLog | 13 | ||||
-rw-r--r-- | eval.c | 68 | ||||
-rw-r--r-- | txr.1 | 64 |
3 files changed, 125 insertions, 20 deletions
@@ -1,3 +1,16 @@ +2015-02-04 Kaz Kylheku <kaz@kylheku.com> + + Symbol macros shadowed by functions under dwim. + + * eval.c (lookup_symac_lisp1, expand_lisp1, + expand_forms_lisp1): New static functions. + (expand): Handle dwim's arguments via + expand_forms_lisp1, rather than as a generic + compound form. + + * txr.1: Revised text in several places to clarify + this expansion rule. + 2015-02-01 Kaz Kylheku <kaz@kylheku.com> Version 103 @@ -275,6 +275,32 @@ static val lookup_symac(val menv, val sym) } } +static val lookup_symac_lisp1(val menv, val sym) +{ + if (nilp(menv)) { + return gethash(top_smb, sym); + } else { + type_check(menv, ENV); + + /* Of course, we are not looking for symbol macros in the operator macro + * name space. Rather, the object of the lookup rule implemented by this + * function is to allow lexical function bindings to shadow symbol macros. + */ + { + val vbinding = assoc(sym, menv->e.vbindings); + if (vbinding) { + return (cdr(vbinding) == special_s) ? nil : vbinding; + } else { + val fbinding = assoc(sym, menv->e.fbindings); + if (fbinding && cdr(fbinding) == special_s) + return nil; + } + + return lookup_symac(menv->e.up_env, sym); + } + } +} + static val lexical_var_p(val menv, val sym) { if (nilp(menv)) { @@ -2188,6 +2214,41 @@ val expand_forms(val form, val menv) } } +static val expand_lisp1(val form, val menv) +{ +tail: + if (bindable(form)) { + val symac_bind = lookup_symac_lisp1(menv, form); + + if (symac_bind) { + val symac = cdr(symac_bind); + if (symac == form) + return form; + form = rlcp_tree(symac, form); + goto tail; + } + return form; + } + + return expand(form, menv); +} + +static val expand_forms_lisp1(val form, val menv) +{ + if (atom(form)) { + return form; + } else { + val f = car(form); + val r = cdr(form); + val ex_f = expand_lisp1(f, menv); + val ex_r = expand_forms_lisp1(r, menv); + + if (ex_f == f && ex_r == r) + return form; + return rlcp(cons(ex_f, ex_r), form); + } +} + static val expand_cond_pairs(val form, val menv) { if (atom(form)) { @@ -2918,6 +2979,13 @@ tail: return expand_macrolet(form, menv); } else if (sym == symacrolet_s) { return expand_symacrolet(form, menv); + } else if (sym == dwim_s) { + val args = rest(form); + val args_ex = expand_forms_lisp1(args, menv); + + if (args == args_ex) + return form; + return rlcp(cons(sym, args_ex), form); } else if ((macro = lookup_mac(menv, sym))) { val mac_expand = expand_macro(form, macro, menv); if (mac_expand == form) @@ -10385,30 +10385,34 @@ and vice versa. If a symbol is bound to both a function and variable in the global namespace, then the variable binding is favored. -Macros do not participate in the special scope conflation. This is apparent -in two ways. Firstly, the space of symbol macros is not folded together with -the space of regular macros. An argument of +Macros do not participate in the special scope conflation, with one +exception. What this means is that the space of symbol macros is not folded +together with the space of regular macros. An argument of .code dwim that is a symbol might be symbol macro, variable or function, but it cannot be interpreted as the name of -a regular macro. Secondly, function bindings cannot shadow symbol macros. If a -function is defined in an inner scope relative to a symbol macro for the -same symbol, that symbol macro still hides the function when it appears in -a +a regular macro. + +The exception is this: from the perspective of a .code dwim -form. Symbol macros are expanded without any consideration for the +form, function bindings can shadow symbol macros. If a +function binding is defined in an inner scope relative to a symbol macro for +the same symbol, +using +.code flet +or +.codn labels , +the function hides the symbol macro. In other words, when +macro expansion processes an argument of a .code dwim -operator, which is not treated specially by the expansion phase. - -Thus, the following is invalid: - -.cblk -[let ((f (+ 2 2))) f] -.cble - -the expression looks for a function or variable called -.codn let , -which does not exist. +form, and that argument is a symbol, it is treated specially +in order to provide a consistent name lookup behavior. If the innermost +binding for that symbol is a function binding, it refers to that +function binding, even if a more outer symbol macro binding exists, +and so the symbol is not expanded using the symbol macro. +By contrast, in an ordinary form, a symbolic argument never resolves +to a function binding. The symbol refers to either a symbol macro or a +variable, whichever is nested closer. How many arguments are required by the .code dwim @@ -11692,11 +11696,22 @@ Note that and .code flet are properly scoped with regard to macros. -During macro expansion, they shadow, macros defined by +During macro expansion, function bindings introduced by these +macro operators shadow macros defined by .code macrolet and .codn defmacro . +Furthermore, function bindings introduced by +.code labels +and +.code flet +also shadow symbol macros defined by +.codn symacrolet , +when those symbol macros occur as arguments of a +.code dwim +form. + See also: the .code macrolet operator. @@ -26117,6 +26132,15 @@ constructs will see whichever of the two bindings is more inner, even though the bindings are active in completely separate phases of processing. +From the perspective of the arguments of a +.code dwim +form, lexical function bindings also shadow symbol macros. +This is consistent with the Lisp-1-style name resolution which +applies inside a +.code dwim +form. Of course, lexical operator macros do not shadow +symbol macros under any circumstances. + .coNP Operator @ tree-bind .synb .mets (tree-bind < macro-style-params < expr << form *) |