summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-12-23 12:09:09 -0800
committerKaz Kylheku <kaz@kylheku.com>2016-12-23 12:09:09 -0800
commitc968457672379368478a3aa064e1d7f2d00280a7 (patch)
tree4eff12ec4accff2f9ae2abc3bec7f9b9b385ec7d /eval.c
parent34fe393aba5271cba73b60fb5fcad646099c7f28 (diff)
downloadtxr-c968457672379368478a3aa064e1d7f2d00280a7.tar.gz
txr-c968457672379368478a3aa064e1d7f2d00280a7.tar.bz2
txr-c968457672379368478a3aa064e1d7f2d00280a7.zip
bugfix: dynamic env handling in parallel binding
In the parallel binding (let ((x s) (s 0) (y s)) ...), both x and y must bind to the prior value of s, not to the new value 0. We have the bug that if s is a special variable, the initialization of y sees the new dynamic environment which contains the new value, so x gets the previous, y gets new. This commit fixes it. * eval.c (reparent_env): New static function. (bindings_helper): Separate logic into two loops, for sequential and parallel binding, so we don't have to repeatedly test this condition in the loop body, and can think separately about each case and streamline it. Nothing new happens under sequential binding; the behavior that is wrong for parallel binding is right for sequential. Under parallel binding, what we do is reset the dynamic environment to the original one prior to each evaluation of an initform. Then if the evaluation changes to a new dynamic environment (a special variable is being bound), we notice this and hook the new environment into a local stack, changing it parent pointer. At the end, we install this stack as the new dynamic env. Thus each init form is evaluated in the original dynamic env. * tests/011/special-1.tl: New tests added.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c75
1 files changed, 57 insertions, 18 deletions
diff --git a/eval.c b/eval.c
index c4cd24cf..8a4c7366 100644
--- a/eval.c
+++ b/eval.c
@@ -572,6 +572,12 @@ static val lookup_symac_lisp1(val menv, val sym)
}
}
+static val reparent_env(val child, val parent)
+{
+ child->e.up_env = parent;
+ return child;
+}
+
static val lexical_var_p(val menv, val sym)
{
if (nilp(menv)) {
@@ -1326,7 +1332,7 @@ static void copy_bh_env_handler(mem_t *ptr, int parent)
static val bindings_helper(val vars, val env, val sequential,
val ret_new_bindings, val ctx)
{
- val iter;
+ val iter, var;
struct bindings_helper_vars v;
list_collect_decl (new_bindings, ptail);
uw_frame_t uw_cc;
@@ -1334,31 +1340,64 @@ static val bindings_helper(val vars, val env, val sequential,
uw_push_cont_copy(&uw_cc, coerce(mem_t *, &v), copy_bh_env_handler);
- for (iter = vars; iter; iter = cdr(iter)) {
- val item = car(iter);
- val var, value = nil;
+ if (sequential) {
+ for (iter = vars; iter; iter = cdr(iter)) {
+ val item = car(iter);
+ val value = nil;
- if (consp(item)) {
- var = pop(&item);
- value = eval(pop(&item), if3(sequential, v.ne, env), ctx);
- } else {
- var = item;
+ if (consp(item)) {
+ var = pop(&item);
+ value = eval(pop(&item), v.ne, ctx);
+ } else {
+ var = item;
+ }
+
+ if (bindable(var)) {
+ val le = make_env(nil, nil, v.ne);
+ val binding = env_vbind(le, var, value);
+ if (ret_new_bindings)
+ ptail = list_collect (ptail, binding);
+ v.ne = le;
+ } else {
+ goto notbindable;
+ }
}
+ } else {
+ val de_in = dyn_env, new_de = de_in;
+
+ for (iter = vars; iter; iter = cdr(iter)) {
+ val item = car(iter);
+ val value = nil;
+
+ if (consp(item)) {
+ var = pop(&item);
+ value = eval(pop(&item), env, ctx);
+ if (dyn_env != de_in) {
+ reparent_env(dyn_env, new_de);
+ new_de = dyn_env;
+ dyn_env = de_in;
+ }
+ } else {
+ var = item;
+ }
- if (bindable(var)) {
- val le = if3(sequential, make_env(nil, nil, v.ne), v.ne);
- val binding = env_vbind(le, var, value);
- if (ret_new_bindings)
- ptail = list_collect (ptail, binding);
- v.ne = le;
- } else {
- eval_error(ctx, lit("~s: ~s is not a bindable symbol"),
- ctx_name(ctx), var, nao);
+ if (bindable(var)) {
+ val binding = env_vbind(v.ne, var, value);
+ if (ret_new_bindings)
+ ptail = list_collect (ptail, binding);
+ } else {
+ goto notbindable;
+ }
}
+ dyn_env = new_de;
}
uw_pop_frame(&uw_cc);
+
return v.ne;
+notbindable:
+ eval_error(ctx, lit("~s: ~s is not a bindable symbol"),
+ ctx_name(ctx), var, nao);
}
static val fbindings_helper(val vars, val env, val lbind, val ctx)