1
1
mirror of https://github.com/kanaka/mal.git synced 2024-08-18 02:00:40 +03:00
mal/impls/vala/step4_if_fn_do.vala
Joel Martin 8a19f60386 Move implementations into impls/ dir
- Reorder README to have implementation list after "learning tool"
  bullet.

- This also moves tests/ and libs/ into impls. It would be preferrable
  to have these directories at the top level.  However, this causes
  difficulties with the wasm implementations which need pre-open
  directories and have trouble with paths starting with "../../". So
  in lieu of that, symlink those directories to the top-level.

- Move the run_argv_test.sh script into the tests directory for
  general hygiene.
2020-02-10 23:50:16 -06:00

222 lines
8.3 KiB
Vala

class Mal.Main: GLib.Object {
static bool eof;
static construct {
eof = false;
}
public static Mal.Val? READ() {
string? line = Readline.readline("user> ");
if (line != null) {
if (line.length > 0)
Readline.History.add(line);
try {
return Reader.read_str(line);
} catch (Mal.Error err) {
GLib.stderr.printf("%s\n", err.message);
return null;
}
} else {
stdout.printf("\n");
eof = true;
return null;
}
}
public static Mal.Val eval_ast(Mal.Val ast, Mal.Env env)
throws Mal.Error {
var roota = new GC.Root(ast); (void)roota;
var roote = new GC.Root(env); (void)roote;
if (ast is Mal.Sym)
return env.get(ast as Mal.Sym);
if (ast is Mal.List) {
var result = new Mal.List.empty();
var root = new GC.Root(result); (void)root;
foreach (var elt in (ast as Mal.List).vs)
result.vs.append(EVAL(elt, env));
return result;
}
if (ast is Mal.Vector) {
var vec = ast as Mal.Vector;
var result = new Mal.Vector.with_size(vec.length);
var root = new GC.Root(result); (void)root;
for (var i = 0; i < vec.length; i++)
result[i] = EVAL(vec[i], env);
return result;
}
if (ast is Mal.Hashmap) {
var result = new Mal.Hashmap();
var root = new GC.Root(result); (void)root;
var map = (ast as Mal.Hashmap).vs;
foreach (var key in map.get_keys())
result.insert(key, EVAL(map[key], env));
return result;
}
return ast;
}
private static Mal.Val define_eval(Mal.Val key, Mal.Val value,
Mal.Env env)
throws Mal.Error {
var rootk = new GC.Root(key); (void)rootk;
var roote = new GC.Root(env); (void)roote;
var symkey = key as Mal.Sym;
if (symkey == null)
throw new Mal.Error.BAD_PARAMS(
"let*: expected a symbol to define");
var val = EVAL(value, env);
env.set(symkey, val);
return val;
}
public static Mal.Val EVAL(Mal.Val ast, Mal.Env env)
throws Mal.Error {
var ast_root = new GC.Root(ast); (void)ast_root;
var env_root = new GC.Root(env); (void)env_root;
GC.Core.maybe_collect();
if (ast is Mal.List) {
unowned GLib.List<Mal.Val> list = (ast as Mal.List).vs;
if (list.first() == null)
return ast;
var first = list.first().data;
if (first is Mal.Sym) {
var sym = first as Mal.Sym;
switch (sym.v) {
case "def!":
if (list.length() != 3)
throw new Mal.Error.BAD_PARAMS(
"def!: expected two values");
return define_eval(list.next.data, list.next.next.data,
env);
case "let*":
if (list.length() != 3)
throw new Mal.Error.BAD_PARAMS(
"let*: expected two values");
var defns = list.nth(1).data;
var newenv = new Mal.Env.within(env);
if (defns is Mal.List) {
for (unowned GLib.List<Mal.Val> iter =
(defns as Mal.List).vs;
iter != null; iter = iter.next.next) {
if (iter.next == null)
throw new Mal.Error.BAD_PARAMS(
"let*: expected an even-length list" +
" of definitions");
define_eval(iter.data, iter.next.data, newenv);
}
} else if (defns is Mal.Vector) {
var vec = defns as Mal.Vector;
if (vec.length % 2 != 0)
throw new Mal.Error.BAD_PARAMS(
"let*: expected an even-length vector" +
" of definitions");
for (var i = 0; i < vec.length; i += 2)
define_eval(vec[i], vec[i+1], newenv);
} else {
throw new Mal.Error.BAD_PARAMS(
"let*: expected a list or vector of definitions");
}
return EVAL(list.nth(2).data, newenv);
case "do":
Mal.Val result = null;
for (list = list.next; list != null; list = list.next)
result = EVAL(list.data, env);
if (result == null)
throw new Mal.Error.BAD_PARAMS(
"do: expected at least one argument");
return result;
case "if":
if (list.length() != 3 && list.length() != 4)
throw new Mal.Error.BAD_PARAMS(
"if: expected two or three arguments");
list = list.next;
var cond = EVAL(list.data, env);
list = list.next;
if (!cond.truth_value()) {
// Skip to the else clause, which defaults to nil.
list = list.next;
if (list == null)
return new Mal.Nil();
}
return EVAL(list.data, env);
case "fn*":
if (list.length() != 3)
throw new Mal.Error.BAD_PARAMS(
"fn*: expected two arguments");
var binds = list.next.data as Mal.Listlike;
var body = list.next.next.data;
if (binds == null)
throw new Mal.Error.BAD_PARAMS(
"fn*: expected a list of parameter names");
for (var iter = binds.iter(); iter.nonempty(); iter.step())
if (!(iter.deref() is Mal.Sym))
throw new Mal.Error.BAD_PARAMS(
"fn*: expected parameter name to be "+
"symbol");
return new Mal.Function(binds, body, env);
}
}
var newlist = eval_ast(ast, env) as Mal.List;
unowned GLib.List<Mal.Val> firstlink = newlist.vs.first();
Mal.Val firstdata = firstlink.data;
newlist.vs.remove_link(firstlink);
if (firstdata is Mal.BuiltinFunction) {
return (firstdata as Mal.BuiltinFunction).call(newlist);
} else if (firstdata is Mal.Function) {
var fn = firstdata as Mal.Function;
var newenv = new Mal.Env.funcall(
fn.env, fn.parameters, newlist);
return EVAL(fn.body, newenv);
} else {
throw new Mal.Error.CANNOT_APPLY(
"bad value at start of list");
}
} else {
return eval_ast(ast, env);
}
}
public static void PRINT(Mal.Val value) {
stdout.printf("%s\n", pr_str(value));
}
public static void rep(Mal.Env env) throws Mal.Error {
Mal.Val? val = READ();
if (val != null) {
val = EVAL(val, env);
PRINT(val);
}
}
public static int main(string[] args) {
var env = new Mal.Env();
var root = new GC.Root(env); (void)root;
Mal.Core.make_ns();
foreach (var key in Mal.Core.ns.get_keys())
env.set(new Mal.Sym(key), Mal.Core.ns[key]);
try {
EVAL(Mal.Reader.read_str("(def! not (fn* (a) (if a false true)))"),
env);
} catch (Mal.Error err) {
assert(false); // shouldn't happen
}
while (!eof) {
try {
rep(env);
} catch (Mal.Error err) {
GLib.stderr.printf("%s\n", err.message);
}
}
return 0;
}
}