1// Copyright 2014 the V8 project authors. All rights reserved. 2// Use of this source code is governed by a BSD-style license that can be 3// found in the LICENSE file. 4 5var TYPE_HEIGHT = 25; 6var DEFAULT_NODE_BUBBLE_RADIUS = 12; 7var NODE_INPUT_WIDTH = 50; 8var MINIMUM_NODE_INPUT_APPROACH = 15 + 2 * DEFAULT_NODE_BUBBLE_RADIUS; 9var MINIMUM_NODE_OUTPUT_APPROACH = 15; 10 11function isNodeInitiallyVisible(node) { 12 return node.cfg; 13} 14 15var Node = { 16 isControl: function() { 17 return this.control; 18 }, 19 isInput: function() { 20 return this.opcode == 'Parameter' || this.opcode.endsWith('Constant'); 21 }, 22 isLive: function() { 23 return this.live !== false; 24 }, 25 isJavaScript: function() { 26 return this.opcode.startsWith('JS'); 27 }, 28 isSimplified: function() { 29 if (this.isJavaScript) return false; 30 return this.opcode.endsWith('Phi') || 31 this.opcode.startsWith('Boolean') || 32 this.opcode.startsWith('Number') || 33 this.opcode.startsWith('String') || 34 this.opcode.startsWith('Change') || 35 this.opcode.startsWith('Object') || 36 this.opcode.startsWith('Reference') || 37 this.opcode.startsWith('Any') || 38 this.opcode.endsWith('ToNumber') || 39 (this.opcode == 'AnyToBoolean') || 40 (this.opcode.startsWith('Load') && this.opcode.length > 4) || 41 (this.opcode.startsWith('Store') && this.opcode.length > 5); 42 }, 43 isMachine: function() { 44 return !(this.isControl() || this.isInput() || 45 this.isJavaScript() || this.isSimplified()); 46 }, 47 getTotalNodeWidth: function() { 48 var inputWidth = this.inputs.length * NODE_INPUT_WIDTH; 49 return Math.max(inputWidth, this.width); 50 }, 51 getTitle: function() { 52 var propsString; 53 if (this.properties === undefined) { 54 propsString = ""; 55 } else if (this.properties === "") { 56 propsString = "no properties"; 57 } else { 58 propsString = "[" + this.properties + "]"; 59 } 60 return this.title + "\n" + propsString + "\n" + this.opinfo; 61 }, 62 getDisplayLabel: function() { 63 var result = this.id + ":" + this.label; 64 if (result.length > 40) { 65 return this.id + ":" + this.opcode; 66 } else { 67 return result; 68 } 69 }, 70 getType: function() { 71 return this.type; 72 }, 73 getDisplayType: function() { 74 var type_string = this.type; 75 if (type_string == undefined) return ""; 76 if (type_string.length > 24) { 77 type_string = type_string.substr(0, 25) + "..."; 78 } 79 return type_string; 80 }, 81 deepestInputRank: function() { 82 var deepestRank = 0; 83 this.inputs.forEach(function(e) { 84 if (e.isVisible() && !e.isBackEdge()) { 85 if (e.source.rank > deepestRank) { 86 deepestRank = e.source.rank; 87 } 88 } 89 }); 90 return deepestRank; 91 }, 92 areAnyOutputsVisible: function() { 93 var visibleCount = 0; 94 this.outputs.forEach(function(e) { if (e.isVisible()) ++visibleCount; }); 95 if (this.outputs.length == visibleCount) return 2; 96 if (visibleCount != 0) return 1; 97 return 0; 98 }, 99 setOutputVisibility: function(v) { 100 var result = false; 101 this.outputs.forEach(function(e) { 102 e.visible = v; 103 if (v) { 104 if (!e.target.visible) { 105 e.target.visible = true; 106 result = true; 107 } 108 } 109 }); 110 return result; 111 }, 112 setInputVisibility: function(i, v) { 113 var edge = this.inputs[i]; 114 edge.visible = v; 115 if (v) { 116 if (!edge.source.visible) { 117 edge.source.visible = true; 118 return true; 119 } 120 } 121 return false; 122 }, 123 getInputApproach: function(index) { 124 return this.y - MINIMUM_NODE_INPUT_APPROACH - 125 (index % 4) * MINIMUM_EDGE_SEPARATION - DEFAULT_NODE_BUBBLE_RADIUS 126 }, 127 getOutputApproach: function(graph, index) { 128 return this.y + this.outputApproach + graph.getNodeHeight(this) + 129 + DEFAULT_NODE_BUBBLE_RADIUS; 130 }, 131 getInputX: function(index) { 132 var result = this.getTotalNodeWidth() - (NODE_INPUT_WIDTH / 2) + 133 (index - this.inputs.length + 1) * NODE_INPUT_WIDTH; 134 return result; 135 }, 136 getOutputX: function() { 137 return this.getTotalNodeWidth() - (NODE_INPUT_WIDTH / 2); 138 }, 139 getFunctionRelativeSourcePosition: function(graph) { 140 return this.pos - graph.sourcePosition; 141 }, 142 hasBackEdges: function() { 143 return (this.opcode == "Loop") || 144 ((this.opcode == "Phi" || this.opcode == "EffectPhi") && 145 this.inputs[this.inputs.length - 1].source.opcode == "Loop"); 146 } 147}; 148