Code coverage report for sc/lang/compiler/codegen/codegen.js

Statements: 100% (85 / 85)      Branches: 100% (22 / 22)      Functions: 100% (18 / 18)      Lines: 100% (85 / 85)      Ignored: none     

All files » sc/lang/compiler/codegen/ » codegen.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 1601     1 1   1 1 1   1 112 111   1 1 1 1       1 111 111 111         111     1 16     1 43     1 473 15   32         458 384     74 63     11     1 88 51   88   88 88 51   88       88   88   88     1 14   14 11 28   11     14     1 175   175 219 88   219     175     1 42   42 49   49 42     49       1 22 22   22   22 22 22   22     1 2 2     1 382 172   210     1 1191   1191 2529 2529     1191     1    
(function(sc) {
  "use strict";
 
  require("../compiler");
  require("./scope");
 
  var slice = [].slice;
  var strlib = sc.libs.strlib;
  var Scope = sc.lang.compiler.Scope;
 
  function CodeGen(parent, opts) {
    if (!parent) {
      initialize(this, opts);
    } else {
      this.parent = parent;
      this.opts  = parent.opts;
      this.state = parent.state;
      this.scope = parent.scope;
    }
  }
 
  function initialize(that, opts) {
    that.parent = null;
    that.opts = opts || {};
    that.state = {
      calledSegmentedMethod: false,
      syncBlockScope: null,
      tempVarId: 0
    };
    that.scope = new Scope(that);
  }
 
  CodeGen.addGenerateMethod = function(name, method) {
    CodeGen.prototype[name] = method;
  };
 
  CodeGen.prototype.compile = function(ast) {
    return this.generate(ast);
  };
 
  CodeGen.prototype.generate = function(node, opts) {
    if (Array.isArray(node)) {
      return [
        "(", this.stitchWith(node, ",", function(item) {
          return this.generate(item, opts);
        }), ")"
      ];
    }
 
    if (node && node.type) {
      return toSourceNodeWhenNeeded(this[node.type](node, opts), node);
    }
 
    if (typeof node === "string") {
      return node.replace(/^(?![_$])/, "$");
    }
 
    return "null";
  };
 
  CodeGen.prototype.withFunction = function(args, func) {
    var argItems = this.stitchWith(args, ",", function(item) {
      return this.generate(item);
    });
    var result = [ "function(", argItems, "){" ];
 
    this.scope.begin();
    for (var i = 0, imax = args.length; i < imax; ++i) {
      this.scope.add("arg", args[i]);
    }
    result.push(
      this.scope.toVariableStatement(),
      func.call(this)
    );
    this.scope.end();
 
    result.push("}");
 
    return result;
  };
 
  CodeGen.prototype.insertArrayElement = function(elements) {
    var result = [ "[", "]" ];
 
    if (elements.length) {
      var items = this.stitchWith(elements, ",", function(item) {
        return this.generate(item);
      });
      result.splice(1, 0, items);
    }
 
    return result;
  };
 
  CodeGen.prototype.stitchWith = function(elements, bond, func) {
    var result = [];
 
    for (var i = 0, imax = elements.length; i < imax; ++i) {
      if (i) {
        result.push(bond);
      }
      result.push(func.call(this, elements[i], i));
    }
 
    return result;
  };
 
  CodeGen.prototype.generateStatements = function(elements) {
    var lastIndex = elements.length - 1;
 
    return elements.map(function(item, i) {
      var stmt = this.generate(item);
 
      if (i === lastIndex) {
        stmt = [ "return ", stmt ];
      }
 
      return [ stmt, ";" ];
    }, this);
  };
 
  CodeGen.prototype.useTemporaryVariable = function(func) {
    var result;
    var tempName = "_ref" + this.state.tempVarId;
 
    this.scope.add("var", tempName);
 
    this.state.tempVarId += 1;
    result = func.call(this, tempName);
    this.state.tempVarId -= 1;
 
    return result;
  };
 
  CodeGen.prototype.throwError = function(obj, messageFormat) {
    var message = strlib.format(messageFormat, slice.call(arguments, 2));
    throw new Error(message);
  };
 
  function toSourceNodeWhenNeeded(generated) {
    if (Array.isArray(generated)) {
      return flattenToString(generated);
    }
    return generated;
  }
 
  function flattenToString(list) {
    var result = "";
 
    for (var i = 0, imax = list.length; i < imax; ++i) {
      var elem = list[i];
      result += Array.isArray(elem) ? flattenToString(elem) : elem;
    }
 
    return result;
  }
 
  sc.lang.compiler.CodeGen = CodeGen;
})(sc);