forked from kanaka/mal
-
Notifications
You must be signed in to change notification settings - Fork 0
/
step7_quote.pl
120 lines (108 loc) · 3.05 KB
/
step7_quote.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use v6;
use lib IO::Path.new($?FILE).dirname;
use reader;
use printer;
use types;
use env;
use core;
sub read ($str) {
return read_str($str);
}
sub eval_ast ($ast, $env) {
given $ast {
when MalSymbol { $env.get($ast.val) || die X::MalNotFound.new(name => $ast.val) }
when MalList { MalList([$ast.map({ eval($_, $env) })]) }
when MalVector { MalVector([$ast.map({ eval($_, $env) })]) }
when MalHashMap { MalHashMap($ast.kv.map({ $^a => eval($^b, $env) }).Hash) }
default { $ast // $NIL }
}
}
sub is_pair ($ast) {
return so $ast ~~ MalList|MalVector && $ast.elems;
}
sub quasiquote ($ast) {
if !is_pair($ast) {
return MalList([MalSymbol('quote'), $ast]);
}
elsif $ast[0] ~~ MalSymbol && $ast[0].val eq 'unquote' {
return $ast[1];
}
elsif is_pair($ast[0]) && $ast[0][0] ~~ MalSymbol && $ast[0][0].val eq 'splice-unquote' {
return MalList([MalSymbol('concat'), $ast[0][1], quasiquote(MalList([$ast[1..*]]))]);
}
else {
return MalList([MalSymbol('cons'), quasiquote($ast[0]), quasiquote(MalList([$ast[1..*]]))]);
}
}
sub eval ($ast is copy, $env is copy) {
loop {
return eval_ast($ast, $env) if $ast !~~ MalList;
return $ast if !$ast.elems;
my ($a0, $a1, $a2, $a3) = $ast.val;
given $a0.val {
when 'def!' {
return $env.set($a1.val, eval($a2, $env));
}
when 'let*' {
my $new_env = MalEnv.new($env);
for |$a1.val -> $key, $value {
$new_env.set($key.val, eval($value, $new_env));
}
$env = $new_env;
$ast = $a2;
}
when 'do' {
eval_ast(MalList([$ast[1..*-2]]), $env);
$ast = $ast[*-1];
}
when 'if' {
if eval($a1, $env) ~~ MalNil|MalFalse {
return $NIL if $a3 ~~ $NIL;
$ast = $a3;
}
else {
$ast = $a2;
}
}
when 'fn*' {
my @binds = $a1 ?? $a1.map(*.val) !! ();
my &fn = -> *@args {
eval($a2, MalEnv.new($env, @binds, @args));
};
return MalFunction($a2, $env, @binds, &fn);
}
when 'quote' { return $a1 }
when 'quasiquote' { $ast = quasiquote($a1) }
default {
my ($func, @args) = eval_ast($ast, $env).val;
return $func.apply(|@args) if $func !~~ MalFunction;
$ast = $func.ast;
$env = MalEnv.new($func.env, $func.params, @args);
}
}
}
}
sub print ($exp) {
return pr_str($exp, True);
}
my $repl_env = MalEnv.new;
sub rep ($str) {
return print(eval(read($str), $repl_env));
}
sub MAIN ($source_file?, *@args) {
$repl_env.set(.key, .value) for %core::ns;
$repl_env.set('eval', MalCode({ eval($^a, $repl_env) }));
$repl_env.set('*ARGV*', MalList([@args.map({ MalString($_) })]));
rep(q{(def! not (fn* (a) (if a false true)))});
rep(q{(def! load-file (fn* (f) (eval (read-string (str "(do " (slurp f) ")")))))});
if ($source_file.defined) {
rep("(load-file \"$source_file\")");
exit;
}
while (my $line = prompt 'user> ').defined {
say rep($line);
CATCH {
when X::MalException { .Str.say }
}
}
}