diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2016-12-20 18:48:53 -0800 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2016-12-20 18:48:53 -0800 |
commit | da23d76b54532d8d172388c1faeb6c97200d2a95 (patch) | |
tree | f4cc82bc6ae26f7573475e213058c3345acd5bc2 /INSTALL | |
parent | 68dec176d0cbd65f8f47b51bb9cbb72d10c199b3 (diff) | |
download | txr-da23d76b54532d8d172388c1faeb6c97200d2a95.tar.gz txr-da23d76b54532d8d172388c1faeb6c97200d2a95.tar.bz2 txr-da23d76b54532d8d172388c1faeb6c97200d2a95.zip |
Different approach for specials in let/let*.
This addresses a problem with the new scheme for handling
specials. If we let specials be bound in the lexical
environment and then do the swizzle into the dynamic
environment using sys:with-dyn-rebinds, that only works
correctly for parallel bindings (and thus also for lambda and
macro parameters). For sequential bindings, it exposes the
possibility that a closure is created during the sequential
binding which captures a would-be special variable while it is
still in the lexical stage. That closure can be thrown out of
there, so the sys:with-dyn-rebinds is never reached which
swizzles the variable.
The new scheme is very simple. When expanding a let, we
tranform (s init) to (s (sys:dvbind s init)) if s is a special
variable. This new sys:dvbind operator binds s to the value of
the init expression in a newly created dynamic environment,
and returns the #:unbound symbol, which is received by the
lexical s. Problem solved. The only thing remains is that the
let special operator must save and restore the dynamic
environment.
There is no need for sys:with-dyn-rebinds around the body
of a let, but we keep that mechanism and approach for handling
specials in argument lists.
* eval.c (dvbind_s): New symbol variale.
(bindings_helper): Lose the env_out argument; return the new
environment. No caller uses the returned bindings any more.
(op_let): Call bindings_helper in initializing expression of
new_env. Save the dyn_env, and restore it after evaluating
the body.
(op_dvbind): New static function.
(expand_vars): Lose the pspecials argument. Perform
the insertion of sys:dvbind.
(do_expand): Simplify the let expander: expand_vars no longer
outputs a list of specials and there is no need to insert
with_dyn_rebinds_s. Add a case for sys:dvbind: assume it
requires no expansion.
(eval_init): Intern sys:dvbind, and bind it as an operator
to the new op_dvbind function.
Diffstat (limited to 'INSTALL')
0 files changed, 0 insertions, 0 deletions