summaryrefslogtreecommitdiffstats
path: root/eval.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-28 06:22:24 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-28 06:22:24 -0700
commitac18f54aca98531e8200d8bcc012555c1f195ec1 (patch)
treec43f2f7517c5c3772afa753c68ef52d566e4e7b3 /eval.c
parent4462afdd0a80cbf8c73faa58fb0540b70bd0d890 (diff)
downloadtxr-ac18f54aca98531e8200d8bcc012555c1f195ec1.tar.gz
txr-ac18f54aca98531e8200d8bcc012555c1f195ec1.tar.bz2
txr-ac18f54aca98531e8200d8bcc012555c1f195ec1.zip
constantp: fully expand; recognize functions.
This patch improves the constantp function dramatically. It now performs a full expansion of its argument, and recognizes all of the constant foldable functions that the compiler recognizes. * eval.c (const_foldable_s): New symbol variable. (const_foldable_hash): New static variable. (constantp_noex): Look up function in the hash table of const foldable functions, including in the case when it appears in a dwim form as in [+ 2 2] which is (dwim + 2 2). In this case, recursively check the arguments for constantp_noex. We get the hash table of foldable functions from the sys:%const-foldable% variable, which comes from an autoloaded module. (constantp): Fully expand the input form, not just m macroexpand. (eval_init): Register the const_foldable_s variable. * lisplib.c (constfun_instantiate, constfun_set_entries): New static functions. (lisplib_init): Register auto-loading of constfun module via new static functions. * stdlib/compiler.tl; Load the constfun module if %const-foldable% is not defined. (%const-foldable-funs%, %const-foldable%): Removed from here. * stdlib/constfun.tl: New file. (%const-foldable-funs%, %const-foldable%): Moved here. * txr.1: Documented changes to constantp.
Diffstat (limited to 'eval.c')
-rw-r--r--eval.c46
1 files changed, 32 insertions, 14 deletions
diff --git a/eval.c b/eval.c
index b383a325..f32608e2 100644
--- a/eval.c
+++ b/eval.c
@@ -104,6 +104,7 @@ val fbind_s, lbind_s, flet_s, labels_s;
val load_path_s, load_recursive_s;
val load_time_s, load_time_lit_s;
val eval_only_s, compile_only_s;
+val const_foldable_s;
val special_s, unbound_s;
val whole_k, form_k, symacro_k;
@@ -115,6 +116,8 @@ val iter_item_f, iter_step_f;
val origin_hash;
+static val const_foldable_hash;
+
val make_env(val vbindings, val fbindings, val up_env)
{
val env = make_obj();
@@ -5277,7 +5280,27 @@ static val macroexpand_lisp1(val form, val menv)
static val constantp_noex(val form)
{
if (consp(form)) {
- return eq(car(form), quote_s);
+ val sym = car(form);
+ val args = cdr(form);
+ if (eq(sym, quote_s))
+ return t;
+ if (!proper_list_p(args))
+ return nil;
+ if (eq(sym, dwim_s)) {
+ sym = us_car(args);
+ args = us_cdr(args);
+ }
+ if (!symbolp(sym))
+ return nil;
+ if (!const_foldable_hash)
+ const_foldable_hash = cdr(lookup_var(nil, const_foldable_s));
+ if (!gethash(const_foldable_hash, sym))
+ return nil;
+ for (; args; args = us_cdr(args)) {
+ if (!constantp_noex(us_car(args)))
+ return nil;
+ }
+ return t;
} else {
if (bindable(form))
return nil;
@@ -5290,21 +5313,15 @@ static val constantp(val form, val env_in)
val env = default_null_arg(env_in);
if (consp(form)) {
- if (car(form) == quote_s) {
+ if (car(form) == quote_s)
return t;
- } else if (macro_form_p(form, env)) {
- return constantp_noex(macroexpand(form, env));
- } else {
- return nil;
- }
+ else
+ return constantp_noex(no_warn_expand(form, env));
} else if (symbolp(form)) {
- if (!bindable(form)) {
+ if (!bindable(form))
return t;
- } else if (macro_form_p(form, env)) {
- return constantp_noex(macroexpand(form, env));
- } else {
- return nil;
- }
+ else
+ return constantp_noex(no_warn_expand(form, env));
} else {
return t;
}
@@ -6421,7 +6438,7 @@ void eval_init(void)
&op_table, &pm_table, &last_form_evaled,
&call_f, &iter_begin_f, &iter_from_binding_f, &iter_more_f,
&iter_item_f, &iter_step_f,
- &unbound_s, &origin_hash, convert(val *, 0));
+ &unbound_s, &origin_hash, &const_foldable_hash, convert(val *, 0));
top_fb = make_hash(t, nil, nil);
top_vb = make_hash(t, nil, nil);
top_mb = make_hash(t, nil, nil);
@@ -6561,6 +6578,7 @@ void eval_init(void)
load_time_lit_s = intern(lit("load-time-lit"), system_package);
eval_only_s = intern(lit("eval-only"), user_package);
compile_only_s = intern(lit("compile-only"), user_package);
+ const_foldable_s = intern(lit("%const-foldable%"), system_package);
qquote_init();