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