├── LICENSE.txt ├── README.md ├── base.js ├── ck ├── ck.js ├── ckeval.js └── editor.html ├── editor.css ├── editor.html ├── editor.js ├── eval.js ├── examples ├── editor.css ├── editor.html ├── editor.js ├── quines.html ├── quines.js ├── veneer.css └── veneer.js ├── immutable.min.js ├── mk.js ├── mk_test.js ├── reader.js ├── resources ├── closebrackets.js ├── codemirror.css ├── codemirror.js ├── matchbrackets.js └── scheme.js ├── veneer.css └── veneer.js /LICENSE.txt: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2015 tca 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # veneer 2 | 3 | [Browser based miniKanren editor and REPL](http://tca.github.io/veneer/examples/editor.html) 4 | -------------------------------------------------------------------------------- /base.js: -------------------------------------------------------------------------------- 1 | function Pair(car, cdr) { 2 | this.car = car; 3 | this.cdr = cdr; 4 | } 5 | function car(x) { return x.car; } 6 | function cdr(x) { return x.cdr; } 7 | function pairp(x) { return x instanceof Pair; } 8 | function cons(car, cdr) { return new Pair(car, cdr); } 9 | 10 | 11 | var gensym_counter = 0; 12 | var symbol_table = new Object(null); 13 | 14 | function Symbol(str) { this.string = str; } 15 | 16 | function gensym(str) { return new Symbol([str, "@@", gensym_counter++].join("")); } 17 | 18 | function intern(string) { 19 | if (symbol_table.hasOwnProperty(string)) { 20 | return symbol_table[string]; 21 | } else { 22 | var sym = new Symbol(string); 23 | symbol_table[string] = sym; 24 | return sym; 25 | } 26 | } 27 | 28 | function nullp(x) { return x === null; } 29 | function procedurep(x) { return x instanceof Function; } 30 | function symbolp(x) { return x instanceof Symbol; } 31 | function numberp(t) { return typeof t === "number"; } 32 | function booleanp(t) { return typeof t === "boolean"; } 33 | function stringp(t) { return typeof t === "string"; } 34 | function constantp(t) { return numberp(t) || booleanp(t) || stringp(t); } 35 | 36 | function not(x) { return x === false; } 37 | 38 | function quotient(a, b) { return Math.trunc(a/b); } 39 | function oddp(n) { return (n % 2) == 1; } 40 | function evenp(n) { return (n % 2) == 0; } 41 | function zerop(n) { return n === 0; } 42 | 43 | function Ref(v) { 44 | this.val = v; 45 | } 46 | Ref.prototype.get = function() { return this.val; }; 47 | Ref.prototype.set = function(v1) { this.val = v1; }; 48 | 49 | function ref(v) { return new Ref(v); } 50 | 51 | function anyp(pred, xs) { 52 | while(xs !== null) { 53 | if (pred(xs.car)) { return xs.car; } 54 | else { xs = xs.cdr; } 55 | } return false; 56 | } 57 | 58 | function memq(x, xs) { 59 | return anyp(function(y) { return x === y; }, xs); 60 | } 61 | 62 | function assp(pred, xs) { 63 | while(xs !== null) { 64 | if (pred(xs.car.car)) { return xs.car; } 65 | else { xs = xs.cdr; } 66 | } return false; 67 | } 68 | 69 | function length(lst) { 70 | var memo = 0; 71 | while(lst !== null) { memo++; lst = lst.cdr; } 72 | return memo; 73 | } 74 | 75 | function nth(xs, n) { 76 | while(--n > 0) { xs = xs.cdr; } 77 | return xs.car; 78 | } 79 | 80 | function reverse_aux(lst, memo) { 81 | while(lst !== null) { 82 | memo = cons(lst.car, memo); 83 | lst = lst.cdr; 84 | } return memo; 85 | } 86 | 87 | function reverse(lst) { 88 | return reverse_aux(lst, null); 89 | } 90 | 91 | function append(as, bs) { 92 | if(as === null) { 93 | return bs; 94 | } else { 95 | return cons(as.car, append(as.cdr, bs)); 96 | } 97 | } 98 | 99 | function make_list(len, elem) { 100 | var accum = null; 101 | while (len-- > 0) { 102 | accum = cons(elem, accum); 103 | } return accum; 104 | } 105 | 106 | function array_slice_list(arr, from) { 107 | var i = arr.length; 108 | var memo = null; 109 | while(i > from) { memo = cons(arr[--i], memo); } 110 | return memo; 111 | } 112 | 113 | function array_to_list(arr) { 114 | var i = arr.length; 115 | var memo = null; 116 | while(i > 0) { memo = cons(arr[--i], memo); } 117 | return memo; 118 | } 119 | 120 | function map(fn, lst) { 121 | var memo = []; 122 | while(lst !== null) { 123 | memo.push(fn(lst.car)); 124 | lst = lst.cdr; 125 | } return array_to_list(memo); 126 | } 127 | 128 | function list() { 129 | return array_to_list(arguments); 130 | } 131 | 132 | function foldl(xs, m, fn) { 133 | while(xs !== null) { 134 | m = fn(m, xs.car); 135 | xs = xs.cdr; 136 | } return m; 137 | } 138 | 139 | function foldr(xs, m, fn) { 140 | return (xs === null) ? m : fn(xs.car, foldr(xs.cdr, m, fn)); 141 | } 142 | 143 | function assq(x, xs) { 144 | while(xs !== null) { 145 | if (x === xs.car.car) { return xs.car; } 146 | else { xs = xs.cdr; } 147 | } return false; 148 | } 149 | 150 | function pretty_print(x) { 151 | if (numberp(x)) { 152 | return x.toString(); 153 | } else if (booleanp(x)) { 154 | return x ? "#t" : "#f"; 155 | } else if (x === null) { 156 | return "()"; 157 | } else if (typeof x === "string") { 158 | return ["\"", x, "\""].join(""); 159 | } else if (symbolp(x)) { 160 | return x.string; 161 | } else if (pairp(x)) { 162 | var cur = x; 163 | var memo = []; 164 | while(true) { 165 | if(pairp(cur.cdr)) { 166 | memo.push(pretty_print(cur.car)); 167 | cur = cur.cdr; 168 | } else if (cur.cdr === null) { 169 | memo.push(pretty_print(cur.car)); 170 | break; 171 | } else { 172 | memo.push(pretty_print(cur.car)); 173 | memo.push("."); 174 | memo.push(pretty_print(cur.cdr)); 175 | break; 176 | } 177 | } return ["(", memo.join(" "), ")"].join(""); 178 | } else if (x && x.toString && typeof x.toString === "function") { 179 | return x.toString(); 180 | } else { 181 | return JSON.stringify(x); 182 | } 183 | } 184 | -------------------------------------------------------------------------------- /ck/ck.js: -------------------------------------------------------------------------------- 1 | function Var(c) { this.c = c; } 2 | function mkvar(c) { return new Var(c); } 3 | function varp(x) { return (x instanceof Var); } 4 | function vareq(x1, x2) { return x1.c === x2.c; } 5 | 6 | function MiniKanrenState(s, c, p, cs) { 7 | this.substitution = s; 8 | this.counter = c; 9 | this.prefix = p; 10 | this.constraint_store = cs; 11 | } 12 | function Mks(s, c, p, cs) { return new MiniKanrenState(s, c, p, cs); } 13 | 14 | var not_found = []; 15 | function walk(u, s) { 16 | var pr; 17 | while(varp(u)) { 18 | pr = s.get(u.c, not_found); 19 | if (pr === not_found) { 20 | return u; 21 | } else { 22 | u = pr; 23 | } 24 | } return u; 25 | } 26 | 27 | function occurs_check(x, v, s) { 28 | var v1 = walk(v, s); 29 | if(varp(v1)) { 30 | return vareq(v1, x); 31 | } else if (pairp(v1)) { 32 | return occurs_check(x, v1.car, s) || occurs_check(x, v1.cdr, s); 33 | } else { 34 | return false; 35 | } 36 | } 37 | 38 | function unify(u, v, mks) { 39 | var s = mks.substitution; 40 | var p = mks.prefix; 41 | var u1 = walk(u, s); 42 | var v1 = walk(v, s); 43 | if (varp(u1) && varp(v1) && vareq(u1, v1)) { return mks; } 44 | else if (varp(u1)) { return ext_s_check_prefix(u1, v1, mks); } 45 | else if (varp(v1)) { return ext_s_check_prefix(v1, u1, mks); } 46 | else if (pairp(u1) && pairp(v1)) { 47 | var mks1 = unify(u1.car, v1.car, mks); 48 | return (mks1 !== false) && unify(u1.cdr, v1.cdr, mks1); 49 | } else { 50 | return (u1 === v1) && mks; 51 | } 52 | } 53 | 54 | function eqeq(u, v) { 55 | return function(mks) { 56 | var mks1 = unify(u, v, mks); 57 | return mks1 === false ? mzero : run_constraints(mks1); 58 | }; 59 | } 60 | 61 | var empty_set = Immutable.Set(); 62 | function ext_disequality(u, v, mks) { 63 | var cs = mks.constraint_store; 64 | return Mks(mks.substitution, mks.counter, mks.prefix, 65 | cs.updateIn([u.c, 1], empty_set, function(ds) { return ds.add(v); })); 66 | } 67 | function rem_disequality(u, v, mks) { 68 | var ds1 = mks.constraint_store.getIn([u.c, 1], false); 69 | if (ds1) { 70 | return Mks(mks.substitution, mks.counter, mks.prefix, mks.constraint_store.setIn([u.c, 1], ds1.delete(v))); 71 | } else { 72 | return mks; 73 | } 74 | } 75 | 76 | // need to do more for =/= between vars? 77 | function disunify(u, v, mks) { 78 | var s = mks.substitution; 79 | var p = mks.prefix; 80 | var u1 = walk(u, s); 81 | var v1 = walk(v, s); 82 | if (varp(u1) && varp(v1) && vareq(u1, v1)) { return false; } 83 | else if (varp(u1)) { return ext_disequality(u1, v1, mks); } 84 | else if (varp(v1)) { return ext_disequality(v1, u1, mks); } 85 | else if (pairp(u1) && pairp(v1)) { 86 | var mks1 = disunify(u1.car, v1.car, mks); 87 | return (mks1 !== false) && disunify(u1.cdr, v1.cdr, mks1); 88 | } else { 89 | if (u1 !== v1) { 90 | if (varp(u)) { 91 | return rem_disequality(u, v1, mks); 92 | } else { return mks; } 93 | } else { 94 | return false; 95 | } 96 | } 97 | } 98 | 99 | function disequality(u, vs, mks) { 100 | var mks1 = mks; 101 | vs.forEach(function(v) { 102 | mks1 = disunify(u, v, mks); 103 | return mks1; 104 | }); 105 | return mks1; 106 | } 107 | 108 | function noteqeq(u, v) { 109 | return function(mks) { 110 | var mks1 = disunify(u, v, mks); 111 | return mks1 !== false ? unit(mks1) : mzero; 112 | }; 113 | } 114 | 115 | var empty_map = Immutable.Map(); 116 | function type_check(v, p, mks) { 117 | var s = mks.substitution; 118 | var cs = mks.constraint_store; 119 | var v1 = walk(v, s); 120 | if (varp(v1)) { 121 | var cs_entry = cs.get(v1.c, false); 122 | var existing_type = cs_entry !== false && cs_entry.get(0, false); 123 | if (existing_type && existing_type !== p) { 124 | if(existing_type === p) { 125 | // types match, dont add extra constraint 126 | return mks; 127 | } else { 128 | // type mismatch; fail 129 | return false; 130 | } 131 | } else { 132 | // add constraint 133 | cs_entry = cs_entry || empty_map; 134 | var cs1 = cs.set(v1.c, cs_entry.set(0, p)); 135 | return Mks(s, mks.counter, mks.prefix, cs1); 136 | } 137 | } else { 138 | // ground; run test 139 | if (p(v1)) { 140 | var cs1 = cs; 141 | if (varp(v)) { 142 | // remove constraint from store; it cannot be violated 143 | var cs_entry = cs.get(v.c, false); 144 | cs1 = cs_entry ? cs.set(v.c, cs_entry.delete(0)) : cs; 145 | } 146 | return Mks(s, mks.counter, mks.prefix, cs1); 147 | } else { 148 | return false; 149 | } 150 | } 151 | } 152 | 153 | function typeo(v, p) { 154 | return function(mks) { 155 | var mks1 = type_check(v, p, mks); 156 | return mks1 === false ? mzero : unit(mks1); 157 | }; 158 | } 159 | 160 | function symbolo(s) { return typeo(s, symbolp); } 161 | function numbero(n) { return typeo(n, numberp); } 162 | 163 | function absent_check(u, v, mks) { 164 | var s = mks.substitution; 165 | 166 | var abs_f = walk(u, s); 167 | var abs_s = walk(v, s); 168 | 169 | // defer until both terms are ground 170 | if (varp(abs_f) || varp(abs_s)) { 171 | var cs2 = mks.constraint_store.updateIn([abs_f.c, 2], function(_) { return abs_s; }); 172 | return vareq(abs_s, abs_f) ? false : Mks(s, mks.counter, mks.prefix, cs2); 173 | } 174 | // split and requeue 175 | else if (pairp(abs_f)) { 176 | var mks1 = Mks(s, mks.counter, mks.prefix, mks.constraint_store.deleteIn([abs_f.c, 2])); 177 | var mks2 = absent_check(abs_f.car, abs_s, mks1); 178 | return mks2 !== false && absent_check(abs_f.car, abs_s, mks2); 179 | } 180 | // both are ground and atomic; check for eqv? 181 | else if (abs_s === abs_f) { 182 | return false; 183 | } else { 184 | // forget constraint, it can never fail 185 | return Mks(s, mks.counter, mks.prefix, mks.constraint_store.deleteIn([abs_f, 2])); 186 | } 187 | } 188 | 189 | function absento(s, f) { 190 | return function(mks) { 191 | var mks1 = absent_check(f, s, mks); 192 | return mks1 !== false ? unit(mks1) : mzero; 193 | }; 194 | } 195 | 196 | function unit(mks) { return cons(mks, mzero); } 197 | var mzero = null; 198 | 199 | function ext_s_check_prefix(x, v, mks) { 200 | var s = mks.substitution; 201 | var p = mks.prefix; 202 | if (occurs_check(x, v, s)) { 203 | return false; 204 | } else { 205 | return Mks(s.set(x.c, v), mks.counter, cons(cons(x, v), p), mks.constraint_store); 206 | } 207 | } 208 | 209 | var ctable = [type_check, disequality, absent_check]; 210 | 211 | function run_constraints(mks) { 212 | var cs = mks.constraint_store; 213 | if (cs.isEmpty()) { return unit(mks); } 214 | 215 | var s = mks.substitution; 216 | var c = mks.counter; 217 | var p = mks.prefix; 218 | var mks1 = Mks(s, c, null, cs); 219 | 220 | // TODO: add loop to run changes added by constraints ? 221 | var relevant_cs; 222 | while(p !== null) { 223 | relevant_cs = cs.get(p.car.car.c, false); 224 | if (relevant_cs) { 225 | relevant_cs.forEach(function(ct, cv) { 226 | mks1 = ctable[cv](p.car.car, ct, mks1); 227 | return mks1; 228 | }); 229 | if(!mks1) { return mzero; } 230 | } 231 | p = p.cdr; 232 | } 233 | return unit(mks1); 234 | } 235 | 236 | function call_fresh(f) { 237 | return function(mks) { 238 | var c = mks.counter; 239 | return f(mkvar(c))(Mks(mks.substitution, (c + 1), mks.prefix, mks.constraint_store)); 240 | }; 241 | } 242 | 243 | function disj(g1, g2) { 244 | return function(mks) { return mplus(g1(mks), g2(mks)); }; 245 | } 246 | function conj(g1, g2) { 247 | return function(mks) { return bind(g1(mks), g2); }; 248 | } 249 | 250 | function mplus($1, $2) { 251 | if ($1 === null) { 252 | return $2; 253 | } else if (procedurep($1)) { 254 | return function() { return mplus($2, $1()); }; 255 | } else { 256 | return cons($1.car, mplus($1.cdr, $2)); 257 | } 258 | } 259 | 260 | function bind($, g) { 261 | if ($ === null) { 262 | return mzero; 263 | } else if (procedurep($)) { 264 | return function() { return bind($(), g); }; 265 | } else { 266 | return mplus(g($.car), bind($.cdr, g)); 267 | } 268 | } 269 | 270 | function pull($) { 271 | while(procedurep($)) { 272 | $ = $(); 273 | } return $; 274 | } 275 | 276 | function reify(v, s) { 277 | var v1 = walk_star(v, s); 278 | return walk_star(v1, reify_s(v1, Immutable.Map())); 279 | } 280 | 281 | function walk_star(v, s) { 282 | var v1 = walk(v, s); 283 | if (varp(v1)) { 284 | return v1; 285 | } else if (pairp(v1)) { 286 | return cons(walk_star(v1.car, s), 287 | walk_star(v1.cdr, s)); 288 | } else { 289 | return v1; 290 | } 291 | } 292 | 293 | function reify_s(v, s) { 294 | var v1 = walk(v, s); 295 | if (varp(v1)) { 296 | return s.set(v1.c, reify_name(s.size)); 297 | } else if (pairp(v1)) { 298 | return reify_s(v1.cdr, reify_s(v1.car, s)); 299 | } else { 300 | return s; 301 | } 302 | } 303 | 304 | function reify_name(n) { 305 | return { toString: function() { return ["_", n].join("."); } }; 306 | } 307 | 308 | function pred_to_tag(p) { 309 | return assq(p, list(cons(symbolp, intern("sym")), 310 | cons(numberp, intern("num")))).cdr; 311 | } 312 | 313 | function query_map(qm, mks) { 314 | var s = mks.substitution; 315 | 316 | var q = reify(qm, s); 317 | 318 | var m = print_constraints(q, s); 319 | 320 | return m; 321 | } 322 | 323 | function print_constraints(sq, s) { 324 | var subs = foldl(sq, [], function(m, a_v) { 325 | return m.concat([a_v.car, ": ", pretty_print(a_v.cdr), "\n"]); 326 | }); 327 | 328 | return subs.join(""); 329 | } 330 | 331 | function build_num(n) { 332 | if((n % 2) === 1) { 333 | return cons(1, build_num(Math.floor((n - 1) / 2))); 334 | } else if (n !== 0 && (n % 2) === 0) { 335 | return cons(0, build_num(Math.floor(n / 2))); 336 | } else { 337 | return null; 338 | } 339 | } 340 | -------------------------------------------------------------------------------- /ck/ckeval.js: -------------------------------------------------------------------------------- 1 | function VeneerVM() { 2 | var toplevel = new Object(null); 3 | 4 | function register_define(exp) { 5 | if (pairp(exp) && exp.car === intern("define")) { 6 | var a = pairp(exp.cdr.car) ? exp.cdr.car.car : exp.cdr.car; 7 | toplevel[a.string] = ref(null); 8 | } 9 | } 10 | 11 | function quote_desugar(exp) { 12 | if (pairp(exp)) { 13 | return meta(list(meta(intern("cons"), {tag:"var"}), quote_desugar(exp.car), quote_desugar(exp.cdr)), {tag:"app-builtin"}); 14 | } else if (exp == null) { 15 | return meta(list(intern("quote"), null), { tag: "quoted" }); 16 | }else if(constantp(exp)) { 17 | return meta(exp, {tag:"const"}); 18 | } else { 19 | return meta(list(intern("quote"), exp), { tag: "quoted" }); 20 | } 21 | } 22 | 23 | function quasiquote_desugar(exp, n, env) { 24 | if (pairp(exp)) { 25 | if (exp.car === intern("unquote")) { 26 | if (n === 1) { 27 | return desugar(exp.cdr.car, env); 28 | } else { 29 | return list(intern("list"), list(intern("quote"), intern("unquote")), 30 | quasiquote_desugar(exp.cdr.car, n - 1, env)); 31 | } 32 | } else if (exp.car === intern("quasiquote")) { 33 | return list(intern("list"), list(intern("quote"), intern("quasiquote")), 34 | quasiquote_desugar(exp.cdr.car, n + 1, env)); 35 | } else { 36 | return list(intern("cons"), quasiquote_desugar(exp.car, n, env), 37 | quasiquote_desugar(exp.cdr, n, env)); 38 | } 39 | } else { 40 | return quote_desugar(exp); 41 | } 42 | } 43 | 44 | function Meta(val, meta) { 45 | this.val = val; 46 | this.meta = meta || {}; 47 | } 48 | function meta(v, meta) { return new Meta(v, meta); } 49 | 50 | function normalize_params(params) { 51 | var ps = null; 52 | var rest = null; 53 | var whole = null; 54 | while(params !== null) { 55 | if(symbolp(params)) { 56 | rest = cons(params, rest); 57 | whole = cons(params, ps); 58 | break; 59 | } else { 60 | ps = cons(params.car, ps); 61 | whole = ps; 62 | params = params.cdr; 63 | } 64 | } return { normal: reverse(ps), rest: rest, whole: reverse(whole) }; 65 | } 66 | 67 | function desugar(exp, env) { 68 | if(exp instanceof Meta) { return exp; } 69 | if(pairp(exp)) { 70 | // application 71 | var app = false; 72 | if(symbolp(exp.car) && (lookup(exp.car, env) || toplevel.hasOwnProperty(exp.car.string)) 73 | || pairp(exp.car)) { 74 | return meta(map(function(e) { return desugar(e, env); }, exp), 75 | { tag: "app" }); 76 | } 77 | // special forms & builtins 78 | switch(exp.car) { 79 | case intern("define"): 80 | var name; 81 | var body; 82 | if(pairp(exp.cdr.car)) { 83 | // (define (x y z) f) => (define x (lambda (y z) f)) 84 | name = exp.cdr.car.car; 85 | body = cons(intern("lambda"), cons(exp.cdr.car.cdr, exp.cdr.cdr)); 86 | } else { 87 | name = exp.cdr.car; 88 | body = exp.cdr.cdr.car; 89 | } 90 | return meta(list(exp.car, name, desugar(body, env)), { tag: "define" }); 91 | case intern("quote"): 92 | var val = quote_desugar(exp.cdr.car); 93 | val.meta.constp = true; 94 | return val; 95 | case intern("quasiquote"): 96 | return desugar(quasiquote_desugar(exp.cdr.car, 1, env), env); 97 | case intern("zzz"): 98 | return meta(list(exp.car, desugar(exp.cdr.car, env)), 99 | { tag: "zzz" }); 100 | case intern("conde"): 101 | var clauses = map(function(row) { return cons(intern("conj"), row); }, exp.cdr); 102 | return desugar(cons(intern("disj"), clauses), env); 103 | case intern("conj"): 104 | if (exp.cdr == null) { throw "error: empty conj"; } 105 | else if (exp.cdr.cdr == null) { 106 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 107 | return e1; 108 | } else { 109 | var e1 = list(intern("zzz"), exp.cdr.car); 110 | var e2 = cons(intern("conj"), exp.cdr.cdr); 111 | return desugar(list(intern("conj/2"), e1, e2), env); 112 | } 113 | case intern("disj"): 114 | if (exp.cdr == null) { throw "error: empty disj"; } 115 | else if (exp.cdr.cdr == null) { 116 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 117 | return e1; 118 | } else { 119 | var e1 = list(intern("zzz"), exp.cdr.car); 120 | var e2 = cons(intern("disj"), exp.cdr.cdr); 121 | return desugar(list(intern("disj/2"), e1, e2), env); 122 | } 123 | case intern("begin"): 124 | return meta(cons(exp.car, map(function(f) { return desugar(f, env); }, exp.cdr)), 125 | { tag: "begin" }); 126 | case intern("fresh"): 127 | var bindings = exp.cdr.car; 128 | var body = exp.cdr.cdr; 129 | if (bindings === null) { 130 | return desugar(cons(intern("conj"), body), env); 131 | } else { 132 | var fn = list(intern("lambda"), bindings, cons(intern("conj"), body)); 133 | var body1 = desugar(fn, env); 134 | var len = desugar(length(bindings), env); 135 | return meta(list(intern("apply/fresh-n"), len, body1), 136 | { tag: "fresh" }); 137 | } 138 | case intern("lambda"): 139 | var bindings = exp.cdr.car; 140 | var body = exp.cdr.cdr; 141 | var norm = normalize_params(bindings); 142 | var env1 = foldl(norm.whole, env, function(e, b) { return cons(cons(b, b), e); }); 143 | var body1 = desugar(cons(intern("begin"), body), env1); 144 | return meta(list(exp.car, norm.whole, body1), 145 | { tag: "lambda", params: norm }); 146 | default: 147 | if(builtins.hasOwnProperty(exp.car.string)) { 148 | return meta(cons(meta(exp.car, { tag: "var" }), map(function(f) { return desugar(f, env); }, exp.cdr)), 149 | { tag: "app-builtin" }); 150 | } else if (exp.car === intern("list")) { 151 | return meta(cons(meta(exp.car, { tag: "var" }), map(function(f) { return desugar(f, env); }, exp.cdr)), 152 | { tag: "app-builtin" }); 153 | } else { 154 | throw new Error("unkown exp: " + pretty_print(exp)); 155 | } 156 | } 157 | } else if(constantp(exp)) { 158 | return meta(exp, { tag: "const" }); 159 | } else if(symbolp(exp)) { 160 | return meta(exp, { tag: "var" }); 161 | } else { 162 | throw new Error("unkown exp: " + pretty_print(exp)); 163 | } 164 | } 165 | 166 | function lookup(x, xs) { 167 | while(xs !== null) { 168 | if (x === xs.car.car) { return xs.car.cdr; } 169 | else { xs = xs.cdr; } 170 | } return false; 171 | } 172 | 173 | function extend_boxed(env, a, v) { 174 | var envu = env.get(); 175 | var existing = lookup(a, envu); 176 | if (existing) { 177 | return existing; 178 | } else { 179 | env.set(cons(cons(a, v), envu)); 180 | return v; 181 | } 182 | } 183 | 184 | function frees(exp, env, lenv, fenv) { 185 | switch(exp.meta.tag) { 186 | case "const": case "quoted": 187 | return exp; 188 | case "define": 189 | var name = exp.val.cdr.car; 190 | var dfenv = ref(null); 191 | // add self to env once mutable vars works, unless toplevel 192 | var ret = list(exp.val.car, name, frees(exp.val.cdr.cdr.car, env, lenv, dfenv)); 193 | if(dfenv.get() === null) { 194 | return meta(ret, exp.meta); 195 | } else { 196 | throw ("free variables in define: " + name.string + pretty_print(map(car, dfenv.get()))); 197 | } 198 | case "lambda": 199 | var bindings = exp.val.cdr.car; 200 | var body = exp.val.cdr.cdr; 201 | var e1 = foldl(bindings, env, function(e, a) { return cons(cons(a, a), e); }); 202 | var inner_frees = ref(null); 203 | var proper_frees = null; 204 | 205 | var re = cons(exp.val.car, cons(bindings, map(function(x) { 206 | return frees(x, e1, bindings, inner_frees); 207 | }, body))); 208 | 209 | map(function(v) { 210 | if (!memq(v.car, bindings)) { 211 | extend_boxed(fenv, v.car, v.cdr); 212 | proper_frees = cons(v.cdr, proper_frees); 213 | } 214 | }, inner_frees.get()); 215 | 216 | exp.meta.frees = proper_frees; 217 | return meta(re, exp.meta); 218 | case "begin": case "zzz": case "fresh": 219 | return meta(cons(exp.val.car, map(function(x) { return frees(x, env, lenv, fenv); }, exp.val.cdr)), 220 | exp.meta); 221 | case "app": case "app-builtin": 222 | return meta(map(function(x) { return frees(x, env, lenv, fenv); }, exp.val), exp.meta); 223 | case "var": 224 | if (memq(exp.val, lenv)) { return exp; } 225 | if (lookup(exp.val, env)) { return meta(extend_boxed(fenv, exp.val, exp.val), exp.meta); } 226 | if (toplevel.hasOwnProperty(exp.val.string)) { return exp; } 227 | if (exp.val === intern("list")) { return exp; } // TODO: remove once varargs added 228 | if (builtins.hasOwnProperty(exp.val.string)) { return exp; } 229 | return meta(extend_boxed(fenv, exp.val, exp.val), exp.meta); 230 | default: 231 | throw new Error("unkown exp: " + pretty_print(exp.val)); 232 | } 233 | } 234 | 235 | // instead of returning a value, it returns a function that fetches a value 236 | // based on the offset of how far it takes to reach the variable 237 | // values in the env are expected to be :: offset -> cenv -> value 238 | function lookup_calc(x, xs) { 239 | var n = 0; 240 | while(xs !== null) { 241 | if (x === xs.car.car) { return xs.car.cdr(n); } 242 | else { n++; xs = xs.cdr; } 243 | } return false; 244 | } 245 | 246 | function augment_cenv(env, name) { 247 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return cenv[offset]; }; }); 248 | return cons(env.car, cons(binding, env.cdr)); 249 | } 250 | 251 | function augment_aenv(env, name) { 252 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return aenv[offset]; }; }); 253 | return cons(cons(binding, env.car), env.cdr); 254 | } 255 | 256 | function augment_aenv_rest(env, name) { 257 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return array_slice_list(aenv, offset); }; }); 258 | return cons(cons(binding, env.car), env.cdr); 259 | } 260 | 261 | function lift_frees(exp) { 262 | var free_env = ref(null); 263 | var exp1 = frees(exp, null, null, free_env); 264 | var e1_c1 = foldl(free_env.get(), cons(null, 0), 265 | function(e_c, a) { 266 | var var1 = mkvar(e_c.cdr); 267 | var retrieve = function(_) { return function(_) { return var1; }; }; 268 | return cons(cons(cons(a.cdr, retrieve), e_c.car), e_c.cdr + 1); }); 269 | return list(exp1, e1_c1.car, Mks(Immutable.Map(), e1_c1.cdr, null, Immutable.Map())); 270 | } 271 | 272 | function eval0(exp, env) { 273 | if(exp.meta.constp) { 274 | exp.meta.constp = false; 275 | var val = eval0(exp, env)(); 276 | return function(cenv) { return val; }; 277 | } 278 | switch(exp.meta.tag) { 279 | case "app": 280 | var clos = eval0(exp.val.car, env); 281 | var args = foldl(reverse(exp.val.cdr), null, function(m, e) { 282 | return cons(eval0(e, env), m); 283 | }); 284 | var len = length(args); 285 | return function(aenv, cenv) { 286 | var clos1 = clos(aenv, cenv); 287 | return clos1.car(build_env(len, args, aenv, cenv), clos1.cdr); 288 | }; 289 | case "define": 290 | var result = eval0(exp.val.cdr.cdr.car, env); 291 | toplevel[exp.val.cdr.car.string].set(result()); 292 | return function(aenv, cenv) { return true; }; 293 | case "quoted": 294 | var val = exp.val.cdr.car; 295 | return function(aenv, cenv) { return val }; 296 | case "begin": 297 | if (exp.val.cdr == null) { throw new Error("empty begin"); } 298 | else if (exp.val.cdr.cdr == null) { 299 | var e1 = eval0(exp.val.cdr.car, env); 300 | return e1; 301 | } else { 302 | return generate_begin_code(length(exp.val.cdr))(exp.val.cdr, env, eval0); 303 | } 304 | case "lambda": 305 | var bindings = exp.val.cdr.car; 306 | var body = exp.val.cdr.cdr.car; 307 | var free_env = foldl(reverse(exp.meta.frees), cons(null, null), augment_cenv); 308 | var env1_rest = foldl(exp.meta.params.rest, free_env, augment_aenv_rest); 309 | var env1 = foldl(reverse(exp.meta.params.normal), env1_rest, augment_aenv); 310 | var len = length(exp.meta.frees); 311 | var free_env1 = map(function(v) { return eval0(meta(v, { tag: "var" }), env); }, exp.meta.frees); 312 | var closure_body = eval0(body, env1); 313 | var closure_env_build = function(aenv, cenv) { return build_env(len, free_env1, aenv, cenv); }; 314 | var closure = function(aenv, cenv) { return cons(closure_body, closure_env_build(aenv, cenv)); }; 315 | return closure; 316 | case "zzz": 317 | var e1 = eval0(exp.val.cdr.car, env); 318 | return function(aenv, cenv) { return function(mks) { return function() { return e1(aenv, cenv)(mks); }; }; }; 319 | case "fresh": 320 | var len = exp.val.cdr.car.val; 321 | var fn = exp.val.cdr.cdr.car; 322 | var closure = eval0(fn, env); 323 | return function(aenv, cenv) { return apply_fresh(len, closure, aenv, cenv); }; 324 | case "app-builtin": 325 | if (exp.val.car.val === intern("list")) { 326 | var args = map(function(e) { return eval0(e, env); }, exp.val.cdr); 327 | return function(aenv, cenv) { 328 | return map(function(a) { return a(aenv, cenv); }, args); 329 | }; 330 | } 331 | return builtins[exp.val.car.val.string](exp.val.cdr, env, eval0); 332 | case "const": 333 | var val = exp.val 334 | return function(aenv, cenv) { return val; }; 335 | case "var": 336 | var v; 337 | v = lookup_calc(exp.val, env.car) || lookup_calc(exp.val, env.cdr); 338 | if (v) { 339 | return v; 340 | } else if(toplevel.hasOwnProperty(exp.val.string)) { 341 | var box = toplevel[exp.val.string]; 342 | return function(aenv, cenv) { return box.get(); }; 343 | } else { 344 | throw new Error("unbound variable: " + pretty_print(exp.val)); 345 | } 346 | default: 347 | throw new Error("unkown exp: " + pretty_print(exp.val)); 348 | } 349 | } 350 | 351 | var builtins = new Object(null); 352 | builtins["cons"] = generate_fn_code("cons", 2); 353 | builtins["conj/2"] = generate_fn_code("conj", 2); 354 | builtins["disj/2"] = generate_fn_code("disj", 2); 355 | builtins["=="] = generate_fn_code("eqeq", 2); 356 | builtins["=/="] = generate_fn_code("noteqeq", 2); 357 | builtins["symbolo"] = generate_fn_code("symbolo", 1); 358 | builtins["numbero"] = generate_fn_code("numbero", 1); 359 | builtins["absento"] = generate_fn_code("absento", 2); 360 | builtins["build-num"] = generate_fn_code("build_num", 1); 361 | 362 | function generate_fn_code(name, arity) { 363 | var c = 0; 364 | var evalers = []; 365 | var callers = []; 366 | 367 | for(c = 0; c < arity; c++) { 368 | evalers = evalers.concat(["var e", c, " = eval0(nth(args, ", c+1, "), env);\n"]); 369 | callers = callers.concat([["e", c, "(aenv, cenv)"].join("")]); 370 | } 371 | 372 | var args_evald = evalers.join(""); 373 | var return_val = ["return function(aenv, cenv) { return ", name, "(", callers.join(", "), "); };"].join(""); 374 | return new Function(["args, env, eval0"], [args_evald, return_val].join("\n")); 375 | } 376 | 377 | function generate_begin_code(arity) { 378 | var c = 0; 379 | var evalers = []; 380 | var callers = []; 381 | 382 | for(c = 0; c < arity; c++) { 383 | evalers = evalers.concat(["var e", c, " = eval0(nth(args, ", c+1, "), env);\n"]); 384 | var prefix = c < arity ? "" : "return "; 385 | callers = callers.concat([[prefix, "e", c, "(aenv, cenv)"].join("")]); 386 | } 387 | 388 | var args_evald = evalers.join(""); 389 | var return_val = ["return function(aenv, cenv) { ", callers.join("; "), " };"].join(""); 390 | return new Function(["args, env, eval0"], [args_evald, return_val].join("\n")); 391 | } 392 | 393 | function veval(exp, env) { 394 | return eval0(exp, env)([], []); 395 | } 396 | 397 | function build_env(len, e, aenv, cenv) { 398 | var new_env = new Array(len); 399 | var i = 0; 400 | while(e !== null) { 401 | new_env[i++] = e.car(aenv, cenv); 402 | e = e.cdr; 403 | } return new_env; 404 | } 405 | 406 | function fresh_n(n, c) { 407 | var m = new Array(n); 408 | while(n-- > 0) { 409 | m[n] = mkvar(c++); 410 | } return [m, c]; 411 | } 412 | 413 | function apply_fresh(len, closure, aenv, cenv) { 414 | var closure1 = closure(aenv, cenv); 415 | var closure_env = closure1.cdr; 416 | var closure_fn = closure1.car; 417 | return function(mks) { 418 | var c = mks.counter; 419 | var e1_c1 = fresh_n(len, c); 420 | var mks1 = Mks(mks.substitution, e1_c1[1], mks.prefix, mks.constraint_store); 421 | return closure_fn(e1_c1[0], closure_env)(mks1); 422 | }; 423 | } 424 | 425 | function map_stream(fn, stream) { 426 | return function() { 427 | var $ = pull(stream); 428 | return ($ == null) ? null : cons(fn($.car), map_stream(fn, $.cdr)); 429 | }; 430 | } 431 | 432 | function query_stream(q, qenv, mks) { 433 | var $ = q(mks); 434 | var run_queries = function(mks) { 435 | var qm = map(function(p) { return cons(p.car.string, p.cdr()()); }, qenv); 436 | return query_map(qm, mks); 437 | }; 438 | return map_stream(run_queries, $); 439 | } 440 | 441 | function stream_generator($) { 442 | var next = $; 443 | return function() { 444 | var cur = next(); 445 | if(cur == null) { return null; } 446 | else { next = cur.cdr; 447 | return cur.car; } 448 | }; 449 | } 450 | 451 | function run_expression(p) { 452 | var desugared = desugar(p, null); 453 | var lifted = lift_frees(desugared); 454 | 455 | var exp = lifted.car; 456 | var env = cons(null, lifted.cdr.car); 457 | var mks = lifted.cdr.cdr.car; 458 | var evald = veval(exp, env); 459 | 460 | if(procedurep(evald)) { 461 | var q$ = query_stream(evald, env.cdr, mks); 462 | return stream_generator(q$); 463 | } else if (env.cdr === null) { 464 | return evald; 465 | } else { 466 | throw "unbound variables: " + pretty_print(map(car, env.cdr)); 467 | } 468 | } 469 | 470 | function run_program(p) { 471 | var stream; 472 | if(p == null) { throw "no program" } 473 | map(register_define, p); 474 | while(p != null) { 475 | stream = run_expression(p.car); 476 | p = p.cdr; 477 | } return stream; 478 | } 479 | 480 | this.reset = function() { toplevel = new Object(null); }; 481 | this.eval = function(ast) { return run_program(ast); }; 482 | this.read_eval = function(text) { return run_program(read_program(text)); }; 483 | } 484 | -------------------------------------------------------------------------------- /ck/editor.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | editor 23 | 24 | 25 | 30 |
31 |
32 | relational interpreter 33 | 136 |
137 |
138 | type inference 139 | 197 |
198 |
199 | numbers 200 | 496 |
497 |
498 | lists 499 | 521 |
522 |
523 | grammar 524 | 570 |
571 |
572 |
573 |
574 | 575 | 576 | -------------------------------------------------------------------------------- /editor.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding:0; 3 | margin: 0; 4 | font-family:courier, monospace; 5 | height:100vh; 6 | background-color:#ccc 7 | } 8 | 9 | #container { height:90%; } 10 | #demos { display:none; } 11 | -------------------------------------------------------------------------------- /editor.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | editor 23 | 24 | 25 | 32 |
33 |
34 | relational interpreter 35 | 138 |
139 |
140 | type inference 141 | 199 |
200 |
201 | numbers 202 | 498 |
499 |
500 | reasoned schemer prelude 501 | 582 |
583 |
584 | grammar 585 | 631 |
632 |
633 |
634 |
635 | 636 | 637 | -------------------------------------------------------------------------------- /editor.js: -------------------------------------------------------------------------------- 1 | function load_editor() { 2 | var veneer = new Veneer_v1(); 3 | 4 | var vm = veneer.new_vm(); 5 | 6 | var ed = veneer.new_editor(document.getElementById("container"), vm); 7 | var editor = ed.editor; 8 | var repl = ed.repl; 9 | var errors = ed.errors; 10 | var dump_button = ed.dump_button; 11 | 12 | var demo = false; 13 | var loaded_file = false; 14 | function setEditorValue(val) { 15 | demo = false; 16 | loaded_file = false; 17 | editor.setValue(val); 18 | } 19 | 20 | var demos = document.getElementsByClassName("demo"); 21 | var demo_picker = document.getElementById("demo_picker"); 22 | var file_values = new Array(demos.length); 23 | for(var i=0; i < demos.length; i++ ) { 24 | var selection = document.createElement("option"); 25 | var name = demos[i].getElementsByClassName("title")[0].textContent; 26 | file_values[i] = demos[i].getElementsByClassName("content")[0].value; 27 | selection.value = i; 28 | selection.appendChild(document.createTextNode(name)); 29 | demo_picker.appendChild(selection); 30 | } 31 | demo_picker.onchange = function() { 32 | load_demo(demo_picker.selectedIndex); 33 | }; 34 | demo_picker.value = "0"; 35 | setEditorValue(file_values[0]); 36 | demo = 0; 37 | 38 | var ls_picker = document.getElementById("localstorage_picker"); 39 | var files = JSON.parse(localStorage.getItem('__index__') || "[]"); 40 | for(var i=0; i < files.length; i++ ) { 41 | var selection = document.createElement("option"); 42 | var name = files[i]; 43 | selection.value = name; 44 | selection.appendChild(document.createTextNode(name)); 45 | ls_picker.appendChild(selection); 46 | } 47 | ls_picker.onchange = function() { 48 | load_file(ls_picker[ls_picker.selectedIndex].value); 49 | }; 50 | document.getElementById("save_file").onclick = save_current; 51 | document.getElementById("new_file").onclick = add_file; 52 | 53 | document.getElementById("create_link").onclick = function() { 54 | var req = new XMLHttpRequest(); 55 | var method = "POST"; 56 | var url = "https://api.github.com/gists"; 57 | 58 | var options = { 59 | "description": "veneer save", 60 | "public": true, 61 | "files": { 62 | "file": { 63 | "content": editor.getValue() 64 | } 65 | } 66 | }; 67 | 68 | req.open(method, url, true); 69 | req.onreadystatechange = function () { 70 | if (req.readyState !== XMLHttpRequest.DONE) { 71 | return; 72 | } 73 | if (req.status === 201) { 74 | var response = JSON.parse(req.responseText); 75 | window.location.hash = response.id; 76 | } 77 | }; 78 | req.send(JSON.stringify(options)); 79 | }; 80 | 81 | 82 | function confirm_overwrite(name) { 83 | return window.confirm("File exists: Are you sure you want to over-write \"" + name + "\""); 84 | } 85 | function pick_filename() { 86 | return window.prompt("save file as:","default"); 87 | } 88 | function confirm_save_progress() { 89 | return window.confirm("Save progress before switching?"); 90 | } 91 | 92 | function load_demo(id) { 93 | var save_status = true; 94 | if (demo !== id) { save_status = save_current(true); } 95 | if(save_status) { 96 | var demo_val = file_values[id]; 97 | setEditorValue(demo_val); 98 | demo = id; 99 | } 100 | } 101 | 102 | function load_file(name) { 103 | var save_status = true; 104 | if (loaded_file !== name) { save_status = save_current(true); } 105 | if(save_status) { 106 | var new_val = localStorage.getItem(name); 107 | setEditorValue(new_val); 108 | loaded_file = name; 109 | } 110 | } 111 | 112 | function save_current(for_change) { 113 | if (loaded_file !== false) { 114 | if(editor.getValue() !== localStorage.getItem(loaded_file)) { 115 | 116 | if(for_change === true && !confirm_save_progress()) { 117 | return true; 118 | } 119 | 120 | loaded_file = confirm_overwrite(loaded_file) ? loaded_file : false; 121 | } 122 | } else if (demo !== false) { 123 | if (editor.getValue() !== file_values[demo]) { 124 | 125 | if(for_change === true && !confirm_save_progress()) { 126 | return true; 127 | } 128 | 129 | loaded_file = window.prompt("save progress as: ", "default") || false; 130 | } else if (for_change === true) { 131 | return true; 132 | } 133 | } 134 | 135 | if(!loaded_file) { return false; } 136 | demo = false; 137 | 138 | if(files.indexOf(loaded_file) < 0) { 139 | files.push(loaded_file); 140 | 141 | var selection = document.createElement("option"); 142 | selection.appendChild(document.createTextNode(loaded_file)); 143 | ls_picker.appendChild(selection); 144 | ls_picker.value = loaded_file; 145 | localStorage.setItem('__index__', JSON.stringify(files)); 146 | }; 147 | localStorage.setItem(loaded_file, editor.getValue()); 148 | return true; 149 | } 150 | 151 | function add_file() { 152 | if(save_current(true)) { setEditorValue(''); } 153 | } 154 | 155 | if(window.location.hash) { 156 | var req = new XMLHttpRequest(); 157 | var method = "GET"; 158 | var url = "https://api.github.com/gists/" + window.location.hash.substring(1); 159 | 160 | req.open(method, url, true); 161 | req.onreadystatechange = function () { 162 | if (req.readyState !== XMLHttpRequest.DONE) { 163 | return; 164 | } 165 | if (req.status === 200) { 166 | var response = JSON.parse(req.responseText); 167 | setEditorValue(response.files.file.content); 168 | } 169 | }; 170 | req.send(); 171 | } 172 | } 173 | -------------------------------------------------------------------------------- /eval.js: -------------------------------------------------------------------------------- 1 | function VeneerVM() { 2 | var toplevel = new Object(null); 3 | 4 | function register_define(exp) { 5 | if (pairp(exp) && exp.car === intern("define")) { 6 | var a = pairp(exp.cdr.car) ? exp.cdr.car.car : exp.cdr.car; 7 | toplevel[a.string] = ref(null); 8 | } 9 | } 10 | 11 | function quote_desugar(exp) { 12 | if (pairp(exp)) { 13 | return meta(list(meta(intern("cons"), {tag:"var"}), quote_desugar(exp.car), quote_desugar(exp.cdr)), {tag:"app-builtin"}); 14 | } else if (exp === null) { 15 | return meta(list(intern("quote"), null), { tag: "quoted" }); 16 | } else if(constantp(exp)) { 17 | return meta(exp, {tag:"const"}); 18 | } else { 19 | return meta(list(intern("quote"), exp), { tag: "quoted" }); 20 | } 21 | } 22 | 23 | function quasiquote_desugar(exp, n, env) { 24 | if (pairp(exp)) { 25 | if (exp.car === intern("unquote")) { 26 | if (n === 1) { 27 | return exp.cdr.car; 28 | } else { 29 | return list(intern("list"), list(intern("quote"), intern("unquote")), 30 | quasiquote_desugar(exp.cdr.car, n - 1, env)); 31 | } 32 | } else if (exp.car === intern("quasiquote")) { 33 | return list(intern("list"), list(intern("quote"), intern("quasiquote")), 34 | quasiquote_desugar(exp.cdr.car, n + 1, env)); 35 | } else { 36 | return list(intern("cons"), quasiquote_desugar(exp.car, n, env), 37 | quasiquote_desugar(exp.cdr, n, env)); 38 | } 39 | } else { 40 | return list(intern("quote"), exp); 41 | } 42 | } 43 | 44 | function Meta(val, meta) { 45 | this.val = val; 46 | this.meta = meta || {}; 47 | } 48 | Meta.prototype.toString = function() { 49 | return pretty_print(this.val); 50 | } 51 | function meta(v, meta) { return new Meta(v, meta); } 52 | 53 | function normalize_params(params) { 54 | var ps = null; 55 | var rest = null; 56 | var whole = null; 57 | while(params !== null) { 58 | if(symbolp(params)) { 59 | rest = cons(params, rest); 60 | whole = cons(params, ps); 61 | break; 62 | } else { 63 | ps = cons(params.car, ps); 64 | whole = ps; 65 | params = params.cdr; 66 | } 67 | } 68 | var normal = reverse(ps); 69 | var whole = reverse(whole); 70 | var min_args = length(normal); 71 | return { normal: normal, rest: rest, whole: whole, min_args: min_args }; 72 | } 73 | 74 | function desugar(exp, env) { 75 | if(pairp(exp)) { 76 | // application 77 | var app = false; 78 | if(symbolp(exp.car) && (lookup(exp.car, env) || toplevel.hasOwnProperty(exp.car.string)) 79 | || pairp(exp.car)) { 80 | return meta(map(function(e) { return desugar(e, env); }, exp), 81 | { tag: "app" }); 82 | } 83 | // special forms & builtins 84 | switch(exp.car) { 85 | case intern("define"): 86 | var name; 87 | var body; 88 | if(pairp(exp.cdr.car)) { 89 | // (define (x y z) f) => (define x (lambda (y z) f)) 90 | name = exp.cdr.car.car; 91 | body = cons(intern("lambda"), cons(exp.cdr.car.cdr, exp.cdr.cdr)); 92 | } else { 93 | name = exp.cdr.car; 94 | body = exp.cdr.cdr.car; 95 | } 96 | return meta(list(exp.car, name, desugar(body, env)), { tag: "define" }); 97 | case intern("quote"): 98 | var val = quote_desugar(exp.cdr.car); 99 | val.meta.constp = true; 100 | return val; 101 | case intern("quasiquote"): 102 | return desugar(quasiquote_desugar(exp.cdr.car, 1, env), env); 103 | case intern("zzz"): 104 | return meta(list(exp.car, desugar(exp.cdr.car, env)), 105 | { tag: "zzz" }); 106 | case intern("conde"): 107 | var clauses = map(function(row) { return cons(intern("conj"), row); }, exp.cdr); 108 | return desugar(cons(intern("disj"), clauses), env); 109 | case intern("conj"): 110 | if (exp.cdr === null) { throw "error: empty conj"; } 111 | else if (exp.cdr.cdr === null) { 112 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 113 | return e1; 114 | } else { 115 | var e1 = list(intern("zzz"), exp.cdr.car); 116 | var e2 = cons(intern("conj"), exp.cdr.cdr); 117 | return desugar(list(intern("conj/2"), e1, e2), env); 118 | } 119 | case intern("disj"): 120 | if (exp.cdr === null) { throw "error: empty disj"; } 121 | else if (exp.cdr.cdr === null) { 122 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 123 | return e1; 124 | } else { 125 | var e1 = list(intern("zzz"), exp.cdr.car); 126 | var e2 = cons(intern("disj"), exp.cdr.cdr); 127 | return desugar(list(intern("disj/2"), e1, e2), env); 128 | } 129 | case intern("conde/dfs"): 130 | var clauses = map(function(row) { return cons(intern("conj/dfs"), row); }, exp.cdr); 131 | return desugar(cons(intern("disj/dfs"), clauses), env); 132 | case intern("conj/dfs"): 133 | if (exp.cdr === null) { throw "error: empty conj"; } 134 | else if (exp.cdr.cdr === null) { 135 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 136 | return e1; 137 | } else { 138 | var e1 = list(intern("zzz"), exp.cdr.car); 139 | var e2 = cons(intern("conj/dfs"), exp.cdr.cdr); 140 | return desugar(list(intern("conj/dfs/2"), e1, e2), env); 141 | } 142 | case intern("disj/dfs"): 143 | if (exp.cdr === null) { throw "error: empty disj"; } 144 | else if (exp.cdr.cdr === null) { 145 | var e1 = desugar(list(intern("zzz"), exp.cdr.car), env); 146 | return e1; 147 | } else { 148 | var e1 = list(intern("zzz"), exp.cdr.car); 149 | var e2 = cons(intern("disj/dfs"), exp.cdr.cdr); 150 | return desugar(list(intern("disj/dfs/2"), e1, e2), env); 151 | } 152 | case intern("begin"): 153 | return meta(cons(exp.car, map(function(f) { return desugar(f, env); }, exp.cdr)), 154 | { tag: "begin" }); 155 | case intern("if"): 156 | return meta(cons(exp.car, map(function(f) { return desugar(f, env); }, exp.cdr)), 157 | { tag: "if" }); 158 | 159 | case intern("and"): 160 | if (exp.cdr === null) { 161 | // (and) => true 162 | return desugar(true, env); 163 | } else if (exp.cdr.cdr === null) { 164 | // (and a) => a 165 | return desugar(exp.cdr.car, env); 166 | } else { 167 | // (and a . r) => (if a (and . r) #f) 168 | return desugar(list(intern("if"), exp.cdr.car, 169 | cons(intern("and"), exp.cdr.cdr), 170 | false), 171 | env); 172 | } 173 | case intern("or"): 174 | if (exp.cdr === null) { 175 | // (or) => false 176 | return desugar(false, env); 177 | } else if (exp.cdr.cdr === null) { 178 | // (or a) => a 179 | return desugar(exp.cdr.car, env); 180 | } else { 181 | // (or a . r) => (let ((tmp a)) (if tmp tmp (or . r))) 182 | var tmp = gensym("tmp"); 183 | return desugar(list(intern("let"), list(list(tmp, exp.cdr.car)), 184 | list(intern("if"), tmp, tmp, cons(intern("or"), exp.cdr.cdr))), 185 | env); 186 | } 187 | case intern("cond"): 188 | if (exp.cdr === null) { 189 | throw "error: empty cond"; 190 | } else if(exp.cdr.cdr === null) { 191 | if(exp.cdr.car.car === intern("else")) { 192 | // (cond (else . x)) => (begin . x) 193 | return desugar(cons(intern("begin"), exp.cdr.car.cdr), env); 194 | } else { 195 | // (cond (a . b)) => (if a (begin . b) #f) 196 | return desugar(list(intern("if"), exp.cdr.car.car, 197 | cons(intern("begin"), exp.cdr.car.cdr), 198 | false), 199 | env); 200 | } 201 | } else { 202 | // (cond (a . b) . r) => (if a (begin . b) (cond . r)) 203 | return desugar(list(intern("if"), exp.cdr.car.car, 204 | cons(intern("begin"), exp.cdr.car.cdr), 205 | cons(intern("cond"), exp.cdr.cdr)), 206 | env); 207 | } 208 | case intern("let"): 209 | // (let ((a b) ...) . body) => ((lambda (a ...) . body) b ...) 210 | return desugar(cons(cons(intern("lambda"), cons(map(car, exp.cdr.car), exp.cdr.cdr)), 211 | map(function(e) { return e.cdr.car; }, exp.cdr.car)), 212 | env); 213 | case intern("fresh"): 214 | var bindings = exp.cdr.car; 215 | var body = exp.cdr.cdr; 216 | if (bindings === null) { 217 | return desugar(cons(intern("conj"), body), env); 218 | } else { 219 | var fn = list(intern("lambda"), bindings, cons(intern("conj"), body)); 220 | var body1 = desugar(fn, env); 221 | var len = desugar(length(bindings), env); 222 | return meta(list(intern("apply/fresh-n"), len, body1), 223 | { tag: "fresh" }); 224 | } 225 | case intern("fresh/dfs"): 226 | var bindings = exp.cdr.car; 227 | var body = exp.cdr.cdr; 228 | if (bindings === null) { 229 | return desugar(cons(intern("conj/dfs"), body), env); 230 | } else { 231 | var fn = list(intern("lambda"), bindings, cons(intern("conj/dfs"), body)); 232 | var body1 = desugar(fn, env); 233 | var len = desugar(length(bindings), env); 234 | return meta(list(intern("apply/fresh-n"), len, body1), 235 | { tag: "fresh" }); 236 | } 237 | case intern("lambda"): 238 | var bindings = exp.cdr.car; 239 | var body = exp.cdr.cdr; 240 | var norm = normalize_params(bindings); 241 | var env1 = foldl(norm.whole, env, function(e, b) { return cons(cons(b, b), e); }); 242 | var body1 = desugar(cons(intern("begin"), body), env1); 243 | return meta(list(exp.car, norm.whole, body1), 244 | { tag: "lambda", params: norm }); 245 | default: 246 | if(builtins.hasOwnProperty(exp.car.string)) { 247 | return meta(cons(meta(exp.car, { tag: "var" }), map(function(f) { return desugar(f, env); }, exp.cdr)), 248 | { tag: "app-builtin" }); 249 | } else if (exp.car === intern("list")) { 250 | return meta(cons(meta(exp.car, { tag: "var" }), map(function(f) { return desugar(f, env); }, exp.cdr)), 251 | { tag: "app-builtin" }); 252 | } else { 253 | throw new Error("unkown exp: " + pretty_print(exp)); 254 | } 255 | } 256 | } else if(constantp(exp)) { 257 | return meta(exp, { tag: "const" }); 258 | } else if(symbolp(exp)) { 259 | return meta(exp, { tag: "var" }); 260 | } else { 261 | throw new Error("unkown exp: " + pretty_print(exp)); 262 | } 263 | } 264 | 265 | function lookup(x, xs) { 266 | while(xs !== null) { 267 | if (x === xs.car.car) { return xs.car.cdr; } 268 | else { xs = xs.cdr; } 269 | } return false; 270 | } 271 | 272 | function extend_boxed(env, a, v) { 273 | var envu = env.get(); 274 | var existing = lookup(a, envu); 275 | if (existing) { 276 | return existing; 277 | } else { 278 | env.set(cons(cons(a, v), envu)); 279 | return v; 280 | } 281 | } 282 | 283 | function frees(exp, env, lenv, fenv) { 284 | switch(exp.meta.tag) { 285 | case "const": case "quoted": 286 | return exp; 287 | case "define": 288 | var name = exp.val.cdr.car; 289 | var dfenv = ref(null); 290 | // add self to env once mutable vars works, unless toplevel 291 | var ret = list(exp.val.car, name, frees(exp.val.cdr.cdr.car, env, lenv, dfenv)); 292 | if(dfenv.get() === null) { 293 | return meta(ret, exp.meta); 294 | } else { 295 | throw ("free variables in define: " + name.string + pretty_print(map(car, dfenv.get()))); 296 | } 297 | case "begin": case "zzz": case "fresh": 298 | return meta(cons(exp.val.car, map(function(x) { return frees(x, env, lenv, fenv); }, exp.val.cdr)), 299 | exp.meta); 300 | case "if": 301 | return meta(cons(exp.val.car, map(function(x) { return frees(x, env, lenv, fenv); }, exp.val.cdr)), 302 | exp.meta); 303 | case "lambda": 304 | var bindings = exp.val.cdr.car; 305 | var body = exp.val.cdr.cdr; 306 | var e1 = foldl(bindings, env, function(e, a) { return cons(cons(a, a), e); }); 307 | var inner_frees = ref(null); 308 | var proper_frees = null; 309 | 310 | var re = cons(exp.val.car, cons(bindings, map(function(x) { 311 | return frees(x, e1, bindings, inner_frees); 312 | }, body))); 313 | 314 | map(function(v) { 315 | if (!memq(v.car, bindings)) { 316 | extend_boxed(fenv, v.car, v.cdr); 317 | proper_frees = cons(v.cdr, proper_frees); 318 | } 319 | }, inner_frees.get()); 320 | 321 | exp.meta.frees = proper_frees; 322 | return meta(re, exp.meta); 323 | case "app": case "app-builtin": 324 | return meta(map(function(x) { return frees(x, env, lenv, fenv); }, exp.val), exp.meta); 325 | case "var": 326 | if (memq(exp.val, lenv)) { return exp; } 327 | if (lookup(exp.val, env)) { return meta(extend_boxed(fenv, exp.val, exp.val), exp.meta); } 328 | if (toplevel.hasOwnProperty(exp.val.string)) { return exp; } 329 | if (exp.val === intern("list")) { return exp; } // TODO: remove once varargs added 330 | if (builtins.hasOwnProperty(exp.val.string)) { return exp; } 331 | return meta(extend_boxed(fenv, exp.val, exp.val), exp.meta); 332 | default: 333 | throw new Error("unkown exp: " + pretty_print(exp.val)); 334 | } 335 | } 336 | 337 | function lookup_calc(x, xs) { 338 | var n = 0; 339 | while(xs !== null) { 340 | if (x === xs.car.car) { return xs.car.cdr(n); } 341 | else { n++; xs = xs.cdr; } 342 | } return false; 343 | } 344 | 345 | function augment_cenv(env, name) { 346 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return cenv[offset]; }; }); 347 | return cons(env.car, cons(binding, env.cdr)); 348 | } 349 | 350 | function augment_aenv(env, name) { 351 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return aenv[offset]; }; }); 352 | return cons(cons(binding, env.car), env.cdr); 353 | } 354 | 355 | function augment_aenv_rest(env, name) { 356 | var binding = cons(name, function (offset) { return function(aenv, cenv) { return array_slice_list(aenv, offset); }; }); 357 | return cons(cons(binding, env.car), env.cdr); 358 | } 359 | 360 | function lift_frees(exp) { 361 | var free_env = ref(null); 362 | var exp1 = frees(exp, null, null, free_env); 363 | var e1_c1 = foldl(free_env.get(), cons(null, 0), 364 | function(e_c, a) { 365 | var var1 = mkvar(e_c.cdr); 366 | var retrieve = function(_) { return function(_) { return var1; }; }; 367 | return cons(cons(cons(a.cdr, retrieve), e_c.car), e_c.cdr + 1); }); 368 | return list(exp1, e1_c1.car, Mks(Immutable.Map(), e1_c1.cdr, null, Immutable.Map(), null, Immutable.Map())); 369 | } 370 | 371 | function eval0(exp, env) { 372 | if(exp.meta.constp) { 373 | exp.meta.constp = false; 374 | var val = eval0(exp, env)(); 375 | return function(cenv) { return val; }; 376 | } 377 | switch(exp.meta.tag) { 378 | case "app": 379 | var clos = eval0(exp.val.car, env); 380 | var args = foldl(reverse(exp.val.cdr), null, function(m, e) { 381 | return cons(eval0(e, env), m); 382 | }); 383 | var len = length(args); 384 | return function(aenv, cenv) { 385 | var clos1 = clos(aenv, cenv); 386 | var args1 = build_env(len, args, aenv, cenv); 387 | if (len < clos1[2]) { 388 | arity_error(clos1[2], len, exp, args1); 389 | } else if (!clos1[3] && len > clos1[2]) { 390 | arity_error(clos1[2], len, exp, args1); 391 | } else { 392 | return clos1[0](args1, clos1[1]); 393 | } 394 | }; 395 | case "define": 396 | var result = eval0(exp.val.cdr.cdr.car, env); 397 | toplevel[exp.val.cdr.car.string].set(result()); 398 | return function(aenv, cenv) { return true; }; 399 | case "quoted": 400 | var val = exp.val.cdr.car; 401 | return function(aenv, cenv) { return val }; 402 | case "begin": 403 | if (exp.val.cdr == null) { throw new Error("empty begin"); } 404 | else if (exp.val.cdr.cdr == null) { 405 | var e1 = eval0(exp.val.cdr.car, env); 406 | return e1; 407 | } else { 408 | return generate_begin_code(length(exp.val.cdr))(exp.val.cdr, env, eval0); 409 | } 410 | case "if": 411 | var t = eval0(exp.val.cdr.car, env); 412 | var c = eval0(exp.val.cdr.cdr.car, env); 413 | var a = eval0(exp.val.cdr.cdr.cdr.car, env); 414 | return function(aenv, cenv) { return t(aenv, cenv) ? c(aenv, cenv) : a(aenv, cenv); }; 415 | case "lambda": 416 | var bindings = exp.val.cdr.car; 417 | var min_args = exp.meta.params.min_args; 418 | var restp = !(exp.meta.params.rest === null); 419 | var body = exp.val.cdr.cdr.car; 420 | var free_env = foldl(reverse(exp.meta.frees), cons(null, null), augment_cenv); 421 | var env1_rest = foldl(exp.meta.params.rest, free_env, augment_aenv_rest); 422 | var env1 = foldl(reverse(exp.meta.params.normal), env1_rest, augment_aenv); 423 | var len = length(exp.meta.frees); 424 | var free_env1 = map(function(v) { return eval0(meta(v, { tag: "var" }), env); }, exp.meta.frees); 425 | var closure_body = eval0(body, env1); 426 | var closure = function(aenv, cenv) { 427 | return [closure_body, build_env(len, free_env1, aenv, cenv), min_args, restp]; 428 | }; 429 | return closure; 430 | case "zzz": 431 | var e1 = eval0(exp.val.cdr.car, env); 432 | return function(aenv, cenv) { return function(mks) { return function() { return e1(aenv, cenv)(mks); }; }; }; 433 | case "fresh": 434 | var len = exp.val.cdr.car.val; 435 | var fn = exp.val.cdr.cdr.car; 436 | var closure = eval0(fn, env); 437 | return function(aenv, cenv) { return apply_fresh(len, closure, aenv, cenv); }; 438 | case "app-builtin": 439 | if (exp.val.car.val === intern("list")) { 440 | var args = map(function(e) { return eval0(e, env); }, exp.val.cdr); 441 | return function(aenv, cenv) { 442 | return map(function(a) { return a(aenv, cenv); }, args); 443 | }; 444 | } 445 | return builtins[exp.val.car.val.string](exp.val.cdr, env, eval0); 446 | case "const": 447 | var val = exp.val 448 | return function(aenv, cenv) { return val; }; 449 | case "var": 450 | var v; 451 | v = lookup_calc(exp.val, env.car) || lookup_calc(exp.val, env.cdr); 452 | if (v) { 453 | return v; 454 | } else if(toplevel.hasOwnProperty(exp.val.string)) { 455 | var box = toplevel[exp.val.string]; 456 | return function(aenv, cenv) { return box.get(); }; 457 | } else { 458 | throw new Error("unbound variable: " + pretty_print(exp.val)); 459 | } 460 | default: 461 | throw new Error("unkown exp: " + pretty_print(exp.val)); 462 | } 463 | } 464 | 465 | var builtins = new Object(null); 466 | builtins["cons"] = generate_fn_code("cons", 2); 467 | builtins["car"] = generate_fn_code("car", 1); 468 | builtins["cdr"] = generate_fn_code("cdr", 1); 469 | builtins["null?"] = generate_fn_code("nullp", 1); 470 | builtins["number?"] = generate_fn_code("numberp", 1); 471 | builtins["+"] = generate_fn_code("+", 2, true); 472 | builtins["-"] = generate_fn_code("-", 2, true); 473 | builtins["*"] = generate_fn_code("*", 2, true); 474 | builtins["/"] = generate_fn_code("/", 2, true); 475 | builtins["quotient"] = generate_fn_code("quotient", 2); 476 | builtins["="] = generate_fn_code("===", 2, true); 477 | builtins[">"] = generate_fn_code(">", 2, true); 478 | builtins["<"] = generate_fn_code("<", 2, true); 479 | builtins[">="] = generate_fn_code(">=", 2, true); 480 | builtins["<="] = generate_fn_code("<=", 2, true); 481 | builtins["odd?"] = generate_fn_code("oddp", 1); 482 | builtins["even?"] = generate_fn_code("evenp", 1); 483 | builtins["zero?"] = generate_fn_code("zerop", 1); 484 | builtins["eq?"] = generate_fn_code("===", 2, true); 485 | builtins["not"] = generate_fn_code("not", 1); 486 | builtins["log"] = generate_fn_code("console.log", 1); 487 | builtins["gensym"] = generate_fn_code("gensym", 1); 488 | builtins["reverse"] = generate_fn_code("reverse", 1); 489 | builtins["append"] = generate_fn_code("append", 2); 490 | builtins["make-list"] = generate_fn_code("make_list", 2); 491 | 492 | builtins["conj/2"] = generate_fn_code("conj", 2); 493 | builtins["disj/2"] = generate_fn_code("disj", 2); 494 | builtins["conj/dfs/2"] = generate_fn_code("conj_dfs", 2); 495 | builtins["disj/dfs/2"] = generate_fn_code("disj_dfs", 2); 496 | builtins["=="] = generate_fn_code("eqeq", 2); 497 | builtins["=/="] = generate_fn_code("noteqeq", 2); 498 | builtins["symbolo"] = generate_fn_code("symbolo", 1); 499 | builtins["numbero"] = generate_fn_code("numbero", 1); 500 | builtins["absento"] = generate_fn_code("absento", 2); 501 | builtins["build-num"] = generate_fn_code("build_num", 1); 502 | builtins["watch"] = generate_fn_code("watch", 2); 503 | 504 | function generate_fn_code(name, arity, infixp) { 505 | var c = 0; 506 | var evalers = []; 507 | var callers = []; 508 | 509 | for(c = 0; c < arity; c++) { 510 | evalers = evalers.concat(["var e", c, " = eval0(args.car, env);\n"]); 511 | evalers = evalers.concat(["args = args.cdr;\n"]); 512 | callers = callers.concat([["e", c, "(aenv, cenv)"].join("")]); 513 | } 514 | var args_evald = evalers.join(""); 515 | 516 | var return_val; 517 | if (infixp === true) { 518 | return_val = ["return function(aenv, cenv) { return ", callers[0], " ", name, " ", callers[1], "; };"].join(""); 519 | } else { 520 | return_val = ["return function(aenv, cenv) { return ", name, "(", callers.join(", "), "); };"].join(""); 521 | } 522 | return new Function(["args, env, eval0"], [args_evald, return_val].join("\n")); 523 | } 524 | 525 | function generate_begin_code(arity) { 526 | var c = 0; 527 | var evalers = []; 528 | var callers = []; 529 | 530 | for(c = 0; c < arity; c++) { 531 | evalers = evalers.concat(["var e", c, " = eval0(nth(args, ", c+1, "), env);\n"]); 532 | var prefix = c < arity ? "" : "return "; 533 | callers = callers.concat([[prefix, "e", c, "(aenv, cenv)"].join("")]); 534 | } 535 | 536 | var args_evald = evalers.join(""); 537 | var return_val = ["return function(aenv, cenv) { ", callers.join("; "), " };"].join(""); 538 | return new Function(["args, env, eval0"], [args_evald, return_val].join("\n")); 539 | } 540 | 541 | function veval(exp, env) { 542 | return eval0(exp, env)([], []); 543 | } 544 | 545 | function build_env(len, e, aenv, cenv) { 546 | var new_env = new Array(len); 547 | var i = 0; 548 | while(e !== null) { 549 | new_env[i++] = e.car(aenv, cenv); 550 | e = e.cdr; 551 | } return new_env; 552 | } 553 | 554 | function fresh_n(n, c) { 555 | var m = new Array(n); 556 | while(n-- > 0) { 557 | m[n] = mkvar(c++); 558 | } return [m, c]; 559 | } 560 | 561 | function apply_fresh(len, closure, aenv, cenv) { 562 | var closure1 = closure(aenv, cenv); 563 | var closure_env = closure1[1]; 564 | var closure_fn = closure1[0]; 565 | return function(mks) { 566 | var c = mks.counter; 567 | var e1_c1 = fresh_n(len, c); 568 | var mks1 = Mks(mks.substitution, e1_c1[1], mks.diseq, mks.types, mks.absentee, mks.watch); 569 | return closure_fn(e1_c1[0], closure_env)(mks1); 570 | }; 571 | } 572 | 573 | function arity_error(expected, recieved, exp, values) { 574 | throw ["Error: Invalid argument count in application", 575 | "\nExpected: ", expected, 576 | "\nRecieved: ", recieved, 577 | "\nExpression: ", pretty_print(exp), 578 | "\nValues:", JSON.stringify(values)].join(""); 579 | } 580 | 581 | function map_stream(fn, stream) { 582 | return function() { 583 | var $ = pull(stream); 584 | return ($ == null) ? null : cons(fn($.car), map_stream(fn, $.cdr)); 585 | }; 586 | } 587 | 588 | function query_stream(q, qenv, mks) { 589 | var $ = q(mks); 590 | var run_queries = function(mks) { 591 | var qm = map(function(p) { return cons(p.car.string, p.cdr()()); }, qenv); 592 | return query_map(qm, mks); 593 | }; 594 | return map_stream(run_queries, $); 595 | } 596 | 597 | function stream_generator($) { 598 | var next = $; 599 | return function() { 600 | var cur = next(); 601 | if(cur == null) { return null; } 602 | else { next = cur.cdr; 603 | return cur.car; } 604 | }; 605 | } 606 | 607 | function run_expression(p) { 608 | var desugared = desugar(p, null); 609 | var lifted = lift_frees(desugared); 610 | 611 | var exp = lifted.car; 612 | var env = cons(null, lifted.cdr.car); 613 | var mks = lifted.cdr.cdr.car; 614 | var evald = veval(exp, env); 615 | 616 | if(procedurep(evald)) { 617 | var q$ = query_stream(evald, env.cdr, mks); 618 | return stream_generator(q$); 619 | } else if (env.cdr === null) { 620 | return evald; 621 | } else { 622 | throw "unbound variables: " + pretty_print(map(car, env.cdr)); 623 | } 624 | } 625 | 626 | function run_program(p) { 627 | var stream; 628 | if(p == null) { throw "no program" } 629 | map(register_define, p); 630 | while(p != null) { 631 | stream = run_expression(p.car); 632 | p = p.cdr; 633 | } return stream; 634 | } 635 | 636 | this.reset = function() { toplevel = new Object(null); }; 637 | this.eval = function(ast) { return run_program(ast); }; 638 | this.read_eval = function(text) { return run_program(read_program(text)); }; 639 | } 640 | -------------------------------------------------------------------------------- /examples/editor.css: -------------------------------------------------------------------------------- 1 | html { height:100%; } 2 | body { 3 | padding:0; 4 | margin: 0; 5 | font-family:courier, monospace; 6 | height:100%; 7 | background-color:#eee; 8 | } 9 | 10 | #container { 11 | position:absolute; 12 | top:0; bottom:0; 13 | height:100%; width:100%; 14 | } 15 | #head { position:absolute; top:0; width:100%; background-color:#ccc;} 16 | #demos { display:none; } 17 | -------------------------------------------------------------------------------- /examples/editor.js: -------------------------------------------------------------------------------- 1 | function load_editor() { 2 | var veneer = new Veneer_v1(); 3 | 4 | var vm = veneer.new_vm(); 5 | 6 | var ed = veneer.new_editor(document.getElementById("container"), vm); 7 | var editor = ed.editor; 8 | var editor_elt = ed.editor_elt; 9 | var repl = ed.repl; 10 | var errors = ed.errors; 11 | var dump_button = ed.dump_button; 12 | 13 | 14 | var font_size = 10; 15 | var font_plus = document.getElementById("font_plus"); 16 | var font_minus = document.getElementById("font_minus"); 17 | font_plus.onclick = function() { 18 | font_size = font_size + 1; 19 | editor_elt.style.fontSize = "" + font_size + "pt"; 20 | repl.style.fontSize = "" + font_size + "pt"; 21 | editor.refresh() 22 | }; 23 | font_minus.onclick = function() { 24 | font_size = font_size - 1; 25 | repl.style.fontSize = "" + font_size + "pt"; 26 | editor_elt.style.fontSize = "" + font_size + "pt"; 27 | editor.refresh(); 28 | }; 29 | 30 | var slider = document.getElementById("slider"); 31 | slider.onchange = function() { 32 | font_size = (parseInt(slider.value) + 16) * 0.5; 33 | editor_elt.style.fontSize = "" + font_size + "pt"; 34 | repl.style.fontSize = "" + font_size + "pt"; 35 | editor.refresh(); 36 | }; 37 | 38 | var demo = false; 39 | function setEditorValue(val) { 40 | demo = false; 41 | editor.setValue(val); 42 | } 43 | 44 | var demos = document.getElementsByClassName("demo"); 45 | var demo_picker = document.getElementById("demo_picker"); 46 | var file_values = new Array(demos.length); 47 | for(var i=0; i < demos.length; i++ ) { 48 | var selection = document.createElement("option"); 49 | var name = demos[i].getElementsByClassName("title")[0].textContent; 50 | file_values[i] = demos[i].getElementsByClassName("content")[0].value; 51 | selection.value = i; 52 | selection.appendChild(document.createTextNode(name)); 53 | demo_picker.appendChild(selection); 54 | } 55 | demo_picker.onchange = function() { 56 | load_demo(demo_picker.selectedIndex); 57 | }; 58 | demo_picker.value = "0"; 59 | setEditorValue(file_values[0]); 60 | demo = 0; 61 | 62 | document.getElementById("create_link").onclick = function() { 63 | var req = new XMLHttpRequest(); 64 | var method = "POST"; 65 | var url = "https://api.github.com/gists"; 66 | 67 | var options = { 68 | "description": "veneer save", 69 | "public": true, 70 | "files": { 71 | "file": { 72 | "content": editor.getValue() 73 | } 74 | } 75 | }; 76 | 77 | req.open(method, url, true); 78 | req.onreadystatechange = function () { 79 | if (req.readyState !== XMLHttpRequest.DONE) { 80 | return; 81 | } 82 | if (req.status === 201) { 83 | var response = JSON.parse(req.responseText); 84 | window.location.hash = response.id; 85 | } 86 | }; 87 | req.send(JSON.stringify(options)); 88 | }; 89 | 90 | function load_demo(id) { 91 | var save_status = true; 92 | if(save_status) { 93 | var demo_val = file_values[id]; 94 | setEditorValue(demo_val); 95 | demo = id; 96 | } 97 | } 98 | 99 | if(window.location.hash) { 100 | var req = new XMLHttpRequest(); 101 | var method = "GET"; 102 | var url = "https://api.github.com/gists/" + window.location.hash.substring(1); 103 | 104 | req.open(method, url, true); 105 | req.onreadystatechange = function () { 106 | if (req.readyState !== XMLHttpRequest.DONE) { 107 | return; 108 | } 109 | if (req.status === 200) { 110 | var response = JSON.parse(req.responseText); 111 | setEditorValue(response.files.file.content); 112 | } 113 | }; 114 | req.send(); 115 | } 116 | } 117 | -------------------------------------------------------------------------------- /examples/quines.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 29 | 30 | relational interpreter 31 | 32 | 33 |

relational interpreter fun

34 |

35 | With a relational interpreter you can run normal scheme functions "backwards" by grounding the output and leaving the inputs fresh. The interpreter will then produce the inputs for the given output. 36 |

37 |
38 | 56 |
57 |

58 | By setting a programs input to also be it's output, you create a program which returns itself, known as a quine. 59 |

60 |
61 | 65 |
66 |

67 | We can also create twines, which are two programs that produce the other. We make each program the output of each other and make sure they are different with a disequality constraint (otherwise we would just get two quines). 68 |

69 |
70 | 77 |
78 |

Now go make some more!

79 |

80 | 81 |

82 | 83 | 84 | 85 | 86 | 174 | 175 | 176 | -------------------------------------------------------------------------------- /examples/quines.js: -------------------------------------------------------------------------------- 1 | function load_page() { 2 | var rel_interp = document.getElementById("rel_interp").value; 3 | var veneer = new Veneer_v1(); 4 | 5 | var vm1 = veneer.new_vm(); 6 | var vm2 = veneer.new_vm(); 7 | var vm3 = veneer.new_vm(); 8 | var editor1 = veneer.new_editor(document.getElementById("ed1"), vm1, rel_interp); 9 | var editor2 = veneer.new_editor(document.getElementById("ed2"), vm2, rel_interp); 10 | var editor3 = veneer.new_editor(document.getElementById("ed3"), vm3, rel_interp); 11 | 12 | editor1.editor.setValue(document.getElementById("code1").value); 13 | editor2.editor.setValue(document.getElementById("code2").value); 14 | editor3.editor.setValue(document.getElementById("code3").value); 15 | } 16 | 17 | -------------------------------------------------------------------------------- /examples/veneer.css: -------------------------------------------------------------------------------- 1 | .CodeMirror-matchingbracket { border:1px solid #ccc; margin:-1px; } 2 | .CodeMirror-nonmatchingbracket { border:1px solid #f00; margin:-1px; } 3 | 4 | .buttons_container { 5 | background-color:#ccc; 6 | position:absolute; 7 | width:100%; 8 | height:1.5em; 9 | top:1.5em; 10 | } 11 | 12 | .editor_container { 13 | position:absolute; 14 | top:3em; 15 | bottom:0; 16 | width:100%; 17 | } 18 | 19 | .CodeMirror { 20 | width:100%; 21 | height:100%; 22 | } 23 | 24 | .repl_container { 25 | display:none; 26 | position:absolute; 27 | top:3em; 28 | bottom:0; 29 | width:100%; 30 | overflow-y:scroll; 31 | } 32 | 33 | .repl { 34 | width:100%; 35 | font-family: monospace, courier; 36 | font-size:10pt; 37 | background-color:#f9f9f9; 38 | } 39 | 40 | .buttons_container div .helptext { font-size:10pt; } 41 | .buttons_container div:hover { color:#000; background-color:#ccc; } 42 | .buttons_container div { 43 | font-size:1.25em; 44 | display:inline; 45 | background-color: #eee; 46 | cursor:pointer; 47 | border-right:3px solid #ccc; 48 | padding:0 .4em; 49 | } 50 | .clear { clear:both; margin:0; padding:0; border-width:0; displat:none; } 51 | 52 | .errors { 53 | position:absolute; 54 | top:.5em; left:.5em; 55 | z-index:100; 56 | width:46%; 57 | margin:0; 58 | padding: 0 2%; 59 | background-color:transparent; 60 | border-width:0; 61 | white-space:pre-wrap; 62 | } 63 | 64 | .error { background-color:#f00; color:#000;} 65 | 66 | .inputbox:last-child { padding:8px 8px; margin-bottom:8px} 67 | 68 | .inputbox { 69 | background-color:#fff; 70 | white-space: pre-wrap; 71 | border:1px solid #ccc; 72 | min-height:1em; 73 | padding:2px 8px; 74 | } 75 | 76 | .result { 77 | white-space: pre-wrap; 78 | background-color: #eee; 79 | margin:0; 80 | padding: 2px 8px; 81 | } 82 | -------------------------------------------------------------------------------- /examples/veneer.js: -------------------------------------------------------------------------------- 1 | function Veneer_v1() { 2 | this.new_editor = function(container, vm, seed) { 3 | container.className += " veneer_container"; 4 | 5 | var errors = document.createElement("div"); 6 | errors.className += "errors"; 7 | container.appendChild(errors); 8 | 9 | var cm_options = { mode: "scheme", 10 | matchBrackets: true, 11 | autoCloseBrackets: true, 12 | lineNumbers: false, 13 | lineWrapping: true}; 14 | 15 | 16 | var editor_container = document.createElement("div"); 17 | editor_container.className += "editor_container"; 18 | container.appendChild(editor_container); 19 | var editor_elt; 20 | var editor = new CodeMirror(function(ed) { 21 | editor_container.appendChild(ed); 22 | editor_elt = ed; 23 | }, cm_options); 24 | 25 | var repl_container = document.createElement("div"); 26 | repl_container.className += "repl_container"; 27 | container.appendChild(repl_container); 28 | 29 | var repl = document.createElement("div"); 30 | repl.className += "repl"; 31 | repl_container.appendChild(repl); 32 | 33 | var buttons_container = document.createElement("div"); 34 | buttons_container.className += "buttons_container"; 35 | container.appendChild(buttons_container); 36 | 37 | var dump_button = document.createElement("div"); 38 | dump_button.className += "dump_button"; 39 | dump_button.appendChild(document.createTextNode("run!")); 40 | dump_button.style.padding = "0 2em"; 41 | buttons_container.appendChild(dump_button); 42 | 43 | var edit_button = document.createElement("div"); 44 | edit_button.className += "edit_button"; 45 | edit_button.appendChild(document.createTextNode("edit")); 46 | buttons_container.appendChild(edit_button); 47 | 48 | var repl_button = document.createElement("div"); 49 | repl_button.className += "repl_button"; 50 | repl_button.appendChild(document.createTextNode("repl")); 51 | buttons_container.appendChild(repl_button); 52 | 53 | var edit_mode = "edit"; 54 | 55 | var vertical_button = document.createElement("div"); 56 | vertical_button.className += "vertical_button"; 57 | vertical_button.appendChild(document.createTextNode("vertical")); 58 | buttons_container.appendChild(vertical_button); 59 | vertical_button.style.float = "right"; 60 | vertical_button.onclick = function() { 61 | editor_container.style.display="block"; 62 | editor_container.style.width="50%" 63 | editor_container.style.top="3em"; 64 | editor_container.style.height="auto"; 65 | editor_container.style.bottom="0"; 66 | 67 | repl_container.style.display="block"; 68 | repl_container.style.width="50%" 69 | repl_container.style.top="3em"; 70 | repl_container.style.right="0"; 71 | repl_container.style.height="auto"; 72 | repl_container.style.bottom="0"; 73 | 74 | edit_mode = "vertical"; 75 | repl_button.style.display = "none"; 76 | edit_button.style.display = "none"; 77 | } 78 | 79 | var present_button = document.createElement("div"); 80 | present_button.className += "present_button"; 81 | present_button.appendChild(document.createTextNode("present")); 82 | buttons_container.appendChild(present_button); 83 | present_button.style.float = "right"; 84 | present_button.onclick = function() { 85 | editor_container.style.display="block"; 86 | editor_container.style.height="auto"; 87 | editor_container.style.width="100%"; 88 | editor_container.style.bottom="0"; 89 | 90 | repl_container.style.display="none"; 91 | repl_container.style.width="100%"; 92 | repl_container.style.height="auto"; 93 | repl_container.style.top="3em"; 94 | repl_container.style.bottom="0"; 95 | 96 | 97 | edit_mode = "edit"; 98 | repl_button.style.display = "inline"; 99 | edit_button.style.display = "inline"; 100 | } 101 | 102 | // var clear = document.createElement("hr"); 103 | // clear.className += "clear"; 104 | // container.appendChild(clear); 105 | 106 | var current_input = repl_getline(false); 107 | function display_error(e) { 108 | var error = document.createElement("div"); 109 | var error_txt = document.createTextNode(e); 110 | error.className += "error"; 111 | error.appendChild(error_txt); 112 | errors.appendChild(error); 113 | setTimeout(function() { errors.removeChild(error); }, 3000); 114 | } 115 | 116 | var getTime; 117 | if (window.performance && window.performance.now) { 118 | getTime = function() { return window.performance.now(); }; 119 | } else { 120 | getTime = function() { return (new Date()).getTime(); }; 121 | } 122 | 123 | function load_input(inputbox) { 124 | inputbox = current_input; 125 | /* var textcontent = inputbox.innerHTML.replace(//ig, '\n') 126 | .replace(/<[p|div]\s/ig, '\n$0') 127 | .replace(/(<([^>]+)>)/ig, ""); */ 128 | 129 | var pass = false; 130 | var textcontent = inputbox.textContent; 131 | try { 132 | var result = vm.read_eval(textcontent); 133 | var result_elt = document.createElement("div"); 134 | result_elt.className += "result"; 135 | 136 | function display_query(generator, parent) { 137 | var answer_text = document.createElement("div"); 138 | var button = document.createElement("button"); 139 | var append_answer = function(focus) { 140 | var answered; 141 | var time_before = getTime(); 142 | var answer_val = generator(); 143 | var run_time = (getTime() - time_before).toFixed(2); 144 | if (answer_val === null) { 145 | answered = "No."; 146 | answer_val = ""; 147 | parent.removeChild(button); 148 | } else { 149 | answered = "Yes."; 150 | } 151 | var answer_val_pp = answered + " (" + run_time + "ms)\n" + answer_val; 152 | var answer_node = document.createTextNode("=> " + answer_val_pp + "\n"); 153 | answer_text.appendChild(answer_node); 154 | 155 | if(focus) { 156 | button.focus(); 157 | current_input.scrollIntoView(false); 158 | } 159 | 160 | return false; 161 | } 162 | button.onclick = function() { return append_answer(true); }; 163 | button.appendChild(document.createTextNode("More answers!")); 164 | parent.appendChild(answer_text); 165 | parent.appendChild(button); 166 | append_answer(false); 167 | return answer_text; 168 | } 169 | 170 | function display_val(val, parent) { 171 | parent.appendChild(document.createTextNode(pretty_print(val))); 172 | } 173 | 174 | procedurep(result) ? display_query(result, result_elt) 175 | : display_val(result, result_elt); 176 | 177 | repl.appendChild(result_elt); 178 | inputbox.contentEditable = false; 179 | repl_getline(true); 180 | return false; 181 | } catch (e) { 182 | if(e instanceof ReaderError) { 183 | inputbox.appendChild(document.createTextNode("\n")); 184 | display_error(e.msg); 185 | pass = true; 186 | throw e.msg; 187 | } else { 188 | display_error(e); 189 | pass = false; 190 | throw e; 191 | } 192 | } finally { return pass; } 193 | } 194 | 195 | function repl_getline(focus) { 196 | var inputbox = document.createElement("div"); 197 | inputbox.className += "inputbox"; 198 | inputbox.contentEditable = true; 199 | inputbox.onkeypress = function(e) { 200 | if (e.keyCode == 13) { 201 | return load_input(inputbox); 202 | } 203 | }; 204 | 205 | repl.appendChild(inputbox); 206 | if (focus) { inputbox.focus(); } 207 | current_input = inputbox; 208 | return inputbox; 209 | } 210 | 211 | var dump_editor = function(event) { 212 | if (edit_mode === "edit") { 213 | edit_mode = "repl"; 214 | editor_container.style.display="none"; 215 | repl_container.style.display="block"; 216 | } 217 | 218 | vm.reset(); 219 | if (!(typeof seed === 'undefined')) { vm.read_eval(seed); } 220 | current_input.textContent = editor.getValue(); 221 | load_input(current_input); 222 | 223 | event.preventDefault(); 224 | return false; 225 | }; 226 | dump_button.onclick = dump_editor; 227 | edit_button.onclick = function(event) { 228 | if (edit_mode === "edit") { 229 | } else { 230 | edit_mode = "edit"; 231 | editor_container.style.display="block"; 232 | repl_container.style.display="none"; 233 | editor.refresh(); 234 | } 235 | }; 236 | repl_button.onclick = function(event) { 237 | if (edit_mode === "edit") { 238 | edit_mode = "repl"; 239 | repl_container.style.display="block"; 240 | editor_container.style.display="none"; 241 | } else if (edit_mode === "repl") { 242 | current_input.scrollIntoView(false); 243 | } 244 | }; 245 | 246 | var shift = false; 247 | 248 | editor.setOption("extraKeys", { 249 | "Shift-Enter": function(cm) { dump_editor();} 250 | }); 251 | 252 | vertical_button.onclick(); 253 | return { editor: editor, editor_elt: editor_elt, repl: repl, errors: errors, dump_button: dump_button }; 254 | }; 255 | 256 | 257 | this.new_vm = function() { return new VeneerVM(); }; 258 | } 259 | -------------------------------------------------------------------------------- /mk.js: -------------------------------------------------------------------------------- 1 | function Var(c) { this.c = c; } 2 | function mkvar(c) { return new Var(c); } 3 | function varp(x) { return (x instanceof Var); } 4 | function vareq(x1, x2) { return x1.c === x2.c; } 5 | 6 | function MiniKanrenState(s, c, d, ty, ab, w) { 7 | this.substitution = s; 8 | this.counter = c; 9 | this.diseq = d; 10 | this.types = ty; 11 | this.absentee = ab; 12 | this.watch = w; 13 | } 14 | function Mks(s, c, d, ty, ab, w) { return new MiniKanrenState(s, c, d, ty, ab, w); } 15 | 16 | var not_found = []; 17 | function walk(u, s) { 18 | var pr; 19 | while(varp(u)) { 20 | pr = s.get(u.c, not_found); 21 | if (pr === not_found) { 22 | return u; 23 | } else { 24 | u = pr; 25 | } 26 | } return u; 27 | } 28 | 29 | function occurs_check(x, v, s) { 30 | var v1 = walk(v, s); 31 | if(varp(v1)) { 32 | return vareq(v1, x); 33 | } else if (pairp(v1)) { 34 | return occurs_check(x, v1.car, s) || occurs_check(x, v1.cdr, s); 35 | } else { 36 | return false; 37 | } 38 | } 39 | 40 | function ext_s_check(x, v, s) { 41 | return occurs_check(x, v, s) ? false : s.set(x.c, v); 42 | } 43 | 44 | function eqeq(u, v) { 45 | return function(mks) { 46 | var s_cs = unify_prefix(u, v, mks.substitution, null); 47 | return s_cs !== false ? normalize_constraint_store(Mks(s_cs.car, mks.counter, mks.diseq, mks.types, mks.absentee, mks.watch), s_cs.cdr) : mzero; 48 | }; 49 | } 50 | 51 | function noteqeq(u, v) { 52 | return function(mks) { 53 | var d = disequality(u, v, mks.substitution); 54 | return d !== false ? unit(Mks(mks.substitution, mks.counter, cons(d, mks.diseq), mks.types, mks.absentee, mks.watch)) : mzero; 55 | }; 56 | } 57 | 58 | function type_check(v, p, s, ty) { 59 | var v1 = walk(v, s); 60 | if (varp(v1)) { 61 | var existing_type = ty.get(v1, false); 62 | if(existing_type && existing_type !== p) { 63 | return false; 64 | } else { // add constraint 65 | return ty.set(v1, p); 66 | } 67 | } else { // ground; run test 68 | return p(v1) ? ty : false; 69 | } 70 | } 71 | 72 | function symbolo(s) { return typeo(s, symbolp); } 73 | function numbero(n) { return typeo(n, numberp); } 74 | 75 | function typeo(v, p) { 76 | return function(mks) { 77 | var s = mks.substitution; 78 | var ty = mks.types; 79 | var ty1 = type_check(v, p, s, ty); 80 | if (!ty1) { 81 | return mzero; 82 | } else if (ty1 === ty) { 83 | return unit(mks); 84 | } else { 85 | var mks1 = Mks(mks.substitution, mks.counter, mks.diseq, ty1, mks.absentee, mks.watch); 86 | return normalize_constraint_store(mks1, null); 87 | } 88 | }; 89 | } 90 | 91 | function absento(s, f) { 92 | return function(mks) { 93 | return normalize_constraint_store(Mks(mks.substitution, mks.counter, mks.diseq, mks.types, cons(cons(s, f), mks.absentee), mks.watch), null); 94 | }; 95 | } 96 | 97 | function unit(mks) { return cons(mks, mzero); } 98 | var mzero = null; 99 | 100 | function unify(u, v, s) { 101 | var u1 = walk(u, s); 102 | var v1 = walk(v, s); 103 | if (varp(u1) && varp(v1) && vareq(u1, v1)) { return s; } 104 | else if (varp(u1)) { return ext_s_check(u1, v1, s); } 105 | else if (varp(v1)) { return ext_s_check(v1, u1, s); } 106 | else if (pairp(u1) && pairp(v1)) { 107 | var s1 = unify(u1.car, v1.car, s); 108 | return (s1 !== false) && unify(u1.cdr, v1.cdr, s1); 109 | } else { 110 | return (u1 === v1) && s; 111 | } 112 | } 113 | 114 | 115 | function ext_s_check_prefix(x, v, s, cs) { 116 | if (occurs_check(x, v, s)) { 117 | return false; 118 | } else { 119 | return cons(s.set(x.c, v), cons(cons(x, v), cs)); 120 | } 121 | } 122 | 123 | function unify_prefix(u, v, s, cs) { 124 | var u1 = walk(u, s); 125 | var v1 = walk(v, s); 126 | if (varp(u1) && varp(v1) && vareq(u1, v1)) { return cons(s, cs); } 127 | else if (varp(u1)) { return ext_s_check_prefix(u1, v1, s, cs); } 128 | else if (varp(v1)) { return ext_s_check_prefix(v1, u1, s, cs); } 129 | else if (pairp(u1) && pairp(v1)) { 130 | var s_cs = unify_prefix(u1.car, v1.car, s, cs); 131 | return (s_cs !== false) && unify_prefix(u1.cdr, v1.cdr, s_cs.car, s_cs.cdr); 132 | } else { 133 | return (u1 === v1) && cons(s, cs); 134 | } 135 | } 136 | 137 | function disequality(u, v, s) { 138 | var s_cs_hat = unify_prefix(u, v, s, null); 139 | if(s_cs_hat !== false) { 140 | var d = s_cs_hat.cdr; 141 | return (d === null) ? false : d; 142 | } 143 | else { 144 | return null; 145 | } 146 | } 147 | 148 | function watch_collect(v, k) { 149 | return cons(v, function(s) { 150 | var v1 = walk(v, s); 151 | if (varp(v1)) { return watch_collect(v1, k); } 152 | else if (pairp(v1)) { 153 | var k1 = function(x, s) { 154 | var k2 = function(y, s) { return k(cons(x,y), s); }; 155 | return cdr(watch_collect(cdr(v1), k2))(s); 156 | }; 157 | return cdr(watch_collect(car(v1), k1))(s); 158 | } 159 | else { return k(v1, s); } 160 | }); 161 | } 162 | 163 | function watch(v, g) { 164 | return function(mks) { 165 | var watch = mks.watch; 166 | var v1 = walk(v, mks.substitution); 167 | 168 | // build collector and run it once 169 | var g1 = function(val) { return g[0]([val], g[1]); }; 170 | var k = function(val, s) { return [g1(val)]; }; 171 | var w = cdr(watch_collect(v1, k))(mks.substitution); 172 | if(pairp(w)) { // still not ground, queue it 173 | watch = watch.update(w.car.c, null, function(ws) { 174 | return cons(w.cdr, ws); 175 | }); 176 | var mks1 = Mks(mks.substitution, mks.counter, mks.diseq, mks.types, mks.absentee, watch); 177 | return unit(mks1); 178 | } else { // done, run goal 179 | return w[0](mks); 180 | } 181 | } 182 | } 183 | 184 | function execute_goals(gs) { 185 | if (gs === null) { 186 | return unit; 187 | } else { 188 | return function(mks) { 189 | return bind(car(gs)(mks), execute_goals(cdr(gs))); 190 | } 191 | } 192 | } 193 | 194 | function normalize_surveilence(mks, prefix) { 195 | var watch = mks.watch; 196 | var done = null; 197 | while(prefix !== null) { 198 | var v = prefix.car.car; 199 | var ws = watch.get(v.c, false); 200 | var wsn = false; 201 | 202 | if (ws) { 203 | while (ws !== null) { 204 | // pick out a watch 205 | var w = car(ws)(mks.substitution); 206 | if(pairp(w)) { // requeue 207 | wsn = true; 208 | watch = watch.update(w.car.c, null, function(ws) { 209 | return cons(w.cdr, ws); 210 | }); 211 | } else { // done 212 | done = cons(w[0], done); 213 | } 214 | ws = ws.cdr; 215 | } 216 | if (wsn === false) { watch = watch.delete(v.c); } 217 | } 218 | prefix = prefix.cdr; 219 | } 220 | 221 | // run all todo 222 | var mks1 = Mks(mks.substitution, mks.counter,mks.diseq, mks.types, mks.absentee, watch); 223 | return execute_goals(done)(mks1); 224 | } 225 | 226 | function normalize_constraint_store(mks, prefix) { 227 | return bind(normalize_surveilence(mks, prefix), normalize_constraint_store1); 228 | } 229 | 230 | 231 | function normalize_constraint_store1(mks) { 232 | var s = mks.substitution; 233 | var c = mks.counter; 234 | var d = mks.diseq; 235 | var dn = null; 236 | var ty = mks.types; 237 | var tyn = Immutable.Map(); 238 | var abs = mks.absentee; 239 | var absn = null; 240 | var ws = mks.watch; 241 | 242 | // Normalize the type constraints 243 | ty.forEach(function(p, v) { 244 | tyn = type_check(v, p, s, tyn); 245 | return tyn; 246 | }); 247 | if (!tyn) { return mzero; } 248 | 249 | // normalize the disequality constraints 250 | while(d !== null) { 251 | var es = d.car; 252 | 253 | if(es !== null) { 254 | var d_hat = disequality(map(car, es), map(cdr, es), s); 255 | 256 | if(d_hat === false) { 257 | return mzero; 258 | } 259 | 260 | dn = cons(d_hat, dn); 261 | } 262 | 263 | d = d.cdr; 264 | } 265 | 266 | // normalize the absento constraints 267 | while(abs !== null) { 268 | var abs_cur = abs.car; 269 | abs = abs.cdr; 270 | 271 | var abs_s = walk(abs_cur.car, s); 272 | var abs_f = walk(abs_cur.cdr, s); 273 | 274 | // defer until both terms are ground 275 | if (varp(abs_f) || varp(abs_s)) { 276 | absn = cons(cons(abs_s, abs_f), absn); 277 | } 278 | // split and requeue 279 | else if (pairp(abs_f)) { 280 | abs = cons(cons(abs_s, abs_f.car), 281 | cons(cons(abs_s, abs_f.cdr), abs )); 282 | } 283 | // both are ground and atomic; check for eqv? 284 | else if (abs_s === abs_f) { 285 | return mzero; 286 | } 287 | // forget constraint, it can never fail 288 | } 289 | 290 | return unit(Mks(s, c, dn, tyn, absn, ws)); 291 | } 292 | 293 | function call_fresh(f) { 294 | return function(mks) { 295 | var c = mks.counter; 296 | return f(mkvar(c))(Mks(mks.substitution, (c + 1), mks.diseq, mks.types, mks.absentee, mks.watch)); 297 | }; 298 | } 299 | 300 | function disj(g1, g2) { 301 | return function(mks) { return mplus(g1(mks), g2(mks)); }; 302 | } 303 | function conj(g1, g2) { 304 | return function(mks) { return bind(g1(mks), g2); }; 305 | } 306 | 307 | function mplus($1, $2) { 308 | if ($1 === null) { 309 | return $2; 310 | } else if (procedurep($1)) { 311 | return function() { return mplus($2, $1()); }; 312 | } else { 313 | return cons($1.car, mplus($1.cdr, $2)); 314 | } 315 | } 316 | 317 | function bind($, g) { 318 | if ($ === null) { 319 | return mzero; 320 | } else if (procedurep($)) { 321 | return function() { return bind($(), g); }; 322 | } else { 323 | return mplus(g($.car), bind($.cdr, g)); 324 | } 325 | } 326 | 327 | 328 | function disj_dfs(g1, g2) { 329 | return function(mks) { return mplus_dfs(g1(mks), g2(mks)); }; 330 | } 331 | 332 | function conj_dfs(g1, g2) { 333 | return function(mks) { return bind_dfs(g1(mks), g2); }; 334 | } 335 | 336 | function mplus_dfs($1, $2) { 337 | if ($1 === null) { 338 | return $2; 339 | } else if (procedurep($1)) { 340 | return function() { return mplus_dfs($1(), $2); }; 341 | } else { 342 | return cons($1.car, mplus_dfs($1.cdr, $2)); 343 | } 344 | } 345 | 346 | function bind_dfs($, g) { 347 | if ($ === null) { 348 | return mzero; 349 | } else if (procedurep($)) { 350 | return function() { return bind_dfs($(), g); }; 351 | } else { 352 | return mplus_dfs(g($.car), bind_dfs($.cdr, g)); 353 | } 354 | } 355 | 356 | function pull($) { 357 | while(procedurep($)) { 358 | $ = $(); 359 | } return $; 360 | } 361 | 362 | function take(n, $) { 363 | if (n <= 0) { 364 | return null; 365 | } else { 366 | var $1 = pull($); 367 | return ($1 === null) ? null : cons($1.car, take(n - 1, $1.cdr)); 368 | } 369 | } 370 | 371 | function take_all($) { 372 | $ = pull($); 373 | return ($ === null) ? null : cons($.car, take_all($.cdr)); 374 | } 375 | 376 | function reify(v, s) { 377 | var v1 = walk_star(v, s); 378 | return walk_star(v1, reify_s(v1, Immutable.Map())); 379 | } 380 | 381 | function walk_star(v, s) { 382 | var v1 = walk(v, s); 383 | if (varp(v1)) { 384 | return v1; 385 | } else if (pairp(v1)) { 386 | return cons(walk_star(v1.car, s), 387 | walk_star(v1.cdr, s)); 388 | } else { 389 | return v1; 390 | } 391 | } 392 | 393 | function reify_s(v, s) { 394 | var v1 = walk(v, s); 395 | if (varp(v1)) { 396 | return s.set(v1.c, reify_name(s.size)); 397 | } else if (pairp(v1)) { 398 | return reify_s(v1.cdr, reify_s(v1.car, s)); 399 | } else { 400 | return s; 401 | } 402 | } 403 | 404 | function reify_name(n) { 405 | return { toString: function() { return ["_", n].join("."); } }; 406 | } 407 | 408 | function pred_to_tag(p) { 409 | return assq(p, list(cons(symbolp, intern("sym")), 410 | cons(numberp, intern("num")))).cdr; 411 | } 412 | 413 | function query_map(qm, mks) { 414 | var s = mks.substitution; 415 | var d = mks.diseq; 416 | var t = mks.types; 417 | // convert the type store to alist 418 | var t1 = t.reduce(function(m, v, k) { return cons(cons(k, v), m); }, null); 419 | 420 | var sq1 = walk_star(qm, s); 421 | var dq1 = walk_star(d, s); 422 | var tq1 = walk_star(t1, s); 423 | var q = walk_star(list(sq1, dq1, tq1), reify_s(list(sq1, dq1, tq1), Immutable.Map())); 424 | 425 | return print_constraints(q.car, q.cdr.car, q.cdr.cdr.car, s); 426 | } 427 | 428 | function print_constraints(sq, dq, tq, s) { 429 | var subs = foldl(sq, [], function(m, a_v) { 430 | return m.concat([a_v.car, ": ", pretty_print(a_v.cdr), "\n"]); 431 | }); 432 | 433 | var present_d = function(dd) { 434 | // dd is a disjunction of disequalities 435 | var diseqs = map(function(a){ return list(intern("=/="), a.car, a.cdr); }, dd); 436 | if (length(diseqs) > 1) { 437 | return pretty_print(cons(intern("or"), diseqs)); 438 | } else if (diseqs !== null) { 439 | return pretty_print(diseqs.car); 440 | } else { 441 | return null; 442 | } 443 | }; 444 | var present_t = function(tt) { 445 | return pretty_print(list(pred_to_tag(tt.cdr), tt.car)); 446 | } 447 | return (dq !== null || tq !== null) ? 448 | (subs 449 | .concat(["{"]) 450 | .concat(intersperse_map(dq, present_d, ", ")) 451 | .concat([", "]) 452 | .concat(intersperse_map(tq, present_t, ", ")) 453 | .concat(["}\n"]).join("")) : 454 | subs.join(""); 455 | } 456 | 457 | function intersperse_map(l, f, sep) { 458 | var m = []; 459 | var r; 460 | while(l !== null) { 461 | r = f(l.car); 462 | m = r === null ? m : m.concat([r, (l.cdr === null ? "" : sep)]); 463 | l = l.cdr; 464 | } 465 | return m; 466 | } 467 | 468 | function build_num(n) { 469 | if((n % 2) === 1) { 470 | return cons(1, build_num(Math.floor((n - 1) / 2))); 471 | } else if (n !== 0 && (n % 2) === 0) { 472 | return cons(0, build_num(Math.floor(n / 2))); 473 | } else { 474 | return null; 475 | } 476 | } 477 | -------------------------------------------------------------------------------- /mk_test.js: -------------------------------------------------------------------------------- 1 | 2 | function fives(x) { 3 | return disj(eqeq(x, 5), 4 | function(a_c) { 5 | return function() { return fives(x)(a_c); } 6 | }); 7 | } 8 | function sixes(x) { 9 | return disj(eqeq(x, 6), 10 | function(a_c) { 11 | return function() { return sixes(x)(a_c); } 12 | }); 13 | } 14 | 15 | 16 | function node_log(x) { return console.log(require('util').inspect(x, true, 10)); } 17 | var empty_state = cons(null, 0); 18 | var test1 = call_fresh(function (q) { return eqeq(q, 5); })(empty_state); 19 | node_log(test1.car); 20 | node_log("-----"); 21 | 22 | var fv; 23 | var sv; 24 | var test2 = function(x) { 25 | fv = x; 26 | return fives(x); 27 | }; 28 | 29 | var res = conj(call_fresh(test2), call_fresh(sixes))(empty_state); 30 | 31 | node_log(assq(fv, take(5, res).car.car)); 32 | 33 | var x = ref(null); 34 | var y = ref(null); 35 | var foo = conj(call_fresh(function(z) { x.set(z); return eqeq(x.get(), y.get()) }), 36 | call_fresh(function(z) { y.set(z); return eqeq(y.get(), 1) })); 37 | 38 | node_log(take(3, foo(empty_state))); 39 | 40 | function appendo(l, s, out) { 41 | return disj(conj(eqeq(null, l), eqeq(s, out)), 42 | call_fresh(function(a) { 43 | return call_fresh(function(d) { 44 | return conj(eqeq(cons(a, d), l), 45 | call_fresh(function(res) { 46 | return conj(function(s_c) { 47 | return function() { 48 | return appendo(d,s,res)(s_c); }; 49 | }, eqeq(cons(a, res), out)); 50 | })); 51 | }); 52 | })); 53 | } 54 | 55 | var call_appendo = 56 | call_fresh(function(q) { 57 | return call_fresh(function(l) { 58 | return call_fresh(function(s) { 59 | return call_fresh(function(out) { 60 | return conj(appendo(l,s,out), 61 | eqeq(list(l, s), q)); }); }); }); }); 62 | 63 | function pp(x) { 64 | if (pairp(x)) { 65 | return ["(", pp(x.car), ".", pp(x.cdr), ")"].join(""); 66 | } else { 67 | return JSON.stringify(x); 68 | } 69 | } 70 | 71 | var test3 = call_fresh(function(a) { return appendo(list(1), a, list(1,2)); }); 72 | var test4 = call_fresh(function(q) { return call_fresh(function(b) { return call_fresh(function(a) { return conj(eqeq(list(a,b), q), 73 | appendo(a, b, list(1,2,3))); }); }); }); 74 | 75 | console.log("--"); 76 | map(function(x) { return node_log(reify_first(x)); }, take(5, test4(empty_state))); 77 | node_log(eqeq(1,1)(empty_state)); 78 | -------------------------------------------------------------------------------- /reader.js: -------------------------------------------------------------------------------- 1 | var eof = ["eof"]; 2 | function eofp(c) { return eof === c; } 3 | 4 | function Stream(string) { 5 | var length = string.length; 6 | this.pos = 0; 7 | this.line = 1; 8 | this.string = string; 9 | this.peek_char = function() { 10 | return (this.pos >= length) ? eof : string[this.pos]; 11 | } 12 | this.read_char = function() { 13 | var old_pos = this.pos; 14 | this.pos++; 15 | if (string[old_pos] == "\n") { this.line++; } 16 | return (old_pos >= length) ? eof : string[old_pos]; 17 | } 18 | this.unread_char = function() { 19 | if (string[this.pos-1] == "\n") { this.line--; } 20 | this.pos--; 21 | } 22 | } 23 | 24 | function whitespacep(c) { return /\s+/.test(c); } 25 | function symbolicp(c) { return /[a-zA-Z0-9=\/\\!@#$%^&*_+=\-?.~<>\:]/.test(c); } 26 | function number_stringp(s) { return /^([0-9]*\.)?[0-9]+$/.test(s); } 27 | 28 | function skip_whitespace(input_stream) { 29 | var c; 30 | while(true) { 31 | c = input_stream.read_char(); 32 | if(!whitespacep(c)) { break; } 33 | } 34 | input_stream.unread_char(); 35 | } 36 | 37 | function skip_whitespace_and_comments(input_stream) { 38 | var c; 39 | while(true) { 40 | c = input_stream.read_char(); 41 | if(c == ";") { 42 | while(!eofp(c)) { 43 | c = input_stream.read_char(); 44 | if(c == "\n") { 45 | break; 46 | } 47 | } 48 | } 49 | else if(!whitespacep(c)) { break; } 50 | } 51 | input_stream.unread_char(); 52 | } 53 | 54 | function read_symbol(input_stream) { 55 | var buffer = [input_stream.read_char()]; 56 | var c; 57 | while(true) { 58 | c = input_stream.read_char(); 59 | if(!eofp(c) && symbolicp(c)) { 60 | buffer.push(c); 61 | } else { 62 | input_stream.unread_char(); 63 | var result = buffer.join(""); 64 | return (number_stringp(result) ? parseFloat(result) : intern(result)); 65 | } 66 | } 67 | } 68 | 69 | function read_hex(input_stream) { 70 | var buffer = [input_stream.read_char()]; 71 | var c; 72 | while(true) { 73 | c = input_stream.read_char(); 74 | if(!eofp(c) && symbolicp(c)) { 75 | buffer.push(c); 76 | } else { 77 | input_stream.unread_char(); 78 | var result = buffer.join(""); 79 | return parseInt(result, 16); 80 | } 81 | } 82 | } 83 | 84 | function read_string(input_stream) { 85 | var buffer = []; 86 | var c; 87 | input_stream.read_char(); 88 | while(c = input_stream.read_char()) { 89 | if(c == '"') { 90 | return buffer.join(""); 91 | } else if (c == "\\") { 92 | buffer.push(input_stream.read_char()); 93 | } else { 94 | buffer.push(c); 95 | } 96 | } 97 | } 98 | 99 | function read_after_whitespace(input_stream) { 100 | skip_whitespace(input_stream); 101 | return read(input_stream); 102 | } 103 | 104 | function read_after_comment(input_stream) { 105 | var c; 106 | while(true) { 107 | c = input_stream.read_char(); 108 | if(c == "\n") { return read(input_stream); } 109 | else if (c === eof) { throw "eof"; } 110 | } 111 | } 112 | 113 | function read_quoted(input_stream) { 114 | input_stream.read_char(); 115 | return cons(intern("quote"), cons(read(input_stream), null)); 116 | } 117 | 118 | function read_quasiquoted(input_stream) { 119 | input_stream.read_char(); 120 | return cons(intern("quasiquote"), cons(read(input_stream), null)); 121 | } 122 | 123 | function read_unquoted(input_stream) { 124 | input_stream.read_char(); 125 | return cons(intern("unquote"), cons(read(input_stream), null)); 126 | } 127 | 128 | function read_list(input_stream) { 129 | input_stream.read_char(); 130 | return read_list_aux(null, input_stream); 131 | } 132 | 133 | function read_list_aux(sexps, input_stream) { 134 | while (true) { 135 | skip_whitespace_and_comments(input_stream); 136 | var c = input_stream.peek_char(); 137 | switch (c) { 138 | case eof: 139 | throw "unexpected eof"; 140 | case ")": 141 | input_stream.read_char(); 142 | return reverse_aux(sexps, null); 143 | case ".": 144 | return read_after_dot(sexps, input_stream); 145 | default: 146 | sexps = cons(read(input_stream), sexps); 147 | } 148 | } 149 | } 150 | 151 | function read_after_dot(sexps, input_stream) { 152 | input_stream.read_char(); 153 | var last = read(input_stream); 154 | skip_whitespace(input_stream); 155 | var c = input_stream.peek_char(); 156 | switch (c) { 157 | case eof: 158 | throw "unexpected eof"; 159 | case ")": 160 | input_stream.read_char(); 161 | break; 162 | default: 163 | throw "too many expressions after dot"; 164 | } 165 | return reverse_aux(sexps, last); 166 | } 167 | 168 | function read_hash(input_stream) { 169 | input_stream.read_char(); 170 | switch (input_stream.read_char()) { 171 | case "f": return false; 172 | case "t": return true; 173 | case "x": return read_hex(input_stream); 174 | case "\\": return input_stream.read_char(); 175 | default: throw "unknown hash code"; 176 | } 177 | } 178 | 179 | var readtable = { 180 | '"': read_string, 181 | "'": read_quoted, 182 | "`": read_quasiquoted, 183 | ",": read_unquoted, 184 | "(": read_list, 185 | ";": read_after_comment, 186 | "#": read_hash 187 | //eof: function(input_stream) { throw eof } 188 | }; 189 | 190 | function reader_for(c) { 191 | var reader = readtable[c]; 192 | if (reader) { return reader; } 193 | if (whitespacep(c)) { return read_after_whitespace; } 194 | if (symbolicp(c)) { return read_symbol; } 195 | return false; 196 | } 197 | 198 | function read(input_stream) { 199 | var c = input_stream.peek_char(); 200 | var reader = reader_for(c); 201 | if(reader) { return reader(input_stream); } 202 | throw "no reader for: " + c; 203 | } 204 | 205 | function read_top(sexps, input_stream) { 206 | while (true) { 207 | skip_whitespace_and_comments(input_stream); 208 | var c = input_stream.peek_char(); 209 | if (c === eof) { 210 | input_stream.read_char(); 211 | return reverse_aux(sexps, null); 212 | } else { 213 | sexps = cons(read(input_stream), sexps); 214 | } 215 | } 216 | } 217 | 218 | function ReaderError(msg) { this.msg = msg; } 219 | function read_program(str) { 220 | try { return read_top(null, new Stream(str)); } 221 | catch(err) { throw new ReaderError(err); } 222 | } 223 | -------------------------------------------------------------------------------- /resources/closebrackets.js: -------------------------------------------------------------------------------- 1 | // CodeMirror, copyright (c) by Marijn Haverbeke and others 2 | // Distributed under an MIT license: http://codemirror.net/LICENSE 3 | 4 | (function(mod) { 5 | if (typeof exports == "object" && typeof module == "object") // CommonJS 6 | mod(require("../../lib/codemirror")); 7 | else if (typeof define == "function" && define.amd) // AMD 8 | define(["../../lib/codemirror"], mod); 9 | else // Plain browser env 10 | mod(CodeMirror); 11 | })(function(CodeMirror) { 12 | var DEFAULT_BRACKETS = "()[]{}''\"\""; 13 | var DEFAULT_EXPLODE_ON_ENTER = "[]{}"; 14 | var SPACE_CHAR_REGEX = /\s/; 15 | 16 | var Pos = CodeMirror.Pos; 17 | 18 | CodeMirror.defineOption("autoCloseBrackets", false, function(cm, val, old) { 19 | if (old != CodeMirror.Init && old) 20 | cm.removeKeyMap("autoCloseBrackets"); 21 | if (!val) return; 22 | var pairs = DEFAULT_BRACKETS, explode = DEFAULT_EXPLODE_ON_ENTER; 23 | if (typeof val == "string") pairs = val; 24 | else if (typeof val == "object") { 25 | if (val.pairs != null) pairs = val.pairs; 26 | if (val.explode != null) explode = val.explode; 27 | } 28 | var map = buildKeymap(pairs); 29 | if (explode) map.Enter = buildExplodeHandler(explode); 30 | cm.addKeyMap(map); 31 | }); 32 | 33 | function charsAround(cm, pos) { 34 | var str = cm.getRange(Pos(pos.line, pos.ch - 1), 35 | Pos(pos.line, pos.ch + 1)); 36 | return str.length == 2 ? str : null; 37 | } 38 | 39 | // Project the token type that will exists after the given char is 40 | // typed, and use it to determine whether it would cause the start 41 | // of a string token. 42 | function enteringString(cm, pos, ch) { 43 | var line = cm.getLine(pos.line); 44 | var token = cm.getTokenAt(pos); 45 | if (/\bstring2?\b/.test(token.type)) return false; 46 | var stream = new CodeMirror.StringStream(line.slice(0, pos.ch) + ch + line.slice(pos.ch), 4); 47 | stream.pos = stream.start = token.start; 48 | for (;;) { 49 | var type1 = cm.getMode().token(stream, token.state); 50 | if (stream.pos >= pos.ch + 1) return /\bstring2?\b/.test(type1); 51 | stream.start = stream.pos; 52 | } 53 | } 54 | 55 | function buildKeymap(pairs) { 56 | var map = { 57 | name : "autoCloseBrackets", 58 | Backspace: function(cm) { 59 | if (cm.getOption("disableInput")) return CodeMirror.Pass; 60 | var ranges = cm.listSelections(); 61 | for (var i = 0; i < ranges.length; i++) { 62 | if (!ranges[i].empty()) return CodeMirror.Pass; 63 | var around = charsAround(cm, ranges[i].head); 64 | if (!around || pairs.indexOf(around) % 2 != 0) return CodeMirror.Pass; 65 | } 66 | for (var i = ranges.length - 1; i >= 0; i--) { 67 | var cur = ranges[i].head; 68 | cm.replaceRange("", Pos(cur.line, cur.ch - 1), Pos(cur.line, cur.ch + 1)); 69 | } 70 | } 71 | }; 72 | var closingBrackets = ""; 73 | for (var i = 0; i < pairs.length; i += 2) (function(left, right) { 74 | closingBrackets += right; 75 | map["'" + left + "'"] = function(cm) { 76 | if (cm.getOption("disableInput")) return CodeMirror.Pass; 77 | var ranges = cm.listSelections(), type, next; 78 | for (var i = 0; i < ranges.length; i++) { 79 | var range = ranges[i], cur = range.head, curType; 80 | var next = cm.getRange(cur, Pos(cur.line, cur.ch + 1)); 81 | if (!range.empty()) { 82 | curType = "surround"; 83 | } else if (left == right && next == right) { 84 | if (cm.getRange(cur, Pos(cur.line, cur.ch + 3)) == left + left + left) 85 | curType = "skipThree"; 86 | else 87 | curType = "skip"; 88 | } else if (left == right && cur.ch > 1 && 89 | cm.getRange(Pos(cur.line, cur.ch - 2), cur) == left + left && 90 | (cur.ch <= 2 || cm.getRange(Pos(cur.line, cur.ch - 3), Pos(cur.line, cur.ch - 2)) != left)) { 91 | curType = "addFour"; 92 | } else if (left == '"' || left == "'") { 93 | if (!CodeMirror.isWordChar(next) && enteringString(cm, cur, left)) curType = "both"; 94 | else return CodeMirror.Pass; 95 | } else if (cm.getLine(cur.line).length == cur.ch || closingBrackets.indexOf(next) >= 0 || SPACE_CHAR_REGEX.test(next)) { 96 | curType = "both"; 97 | } else { 98 | return CodeMirror.Pass; 99 | } 100 | if (!type) type = curType; 101 | else if (type != curType) return CodeMirror.Pass; 102 | } 103 | 104 | cm.operation(function() { 105 | if (type == "skip") { 106 | cm.execCommand("goCharRight"); 107 | } else if (type == "skipThree") { 108 | for (var i = 0; i < 3; i++) 109 | cm.execCommand("goCharRight"); 110 | } else if (type == "surround") { 111 | var sels = cm.getSelections(); 112 | for (var i = 0; i < sels.length; i++) 113 | sels[i] = left + sels[i] + right; 114 | cm.replaceSelections(sels, "around"); 115 | } else if (type == "both") { 116 | cm.replaceSelection(left + right, null); 117 | cm.execCommand("goCharLeft"); 118 | } else if (type == "addFour") { 119 | cm.replaceSelection(left + left + left + left, "before"); 120 | cm.execCommand("goCharRight"); 121 | } 122 | }); 123 | }; 124 | if (left != right) map["'" + right + "'"] = function(cm) { 125 | var ranges = cm.listSelections(); 126 | for (var i = 0; i < ranges.length; i++) { 127 | var range = ranges[i]; 128 | if (!range.empty() || 129 | cm.getRange(range.head, Pos(range.head.line, range.head.ch + 1)) != right) 130 | return CodeMirror.Pass; 131 | } 132 | cm.execCommand("goCharRight"); 133 | }; 134 | })(pairs.charAt(i), pairs.charAt(i + 1)); 135 | return map; 136 | } 137 | 138 | function buildExplodeHandler(pairs) { 139 | return function(cm) { 140 | if (cm.getOption("disableInput")) return CodeMirror.Pass; 141 | var ranges = cm.listSelections(); 142 | for (var i = 0; i < ranges.length; i++) { 143 | if (!ranges[i].empty()) return CodeMirror.Pass; 144 | var around = charsAround(cm, ranges[i].head); 145 | if (!around || pairs.indexOf(around) % 2 != 0) return CodeMirror.Pass; 146 | } 147 | cm.operation(function() { 148 | cm.replaceSelection("\n\n", null); 149 | cm.execCommand("goCharLeft"); 150 | ranges = cm.listSelections(); 151 | for (var i = 0; i < ranges.length; i++) { 152 | var line = ranges[i].head.line; 153 | cm.indentLine(line, null, true); 154 | cm.indentLine(line + 1, null, true); 155 | } 156 | }); 157 | }; 158 | } 159 | }); 160 | -------------------------------------------------------------------------------- /resources/codemirror.css: -------------------------------------------------------------------------------- 1 | /* BASICS */ 2 | 3 | .CodeMirror { 4 | /* Set height, width, borders, and global font properties here */ 5 | font-family: monospace; 6 | height: 300px; 7 | } 8 | 9 | /* PADDING */ 10 | 11 | .CodeMirror-lines { 12 | padding: 4px 0; /* Vertical padding around content */ 13 | } 14 | .CodeMirror pre { 15 | padding: 0 4px; /* Horizontal padding of content */ 16 | } 17 | 18 | .CodeMirror-scrollbar-filler, .CodeMirror-gutter-filler { 19 | background-color: white; /* The little square between H and V scrollbars */ 20 | } 21 | 22 | /* GUTTER */ 23 | 24 | .CodeMirror-gutters { 25 | border-right: 1px solid #ddd; 26 | background-color: #f7f7f7; 27 | white-space: nowrap; 28 | } 29 | .CodeMirror-linenumbers {} 30 | .CodeMirror-linenumber { 31 | padding: 0 3px 0 5px; 32 | min-width: 20px; 33 | text-align: right; 34 | color: #999; 35 | -moz-box-sizing: content-box; 36 | box-sizing: content-box; 37 | } 38 | 39 | .CodeMirror-guttermarker { color: black; } 40 | .CodeMirror-guttermarker-subtle { color: #999; } 41 | 42 | /* CURSOR */ 43 | 44 | .CodeMirror div.CodeMirror-cursor { 45 | border-left: 1px solid black; 46 | } 47 | /* Shown when moving in bi-directional text */ 48 | .CodeMirror div.CodeMirror-secondarycursor { 49 | border-left: 1px solid silver; 50 | } 51 | .CodeMirror.cm-fat-cursor div.CodeMirror-cursor { 52 | width: auto; 53 | border: 0; 54 | background: #7e7; 55 | } 56 | .CodeMirror.cm-fat-cursor div.CodeMirror-cursors { 57 | z-index: 1; 58 | } 59 | 60 | .cm-animate-fat-cursor { 61 | width: auto; 62 | border: 0; 63 | -webkit-animation: blink 1.06s steps(1) infinite; 64 | -moz-animation: blink 1.06s steps(1) infinite; 65 | animation: blink 1.06s steps(1) infinite; 66 | } 67 | @-moz-keyframes blink { 68 | 0% { background: #7e7; } 69 | 50% { background: none; } 70 | 100% { background: #7e7; } 71 | } 72 | @-webkit-keyframes blink { 73 | 0% { background: #7e7; } 74 | 50% { background: none; } 75 | 100% { background: #7e7; } 76 | } 77 | @keyframes blink { 78 | 0% { background: #7e7; } 79 | 50% { background: none; } 80 | 100% { background: #7e7; } 81 | } 82 | 83 | /* Can style cursor different in overwrite (non-insert) mode */ 84 | div.CodeMirror-overwrite div.CodeMirror-cursor {} 85 | 86 | .cm-tab { display: inline-block; text-decoration: inherit; } 87 | 88 | .CodeMirror-ruler { 89 | border-left: 1px solid #ccc; 90 | position: absolute; 91 | } 92 | 93 | /* DEFAULT THEME */ 94 | 95 | .cm-s-default .cm-keyword {color: #708;} 96 | .cm-s-default .cm-atom {color: #219;} 97 | .cm-s-default .cm-number {color: #164;} 98 | .cm-s-default .cm-def {color: #00f;} 99 | .cm-s-default .cm-variable, 100 | .cm-s-default .cm-punctuation, 101 | .cm-s-default .cm-property, 102 | .cm-s-default .cm-operator {} 103 | .cm-s-default .cm-variable-2 {color: #05a;} 104 | .cm-s-default .cm-variable-3 {color: #085;} 105 | .cm-s-default .cm-comment {color: #a50;} 106 | .cm-s-default .cm-string {color: #a11;} 107 | .cm-s-default .cm-string-2 {color: #f50;} 108 | .cm-s-default .cm-meta {color: #555;} 109 | .cm-s-default .cm-qualifier {color: #555;} 110 | .cm-s-default .cm-builtin {color: #30a;} 111 | .cm-s-default .cm-bracket {color: #997;} 112 | .cm-s-default .cm-tag {color: #170;} 113 | .cm-s-default .cm-attribute {color: #00c;} 114 | .cm-s-default .cm-header {color: blue;} 115 | .cm-s-default .cm-quote {color: #090;} 116 | .cm-s-default .cm-hr {color: #999;} 117 | .cm-s-default .cm-link {color: #00c;} 118 | 119 | .cm-negative {color: #d44;} 120 | .cm-positive {color: #292;} 121 | .cm-header, .cm-strong {font-weight: bold;} 122 | .cm-em {font-style: italic;} 123 | .cm-link {text-decoration: underline;} 124 | .cm-strikethrough {text-decoration: line-through;} 125 | 126 | .cm-s-default .cm-error {color: #f00;} 127 | .cm-invalidchar {color: #f00;} 128 | 129 | /* Default styles for common addons */ 130 | 131 | div.CodeMirror span.CodeMirror-matchingbracket {color: #0f0;} 132 | div.CodeMirror span.CodeMirror-nonmatchingbracket {color: #f22;} 133 | .CodeMirror-matchingtag { background: rgba(255, 150, 0, .3); } 134 | .CodeMirror-activeline-background {background: #e8f2ff;} 135 | 136 | /* STOP */ 137 | 138 | /* The rest of this file contains styles related to the mechanics of 139 | the editor. You probably shouldn't touch them. */ 140 | 141 | .CodeMirror { 142 | line-height: 1; 143 | position: relative; 144 | overflow: hidden; 145 | background: white; 146 | color: black; 147 | } 148 | 149 | .CodeMirror-scroll { 150 | overflow: scroll !important; /* Things will break if this is overridden */ 151 | /* 30px is the magic margin used to hide the element's real scrollbars */ 152 | /* See overflow: hidden in .CodeMirror */ 153 | margin-bottom: -30px; margin-right: -30px; 154 | padding-bottom: 30px; 155 | height: 100%; 156 | outline: none; /* Prevent dragging from highlighting the element */ 157 | position: relative; 158 | -moz-box-sizing: content-box; 159 | box-sizing: content-box; 160 | } 161 | .CodeMirror-sizer { 162 | position: relative; 163 | border-right: 30px solid transparent; 164 | -moz-box-sizing: content-box; 165 | box-sizing: content-box; 166 | } 167 | 168 | /* The fake, visible scrollbars. Used to force redraw during scrolling 169 | before actuall scrolling happens, thus preventing shaking and 170 | flickering artifacts. */ 171 | .CodeMirror-vscrollbar, .CodeMirror-hscrollbar, .CodeMirror-scrollbar-filler, .CodeMirror-gutter-filler { 172 | position: absolute; 173 | z-index: 6; 174 | display: none; 175 | } 176 | .CodeMirror-vscrollbar { 177 | right: 0; top: 0; 178 | overflow-x: hidden; 179 | overflow-y: scroll; 180 | } 181 | .CodeMirror-hscrollbar { 182 | bottom: 0; left: 0; 183 | overflow-y: hidden; 184 | overflow-x: scroll; 185 | } 186 | .CodeMirror-scrollbar-filler { 187 | right: 0; bottom: 0; 188 | } 189 | .CodeMirror-gutter-filler { 190 | left: 0; bottom: 0; 191 | } 192 | 193 | .CodeMirror-gutters { 194 | position: absolute; left: 0; top: 0; 195 | z-index: 3; 196 | } 197 | .CodeMirror-gutter { 198 | white-space: normal; 199 | height: 100%; 200 | -moz-box-sizing: content-box; 201 | box-sizing: content-box; 202 | display: inline-block; 203 | margin-bottom: -30px; 204 | /* Hack to make IE7 behave */ 205 | *zoom:1; 206 | *display:inline; 207 | } 208 | .CodeMirror-gutter-wrapper { 209 | position: absolute; 210 | z-index: 4; 211 | height: 100%; 212 | } 213 | .CodeMirror-gutter-elt { 214 | position: absolute; 215 | cursor: default; 216 | z-index: 4; 217 | } 218 | 219 | .CodeMirror-lines { 220 | cursor: text; 221 | min-height: 1px; /* prevents collapsing before first draw */ 222 | } 223 | .CodeMirror pre { 224 | /* Reset some styles that the rest of the page might have set */ 225 | -moz-border-radius: 0; -webkit-border-radius: 0; border-radius: 0; 226 | border-width: 0; 227 | background: transparent; 228 | font-family: inherit; 229 | font-size: inherit; 230 | margin: 0; 231 | white-space: pre; 232 | word-wrap: normal; 233 | line-height: inherit; 234 | color: inherit; 235 | z-index: 2; 236 | position: relative; 237 | overflow: visible; 238 | } 239 | .CodeMirror-wrap pre { 240 | word-wrap: break-word; 241 | white-space: pre-wrap; 242 | word-break: normal; 243 | } 244 | 245 | .CodeMirror-linebackground { 246 | position: absolute; 247 | left: 0; right: 0; top: 0; bottom: 0; 248 | z-index: 0; 249 | } 250 | 251 | .CodeMirror-linewidget { 252 | position: relative; 253 | z-index: 2; 254 | overflow: auto; 255 | } 256 | 257 | .CodeMirror-widget {} 258 | 259 | .CodeMirror-measure { 260 | position: absolute; 261 | width: 100%; 262 | height: 0; 263 | overflow: hidden; 264 | visibility: hidden; 265 | } 266 | .CodeMirror-measure pre { position: static; } 267 | 268 | .CodeMirror div.CodeMirror-cursor { 269 | position: absolute; 270 | border-right: none; 271 | width: 0; 272 | } 273 | 274 | div.CodeMirror-cursors { 275 | visibility: hidden; 276 | position: relative; 277 | z-index: 3; 278 | } 279 | .CodeMirror-focused div.CodeMirror-cursors { 280 | visibility: visible; 281 | } 282 | 283 | .CodeMirror-selected { background: #d9d9d9; } 284 | .CodeMirror-focused .CodeMirror-selected { background: #d7d4f0; } 285 | .CodeMirror-crosshair { cursor: crosshair; } 286 | 287 | .cm-searching { 288 | background: #ffa; 289 | background: rgba(255, 255, 0, .4); 290 | } 291 | 292 | /* IE7 hack to prevent it from returning funny offsetTops on the spans */ 293 | .CodeMirror span { *vertical-align: text-bottom; } 294 | 295 | /* Used to force a border model for a node */ 296 | .cm-force-border { padding-right: .1px; } 297 | 298 | @media print { 299 | /* Hide the cursor when printing */ 300 | .CodeMirror div.CodeMirror-cursors { 301 | visibility: hidden; 302 | } 303 | } 304 | 305 | /* See issue #2901 */ 306 | .cm-tab-wrap-hack:after { content: ''; } 307 | 308 | /* Help users use markselection to safely style text background */ 309 | span.CodeMirror-selectedtext { background: none; } 310 | -------------------------------------------------------------------------------- /resources/matchbrackets.js: -------------------------------------------------------------------------------- 1 | // CodeMirror, copyright (c) by Marijn Haverbeke and others 2 | // Distributed under an MIT license: http://codemirror.net/LICENSE 3 | 4 | (function(mod) { 5 | if (typeof exports == "object" && typeof module == "object") // CommonJS 6 | mod(require("../../lib/codemirror")); 7 | else if (typeof define == "function" && define.amd) // AMD 8 | define(["../../lib/codemirror"], mod); 9 | else // Plain browser env 10 | mod(CodeMirror); 11 | })(function(CodeMirror) { 12 | var ie_lt8 = /MSIE \d/.test(navigator.userAgent) && 13 | (document.documentMode == null || document.documentMode < 8); 14 | 15 | var Pos = CodeMirror.Pos; 16 | 17 | var matching = {"(": ")>", ")": "(<", "[": "]>", "]": "[<", "{": "}>", "}": "{<"}; 18 | 19 | function findMatchingBracket(cm, where, strict, config) { 20 | var line = cm.getLineHandle(where.line), pos = where.ch - 1; 21 | var match = (pos >= 0 && matching[line.text.charAt(pos)]) || matching[line.text.charAt(++pos)]; 22 | if (!match) return null; 23 | var dir = match.charAt(1) == ">" ? 1 : -1; 24 | if (strict && (dir > 0) != (pos == where.ch)) return null; 25 | var style = cm.getTokenTypeAt(Pos(where.line, pos + 1)); 26 | 27 | var found = scanForBracket(cm, Pos(where.line, pos + (dir > 0 ? 1 : 0)), dir, style || null, config); 28 | if (found == null) return null; 29 | return {from: Pos(where.line, pos), to: found && found.pos, 30 | match: found && found.ch == match.charAt(0), forward: dir > 0}; 31 | } 32 | 33 | // bracketRegex is used to specify which type of bracket to scan 34 | // should be a regexp, e.g. /[[\]]/ 35 | // 36 | // Note: If "where" is on an open bracket, then this bracket is ignored. 37 | // 38 | // Returns false when no bracket was found, null when it reached 39 | // maxScanLines and gave up 40 | function scanForBracket(cm, where, dir, style, config) { 41 | var maxScanLen = (config && config.maxScanLineLength) || 10000; 42 | var maxScanLines = (config && config.maxScanLines) || 1000; 43 | 44 | var stack = []; 45 | var re = config && config.bracketRegex ? config.bracketRegex : /[(){}[\]]/; 46 | var lineEnd = dir > 0 ? Math.min(where.line + maxScanLines, cm.lastLine() + 1) 47 | : Math.max(cm.firstLine() - 1, where.line - maxScanLines); 48 | for (var lineNo = where.line; lineNo != lineEnd; lineNo += dir) { 49 | var line = cm.getLine(lineNo); 50 | if (!line) continue; 51 | var pos = dir > 0 ? 0 : line.length - 1, end = dir > 0 ? line.length : -1; 52 | if (line.length > maxScanLen) continue; 53 | if (lineNo == where.line) pos = where.ch - (dir < 0 ? 1 : 0); 54 | for (; pos != end; pos += dir) { 55 | var ch = line.charAt(pos); 56 | if (re.test(ch) && (style === undefined || cm.getTokenTypeAt(Pos(lineNo, pos + 1)) == style)) { 57 | var match = matching[ch]; 58 | if ((match.charAt(1) == ">") == (dir > 0)) stack.push(ch); 59 | else if (!stack.length) return {pos: Pos(lineNo, pos), ch: ch}; 60 | else stack.pop(); 61 | } 62 | } 63 | } 64 | return lineNo - dir == (dir > 0 ? cm.lastLine() : cm.firstLine()) ? false : null; 65 | } 66 | 67 | function matchBrackets(cm, autoclear, config) { 68 | // Disable brace matching in long lines, since it'll cause hugely slow updates 69 | var maxHighlightLen = cm.state.matchBrackets.maxHighlightLineLength || 1000; 70 | var marks = [], ranges = cm.listSelections(); 71 | for (var i = 0; i < ranges.length; i++) { 72 | var match = ranges[i].empty() && findMatchingBracket(cm, ranges[i].head, false, config); 73 | if (match && cm.getLine(match.from.line).length <= maxHighlightLen) { 74 | var style = match.match ? "CodeMirror-matchingbracket" : "CodeMirror-nonmatchingbracket"; 75 | marks.push(cm.markText(match.from, Pos(match.from.line, match.from.ch + 1), {className: style})); 76 | if (match.to && cm.getLine(match.to.line).length <= maxHighlightLen) 77 | marks.push(cm.markText(match.to, Pos(match.to.line, match.to.ch + 1), {className: style})); 78 | } 79 | } 80 | 81 | if (marks.length) { 82 | // Kludge to work around the IE bug from issue #1193, where text 83 | // input stops going to the textare whever this fires. 84 | if (ie_lt8 && cm.state.focused) cm.display.input.focus(); 85 | 86 | var clear = function() { 87 | cm.operation(function() { 88 | for (var i = 0; i < marks.length; i++) marks[i].clear(); 89 | }); 90 | }; 91 | if (autoclear) setTimeout(clear, 800); 92 | else return clear; 93 | } 94 | } 95 | 96 | var currentlyHighlighted = null; 97 | function doMatchBrackets(cm) { 98 | cm.operation(function() { 99 | if (currentlyHighlighted) {currentlyHighlighted(); currentlyHighlighted = null;} 100 | currentlyHighlighted = matchBrackets(cm, false, cm.state.matchBrackets); 101 | }); 102 | } 103 | 104 | CodeMirror.defineOption("matchBrackets", false, function(cm, val, old) { 105 | if (old && old != CodeMirror.Init) 106 | cm.off("cursorActivity", doMatchBrackets); 107 | if (val) { 108 | cm.state.matchBrackets = typeof val == "object" ? val : {}; 109 | cm.on("cursorActivity", doMatchBrackets); 110 | } 111 | }); 112 | 113 | CodeMirror.defineExtension("matchBrackets", function() {matchBrackets(this, true);}); 114 | CodeMirror.defineExtension("findMatchingBracket", function(pos, strict, config){ 115 | return findMatchingBracket(this, pos, strict, config); 116 | }); 117 | CodeMirror.defineExtension("scanForBracket", function(pos, dir, style, config){ 118 | return scanForBracket(this, pos, dir, style, config); 119 | }); 120 | }); 121 | -------------------------------------------------------------------------------- /resources/scheme.js: -------------------------------------------------------------------------------- 1 | // CodeMirror, copyright (c) by Marijn Haverbeke and others 2 | // Distributed under an MIT license: http://codemirror.net/LICENSE 3 | 4 | /** 5 | * Author: Koh Zi Han, based on implementation by Koh Zi Chun 6 | */ 7 | 8 | (function(mod) { 9 | if (typeof exports == "object" && typeof module == "object") // CommonJS 10 | mod(require("../../lib/codemirror")); 11 | else if (typeof define == "function" && define.amd) // AMD 12 | define(["../../lib/codemirror"], mod); 13 | else // Plain browser env 14 | mod(CodeMirror); 15 | })(function(CodeMirror) { 16 | "use strict"; 17 | 18 | CodeMirror.defineMode("scheme", function () { 19 | var BUILTIN = "builtin", COMMENT = "comment", STRING = "string", 20 | ATOM = "atom", NUMBER = "number", BRACKET = "bracket"; 21 | var INDENT_WORD_SKIP = 2; 22 | 23 | function makeKeywords(str) { 24 | var obj = {}, words = str.split(" "); 25 | for (var i = 0; i < words.length; ++i) obj[words[i]] = true; 26 | return obj; 27 | } 28 | 29 | var keywords = makeKeywords("λ case-lambda call/cc class define-class exit-handler field import inherit init-field interface let*-values let-values let/ec mixin opt-lambda override protect provide public rename require require-for-syntax syntax syntax-case syntax-error unit/sig unless when with-syntax and begin call-with-current-continuation call-with-input-file call-with-output-file case cond define define-syntax delay do dynamic-wind else for-each if lambda let let* let-syntax letrec letrec-syntax map or syntax-rules abs acos angle append apply asin assoc assq assv atan boolean? caar cadr call-with-input-file call-with-output-file call-with-values car cdddar cddddr cdr ceiling char->integer char-alphabetic? char-ci<=? char-ci=? char-ci>? char-downcase char-lower-case? char-numeric? char-ready? char-upcase char-upper-case? char-whitespace? char<=? char=? char>? char? close-input-port close-output-port complex? cons cos current-input-port current-output-port denominator display eof-object? eq? equal? eqv? eval even? exact->inexact exact? exp expt #f floor force gcd imag-part inexact->exact inexact? input-port? integer->char integer? interaction-environment lcm length list list->string list->vector list-ref list-tail list? load log magnitude make-polar make-rectangular make-string make-vector max member memq memv min modulo negative? newline not null-environment null? number->string number? numerator odd? open-input-file open-output-file output-port? pair? peek-char port? positive? procedure? quasiquote quote quotient rational? rationalize read read-char real-part real? remainder reverse round scheme-report-environment set! set-car! set-cdr! sin sqrt string string->list string->number string->symbol string-append string-ci<=? string-ci=? string-ci>? string-copy string-fill! string-length string-ref string-set! string<=? string=? string>? string? substring symbol->string symbol? #t tan transcript-off transcript-on truncate values vector vector->list vector-fill! vector-length vector-ref vector-set! with-input-from-file with-output-to-file write write-char zero?"); 30 | var indentKeys = makeKeywords("define let letrec let* lambda fresh"); 31 | 32 | function stateStack(indent, type, prev) { // represents a state stack object 33 | this.indent = indent; 34 | this.type = type; 35 | this.prev = prev; 36 | } 37 | 38 | function pushStack(state, indent, type) { 39 | state.indentStack = new stateStack(indent, type, state.indentStack); 40 | } 41 | 42 | function popStack(state) { 43 | state.indentStack = state.indentStack.prev; 44 | } 45 | 46 | var binaryMatcher = new RegExp(/^(?:[-+]i|[-+][01]+#*(?:\/[01]+#*)?i|[-+]?[01]+#*(?:\/[01]+#*)?@[-+]?[01]+#*(?:\/[01]+#*)?|[-+]?[01]+#*(?:\/[01]+#*)?[-+](?:[01]+#*(?:\/[01]+#*)?)?i|[-+]?[01]+#*(?:\/[01]+#*)?)(?=[()\s;"]|$)/i); 47 | var octalMatcher = new RegExp(/^(?:[-+]i|[-+][0-7]+#*(?:\/[0-7]+#*)?i|[-+]?[0-7]+#*(?:\/[0-7]+#*)?@[-+]?[0-7]+#*(?:\/[0-7]+#*)?|[-+]?[0-7]+#*(?:\/[0-7]+#*)?[-+](?:[0-7]+#*(?:\/[0-7]+#*)?)?i|[-+]?[0-7]+#*(?:\/[0-7]+#*)?)(?=[()\s;"]|$)/i); 48 | var hexMatcher = new RegExp(/^(?:[-+]i|[-+][\da-f]+#*(?:\/[\da-f]+#*)?i|[-+]?[\da-f]+#*(?:\/[\da-f]+#*)?@[-+]?[\da-f]+#*(?:\/[\da-f]+#*)?|[-+]?[\da-f]+#*(?:\/[\da-f]+#*)?[-+](?:[\da-f]+#*(?:\/[\da-f]+#*)?)?i|[-+]?[\da-f]+#*(?:\/[\da-f]+#*)?)(?=[()\s;"]|$)/i); 49 | var decimalMatcher = new RegExp(/^(?:[-+]i|[-+](?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*)i|[-+]?(?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*)@[-+]?(?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*)|[-+]?(?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*)[-+](?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*)?i|(?:(?:(?:\d+#+\.?#*|\d+\.\d*#*|\.\d+#*|\d+)(?:[esfdl][-+]?\d+)?)|\d+#*\/\d+#*))(?=[()\s;"]|$)/i); 50 | 51 | function isBinaryNumber (stream) { 52 | return stream.match(binaryMatcher); 53 | } 54 | 55 | function isOctalNumber (stream) { 56 | return stream.match(octalMatcher); 57 | } 58 | 59 | function isDecimalNumber (stream, backup) { 60 | if (backup === true) { 61 | stream.backUp(1); 62 | } 63 | return stream.match(decimalMatcher); 64 | } 65 | 66 | function isHexNumber (stream) { 67 | return stream.match(hexMatcher); 68 | } 69 | 70 | return { 71 | startState: function () { 72 | return { 73 | indentStack: null, 74 | indentation: 0, 75 | mode: false, 76 | sExprComment: false 77 | }; 78 | }, 79 | 80 | token: function (stream, state) { 81 | if (state.indentStack == null && stream.sol()) { 82 | // update indentation, but only if indentStack is empty 83 | state.indentation = stream.indentation(); 84 | } 85 | 86 | // skip spaces 87 | if (stream.eatSpace()) { 88 | return null; 89 | } 90 | var returnType = null; 91 | 92 | switch(state.mode){ 93 | case "string": // multi-line string parsing mode 94 | var next, escaped = false; 95 | while ((next = stream.next()) != null) { 96 | if (next == "\"" && !escaped) { 97 | 98 | state.mode = false; 99 | break; 100 | } 101 | escaped = !escaped && next == "\\"; 102 | } 103 | returnType = STRING; // continue on in scheme-string mode 104 | break; 105 | case "comment": // comment parsing mode 106 | var next, maybeEnd = false; 107 | while ((next = stream.next()) != null) { 108 | if (next == "#" && maybeEnd) { 109 | 110 | state.mode = false; 111 | break; 112 | } 113 | maybeEnd = (next == "|"); 114 | } 115 | returnType = COMMENT; 116 | break; 117 | case "s-expr-comment": // s-expr commenting mode 118 | state.mode = false; 119 | if(stream.peek() == "(" || stream.peek() == "["){ 120 | // actually start scheme s-expr commenting mode 121 | state.sExprComment = 0; 122 | }else{ 123 | // if not we just comment the entire of the next token 124 | stream.eatWhile(/[^/s]/); // eat non spaces 125 | returnType = COMMENT; 126 | break; 127 | } 128 | default: // default parsing mode 129 | var ch = stream.next(); 130 | 131 | if (ch == "\"") { 132 | state.mode = "string"; 133 | returnType = STRING; 134 | 135 | } else if (ch == "'") { 136 | returnType = ATOM; 137 | } else if (ch == '#') { 138 | if (stream.eat("|")) { // Multi-line comment 139 | state.mode = "comment"; // toggle to comment mode 140 | returnType = COMMENT; 141 | } else if (stream.eat(/[tf]/i)) { // #t/#f (atom) 142 | returnType = ATOM; 143 | } else if (stream.eat(';')) { // S-Expr comment 144 | state.mode = "s-expr-comment"; 145 | returnType = COMMENT; 146 | } else { 147 | var numTest = null, hasExactness = false, hasRadix = true; 148 | if (stream.eat(/[ei]/i)) { 149 | hasExactness = true; 150 | } else { 151 | stream.backUp(1); // must be radix specifier 152 | } 153 | if (stream.match(/^#b/i)) { 154 | numTest = isBinaryNumber; 155 | } else if (stream.match(/^#o/i)) { 156 | numTest = isOctalNumber; 157 | } else if (stream.match(/^#x/i)) { 158 | numTest = isHexNumber; 159 | } else if (stream.match(/^#d/i)) { 160 | numTest = isDecimalNumber; 161 | } else if (stream.match(/^[-+0-9.]/, false)) { 162 | hasRadix = false; 163 | numTest = isDecimalNumber; 164 | // re-consume the intial # if all matches failed 165 | } else if (!hasExactness) { 166 | stream.eat('#'); 167 | } 168 | if (numTest != null) { 169 | if (hasRadix && !hasExactness) { 170 | // consume optional exactness after radix 171 | stream.match(/^#[ei]/i); 172 | } 173 | if (numTest(stream)) 174 | returnType = NUMBER; 175 | } 176 | } 177 | } else if (/^[-+0-9.]/.test(ch) && isDecimalNumber(stream, true)) { // match non-prefixed number, must be decimal 178 | returnType = NUMBER; 179 | } else if (ch == ";") { // comment 180 | stream.skipToEnd(); // rest of the line is a comment 181 | returnType = COMMENT; 182 | } else if (ch == "(" || ch == "[") { 183 | var keyWord = ''; var indentTemp = stream.column(), letter; 184 | /** 185 | Either 186 | (indent-word .. 187 | (non-indent-word .. 188 | (;something else, bracket, etc. 189 | */ 190 | 191 | while ((letter = stream.eat(/[^\s\(\[\;\)\]]/)) != null) { 192 | keyWord += letter; 193 | } 194 | 195 | if (keyWord.length > 0 && indentKeys.propertyIsEnumerable(keyWord)) { // indent-word 196 | 197 | pushStack(state, indentTemp + INDENT_WORD_SKIP, ch); 198 | } else { // non-indent word 199 | // we continue eating the spaces 200 | stream.eatSpace(); 201 | if (stream.eol() || stream.peek() == ";") { 202 | // nothing significant after 203 | // we restart indentation 1 space after 204 | pushStack(state, indentTemp + 1, ch); 205 | } else { 206 | pushStack(state, indentTemp + stream.current().length, ch); // else we match 207 | } 208 | } 209 | stream.backUp(stream.current().length - 1); // undo all the eating 210 | 211 | if(typeof state.sExprComment == "number") state.sExprComment++; 212 | 213 | returnType = BRACKET; 214 | } else if (ch == ")" || ch == "]") { 215 | returnType = BRACKET; 216 | if (state.indentStack != null && state.indentStack.type == (ch == ")" ? "(" : "[")) { 217 | popStack(state); 218 | 219 | if(typeof state.sExprComment == "number"){ 220 | if(--state.sExprComment == 0){ 221 | returnType = COMMENT; // final closing bracket 222 | state.sExprComment = false; // turn off s-expr commenting mode 223 | } 224 | } 225 | } 226 | } else { 227 | stream.eatWhile(/[\w\$_\-!$%&*+\.\/:<=>?@\^~]/); 228 | 229 | if (keywords && keywords.propertyIsEnumerable(stream.current())) { 230 | returnType = BUILTIN; 231 | } else returnType = "variable"; 232 | } 233 | } 234 | return (typeof state.sExprComment == "number") ? COMMENT : returnType; 235 | }, 236 | 237 | indent: function (state) { 238 | if (state.indentStack == null) return state.indentation; 239 | return state.indentStack.indent; 240 | }, 241 | 242 | lineComment: ";;" 243 | }; 244 | }); 245 | 246 | CodeMirror.defineMIME("text/x-scheme", "scheme"); 247 | 248 | }); 249 | -------------------------------------------------------------------------------- /veneer.css: -------------------------------------------------------------------------------- 1 | .CodeMirror-matchingbracket { border:1px solid #ccc; margin:-1px; } 2 | .CodeMirror-nonmatchingbracket { border:1px solid #f00; margin:-1px; } 3 | 4 | .CodeMirror { float:left; height:100%; width:50%; } 5 | 6 | .clear { clear:both; margin:0; padding:0; border-width:0; visibility:hidden; } 7 | 8 | .repl { 9 | font-family: monospace, courier; 10 | font-size:13px; 11 | float:left; 12 | height:100%; 13 | width:50%; 14 | overflow-y:scroll; 15 | margin:0; 16 | padding: 0; 17 | background-color:#f9f9f9; 18 | border-width:0; 19 | } 20 | 21 | .errors { 22 | position:absolute; 23 | top:.5em; left:.5em; 24 | z-index:100; 25 | width:46%; 26 | margin:0; 27 | padding: 0 2%; 28 | background-color:transparent; 29 | border-width:0; 30 | } 31 | 32 | .error { background-color:#f00; color:#000; } 33 | 34 | .inputbox:last-child { margin-bottom:1em; } 35 | 36 | .inputbox { 37 | background-color:#fff; 38 | white-space: pre-wrap; 39 | display:block; 40 | border:1px solid #ccc; 41 | min-height:1em; 42 | padding:2px 8px; 43 | } 44 | 45 | .result { 46 | white-space: pre-wrap; 47 | background-color: #eee; 48 | margin:0; 49 | padding: 2px 8px; 50 | } 51 | 52 | .dump_button { 53 | z-index:99; 54 | font-size:36pt; 55 | position:absolute; right:50%; 56 | margin-right:16px; 57 | background-color: #eee; 58 | border: 3px solid #ccc; 59 | padding:0 .4em; 60 | cursor:pointer; 61 | } 62 | 63 | .dump_button .helptext { font-size:10pt; } 64 | 65 | .dump_button:hover { color:#fff; } 66 | -------------------------------------------------------------------------------- /veneer.js: -------------------------------------------------------------------------------- 1 | function Veneer_v1() { 2 | this.new_editor = function(container, vm, seed) { 3 | var errors = document.createElement("div"); 4 | errors.className += "errors"; 5 | container.appendChild(errors); 6 | 7 | var options = { mode: "scheme", 8 | matchBrackets: true, 9 | autoCloseBrackets: true, 10 | lineNumbers: true}; 11 | var editor = new CodeMirror(function(ed) { 12 | container.appendChild(ed); 13 | return ed; 14 | }, options); 15 | 16 | var repl = document.createElement("div"); 17 | repl.className += "repl"; 18 | container.appendChild(repl); 19 | 20 | var dump_button = document.createElement("div"); 21 | dump_button.className += "dump_button"; 22 | dump_button.appendChild(document.createTextNode("run!")); 23 | container.appendChild(dump_button); 24 | 25 | var clear = document.createElement("hr"); 26 | clear.className += "clear"; 27 | container.appendChild(clear); 28 | 29 | var current_input = repl_getline(false); 30 | function display_error(e) { 31 | var error = document.createElement("div"); 32 | var error_txt = document.createTextNode(e); 33 | error.className += "error"; 34 | error.appendChild(error_txt); 35 | errors.appendChild(error); 36 | setTimeout(function() { errors.removeChild(error); }, 3000); 37 | } 38 | 39 | var getTime; 40 | if (window.performance && window.performance.now) { 41 | getTime = function() { return window.performance.now(); }; 42 | } else { 43 | getTime = function() { return (new Date()).getTime(); }; 44 | } 45 | 46 | function load_input(inputbox) { 47 | inputbox = current_input; 48 | /* var textcontent = inputbox.innerHTML.replace(//ig, '\n') 49 | .replace(/<[p|div]\s/ig, '\n$0') 50 | .replace(/(<([^>]+)>)/ig, ""); */ 51 | 52 | var pass = false; 53 | var textcontent = inputbox.textContent; 54 | try { 55 | var result = vm.read_eval(textcontent); 56 | var result_elt = document.createElement("div"); 57 | result_elt.className += "result"; 58 | 59 | function display_query(generator, parent) { 60 | var answer_text = document.createElement("div"); 61 | var button = document.createElement("button"); 62 | var append_answer = function(focus) { 63 | var answered; 64 | var time_before = getTime(); 65 | var answer_val = generator(); 66 | var run_time = (getTime() - time_before).toFixed(2); 67 | if (answer_val === null) { 68 | answered = "No."; 69 | answer_val = ""; 70 | parent.removeChild(button); 71 | } else { 72 | answered = "Yes."; 73 | } 74 | var answer_val_pp = answered + " (" + run_time + "ms)\n" + answer_val; 75 | var answer_node = document.createTextNode("=> " + answer_val_pp + "\n"); 76 | answer_text.appendChild(answer_node); 77 | 78 | if(focus) { 79 | button.focus(); 80 | current_input.scrollIntoView(false); 81 | } 82 | 83 | return false; 84 | } 85 | button.onclick = function() { return append_answer(true); }; 86 | button.appendChild(document.createTextNode("More answers!")); 87 | parent.appendChild(answer_text); 88 | parent.appendChild(button); 89 | append_answer(false); 90 | return answer_text; 91 | } 92 | 93 | function display_val(val, parent) { 94 | parent.appendChild(document.createTextNode(pretty_print(val))); 95 | } 96 | 97 | procedurep(result) ? display_query(result, result_elt) 98 | : display_val(result, result_elt); 99 | 100 | repl.appendChild(result_elt); 101 | inputbox.contentEditable = false; 102 | repl_getline(true); 103 | return false; 104 | } catch (e) { 105 | if(e instanceof ReaderError) { 106 | inputbox.appendChild(document.createTextNode("\n")); 107 | display_error(e.msg); 108 | pass = true; 109 | throw e.msg; 110 | } else { 111 | display_error(e); 112 | pass = false; 113 | throw e; 114 | } 115 | } finally { return pass; } 116 | } 117 | 118 | function repl_getline(focus) { 119 | var inputbox = document.createElement("pre"); 120 | inputbox.className += "inputbox"; 121 | inputbox.contentEditable = true; 122 | inputbox.onkeypress = function(e) { 123 | if (e.keyCode == 13) { 124 | return load_input(inputbox); 125 | } 126 | }; 127 | 128 | repl.appendChild(inputbox); 129 | if (focus) { inputbox.focus(); } 130 | current_input = inputbox; 131 | return inputbox; 132 | } 133 | 134 | 135 | var dump_editor = function() { 136 | vm.reset(); 137 | if (!(typeof seed === 'undefined')) { vm.read_eval(seed); } 138 | current_input.textContent = editor.getValue(); 139 | load_input(current_input); 140 | return false; 141 | } 142 | dump_button.onclick = dump_editor; 143 | 144 | var shift = false; 145 | 146 | editor.setOption("extraKeys", { 147 | "Shift-Enter": function(cm) { dump_editor();} 148 | }); 149 | 150 | return { editor: editor, repl: repl, errors: errors, dump_button: dump_button }; 151 | }; 152 | 153 | 154 | this.new_vm = function() { return new VeneerVM(); }; 155 | } 156 | --------------------------------------------------------------------------------