1
0
mirror of https://github.com/janet-lang/janet synced 2025-01-26 15:16:51 +00:00

Add macroexpand and macroexpand1

This commit is contained in:
Calvin Rose 2018-03-18 14:01:58 -04:00
parent 9461eb8b74
commit 855787b292
8 changed files with 127 additions and 12 deletions

View File

@ -1,6 +1,10 @@
# Bootstrap the dst environment # Bootstrap the dst environment
# Copyright 2018 (C) Calvin Rose # Copyright 2018 (C) Calvin Rose
(var *env*
"The current environment. Is dynamically bound."
_env)
(def defn :macro (def defn :macro
"Define a function" "Define a function"
(fn [name & more] (fn [name & more]
@ -18,7 +22,7 @@
(def defmacro :macro (def defmacro :macro
"Define a macro." "Define a macro."
(do (do
(def defn* (get (get _env 'defn) 'value)) (def defn* (get (get _env 'defn) :value))
(fn [name & more] (fn [name & more]
(def args (array-concat [] name :macro more)) (def args (array-concat [] name :macro more))
(apply1 defn* args)))) (apply1 defn* args))))
@ -355,11 +359,95 @@ If no match is found, returns nil"
(foreach s (fn [x] (put tab x true))) (foreach s (fn [x] (put tab x true)))
(keys tab)) (keys tab))
(defn macroexpand1
"Expand macros in a form, but do not recursively expand macros."
[x]
(defn doarray [a]
(def len (length a))
(def newa [])
(for [i 0 len]
(array-push newa (macroexpand1 (get a i))))
newa)
(defn dotable [t]
(def newt @{})
(var key (next t nil))
(while (not= nil key)
(put newt (macroexpand1 key) (macroexpand1 (get t key)))
(:= key (next t key)))
newt)
(defn expandlast [t]
(def len (length t))
(def last (get t (- len 1)))
(tuple-append (tuple-slice t 0 -2) (macroexpand1 last)))
(defn expandall [t]
(def args (doarray (tuple-slice t 1)))
(apply tuple (get t 0) args))
(defn expandfn [t]
(def args (doarray (tuple-slice t 2)))
(apply tuple 'fn (get t 1) args))
(def specs {
':= expandlast
'ast-quote identity
'def expandlast
'do expandall
'fn expandfn
'if expandall
'quote identity
'var expandlast
'while expandall
})
(defn dotup [t]
(def h (get t 0))
(def s (get specs h))
(def entry (get *env* h))
(def m (get entry :value))
(def m? (get entry :macro))
(cond
s (s t)
m? (apply1 m (tuple-slice t 1))
(apply1 tuple (doarray t))))
(defn doarray* [a]
(def res (doarray a))
(if (= (apply tuple res) (apply tuple a)) a res))
(defn dotable* [t]
(def res (dotable t))
(if (= (table-to-struct res) (table-to-struct t)) t res))
(def ux (ast-unwrap1 x))
(select (type ux)
:tuple (dotup ux)
:array (doarray* ux)
:struct (table-to-struct (dotable ux))
:table (dotable* ux)
ux))
(defn macroexpand
"Expand macros completely."
[x]
(var previous x)
(var current (macroexpand1 x))
(var counter 0)
(while (not= current previous)
(if (> (++ counter) 200)
(error "macro expansion too nested"))
(:= previous current)
(:= current (macroexpand1 current)))
current)
(defn make-env [parent safe] (defn make-env [parent safe]
(def parent (if parent parent _env)) (def parent (if parent parent _env))
(def newenv (setproto @{} parent)) (def newenv (setproto @{} parent))
(if (not safe) (if (not safe)
(put newenv '_env @{'value newenv})) (put newenv '_env @{:value newenv}))
newenv) newenv)
(def run-context (def run-context
@ -444,9 +532,12 @@ onvalue."
(put loading path true) (put loading path true)
(def f (file-open path)) (def f (file-open path))
(defn chunks [buf] (file-read f 1024 buf)) (defn chunks [buf] (file-read f 1024 buf))
(def oldenv *env*)
(:= *env* newenv)
(run-context newenv chunks identity (run-context newenv chunks identity
(fn [t x] (print (string t " error: " x)))) (fn [t x] (print (string t " error: " x))))
(file-close f) (file-close f)
(:= *env* oldenv)
(put loading path nil) (put loading path nil)
newenv))))) newenv)))))
@ -469,7 +560,7 @@ onvalue."
(file-flush stdout) (file-flush stdout)
(file-read stdin :line buf)) (file-read stdin :line buf))
(defn onvalue [x] (defn onvalue [x]
(put newenv '_ @{'value x}) (put newenv '_ @{:value x})
(pp x)) (pp x))
(run-context newenv (if getchunk getchunk chunks) onvalue (run-context newenv (if getchunk getchunk chunks) onvalue
(fn [t x] (print (string t " error: " x))))) (fn [t x] (print (string t " error: " x)))))

View File

@ -263,7 +263,7 @@ DstSlot dstc_resolve(
dstc_error(c, ast, dst_formatc("unknown symbol %q", sym)); dstc_error(c, ast, dst_formatc("unknown symbol %q", sym));
return dstc_cslot(dst_wrap_nil()); return dstc_cslot(dst_wrap_nil());
} }
ref = dst_get(check, dst_csymbolv("ref")); ref = dst_get(check, dst_csymbolv(":ref"));
if (dst_checktype(ref, DST_ARRAY)) { if (dst_checktype(ref, DST_ARRAY)) {
DstSlot ret = dstc_cslot(ref); DstSlot ret = dstc_cslot(ref);
/* TODO save type info */ /* TODO save type info */
@ -271,7 +271,7 @@ DstSlot dstc_resolve(
ret.flags &= ~DST_SLOT_CONSTANT; ret.flags &= ~DST_SLOT_CONSTANT;
return ret; return ret;
} else { } else {
Dst value = dst_get(check, dst_csymbolv("value")); Dst value = dst_get(check, dst_csymbolv(":value"));
return dstc_cslot(value); return dstc_cslot(value);
} }
} }
@ -819,7 +819,7 @@ recur:
for (;;) { for (;;) {
if (dst_checktype(entry, DST_NIL)) break; if (dst_checktype(entry, DST_NIL)) break;
if (dst_checktype(dst_get(entry, dst_csymbolv(":macro")), DST_NIL)) break; if (dst_checktype(dst_get(entry, dst_csymbolv(":macro")), DST_NIL)) break;
fn = dst_get(entry, dst_csymbolv("value")); fn = dst_get(entry, dst_csymbolv(":value"));
if (!dst_checktype(fn, DST_FUNCTION)) break; if (!dst_checktype(fn, DST_FUNCTION)) break;
if (macrorecur++ > DST_RECURSION_GUARD) { if (macrorecur++ > DST_RECURSION_GUARD) {
dstc_cerror(c, ast, "macro expansion recursed too deeply"); dstc_cerror(c, ast, "macro expansion recursed too deeply");

View File

@ -228,7 +228,7 @@ static void varleaf(
reftab->proto = attr; reftab->proto = attr;
DstArray *ref = dst_array(1); DstArray *ref = dst_array(1);
dst_array_push(ref, dst_wrap_nil()); dst_array_push(ref, dst_wrap_nil());
dst_table_put(reftab, dst_csymbolv("ref"), dst_wrap_array(ref)); dst_table_put(reftab, dst_csymbolv(":ref"), dst_wrap_array(ref));
dst_table_put(c->env, dst_wrap_symbol(sym), dst_wrap_table(reftab)); dst_table_put(c->env, dst_wrap_symbol(sym), dst_wrap_table(reftab));
refslot = dstc_cslot(dst_wrap_array(ref)); refslot = dstc_cslot(dst_wrap_array(ref));
refarrayslot = refslot; refarrayslot = refslot;
@ -266,7 +266,7 @@ static void defleaf(
DstTable *tab = dst_table(2); DstTable *tab = dst_table(2);
tab->proto = attr; tab->proto = attr;
int32_t tableindex, valsymindex, valueindex; int32_t tableindex, valsymindex, valueindex;
DstSlot valsym = dstc_cslot(dst_csymbolv("value")); DstSlot valsym = dstc_cslot(dst_csymbolv(":value"));
DstSlot tabslot = dstc_cslot(dst_wrap_table(tab)); DstSlot tabslot = dstc_cslot(dst_wrap_table(tab));
/* Add env entry to env */ /* Add env entry to env */

View File

@ -109,6 +109,7 @@ DstTable *dst_stl_env() {
dst_lib_array(args); dst_lib_array(args);
dst_lib_tuple(args); dst_lib_tuple(args);
dst_lib_buffer(args); dst_lib_buffer(args);
dst_lib_table(args);
dst_lib_parse(args); dst_lib_parse(args);
dst_lib_compile(args); dst_lib_compile(args);
dst_lib_asm(args); dst_lib_asm(args);

View File

@ -235,4 +235,25 @@ void dst_table_merge_struct(DstTable *table, const DstKV *other) {
dst_table_mergekv(table, other, dst_struct_capacity(other)); dst_table_mergekv(table, other, dst_struct_capacity(other));
} }
static int cfun_tostruct(DstArgs args) {
DstTable *t;
if (args.n != 1 || !dst_checktype(args.v[0], DST_TABLE)) {
return dst_throw(args, "expected table");
}
t = dst_unwrap_table(args.v[0]);
return dst_return(args, dst_wrap_struct(dst_table_to_struct(t)));
}
static const DstReg cfuns[] = {
{"table-to-struct", cfun_tostruct},
{NULL, NULL}
};
/* Load the table module */
int dst_lib_table(DstArgs args) {
DstTable *env = dst_env_arg(args);
dst_env_cfuns(env, cfuns);
return 0;
}
#undef dst_table_maphash #undef dst_table_maphash

View File

@ -112,7 +112,7 @@ int dst_cstrcmp(const uint8_t *str, const char *other) {
/* Add a module definition */ /* Add a module definition */
void dst_env_def(DstTable *env, const char *name, Dst val) { void dst_env_def(DstTable *env, const char *name, Dst val) {
DstTable *subt = dst_table(1); DstTable *subt = dst_table(1);
dst_table_put(subt, dst_csymbolv("value"), val); dst_table_put(subt, dst_csymbolv(":value"), val);
dst_table_put(env, dst_csymbolv(name), dst_wrap_table(subt)); dst_table_put(env, dst_csymbolv(name), dst_wrap_table(subt));
} }
@ -121,7 +121,7 @@ void dst_env_var(DstTable *env, const char *name, Dst val) {
DstArray *array = dst_array(1); DstArray *array = dst_array(1);
DstTable *subt = dst_table(1); DstTable *subt = dst_table(1);
dst_array_push(array, val); dst_array_push(array, val);
dst_table_put(subt, dst_csymbolv("ref"), dst_wrap_array(array)); dst_table_put(subt, dst_csymbolv(":ref"), dst_wrap_array(array));
dst_table_put(env, dst_csymbolv(name), dst_wrap_table(subt)); dst_table_put(env, dst_csymbolv(name), dst_wrap_table(subt));
} }
@ -139,11 +139,11 @@ Dst dst_env_resolve(DstTable *env, const char *name) {
Dst ref; Dst ref;
Dst entry = dst_table_get(env, dst_csymbolv(name)); Dst entry = dst_table_get(env, dst_csymbolv(name));
if (dst_checktype(entry, DST_NIL)) return dst_wrap_nil(); if (dst_checktype(entry, DST_NIL)) return dst_wrap_nil();
ref = dst_get(entry, dst_csymbolv("ref")); ref = dst_get(entry, dst_csymbolv(":ref"));
if (dst_checktype(ref, DST_ARRAY)) { if (dst_checktype(ref, DST_ARRAY)) {
return dst_getindex(ref, 0); return dst_getindex(ref, 0);
} }
return dst_get(entry, dst_csymbolv("value")); return dst_get(entry, dst_csymbolv(":value"));
} }
/* Get module from the arguments passed to library */ /* Get module from the arguments passed to library */

View File

@ -112,6 +112,7 @@ int dst_lib_math(DstArgs args);
int dst_lib_array(DstArgs args); int dst_lib_array(DstArgs args);
int dst_lib_tuple(DstArgs args); int dst_lib_tuple(DstArgs args);
int dst_lib_buffer(DstArgs args); int dst_lib_buffer(DstArgs args);
int dst_lib_table(DstArgs args);
/* Useful for compiler */ /* Useful for compiler */
Dst dst_op_add(Dst lhs, Dst rhs); Dst dst_op_add(Dst lhs, Dst rhs);

View File

@ -152,6 +152,7 @@ static Dst astunwrap_table(DstTable *other) {
} }
if (!prescan) return dst_wrap_table(other); if (!prescan) return dst_wrap_table(other);
table = dst_table(other->capacity); table = dst_table(other->capacity);
table->proto = other->proto;
iter = NULL; iter = NULL;
while ((iter = dst_table_next(other, iter))) { while ((iter = dst_table_next(other, iter))) {
if (iter == prescan) break; if (iter == prescan) break;