summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-05-14 06:11:59 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-05-14 06:11:59 -0700
commit07c6d46b1cc8300042832c9a01fe57cdfb130db3 (patch)
treea9d8e199efb95507f88d7191520ceda2b62bc8ca /eval.c
parent28304a063238826c7c8d1e9cad771a5d23d8b37a (diff)
downloadtxr-07c6d46b1cc8300042832c9a01fe57cdfb130db3.tar.gz
txr-07c6d46b1cc8300042832c9a01fe57cdfb130db3.tar.bz2
txr-07c6d46b1cc8300042832c9a01fe57cdfb130db3.zip
compiler: better code for global var definitions.
* eval.c (rt_defvarl): More accurate self string. (rt_defv): New static function: like rt_defvarl but ensures that the new variable has a binding cell, and returns that cell instead of the hash cell. (op_defvarl): Take advantage of rt_defv to not have to cons up the binding cell. (eval_init): Register sys:rt-defv intrinsic. * parser.c (read_file_common): Compiled files are now version 7, so we must recognize them. We still load version 6 files because rt:defvarl still exists for them. * share/txr/stdlib/compiler.tl (expand-defvarl): Improve the generated code in two ways. Firstly, use the new sys:rt-defv, which returns the binding cell, so that the value can be stored into it with rplacd without having to cons up anything. Secondly, if there is no value expression, don't emit the code to do the assignment. (%tlo-ver%): Bump compiled file version to (7 0). * txr.1: Add note about TXR 260 loading version 7 and 6.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c26
1 files changed, 22 insertions, 4 deletions
diff --git a/eval.c b/eval.c
index 47561afb..0f6600c6 100644
--- a/eval.c
+++ b/eval.c
@@ -1965,7 +1965,7 @@ static val op_or(val form, val env)
static val rt_defvarl(val sym)
{
- val self = lit("defvar");
+ val self = lit("sys:defvarl");
val new_p;
val cell = (lisplib_try_load(sym),
gethash_c(self, top_vb, sym, mkcloc(new_p)));
@@ -1980,15 +1980,32 @@ static val rt_defvarl(val sym)
return nil;
}
+static val rt_defv(val sym)
+{
+ val self = lit("sys:defv");
+ val new_p;
+ val cell = (lisplib_try_load(sym),
+ gethash_c(self, top_vb, sym, mkcloc(new_p)));
+
+ if (new_p) {
+ uw_purge_deferred_warning(cons(var_s, sym));
+ uw_purge_deferred_warning(cons(sym_s, sym));
+ remhash(top_smb, sym);
+ return sys_rplacd(cell, cons(sym, nil));
+ }
+
+ return nil;
+}
+
static val op_defvarl(val form, val env)
{
val args = rest(form);
val sym = first(args);
- val cell = rt_defvarl(sym);
+ val bind = rt_defv(sym);
- if (cell) {
+ if (bind) {
val value = eval(second(args), env, form);
- rplacd(cell, cons(sym, value));
+ rplacd(bind, value);
}
return sym;
@@ -7180,6 +7197,7 @@ void eval_init(void)
reg_varl(intern(lit("cptr-null"), user_package), cptr(0));
reg_fun(intern(lit("rt-defvarl"), system_package), func_n1(rt_defvarl));
+ reg_fun(intern(lit("rt-defv"), system_package), func_n1(rt_defv));
reg_fun(intern(lit("rt-defun"), system_package), func_n2(rt_defun));
reg_fun(intern(lit("rt-defmacro"), system_package), func_n3(rt_defmacro));
reg_fun(intern(lit("rt-defsymacro"), system_package), func_n2(rt_defsymacro));