diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2014-07-10 07:23:06 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2014-07-10 07:23:06 -0700 |
commit | 157145753623fde988c5b4664168a0cbd6282503 (patch) | |
tree | 171a451e01a7430254166bd7840da50a40d999f4 /eval.c | |
parent | 1ab3edf06ab0fdc2072c855392fa505e903846a4 (diff) | |
download | txr-157145753623fde988c5b4664168a0cbd6282503.tar.gz txr-157145753623fde988c5b4664168a0cbd6282503.tar.bz2 txr-157145753623fde988c5b4664168a0cbd6282503.zip |
Implementing local function binding constructs.
* eval.c (fbind_s, lbind_s, flet_s, labels_s): New symbol globals.
(env_fb_to_fb): New static function.
(lookup_mac): Implement the same shadow check for function macros
that is done for symbol macros, because we now have local functions
that can shadow local macros.
(fbindings_helper, op_fbind, make_fun_shadowing_env,
expand_fbind_vars, me_flet_labels): New static functions.
(expand): Add cases for fbind and lbind.
(eval_init): Intern the four new symbols.
Register sys:fbind and sys:lbind operators.
Register flet and labels macros.
* txr.1: Documented flet and labels.
Diffstat (limited to 'eval.c')
-rw-r--r-- | eval.c | 121 |
1 files changed, 118 insertions, 3 deletions
@@ -85,6 +85,7 @@ val hash_lit_s, hash_construct_s; val vector_lit_s, vector_list_s; val macro_time_s, with_saved_vars_s, macrolet_s; val defsymacro_s, symacrolet_s, prof_s; +val fbind_s, lbind_s, flet_s, labels_s; val special_s, whole_k; @@ -130,6 +131,13 @@ val env_vbind(val env, val sym, val obj) return cell; } +static void env_vb_to_fb(val env) +{ + type_check(env, ENV); + env->e.fbindings = env->e.vbindings; + env->e.vbindings = nil; +} + noreturn static val eval_error(val form, val fmt, ...) { va_list vl; @@ -244,8 +252,8 @@ static val lookup_mac(val menv, val sym) { val binding = assoc(sym, menv->e.fbindings); - if (binding) - return binding; + if (binding) /* special_s: see make_fun_shadowing_env */ + return (cdr(binding) == special_s) ? nil : binding; return lookup_mac(menv->e.up_env, sym); } } @@ -1062,6 +1070,28 @@ static val bindings_helper(val vars, val env, val sequential, return new_bindings; } +static val fbindings_helper(val vars, val env, val lbind, val ctx_form) +{ + val iter; + val nenv = make_env(nil, nil, env); + val lenv = if3(lbind, nenv, env); + + for (iter = vars; iter; iter = cdr(iter)) { + val item = car(iter); + val var = pop(&item); + val value = eval(pop(&item), lenv, ctx_form); + + if (bindable(var)) { + (void) env_fbind(nenv, var, value); + } else { + eval_error(ctx_form, lit("~s: ~s is not a bindable symbol"), + car(ctx_form), var, nao); + } + } + + return nenv; +} + static val op_progn(val form, val env) { return eval_progn(rest(form), env, form); @@ -1083,6 +1113,16 @@ static val op_let(val form, val env) return eval_progn(body, new_env, form); } +static val op_fbind(val form, val env) +{ + val oper = first(form); + val args = rest(form); + val vars = first(args); + val body = rest(args); + val new_env = fbindings_helper(vars, env, eq(oper, lbind_s), form); + return eval_progn(body, new_env, form); +} + static val op_each(val form, val env) { uses_or2; @@ -1412,6 +1452,13 @@ static val make_var_shadowing_env(val menv, val vars) } } +static val make_fun_shadowing_env(val menv, val funcs) +{ + val env = make_var_shadowing_env(menv, funcs); + env_vb_to_fb(env); + return env; +} + static val op_tree_case(val form, val env) { val cases = form; @@ -2253,7 +2300,36 @@ static val expand_vars(val vars, val menv, val form, if (init == init_ex && rest_vars == rest_vars_ex) return vars; return rlcp(cons(cons(var, init_ex), rest_vars_ex), vars); - } + } + } +} + +static val expand_fbind_vars(val vars, val menv, val form) +{ + val sym; + + if (nilp(vars)) { + return nil; + } else if (atom(vars)) { + eval_error(form, lit("~a is an invalid function binding syntax"), + vars, nao); + return vars; + } else if (symbolp(sym = car(vars))) { + eval_error(form, lit("symbols in this construct require initforms"), nao); + } else { + cons_bind (var, init, sym); + val rest_vars = rest(vars); + /* This var's init form sees a previous macro whose name is + the same as the symbol, so menv is used. */ + val init_ex = rlcp(expand_forms(init, menv), init); + /* The initforms of subsequent vars in a sequential binding + do not see a previous symbol macro; they see the var. */ + val rest_vars_ex = rlcp(expand_fbind_vars(rest_vars, menv, form), + rest_vars); + + if (init == init_ex && rest_vars == rest_vars_ex) + return vars; + return rlcp(cons(cons(var, init_ex), rest_vars_ex), vars); } } @@ -2463,6 +2539,25 @@ static val me_ret(val form, val menv) return cons(op_s, cons(identity_s, rest(form))); } +static val me_flet_labels(val form, val menv) +{ + val body = form; + val sym = pop(&body); + val funcs = pop(&body); + list_collect_decl (lambdas, ptail); + + for (; funcs; funcs = cdr(funcs)) { + val func = car(funcs); + val name = pop(&func); + val params = pop(&func); + val lambda = cons(lambda_s, cons(params, func)); + ptail = list_collect (ptail, cons(name, cons(lambda, nil))); + } + + return cons(if3(eq(sym, flet_s), fbind_s, lbind_s), + cons(lambdas, body)); +} + static val expand_catch_clause(val form, val menv) { val sym = first(form); @@ -2544,6 +2639,18 @@ tail: val basic_form = rlcp(cons(sym, cons(vars_ex, body_ex)), form); return expand_save_specials(basic_form, specials_p); } + } else if (sym == fbind_s || sym == lbind_s) { + val body = rest(rest(form)); + val funcs = second(form); + val new_menv = make_fun_shadowing_env(menv, funcs); + val body_ex = expand_forms(body, new_menv); + val funcs_ex = expand_fbind_vars(funcs, + sym == lbind_s ? new_menv : menv, form); + if (body == body_ex && funcs == funcs_ex) { + return form; + } else { + return rlcp(cons(sym, cons(funcs_ex, body_ex)), form); + } } else if (sym == block_s || sym == return_from_s) { val name = second(form); val body = rest(rest(form)); @@ -3232,6 +3339,10 @@ void eval_init(void) let_s = intern(lit("let"), user_package); let_star_s = intern(lit("let*"), user_package); lambda_s = intern(lit("lambda"), user_package); + fbind_s = intern(lit("fbind"), system_package); + lbind_s = intern(lit("lbind"), system_package); + flet_s = intern(lit("flet"), user_package); + labels_s = intern(lit("labels"), user_package); call_s = intern(lit("call"), user_package); cond_s = intern(lit("cond"), user_package); if_s = intern(lit("if"), user_package); @@ -3307,6 +3418,8 @@ void eval_init(void) reg_op(append_each_s, op_each); reg_op(append_each_star_s, op_each); reg_op(let_star_s, op_let); + reg_op(fbind_s, op_fbind); + reg_op(lbind_s, op_fbind); reg_op(lambda_s, op_lambda); reg_op(fun_s, op_fun); reg_op(cond_s, op_cond); @@ -3357,6 +3470,8 @@ void eval_init(void) reg_mac(intern(lit("while"), user_package), me_while); reg_mac(intern(lit("until"), user_package), me_until); reg_mac(quasilist_s, me_quasilist); + reg_mac(flet_s, me_flet_labels); + reg_mac(labels_s, me_flet_labels); reg_fun(cons_s, func_n2(cons)); reg_fun(intern(lit("make-lazy-cons"), user_package), func_n1(make_lazy_cons)); |