DISABLE FDs (REMOVE ME).
[jackhill/mal.git] / pike / stepA_mal.pike
CommitLineData
a04e7a78
DM
1import .Env;
2import .Printer;
3import .Reader;
4import .Readline;
5import .Types;
6
7Val READ(string str)
8{
9 return read_str(str);
10}
11
12bool is_pair(Val e)
13{
14 return e.is_sequence && !e.emptyp();
15}
16
17Val quasiquote(Val ast)
18{
19 if(!is_pair(ast)) return List(({ Symbol("quote"), ast }));
20 Val ast0 = ast.data[0];
21 if(ast0.mal_type == MALTYPE_SYMBOL && ast0.value == "unquote") return ast.data[1];
22 if(is_pair(ast0) && ast0.data[0].mal_type == MALTYPE_SYMBOL && ast0.data[0].value == "splice-unquote")
23 {
24 return List(({ Symbol("concat"), ast0.data[1], quasiquote(ast.rest()) }));
25 }
26 return List(({ Symbol("cons"), quasiquote(ast0), quasiquote(ast.rest()) }));
27}
28
29bool is_macro_call(Val ast, Env env)
30{
31 if(ast.mal_type == MALTYPE_LIST &&
32 !ast.emptyp() &&
33 ast.data[0].mal_type == MALTYPE_SYMBOL &&
34 env.find(ast.data[0]))
35 {
36 Val v = env.get(ast.data[0]);
37 if(objectp(v) && v.macro) return true;
38 }
39 return false;
40}
41
42Val macroexpand(Val ast, Env env)
43{
44 while(is_macro_call(ast, env))
45 {
46 Val macro = env.get(ast.data[0]);
47 ast = macro(@ast.data[1..]);
48 }
49 return ast;
50}
51
52Val eval_ast(Val ast, Env env)
53{
54 switch(ast.mal_type)
55 {
56 case MALTYPE_SYMBOL:
57 return env.get(ast);
58 case MALTYPE_LIST:
59 return List(map(ast.data, lambda(Val e) { return EVAL(e, env); }));
60 case MALTYPE_VECTOR:
61 return Vector(map(ast.data, lambda(Val e) { return EVAL(e, env); }));
62 case MALTYPE_MAP:
63 array(Val) elements = ({ });
64 foreach(ast.data; Val k; Val v)
65 {
66 elements += ({ k, EVAL(v, env) });
67 }
68 return Map(elements);
69 default:
70 return ast;
71 }
72}
73
74Val EVAL(Val ast, Env env)
75{
76 while(true)
77 {
78 if(ast.mal_type != MALTYPE_LIST) return eval_ast(ast, env);
79 ast = macroexpand(ast, env);
80 if(ast.mal_type != MALTYPE_LIST) return eval_ast(ast, env);
81 if(ast.emptyp()) return ast;
82 if(ast.data[0].mal_type == MALTYPE_SYMBOL) {
83 switch(ast.data[0].value)
84 {
85 case "def!":
86 return env.set(ast.data[1], EVAL(ast.data[2], env));
87 case "let*":
88 Env let_env = Env(env);
89 Val ast1 = ast.data[1];
90 for(int i = 0; i < sizeof(ast1.data); i += 2)
91 {
92 let_env.set(ast1.data[i], EVAL(ast1.data[i + 1], let_env));
93 }
94 env = let_env;
95 ast = ast.data[2];
96 continue; // TCO
97 case "quote":
98 return ast.data[1];
99 case "quasiquote":
100 ast = quasiquote(ast.data[1]);
101 continue; // TCO
102 case "defmacro!":
103 Val macro = EVAL(ast.data[2], env).clone_as_macro();
104 return env.set(ast.data[1], macro);
105 case "macroexpand":
106 return macroexpand(ast.data[1], env);
107 case "try*":
108 if(ast.count() < 3) return EVAL(ast.data[1], env);
109 if(mixed err = catch { return EVAL(ast.data[1], env); } )
110 {
111 Val err_val;
112 if(objectp(err)) err_val = err;
113 else if(stringp(err)) err_val = String(err);
114 else if(arrayp(err)) err_val = String(err[0]);
115 Val catch_clause = ast.data[2];
116 Env catch_env = Env(env);
117 catch_env.set(catch_clause.data[1], err_val);
118 return EVAL(catch_clause.data[2], catch_env);
119 }
120 case "do":
121 Val result;
122 foreach(ast.data[1..(sizeof(ast.data) - 2)], Val element)
123 {
124 result = EVAL(element, env);
125 }
126 ast = ast.data[-1];
127 continue; // TCO
128 case "if":
129 Val cond = EVAL(ast.data[1], env);
130 if(cond.mal_type == MALTYPE_FALSE || cond.mal_type == MALTYPE_NIL)
131 {
132 if(sizeof(ast.data) > 3)
133 ast = ast.data[3];
134 else
135 return MAL_NIL;
136 }
137 else
138 ast = ast.data[2];
139 continue; // TCO
140 case "fn*":
141 return Fn(ast.data[2], ast.data[1], env,
142 lambda(Val ... a) { return EVAL(ast.data[2], Env(env, ast.data[1], List(a))); });
143 }
144 }
145 Val evaled_ast = eval_ast(ast, env);
146 Val f = evaled_ast.data[0];
147 switch(f.mal_type)
148 {
149 case MALTYPE_BUILTINFN:
150 return f(@evaled_ast.data[1..]);
151 case MALTYPE_FN:
152 ast = f.ast;
153 env = Env(f.env, f.params, List(evaled_ast.data[1..]));
154 continue; // TCO
155 default:
156 throw("Unknown function type");
157 }
158 }
159}
160
161string PRINT(Val exp)
162{
163 return pr_str(exp, true);
164}
165
166string rep(string str, Env env)
167{
168 return PRINT(EVAL(READ(str), env));
169}
170
171int main(int argc, array argv)
172{
173 Env repl_env = Env(0);
174 foreach(.Core.NS(); Val k; Val v) repl_env.set(k, v);
175 repl_env.set(Symbol("eval"), BuiltinFn("eval", lambda(Val a) { return EVAL(a, repl_env); }));
176 repl_env.set(Symbol("*ARGV*"), List(map(argv[2..], String)));
177 rep("(def! *host-language* \"pike\")", repl_env);
178 rep("(def! not (fn* (a) (if a false true)))", repl_env);
179 rep("(def! load-file (fn* (f) (eval (read-string (str \"(do \" (slurp f) \"\nnil)\")))))", repl_env);
180 rep("(defmacro! cond (fn* (& xs) (if (> (count xs) 0) (list 'if (first xs) (if (> (count xs) 1) (nth xs 1) (throw \"odd number of forms to cond\")) (cons 'cond (rest (rest xs)))))))", repl_env);
181 if(argc >= 2)
182 {
183 rep("(load-file \"" + argv[1] + "\")", repl_env);
184 return 0;
185 }
186 rep("(println (str \"Mal [\" \*host-language\* \"]\"))", repl_env);
187 while(1)
188 {
189 string line = readline("user> ");
190 if(!line) break;
191 if(strlen(line) == 0) continue;
192 if(mixed err = catch { write(({ rep(line, repl_env), "\n" })); } )
193 {
194 if(arrayp(err))
195 {
196 err = err[0];
197 }
198 else if(objectp(err))
199 {
200 err = err.to_string(true);
201 }
202 write(({ "Error: ", err, "\n" }));
203 }
204 }
205 write("\n");
206 return 0;
207}