summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-10-14 06:49:33 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-10-14 06:49:33 -0700
commit9cef0547ba0ec81ce6051bb1cba9db5671e08e64 (patch)
treec7f9cf081f07026c01d8aba741c78e67d3b376eb /eval.c
parentd34d09ef504ba2728821ce5b806bb59f22f378d7 (diff)
downloadtxr-9cef0547ba0ec81ce6051bb1cba9db5671e08e64.tar.gz
txr-9cef0547ba0ec81ce6051bb1cba9db5671e08e64.tar.bz2
txr-9cef0547ba0ec81ce6051bb1cba9db5671e08e64.zip
New way of handling exceptions without unwinding.
* eval.c (handler_bind_s): New symbol variable. (op_handler_bind): New static function. (do_expand): Traverse handler-bind forms. (eval_init): Initialize handler_bind_s variable and register handler-bind operator. * lisplib.c (except_set_entries, except_instantiate): New functions. (lisplib_init): Register new functions in dl_table. * parser.c (intr_s): New symbol variable. (repl_intr): Throw exception of type intr, rather than error. This way we can interrupt accidental exception handling loops involving exceptions derived from error. (parse_init): Initialize intr_s. * share/txr/stdlib/except.tl: New file, defines handle macro. * unwind.c (uw_push_handler): New function. (invoke_handler): New static function. (uw_throw): Search loop looks for and processes handlers in addition to catches. * unwind.h (uw_frtype_t): New enum member, UW_HANDLE. (struct uw_catch): Move member visible so it is in the same position as in struct uw_handler. (struct uw_handler): New struct type. (union uw_frame): New member ha of type struct uw_handler. (uw_push_handler): Function declared. * txr.1: Added introductory paragraphs to Exception Handling section. Documented handler-bind and handle. Some minor errors corrected.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c41
1 files changed, 40 insertions, 1 deletions
diff --git a/eval.c b/eval.c
index f16be7ec..8fa4120a 100644
--- a/eval.c
+++ b/eval.c
@@ -73,7 +73,7 @@ val eval_initing;
val eval_error_s;
val dwim_s, progn_s, prog1_s, let_s, let_star_s, lambda_s, call_s;
-val cond_s, if_s, iflet_s, when_s;
+val handler_bind_s, cond_s, if_s, iflet_s, when_s;
val defvar_s, defvarl_s, defparm_s, defparml_s, defun_s, defmacro_s;
val tree_case_s, tree_bind_s;
val sys_mark_special_s;
@@ -1911,6 +1911,24 @@ static val op_catch(val form, val env)
return result;
}
+static val op_handler_bind(val form, val env)
+{
+ val args = rest(form);
+ val fun = pop(&args);
+ val handle_syms = pop(&args);
+ val body = args;
+ val result;
+ uw_frame_t uw_handler;
+
+ uw_push_handler(&uw_handler, handle_syms, eval(fun, env, form));
+
+ result = eval_progn(body, env, form);
+
+ uw_pop_frame(&uw_handler);
+
+ return result;
+}
+
static val subst_vars(val forms, val env)
{
list_collect_decl(out, iter);
@@ -3152,6 +3170,25 @@ tail:
return rlcp(cons(sym, quasi_ex), form);
} else if (sym == catch_s) {
return expand_catch(rest(form), menv);
+ } else if (sym == handler_bind_s) {
+ val args = rest(form);
+ val fun = pop(&args);
+ val handle_syms = pop(&args);
+ val body = args;
+ val fun_ex = expand(fun, menv);
+ val body_ex = expand_forms(body, menv);
+
+ if (!cddr(form))
+ eval_error(form, lit("~s: missing arguments"), sym, nao);
+
+ if (fun == fun_ex && body == body_ex)
+ return form;
+
+ return rlcp(cons(sym, cons(if3(fun == fun_ex,
+ fun, fun_ex),
+ cons(handle_syms,
+ if3(body == body_ex,
+ body, body_ex)))), form);
} else if (sym == macro_time_s) {
val args = rest(form);
val args_ex = expand_forms(args, menv);
@@ -4076,6 +4113,7 @@ void eval_init(void)
flet_s = intern(lit("flet"), user_package);
labels_s = intern(lit("labels"), user_package);
call_s = intern(lit("call"), user_package);
+ handler_bind_s = intern(lit("handler-bind"), user_package);
cond_s = intern(lit("cond"), user_package);
caseq_s = intern(lit("caseq"), user_package);
caseql_s = intern(lit("caseql"), user_package);
@@ -4214,6 +4252,7 @@ void eval_init(void)
reg_op(dwim_s, op_dwim);
reg_op(quasi_s, op_quasi_lit);
reg_op(catch_s, op_catch);
+ reg_op(handler_bind_s, op_handler_bind);
reg_op(with_saved_vars_s, op_with_saved_vars);
reg_op(prof_s, op_prof);