diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-10-14 06:49:33 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-10-14 06:49:33 -0700 |
commit | 9cef0547ba0ec81ce6051bb1cba9db5671e08e64 (patch) | |
tree | c7f9cf081f07026c01d8aba741c78e67d3b376eb /eval.c | |
parent | d34d09ef504ba2728821ce5b806bb59f22f378d7 (diff) | |
download | txr-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.c | 41 |
1 files changed, 40 insertions, 1 deletions
@@ -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); |