]> gitweb.factorcode.org Git - factor.git/commitdiff
fjsc: move to CPS compiling javascript
authorchris.double <chris.double@double.co.nz>
Sat, 16 Dec 2006 13:46:04 +0000 (13:46 +0000)
committerchris.double <chris.double@double.co.nz>
Sat, 16 Dec 2006 13:46:04 +0000 (13:46 +0000)
apps/furnace-fjsc/resources/bootstrap.js
libs/fjsc/fjsc.factor
libs/fjsc/tests.factor

index 5e3f63f50efd677eff732aecadf7127cf215b93c..dc822e654ae0428f4587846b400197beb0cf5595 100644 (file)
-function Factor() {
-  var self = this;
-  this.form = false;
-  this.data_stack = [ ];  
-  this.words = { 
-    dup: function() { self.fjsc_dup(); },
-    drop: function() { self.fjsc_drop(); },
-    nip: function() { self.fjsc_nip(); },
-    over: function() { self.fjsc_over(); },
-    swap: function() { self.fjsc_swap(); },
-    "+": function() { self.fjsc_plus(); },
-    "-": function() { self.fjsc_minus(); },
-    "*": function() { self.fjsc_times(); },
-    "/": function() { self.fjsc_divide(); },
-    ".": function() { self.fjsc_dot(); },
-    "call": function() { self.fjsc_call(); },
-    "execute": function() { self.fjsc_call(); },
-    "map": function() { self.fjsc_map(); },
-    "reduce": function() { self.fjsc_reduce(); },
-    "clear": function() { self.fjsc_clear(); },
-    "if": function() { self.fjsc_if(); },
-    "=": function() { self.fjsc_equals(); },
-    "f": function() { self.fjsc_false(); },
-    "t": function() { self.fjsc_true(); },
-    "empty?": function() { self.fjsc_is_empty(); },
-    "window": function() { self.fjsc_window(); },
-    "run-file": function() { self.fjsc_run_file(); },
-    "http-get": function() { self.fjsc_http_get(); },
-    "bootstrap": function() { self.fjsc_bootstrap(); }
-  };  
+function Word(name, source, func) {
+  this.name = name;
+  this.source = source;
+  this.func = func;
 }
 
-Factor.prototype.server_eval = function(text) {
-   var self = this;
-   var callback = {
-      success: function(o) {
-        var v = o.responseText;
-        eval(v)
-        self.display_datastack();
-        document.getElementById('compiled').innerHTML="<pre>" + v + "</pre>";
-        document.getElementById('code').value="";
-      }
-   };
-   this.form.code.value=text;
-   YAHOO.util.Connect.setForm(this.form);
-   YAHOO.util.Connect.asyncRequest('POST', "/responder/fjsc/compile", callback);
+Word.prototype.execute = function(world, next) {
+  this.func(world,next);
 }
 
-Factor.prototype.fjsc_eval = function(form) {
-   this.form = form;
-   this.server_eval(form.code.value);
+function Stack() {
+  this.stack = [];
 }
 
-Factor.prototype.display_datastack = function() {
-   var html=[];
-   html.push("<table border='1'>")
-   for(var i = 0; i < this.data_stack.length; ++i) {
-      html.push("<tr><td>")
-      html.push(this.data_stack[i])
-      html.push("</td></tr>")
-   }
-   html.push("</table>")
-   document.getElementById('stack').innerHTML=html.join("");
+Stack.prototype.push = function(v,world,next) {
+  this.stack.push(v);
+  next(world);
 }
 
-Factor.prototype.fjsc_dup = function() {
-  var stack = this.data_stack;
-   var v = stack.pop();
-   stack.push(v);
-   stack.push(v);
+Stack.prototype.pop = function(world,next) {
+  this.stack.pop();
+  next(world);
 }
 
-Factor.prototype.fjsc_drop = function() {
-  this.data_stack.pop();
+function Factor() {
+  this.words = { };
+  this.data_stack = new Stack();
+  this.form = false ;
+  this.next = false;
 }
 
-Factor.prototype.fjsc_nip = function() {
-  var stack = this.data_stack;
-  var v = stack.pop();
-  stack.pop();
-  stack.push(v);
-}
+var factor = new Factor();
 
-Factor.prototype.fjsc_plus = function() {
-  var stack = this.data_stack;
-  var v1 = stack.pop();
-  var v2 = stack.pop();
-  stack.push(v1+v2);
-}
+factor.words["dup"] = new Word("dup", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  stack[stack.length] = stack[stack.length-1];
+  next(world);
+});
 
-Factor.prototype.fjsc_minus = function() {
-  var stack = this.data_stack;
-  var v1 = stack.pop();
-  var v2 = stack.pop();
-  stack.push(v2-v1);
-}
+factor.words["drop"] = new Word("drop", "primitive", function(world, next) {
+  world.data_stack.stack.pop();
+  next(world);
+});
 
-Factor.prototype.fjsc_times = function() {
-  var stack = this.data_stack;
+factor.words["nip"] = new Word("nip", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  stack[stack.length-2] = stack[stack.length-1];
+  stack.pop();
+  next(world);
+});
+
+factor.words["over"] = new Word("over", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  stack[stack.length] = stack[stack.length-2];
+  next(world);
+});
+
+factor.words["swap"] = new Word("swap", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  var temp = stack[stack.length-2];
+  stack[stack.length-2] = stack[stack.length-1];
+  stack[stack.length-1] = temp;
+  next(world);
+});
+
+factor.words["*"] = new Word("*", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  stack.push(stack.pop() * stack.pop());
+  next(world);
+});
+
+factor.words["+"] = new Word("+", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
+  stack.push(stack.pop() + stack.pop());
+  next(world);
+});
+
+factor.words["-"] = new Word("-", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
   var v1 = stack.pop();
   var v2 = stack.pop();
-  stack.push(v1*v2);
-}
+  stack.push(v2 - v1);
+  next(world);
+});
 
-Factor.prototype.fjsc_divide = function() {
-  var stack = this.data_stack;
+factor.words["/"] = new Word("/", "primitive", function(world, next) {
+  var stack = world.data_stack.stack;
   var v1 = stack.pop();
   var v2 = stack.pop();
-  stack.push(v2/v1);
-}
-
-Factor.prototype.fjsc_dot = function() {
-  alert(this.data_stack.pop());
-}
-
-Factor.prototype.fjsc_call = function() {
-  (this.data_stack.pop())();
-}
-
-Factor.prototype.fjsc_map = function() {
-  var quot = this.data_stack.pop();
-  var seq = this.data_stack.pop();
-  var result = [ ];
-  for(var i=0;i<seq.length;++i) {  
-    this.data_stack.push(seq[i]);
-    (quot)();
-    result[i]=this.data_stack.pop();
-  }
-  this.data_stack.push(result);
-}
-
-Factor.prototype.fjsc_reduce = function() {
-  var quot = this.data_stack.pop();
-  var prev = this.data_stack.pop();
-  var seq = this.data_stack.pop();
-  for(var i=0;i<seq.length;++i) {  
-    this.data_stack.push(prev);
-    this.data_stack.push(seq[i]);
-    (quot)();
-    prev=this.data_stack.pop();
-  }
-  this.data_stack.push(prev);
-}
-
-Factor.prototype.fjsc_if = function() {
-  var else_quot = this.data_stack.pop();
-  var then_quot = this.data_stack.pop();
-  var condition = this.data_stack.pop();
+  stack.push(v2 / v1);
+  next(world);
+});
+
+factor.words["."] = new Word(".", "primitive", function(world, next) {
+  alert(world.data_stack.stack.pop());
+  next(world);
+});
+
+factor.words["call"] = new Word("call", "primitive", function(world, next) {
+  var quot = world.data_stack.stack.pop();
+  quot.execute(world, next);
+});
+
+factor.words["execute"] = new Word("execute", "primitive", function(world, next) {
+  var quot = world.data_stack.stack.pop();
+  quot.execute(world, next);
+});
+
+factor.words["clear"] = new Word("clear", "primitive", function(world, next) {
+  world.data_stack.stack = [];
+  next(world);
+});
+
+factor.words["square"] = new Word("square", "primitive", function(world, next) {  
+  var stack = world.data_stack.stack;
+  stack[stack.length-1] = stack[stack.length-1] * stack[stack.length-1];
+  next(world);
+});
+
+factor.words["if"] = new Word("if", "primitive", function(world, next) {  
+  var stack = world.data_stack.stack;
+  var else_quot = stack.pop();
+  var then_quot = stack.pop();
+  var condition = stack.pop();
   if(condition) {
-    (then_quot)();
+    then_quot.execute(world, next);
   } else {
-    (else_quot)();
+    else_quot.execute(world, next);
   }
-}
-
-Factor.prototype.fjsc_equals = function() {
-  var v1 = this.data_stack.pop();
-  var v2 = this.data_stack.pop();
-  this.data_stack.push(v1==v2);
-}
-
-Factor.prototype.fjsc_clear = function() {
-  factor.data_stack = [ ]
-}
-
-Factor.prototype.fjsc_false = function() {
-  factor.data_stack.push(false);
-}
-
-Factor.prototype.fjsc_true = function() {
-  factor.data_stack.push(true);
-}
-
-Factor.prototype.fjsc_is_empty = function() {
-  factor.data_stack.push(factor.data_stack.pop().length==0);
-}
-
-Factor.prototype.fjsc_over = function() {
-   var stack = this.data_stack;
-   stack.push(stack[stack.length-2]);
-}
-
-Factor.prototype.fjsc_swap = function() {
-   var stack = this.data_stack; 
-   var len = stack.length;
-   var temp = stack[len-2];   
-   stack[len-2] = stack[len-1];
-   stack[len-1] = temp;
-}
-
-Factor.prototype.fjsc_window = function() {
-   var stack = this.data_stack;
-   stack.push(window);
-}
-
-Factor.prototype.fjsc_run_file = function() {
+});
+
+factor.words["f"] = new Word("f", "primitive", function(world, next) {  
+  world.data_stack.stack.push(false);
+  next(world);
+});
+
+factor.words["t"] = new Word("t", "primitive", function(world, next) {  
+  world.data_stack.stack.push(true);
+  next(world);
+});
+
+factor.words["window"] = new Word("window", "primitive", function(world, next) {  
+  world.data_stack.stack.push(window);
+  next(world);
+});
+
+factor.words["bootstrap"] = new Word("bootstrap", "primitive", function(world, next) {  
+  world.data_stack.stack.push("/responder/fjsc-resources/bootstrap.factor");
+  world.words["run-file"].execute(world, next);
+});
+
+factor.words["run-file"] = new Word("run-file", "primitive", function(world, next) {  
+  var stack = world.data_stack.stack;
+  var url = stack.pop();
+  var callback = {
+    success: function(o) {
+      var result = o.responseText;
+      world.server_eval(result, world, next);
+    },
+    failure: function(o) {
+      alert('run-file failed');
+      next(world);
+    }
+  };
+  YAHOO.util.Connect.asyncRequest('GET', url, callback, null);
+});
+
+Factor.prototype.define_word = function(name, source, func, world, next) {
+  factor.words[name] = new Word(name, source, function(world, next) {
+    var old = world.next;
+    world.next = function(world) {
+      world.next = old;
+      next(world);
+    }
+    func(world);
+  });
+  next(world);
+}
+
+Factor.prototype.make_quotation = function(source, func) {
+  return new Word("quotation", source, function(world, next) {
+    var old = world.next;
+    world.next = function(world) {
+      world.next = old;
+      next(world);
+    }
+    func(world);
+  });
+}
+
+Factor.prototype.call_alien = function(has_return,method_name, object, args, world, next) {
+  var v = object[method_name].apply(object, args);
+  if(has_return)
+    world.data_stack.stack.push(v);
+  next(world);
+}
+
+
+Factor.prototype.server_eval = function(text, world, next) {
    var self = this;
-   var stack = this.data_stack;
-   var url = stack.pop();
    var callback = {
-     success: function(o) {
-       var result = o.responseText;
-       self.server_eval(result);
-     },
-     failure: function(o) {
-       alert('run-file failed');
-     }
+      success: function(o) {
+        var v = o.responseText;
+        document.getElementById('compiled').innerHTML="<pre>" + v + "</pre>";
+        document.getElementById('code').value="";
+        var func = eval(v);
+         factor.next = function() { self.display_datastack(); } 
+        func(factor);
+         if(world && next) 
+           next(world);
+      }
    };
-
-   YAHOO.util.Connect.asyncRequest('GET', url, callback, null);
+   this.form.code.value=text;
+   YAHOO.util.Connect.setForm(this.form);
+   YAHOO.util.Connect.asyncRequest('POST', "/responder/fjsc/compile", callback);
 }
 
-Factor.prototype.fjsc_http_get = function() {
-   var self = this;
-   var stack = this.data_stack;
-   var url = stack.pop();
-   var callback = {
-     success: function(o) {
-       var result = o.responseText;
-       self.data_stack.push(result);
-       self.display_datastack();
-     },
-     failure: function(o) {
-       alert('http-get failed');
-     }
-   };
-
-   YAHOO.util.Connect.asyncRequest('GET', url, callback, null);
+Factor.prototype.fjsc_eval = function(form) {
+   this.form = form;
+   this.server_eval(form.code.value);
 }
 
-
-Factor.prototype.fjsc_bootstrap = function() {
-   this.data_stack.push("/responder/fjsc-resources/bootstrap.factor");
-   this.fjsc_run_file();
+Factor.prototype.display_datastack = function() {
+   var html=[];
+   html.push("<table border='1'>")
+   for(var i = 0; i < this.data_stack.stack.length; ++i) {
+      html.push("<tr><td>")
+      html.push(this.data_stack.stack[i])
+      html.push("</td></tr>")
+   }
+   html.push("</table>")
+   document.getElementById('stack').innerHTML=html.join("");
 }
 
-var factor = new Factor();
\ No newline at end of file
index 40a33a71057ca8e822324008d8642e6b64f97250..61f86ae4e9fca05a16c4d2820c5097c600855e6b 100644 (file)
@@ -99,9 +99,9 @@ LAZY: 'atom' ( -- parser )
   'identifier' 'number' <|> 'string' <|> ;
 
 LAZY: 'alien' ( -- parser )
-  'array' [ ast-array-elements ast-expression-values [ ast-string-value ] map ] <@
+  'array' [ ast-array-elements ast-expression-values ] <@
   'string' [ ast-string-value ] <@ <&>
-  'array' [ ast-array-elements ast-expression-values [ ast-string-value ] map ] <@ <:&>
+  'array' [ ast-array-elements ast-expression-values ] <@ <:&>
   "alien-invoke" token sp <& [ first3 <ast-alien> ] <@ ;
 
 LAZY: 'comment' ( -- parser )
@@ -112,12 +112,12 @@ LAZY: 'comment' ( -- parser )
 
 LAZY: 'expression' ( -- parser )
   'comment' 
-  'define' sp <|>
-  'word' sp <|>
   'alien' sp <|>
-  'atom' sp <|> 
   'quotation' sp <|> 
   'array' sp <|>
+  'define' sp <|>
+  'word' sp <|>
+  'atom' sp <|> 
   <*> [ <ast-expression> ] <@ ;
 
 LAZY: 'statement' ( -- parser )
@@ -130,9 +130,9 @@ M: ast-number (literal)
   ast-number-value number>string , ;
 
 M: ast-number (compile) 
-  "factor.data_stack.push(" ,
+  "world.data_stack.push(" ,
   (literal)  
-  ")" , ;
+  ",world," , ;
 
 M: ast-string (literal) 
   "'" ,
@@ -142,30 +142,30 @@ M: ast-string (literal)
 M: ast-string (compile) 
   "factor.data_stack.push(" ,
   (literal)
-  ")" , ;
+  ",world," , ;
 
 M: ast-identifier (literal) 
-  "factor.words[\"" , ast-identifier-value , "\"]" ,  ;
+  "world.words[\"" , ast-identifier-value , "\"]" ,  ;
 
 M: ast-identifier (compile) 
-  (literal) "();" ,  ;
+  (literal) ".execute(world, " ,  ;
 
 M: ast-define (compile) 
-  "factor.words[\"" , 
+  "world.define_word(\"" , 
   dup ast-define-name ast-identifier-value , 
-  "\"]=function() { " ,  
+  "\",\"source\"," ,
   ast-define-expression (compile)
-  "}" , ;
+  ",world," , ;
 
 M: ast-quotation (literal)   
-  "function() { " ,  
+  "world.make_quotation(\"source\"," ,
   ast-quotation-expression (compile)
-  "}" , ;
+  ")" , ;
 
 M: ast-quotation (compile)   
-  "factor.data_stack.push(" ,  
-  (literal)
-  ")" , ;
+  "world.data_stack.push(world.make_quotation(\"source\"," ,
+  ast-quotation-expression (compile)
+  "),world," , ;
 
 M: ast-array (literal)   
   "[" ,  
@@ -173,30 +173,43 @@ M: ast-array (literal)
   "]" , ;
 
 M: ast-array (compile)   
-  "factor.data_stack.push(" ,  
-  (literal)
-  ")" , ;
+  "world.data_stack.push(" , (literal) ",world," , ;
+
 
 M: ast-expression (literal)
   ast-expression-values [
     (literal) 
   ] each ;
 
+: do-expressions ( seq -- )
+  dup empty? not [
+    unclip
+    dup ast-comment? not [
+      "function(world) {" ,
+      (compile) 
+      do-expressions
+      ")}" ,
+    ] [
+      drop do-expressions
+    ] if
+  ] [
+    drop "world.next" ,
+  ] if  ;
+  
 M: ast-expression (compile)
-  ast-expression-values [ (compile) ] [ ";" , ] interleave ;
+  ast-expression-values do-expressions ;
 
 M: ast-alien (compile)
+  "world.call_alien(" ,
   dup ast-alien-return empty? not [
-    "factor.data_stack.push(" ,
-  ] when
-  dup ast-alien-method ,
-  ".apply(" ,
-  "factor.data_stack.pop(), [" ,
-  dup ast-alien-args [ drop "factor.data_stack.pop()" , ] [ "," , ] interleave 
-  "])" ,
-  ast-alien-return empty? not [
-    ")" ,
-  ] when ;
+    "true," ,
+  ] [
+    "false," ,
+  ] if
+  dup ast-alien-method "\"" , , "\"," ,
+  "factor.data_stack.stack.pop(), [" ,
+  ast-alien-args [ drop "factor.data_stack.stack.pop()" , ] [ "," , ] interleave 
+  "],world," , ;
 
 M: ast-word (literal)   
   "factor.words[\"" , 
@@ -206,19 +219,32 @@ M: ast-word (literal)
 M: ast-word (compile)
   "factor.data_stack.push(" ,
   (literal)
-  ")" , ;
+  ",world," , ;
   
 M: ast-comment (compile)
-  drop "/* */" , ;
+  drop ;
 
 M: ast-stack-effect (compile)
   drop ;
 
 : fjsc-compile ( ast -- string )
   [
-    [ (compile) ] { } make [ write ] each
+    [ 
+      "(" ,
+      (compile) 
+      ")" ,
+    ] { } make [ write ] each
   ] string-out ;
   
+: fjsc-compile* ( string -- string )
+  'statement' parse car parse-result-parsed fjsc-compile ;
+
+: fc* ( string -- string )
+  [
+  'statement' parse car parse-result-parsed ast-expression-values do-expressions 
+  ] { } make [ write ] each ;
+  
+
 : fjsc-literal ( ast -- string )
   [
     [ (literal) ] { } make [ write ] each
index ba467021f89ac62b3481abc3c7b5421ff61e1ef0..0c3b1dc20cb0b6fcf1300d1c3ca87f1fa3bff853 100644 (file)
@@ -16,34 +16,6 @@ IN: temporary
   "{ 55 2abc1 100 }" 'array' parse car parse-result-parsed
 ] unit-test
 
-{ "factor.words[\"alert\"]();" } [
-  "alert" 'identifier' parse car parse-result-parsed fjsc-compile 
-] unit-test
-
-{ "factor.data_stack.push(123);factor.words[\"alert\"]();" } [
-  "123 alert" 'expression' parse car parse-result-parsed fjsc-compile 
-] unit-test
-
-{ "factor.data_stack.push(123);factor.data_stack.push('hello');factor.words[\"alert\"]();" } [
-  "123 \"hello\" alert" 'expression' parse car parse-result-parsed fjsc-compile 
-] unit-test
-{ "factor.words[\"foo\"]=function() { factor.data_stack.push(123);factor.data_stack.push('hello')}" } [
-  ": foo 123 \"hello\" ;" 'define' parse car parse-result-parsed fjsc-compile 
-] unit-test
-
-{ "factor.words[\"foo\"]=function() { factor.data_stack.push(123);factor.data_stack.push('hello')}" } [
-  ": foo 123 \"hello\" ;" 'expression' parse car parse-result-parsed fjsc-compile 
-] unit-test
-
-{ "alert.apply(factor.data_stack.pop(), [factor.data_stack.pop()])" } [
-  "{ } \"alert\" { \"string\" } alien-invoke" 'expression' parse car parse-result-parsed fjsc-compile
-] unit-test
-
-{ "factor.data_stack.push(alert.apply(factor.data_stack.pop(), [factor.data_stack.pop()]))" } [
-  "{ \"string\" } \"alert\" { \"string\" } alien-invoke" 'expression' parse car parse-result-parsed fjsc-compile
-] unit-test
-
 { T{ ast-stack-effect f { } { "d" "e" "f" } } } [
   "( -- d e f )" 'stack-effect' parse car parse-result-parsed 
 ] unit-test