function Word(vocab, name, source, func) { this.vocab = vocab; this.name = name; this.source = source; this.func = func; } Word.prototype.execute = function(next) { this.func(next); } Word.prototype.toString = function() { var html = []; html.push("") html.push(this.name); html.push(""); return html.join(""); } function Continuation() { this.data_stack = [ ]; this.retain_stack = [ ]; this.next = false; this.nesting = 0; } Continuation.prototype.clone = function() { var c = new Continuation(); c.data_stack = this.data_stack.slice(0); c.retain_stack = this.retain_stack.slice(0); c.nesting = this.nesting; c.next = this.next; return c; } function Factor() { this.vocabs = { scratchpad: { } }; this.in_vocab = "scratchpad"; this.using_vocabs = [ "scratchpad", "kernel","math","sequences","parser","alien","browser-dom", "words" ]; this.cont = new Continuation(); } var factor = new Factor(); Factor.prototype.call_next = function(next) { if(next) { if(this.cont.nesting++ > 150) { this.cont.nesting = 0; setTimeout(next, 0); } else { next(); } } } Factor.prototype.push_data = function(v, next) { factor.cont.data_stack.push(v); factor.call_next(next); } Factor.prototype.get_word = function(vocab,name) { return factor.vocabs[vocab][name]; } Factor.prototype.find_word = function(name) { for(var v = 0; v") for(var i = 0; i < this.cont.data_stack.length; ++i) { html.push("") html.push(this.cont.data_stack[i]) html.push("") } html.push("") document.getElementById('stack').innerHTML=html.join(""); } /* Kernel Vocabulary */ factor.add_word("kernel","dup", "primitive", function(next) { var stack = factor.cont.data_stack; stack[stack.length] = stack[stack.length-1]; factor.call_next(next); }); factor.add_word("kernel", "drop", "primitive", function(next) { factor.cont.data_stack.pop(); factor.call_next(next); }); factor.add_word("kernel", "nip", "primitive", function(next) { var stack = factor.cont.data_stack; stack[stack.length-2] = stack[stack.length-1]; stack.pop(); factor.call_next(next); }); factor.add_word("kernel", "over", "primitive", function(next) { var stack = factor.cont.data_stack; stack[stack.length] = stack[stack.length-2]; factor.call_next(next); }); factor.add_word("kernel", "swap", "primitive", function(next) { var stack = factor.cont.data_stack; var temp = stack[stack.length-2]; stack[stack.length-2] = stack[stack.length-1]; stack[stack.length-1] = temp; factor.call_next(next); }); factor.add_word("kernel", ">r", "primitive", function(next) { var data_stack = factor.cont.data_stack; var retain_stack = factor.cont.retain_stack; retain_stack.push(data_stack.pop()); factor.call_next(next); }); factor.add_word("kernel", "r>", "primitive", function(next) { var data_stack = factor.cont.data_stack; var retain_stack = factor.cont.retain_stack; data_stack.push(retain_stack.pop()); factor.call_next(next); }); factor.add_word("kernel", "call", "primitive", function(next) { var quot = factor.cont.data_stack.pop(); quot.execute(next); }); factor.add_word("kernel", "execute", "primitive", function(next) { var quot = factor.cont.data_stack.pop(); quot.execute(next); }); factor.add_word("kernel", "clear", "primitive", function(next) { factor.cont.data_stack = []; factor.cont.retain_stack = []; factor.call_next(next); }); factor.add_word("kernel", "if", "primitive", function(next) { var stack = factor.cont.data_stack; var else_quot = stack.pop(); var then_quot = stack.pop(); var condition = stack.pop(); if(condition) { then_quot.execute(next); } else { else_quot.execute(next); } }); factor.add_word("kernel", "f", "primitive", function(next) { factor.cont.data_stack.push(false); factor.call_next(next); }); factor.add_word("kernel", "t", "primitive", function(next) { factor.cont.data_stack.push(true); factor.call_next(next); }); factor.add_word("kernel", "=", "primitive", function(next) { var stack = factor.cont.data_stack; stack.push(stack.pop()==stack.pop()); factor.call_next(next); }); factor.add_word("kernel", "bootstrap", "primitive", function(next) { factor.cont.data_stack.push("/responder/fjsc-resources/bootstrap.factor"); factor.get_word("parser", "run-file").execute(next); }); factor.add_word("kernel", "callcc0", "primitive", function(next) { var data_stack = factor.cont.data_stack; var quot = data_stack.pop(); var new_cont = factor.cont.clone(); var old_next = factor.cont.next; factor.cont.next = function() { factor.cont.next = old_next; factor.call_next(next); } new_cont.data_stack.push(factor.cont); factor.cont = new_cont; quot.execute(next); }); factor.add_word("kernel", "callcc1", "primitive", function(next) { factor.get_word("kernel", "callcc0").execute(next); }); factor.add_word("kernel", "continue", "primitive", function(next) { var data_stack = factor.cont.data_stack; var cont = data_stack.pop(); factor.cont = cont.clone(); factor.call_next(cont.next); }); factor.add_word("kernel", "continue-with", "primitive", function(next) { var data_stack = factor.cont.data_stack; var cont = data_stack.pop(); var data = data_stack.pop(); factor.cont = cont.clone(); factor.cont.data_stack.push(data); factor.call_next(cont.next); }); factor.add_word("kernel", "in", "primitive", function(next) { var stack = factor.cont.data_stack; var vocab = stack.pop(); var v = factor.vocabs[vocab]; if(!v) { v = { }; factor.vocabs[vocab] = v; } factor.in_vocab = vocab; factor.call_next(next); }); factor.add_word("kernel", "current-vocab", "primitive", function(next) { var stack = factor.cont.data_stack; stack.push(factor.in_vocab); factor.call_next(next); }); factor.add_word("kernel", "use", "primitive", function(next) { var stack = factor.cont.data_stack; var vocab = stack.pop(); var v = factor.vocabs[vocab]; if(!v) { v = { }; factor.vocabs[vocab] = v; } factor.using_vocabs.push(vocab); factor.call_next(next); }); factor.add_word("kernel", "using", "primitive", function(next) { var stack = factor.cont.data_stack; var vocabs = stack.pop(); factor.using_vocabs = []; for(var i=0;ifunction", "primitive", function(next) { var stack = factor.cont.data_stack; var word = stack.pop(); stack.push(function() { word.func(function() { }) }); factor.call_next(next); }); factor.add_word("kernel", "curry", "primitive", function(next) { var stack = factor.cont.data_stack; var quot = stack.pop(); var value = stack.pop(); stack.push(factor.make_quotation("quotation", function(next) { factor.cont.data_stack.push(value); quot.execute(factor.cont.next); })); factor.call_next(next); }); /* Math vocabulary */ factor.add_word("math", "*", "primitive", function(next) { var stack = factor.cont.data_stack; stack.push(stack.pop() * stack.pop()); factor.call_next(next); }); factor.add_word("math", "+", "primitive", function(next) { var stack = factor.cont.data_stack; stack.push(stack.pop() + stack.pop()); factor.call_next(next); }); factor.add_word("math", "-", "primitive", function(next) { var stack = factor.cont.data_stack; var v1 = stack.pop(); var v2 = stack.pop(); stack.push(v2 - v1); factor.call_next(next); }); factor.add_word("math", "/", "primitive", function(next) { var stack = factor.cont.data_stack; var v1 = stack.pop(); var v2 = stack.pop(); stack.push(v2 / v1); factor.call_next(next); }); factor.add_word("prettyprint", ".", "primitive", function(next) { alert(factor.cont.data_stack.pop()); factor.call_next(next); }); factor.add_word("parser", "run-file", "primitive", function(next) { var stack = factor.cont.data_stack; var url = stack.pop(); $.get(url, function(result) { factor.server_eval(result, next); }); }); factor.add_word("alien", "alien-invoke", "primitive", function(next) { var stack = factor.cont.data_stack; var arg_types = stack.pop(); var method_name = stack.pop(); var library_name = stack.pop(); var return_values = stack.pop(); var obj = stack.pop(); var args = [ ]; for(var i = 0; i < arg_types.length; ++i) { args[i] = stack.pop(); } var v = obj[method_name].apply(obj, args.reverse()); if(return_values.length > 0) stack.push(v); factor.call_next(next); }); factor.add_word("alien", "alien-property", "primitive", function(next) { var stack = factor.cont.data_stack; var obj = stack.pop(); var property_name = stack.pop(); var v = obj[property_name]; stack.push(v); factor.call_next(next); }); factor.add_word("alien", "set-alien-property", "primitive", function(next) { var stack = factor.cont.data_stack; var obj = stack.pop(); var property_name = stack.pop(); var data = stack.pop(); obj[property_name] = v; factor.call_next(next); }); factor.add_word("words", "vocabs", "primitive", function(next) { var stack = factor.cont.data_stack; var result = []; for(v in factor.vocabs) { result.push(v); } stack.push(result); factor.call_next(next); }); factor.add_word("words", "words", "primitive", function(next) { var stack = factor.cont.data_stack; var vocab = factor.vocabs[stack.pop()]; var result = []; for(w in vocab) { result.push(vocab[w]); } stack.push(result); factor.call_next(next); }); factor.add_word("words", "all-words", "primitive", function(next) { var stack = factor.cont.data_stack; var result = []; for(v in factor.vocabs) { for(w in factor.vocabs[v]) { result.push(factor.vocabs[v][w]); } } stack.push(result); factor.call_next(next); }); /* Sequences vocabulary */ factor.add_word("sequences", "nth", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); var index = stack.pop(); stack.push(seq[index]); factor.call_next(next); }); factor.add_word("sequences", "first", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[0]); factor.call_next(next); }); factor.add_word("sequences", "second", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[1]); factor.call_next(next); }); factor.add_word("sequences", "third", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[2]); factor.call_next(next); }); factor.add_word("sequences", "fourth", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[0]); factor.call_next(next); }); factor.add_word("sequences", "first2", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[0],seq[1]); factor.call_next(next); }); factor.add_word("sequences", "first3", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[0],seq[1],seq[2]); factor.call_next(next); }); factor.add_word("sequences", "first4", "primitive", function(next) { var stack = factor.cont.data_stack; var seq = stack.pop(); stack.push(seq[0],seq[1],seq[2],seq[3]); factor.call_next(next); }); factor.add_word("sequences", "each", "primitive", function(next) { var stack = factor.cont.data_stack; var quot = stack.pop(); var seq = stack.pop(); for(var i=0;i