OLD | NEW |
| (Empty) |
1 part of lisp; | |
2 | |
3 /** | |
4 * The native functions. | |
5 */ | |
6 class Natives { | |
7 | |
8 /** Imports the native functions into the [environment]. */ | |
9 static Environment import(Environment environment) { | |
10 | |
11 // basic functions | |
12 environment.define(new Name('define'), _define); | |
13 environment.define(new Name('lambda'), _lambda); | |
14 environment.define(new Name('quote'), _quote); | |
15 environment.define(new Name('eval'), _eval); | |
16 environment.define(new Name('apply'), _apply); | |
17 environment.define(new Name('let'), _let); | |
18 environment.define(new Name('set!'), _set); | |
19 environment.define(new Name('print'), _print); | |
20 | |
21 // control structures | |
22 environment.define(new Name('if'), _if); | |
23 environment.define(new Name('while'), _while); | |
24 environment.define(new Name('and'), _and); | |
25 environment.define(new Name('or'), _or); | |
26 environment.define(new Name('not'), _not); | |
27 | |
28 // arithmetic operators | |
29 environment.define(new Name('+'), _plus); | |
30 environment.define(new Name('-'), _minus); | |
31 environment.define(new Name('*'), _multiply); | |
32 environment.define(new Name('/'), _divide); | |
33 environment.define(new Name('%'), _modulo); | |
34 | |
35 // arithmetic comparators | |
36 environment.define(new Name('<'), _smaller); | |
37 environment.define(new Name('<='), _smallerOrEqual); | |
38 environment.define(new Name('='), _equal); | |
39 environment.define(new Name('!='), _notEqual); | |
40 environment.define(new Name('>'), _larger); | |
41 environment.define(new Name('>='), _largerOrEqual); | |
42 | |
43 // list operators | |
44 environment.define(new Name('cons'), _cons); | |
45 environment.define(new Name('car'), _car); | |
46 environment.define(new Name('car!'), _carSet); | |
47 environment.define(new Name('cdr'), _cdr); | |
48 environment.define(new Name('cdr!'), _cdrSet); | |
49 | |
50 return environment; | |
51 } | |
52 | |
53 static _define(Environment env, args) { | |
54 if (args.head is Name) { | |
55 return env.define(args.head, evalList(env, args.tail)); | |
56 } else if (args.head.head is Name) { | |
57 return env.define( | |
58 args.head.head, _lambda(env, new Cons(args.head.tail, args.tail))); | |
59 } else { | |
60 throw new ArgumentError('Invalid define: $args'); | |
61 } | |
62 } | |
63 | |
64 static _lambda(Environment lambda_env, lambda_args) { | |
65 return (Environment env, args) { | |
66 var inner = lambda_env.create(); | |
67 var names = lambda_args.head; | |
68 var values = evalArguments(env, args); | |
69 while (names != null && values != null) { | |
70 inner.define(names.head, values.head); | |
71 names = names.tail; | |
72 values = values.tail; | |
73 } | |
74 return evalList(inner, lambda_args.tail); | |
75 }; | |
76 } | |
77 | |
78 static _quote(Environment env, args) { | |
79 return args; | |
80 } | |
81 | |
82 static _eval(Environment env, args) { | |
83 return eval(env.create(), eval(env, args.head)); | |
84 } | |
85 | |
86 static _apply(Environment env, args) { | |
87 return eval(env, args.head)(env.create(), args.tail); | |
88 } | |
89 | |
90 static _let(Environment env, args) { | |
91 var inner = env.create(); | |
92 var binding = args.head; | |
93 while (binding != null) { | |
94 inner.define(binding.head.head, eval(env, binding.head.tail.head)); | |
95 binding = binding.tail; | |
96 } | |
97 return evalList(inner, args.tail); | |
98 } | |
99 | |
100 static _set(Environment env, args) { | |
101 return env[args.head] = eval(env, args.tail.head); | |
102 } | |
103 | |
104 static _print(Environment env, args) { | |
105 var buffer = new StringBuffer(); | |
106 while (args != null) { | |
107 buffer.write(eval(env, args.head)); | |
108 args = args.tail; | |
109 } | |
110 print(buffer); | |
111 return null; | |
112 } | |
113 | |
114 static _if(Environment env, args) { | |
115 var condition = eval(env, args.head); | |
116 if (condition) { | |
117 if (args.tail != null) { | |
118 return eval(env, args.tail.head); | |
119 } | |
120 } else { | |
121 if (args.tail != null && args.tail.tail != null) { | |
122 return eval(env, args.tail.tail.head); | |
123 } | |
124 } | |
125 return null; | |
126 } | |
127 | |
128 static _while(Environment env, args) { | |
129 var result = null; | |
130 while (eval(env, args.head)) { | |
131 result = evalList(env, args.tail); | |
132 } | |
133 return result; | |
134 } | |
135 | |
136 static _and(Environment env, args) { | |
137 while (args != null) { | |
138 if (!eval(env, args.head)) { | |
139 return false; | |
140 } | |
141 args = args.tail; | |
142 } | |
143 return true; | |
144 } | |
145 | |
146 static _or(Environment env, args) { | |
147 while (args != null) { | |
148 if (eval(env, args.head)) { | |
149 return true; | |
150 } | |
151 args = args.tail; | |
152 } | |
153 return false; | |
154 } | |
155 | |
156 static _not(Environment env, args) { | |
157 return !eval(env, args.head); | |
158 } | |
159 | |
160 static _plus(Environment env, args) { | |
161 var value = eval(env, args.head); | |
162 for (args = args.tail; args != null; args = args.tail) { | |
163 value += eval(env, args.head); | |
164 } | |
165 return value; | |
166 } | |
167 | |
168 static _minus(Environment env, args) { | |
169 var value = eval(env, args.head); | |
170 if (args.tail == null) { | |
171 return -value; | |
172 } | |
173 for (args = args.tail; args != null; args = args.tail) { | |
174 value -= eval(env, args.head); | |
175 } | |
176 return value; | |
177 } | |
178 | |
179 static _multiply(Environment env, args) { | |
180 var value = eval(env, args.head); | |
181 for (args = args.tail; args != null; args = args.tail) { | |
182 value *= eval(env, args.head); | |
183 } | |
184 return value; | |
185 } | |
186 | |
187 static _divide(Environment env, args) { | |
188 var value = eval(env, args.head); | |
189 for (args = args.tail; args != null; args = args.tail) { | |
190 value /= eval(env, args.head); | |
191 } | |
192 return value; | |
193 } | |
194 | |
195 static _modulo(Environment env, args) { | |
196 var value = eval(env, args.head); | |
197 for (args = args.tail; args != null; args = args.tail) { | |
198 value %= eval(env, args.head); | |
199 } | |
200 return value; | |
201 } | |
202 | |
203 static _smaller(Environment env, args) { | |
204 return eval(env, args.head) < eval(env, args.tail.head); | |
205 } | |
206 | |
207 static _smallerOrEqual(Environment env, args) { | |
208 return eval(env, args.head) <= eval(env, args.tail.head); | |
209 } | |
210 | |
211 static _equal(Environment env, args) { | |
212 return eval(env, args.head) == eval(env, args.tail.head); | |
213 } | |
214 | |
215 static _notEqual(Environment env, args) { | |
216 return eval(env, args.head) != eval(env, args.tail.head); | |
217 } | |
218 | |
219 static _larger(Environment env, args) { | |
220 return eval(env, args.head) > eval(env, args.tail.head); | |
221 } | |
222 | |
223 static _largerOrEqual(Environment env, args) { | |
224 return eval(env, args.head) >= eval(env, args.tail.head); | |
225 } | |
226 | |
227 static _cons(Environment env, args) { | |
228 return new Cons(eval(env, args.head), eval(env, args.tail.head)); | |
229 } | |
230 | |
231 static _car(Environment env, args) { | |
232 var cons = eval(env, args.head); | |
233 return cons is Cons ? cons.head : null; | |
234 } | |
235 | |
236 static _carSet(Environment env, args) { | |
237 var cons = eval(env, args.head); | |
238 if (cons is Cons) { | |
239 cons.head = eval(env, args.tail.head); | |
240 } | |
241 return cons; | |
242 } | |
243 | |
244 static _cdr(Environment env, args) { | |
245 var cons = eval(env, args.head); | |
246 return cons is Cons ? cons.tail : null; | |
247 } | |
248 | |
249 static _cdrSet(Environment env, args) { | |
250 var cons = eval(env, args.head); | |
251 if (cons is Cons) { | |
252 cons.tail = eval(env, args.tail.head); | |
253 } | |
254 return cons; | |
255 } | |
256 } | |
OLD | NEW |