summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-10-28 20:18:20 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-10-28 20:18:20 -0700
commita5ef086fc33cfbfce7b03bad291efa28acf739b2 (patch)
treeb6658f3ecbc98054217f72a166b02bfdb0b029e0 /eval.c
parent67af4be97a2ea8700a841feb893a1f1747987843 (diff)
downloadtxr-a5ef086fc33cfbfce7b03bad291efa28acf739b2.tar.gz
txr-a5ef086fc33cfbfce7b03bad291efa28acf739b2.tar.bz2
txr-a5ef086fc33cfbfce7b03bad291efa28acf739b2.zip
Implementing sys:abscond-from operator.
* eval.c (sys_abscond_from_s): New symbol variable. (op_abscond_from): New static function. (do_expand): Handle abscond-from like return-from. (eval_init): Initialize sys_abscond_from_s and register sys:abscond-from operator. * share/txr/stdlib/yield.tl (yield-from): Use sys:abscond-from instead of return-from, to avoid tearing down the continuation's resources that it may need when restarted. * txr.1: Documented sys:abscond-from and added a mention to the Delimited Continuations introduction. * unwind.c (uw_abscond_to_exit_point): New static function. (uw_block_abscond): New function. * unwind.h (uw_block_abscond): Declared.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c18
1 files changed, 16 insertions, 2 deletions
diff --git a/eval.c b/eval.c
index 4c2f6411..052c7994 100644
--- a/eval.c
+++ b/eval.c
@@ -84,7 +84,7 @@ val setq_s, inc_s, zap_s;
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;
+val uw_protect_s, return_s, return_from_s, sys_abscond_from_s;
val list_s, append_s, 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;
@@ -1858,6 +1858,16 @@ static val op_return_from(val form, val env)
abort();
}
+static val op_abscond_from(val form, val env)
+{
+ val name = second(form);
+ val retval = eval(third(form), env, form);
+ uw_block_abscond(name, retval);
+ eval_error(form, lit("sys:abscond-from: no block named ~s is visible"),
+ name, nao);
+ abort();
+}
+
static val op_dwim(val form, val env)
{
val argexps = rest(form);
@@ -3030,7 +3040,9 @@ tail:
} else {
return rlcp(cons(sym, cons(funcs_ex, body_ex)), form);
}
- } else if (sym == block_s || sym == return_from_s) {
+ } else if (sym == block_s || sym == return_from_s ||
+ sym == sys_abscond_from_s)
+ {
val name = second(form);
val body = rest(rest(form));
val body_ex = expand_progn(body, menv);
@@ -4157,6 +4169,7 @@ void eval_init(void)
uw_protect_s = intern(lit("unwind-protect"), user_package);
return_s = intern(lit("return"), user_package);
return_from_s = intern(lit("return-from"), user_package);
+ sys_abscond_from_s = intern(lit("abscond-from"), system_package);
gethash_s = intern(lit("gethash"), user_package);
car_s = intern(lit("car"), user_package);
cdr_s = intern(lit("cdr"), user_package);
@@ -4252,6 +4265,7 @@ void eval_init(void)
reg_op(block_s, op_block);
reg_op(return_s, op_return);
reg_op(return_from_s, op_return_from);
+ reg_op(sys_abscond_from_s, op_abscond_from);
reg_op(dwim_s, op_dwim);
reg_op(quasi_s, op_quasi_lit);
reg_op(catch_s, op_catch);