summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-09-02 06:20:42 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-09-02 06:20:42 -0700
commitbc987afb87416418841573cd858258438be9f2ea (patch)
tree7d5586370f059e97a7315b1ed36fd37caf093bd0 /eval.c
parent9272b1d59ac9f7fbe05e4fd19a1a28b180732235 (diff)
downloadtxr-bc987afb87416418841573cd858258438be9f2ea.tar.gz
txr-bc987afb87416418841573cd858258438be9f2ea.tar.bz2
txr-bc987afb87416418841573cd858258438be9f2ea.zip
load: new *load-hooks* feature.
*load-hooks* lets a .txr, .tl or .tlo file specify actions to be taken when the loading of that file completes, whether normally or via an exception. They are also honored by process exit. For instance, with this, we can have a Lisp file that behaves like a script which cleans up after itself (e.g. removing temporary files) even if it is not run as a stand-alone program, but invoked via (load ...). Because it's not a stand-alone program, it cannot simply use the at-exit-call mechanism. The unwind-protect operator could be used, but it's inconvenient because it protects a single form. The *load-hooks* feature in effect protects all the top level forms of a load, similarly to unwind-protect. Also, unwind-protect does not guard against a process exit. (However, *load-hooks* does not guard against an abnormal exit, only normal termination). * eval.c (load_hooks_s): New symbol variable. (run_load_hooks): New function. (run_load_hooks_atexit): New static function. (load): bind *load-hooks* to nil around load. Implement the hooks processing via run_load_hooks, taking care to pass the load-time dynamic environment that has already been undone. (eval_init): Initialize load_hooks_s and register the *load-hooks* variable. Register run_load_hooks_atexit with atexit, so the current value of *load-hooks* is processed on process exit. * eval.h (load_hooks_s, run_load_hooks): Declared. * match.c (v_load): Similar changes as in load. * txr.c (txr_main): Run the load hooks with run_load_hooks immediately after processing the .txr or .tl file, before entering the listener. * tests/019/load-hook.tl: New directory and file * tests/load-hook.tl: New file. * txr.1: Documented. * stdlib/doc-syms.tl: Updated.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c27
1 files changed, 25 insertions, 2 deletions
diff --git a/eval.c b/eval.c
index 497644b0..8a9fada6 100644
--- a/eval.c
+++ b/eval.c
@@ -102,7 +102,7 @@ val vector_lit_s, vec_list_s, tree_lit_s, tree_construct_s;
val macro_time_s, macrolet_s;
val defsymacro_s, symacrolet_s, prof_s, switch_s, struct_s;
val fbind_s, lbind_s, flet_s, labels_s;
-val load_path_s, load_recursive_s;
+val load_path_s, load_hooks_s, load_recursive_s;
val load_time_s, load_time_lit_s;
val eval_only_s, compile_only_s;
val const_foldable_s;
@@ -4579,6 +4579,23 @@ static val me_load_for(val form, val menv)
return cons(rt_load_for_s, out);
}
+void run_load_hooks(val load_dyn_env)
+{
+ val hooks_binding = lookup_var(load_dyn_env, load_hooks_s);
+ val hooks = cdr(hooks_binding);
+
+ if (hooks) {
+ for (; hooks; hooks = cdr(hooks))
+ funcall(car(hooks));
+ rplacd(hooks_binding, nil);
+ }
+}
+
+static void run_load_hooks_atexit(void)
+{
+ run_load_hooks(dyn_env);
+}
+
val load(val target)
{
val self = lit("load");
@@ -4592,6 +4609,7 @@ val load(val target)
val name, stream;
val txr_lisp_p = t;
val saved_dyn_env = dyn_env;
+ val load_dyn_env = make_env(nil, nil, dyn_env);
val rec = cdr(lookup_var(saved_dyn_env, load_recursive_s));
open_txr_file(path, &txr_lisp_p, &name, &stream, self);
@@ -4603,11 +4621,12 @@ val load(val target)
uw_simple_catch_begin;
- dyn_env = make_env(nil, nil, dyn_env);
+ dyn_env = load_dyn_env;
env_vbind(dyn_env, load_path_s, if3(opt_compat && opt_compat <= 215,
path,
stream_get_prop(stream, name_k)));
env_vbind(dyn_env, load_recursive_s, t);
+ env_vbind(dyn_env, load_hooks_s, nil);
env_vbind(dyn_env, package_s, cur_package);
if (txr_lisp_p == t) {
@@ -4653,6 +4672,7 @@ val load(val target)
uw_unwind {
close_stream(stream, nil);
+ run_load_hooks(load_dyn_env);
if (!rec)
uw_dump_deferred_warnings(std_null);
}
@@ -6667,6 +6687,7 @@ void eval_init(void)
switch_s = intern(lit("switch"), system_package);
struct_s = intern(lit("struct"), user_package);
load_path_s = intern(lit("*load-path*"), user_package);
+ load_hooks_s = intern(lit("*load-hooks*"), user_package);
load_recursive_s = intern(lit("*load-recursive*"), system_package);
load_time_s = intern(lit("load-time"), user_package);
load_time_lit_s = intern(lit("load-time-lit"), system_package);
@@ -6983,6 +7004,7 @@ void eval_init(void)
reg_var(load_path_s, nil);
reg_symacro(intern(lit("self-load-path"), user_package), load_path_s);
reg_var(load_recursive_s, nil);
+ reg_var(load_hooks_s, nil);
reg_fun(intern(lit("expand"), user_package), func_n2o(no_warn_expand, 1));
reg_fun(intern(lit("expand*"), user_package), func_n2o(expand, 1));
reg_fun(intern(lit("expand-with-free-refs"), user_package),
@@ -7373,6 +7395,7 @@ void eval_init(void)
uw_register_subtype(eval_error_s, error_s);
uw_register_subtype(case_error_s, error_s);
+ atexit(run_load_hooks_atexit);
lisplib_init();
}