ic-explorer.html 11.7 KB
Newer Older
1
<html>
2 3 4 5
<!--
Copyright 2016 the V8 project authors. All rights reserved.  Use of this source
code is governed by a BSD-style license that can be found in the LICENSE file.
-->
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
<head>
  <style>
    .entry-details {}
    
    .entry-details TD {}
    
    .details {
      width: 2em;
      border: 1px black dotted;
    }
    
    .count {
      text-align: right;
      width: 5em;
      font-family: monospace;
    }
    
    .percentage {
      text-align: right;
      width: 5em;
      font-family: monospace;
    }
    
    .key {
      padding-left: 1em;
32
    }
33 34 35 36 37 38 39 40 41 42
    
    .drilldown-group-title {
      font-weight: bold;
      padding: 0.5em 0 0.2em 0;
    }
  </style>
  <script>
    "use strict"
    var entries = [];

43 44 45 46 47
    var properties = ['type', 'category', 'file', 'filePosition', 'state',
      'key', 'isNative', 'map', 'propertiesMode', 'numberOfOwnProperties',
      'instanceType'
    ]

48 49 50 51 52 53 54 55 56 57
    class Entry {
      constructor(id, line) {
        this.id = id;
        this.line = line;
        var parts = line.split(" ");
        if (parts.length < 6) return
        this.isValid = false;
        if (parts[0][0] !== "[") return;
        if (parts[1] === "patching") return;
        this.type = parts[0].substr(1);
58 59 60 61 62
        this.category = "unknown";
        this.map = "unknown";
        this.propertiesMode = "unknown";
        this.numberOfOwnProperties = 0;
        this.instanceType = "unknown";
63 64 65 66 67 68 69 70 71 72 73 74 75 76
        if (this.type.indexOf("Store") !== -1) {
          this.category = "Store";
        } else if (this.type.indexOf("Load") !== -1) {
          this.category = "Load";
        }
        if (this.type.length == 0) return;
        if (this.type.indexOf('BinaryOpIC(') === 0) {
          this.type = "BinaryOpIC";
          var split = parts[0].split('(');
          this.state = "(" + split[1] + " => " + parts[2];
          var offset = this.parsePositionAndFile(parts, 6);
          if (offset == -1) return
          if (this.file === undefined) return
          this.file = this.file.slice(0, -1);
77
        } else {
78 79 80
          var offset = this.parsePositionAndFile(parts, 2);
          if (offset == -1) return
          this.state = parts[++offset];
81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
          var mapPart = parts[offset + 1];
          if (mapPart !== undefined && mapPart.startsWith("map=")) {
            if (mapPart[4] == "(") {
              if (mapPart.endsWith(")")) {
                this.map = mapPart.substr(5, mapPart.length-6);
              } else {
                this.map = mapPart.substr(5);
              }
              offset++;
              offset = this.parseMapProperties(parts, offset);
            } else {
              this.map = mapPart.substr(4);
              offset++;
            }
            if (this.map == "(nil)") this.map = "unknown";
          } 
97 98 99 100 101 102 103 104 105
          if (this.type !== "CompareIC") {
            // if there is no address we have a smi key
            var address = parts[++offset];
            if (address !== undefined && address.indexOf("0x") === 0) {
              this.key = parts.slice(++offset).join(" ");
            } else {
              this.key = address;
            }
          }
106
        }
107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
        this.filePosition = this.file + " " + this.position;
        if (this.key) {
          var isStringKey = false
          if (this.key.indexOf("<String[") === 0) {
            isStringKey = true;
            this.key = "\"" + this.key.slice(this.key.indexOf(']') + 3);
          } else if (this.key.indexOf("<") === 0) {
            this.key = this.key.slice(1);
          }
          if (this.key.endsWith(">]")) {
            this.key = this.key.slice(0, -2);
          } else if (this.key.endsWith("]")) {
            this.key = this.key.slice(0, -1);
          }
          if (isStringKey) {
            this.key = this.key + "\"";
          }
        }
        this.isValid = true;
126
      }
127

128 129 130 131 132 133 134 135 136 137 138
      parseMapProperties(parts, offset) {
        var next = parts[++offset];
        if (!next.startsWith('dict')) return offset;
        this.propertiesMode = 
            next.substr(5) == "0" ? "fast" : "slow";
        this.numberOfOwnProperties  = parts[++offset].substr(4);
        next = parts[++offset];
        this.instanceType = next.substr(5, next.length-6);
        return offset;
      }

139 140 141 142 143 144 145 146 147 148 149 150 151 152
      parsePositionAndFile(parts, start) {
        // find the position of 'at' in the parts array.
        var offset = start;
        for (var i = start + 1; i < parts.length; i++) {
          offset++;
          if (parts[i] == 'at') break;
        }
        if (parts[offset] !== 'at') return -1;
        this.position = parts.slice(start, offset).join(' ');
        offset += 1;
        this.isNative = parts[offset] == "native"
        offset += this.isNative ? 1 : 0;
        this.file = parts[offset];
        return offset;
153 154 155
      }
    }

156 157
    function loadFile() {
      var files = document.getElementById("uploadInput").files;
158

159 160
      var file = files[0];
      var reader = new FileReader();
161

162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184
      reader.onload = function(evt) {
        entries = [];
        var end = this.result.length;
        var current = 0;
        var next = 0;
        var line;
        var i = 0;
        var entry;
        while (current < end) {
          next = this.result.indexOf("\n", current);
          if (next === -1) break;
          i++;
          line = this.result.substring(current, next);
          current = next + 1;
          entry = new Entry(i, line);
          if (entry.isValid) entries.push(entry);
        }

        document.getElementById("count").innerHTML = i;
        updateTable();
      }
      reader.readAsText(file);
      initGroupKeySelect();
185 186 187
    }


188 189 190 191 192 193 194 195 196 197 198 199 200 201
    class Group {
      constructor(property, key, entry) {
        this.property = property;
        this.key = key;
        this.count = 1;
        this.entries = [entry];
        this.percentage = undefined;
        this.groups = undefined;
      }

      add(entry) {
        this.count++;
        this.entries.push(entry)
      }
202

203 204 205 206 207 208 209 210
      createSubGroups() {
        this.groups = {};
        for (var i = 0; i < properties.length; i++) {
          var subProperty = properties[i];
          if (this.property == subProperty) continue;
          this.groups[subProperty] = groupBy(this.entries, subProperty);
        }
      }
211 212
    }

213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238
    function groupBy(entries, property) {
      var accumulator = {};
      accumulator.__proto__ = null;
      var length = entries.length;
      for (var i = 0; i < length; i++) {
        var entry = entries[i];
        var key = entry[property];
        if (accumulator[key] == undefined) {
          accumulator[key] = new Group(property, key, entry)
        } else {
          var group = accumulator[key];
          if (group.entries == undefined) console.log([group, entry]);
          group.add(entry)
        }
      }
      var result = []
      for (var key in accumulator) {
        var group = accumulator[key];
        group.percentage = Math.round(group.count / length * 100 * 100) / 100;
        result.push(group);
      }
      result.sort((a, b) => {
        return b.count - a.count
      });
      return result;
    }
239 240 241



242
    function escapeHtml(unsafe) {
243
      if (!unsafe) return "";
244 245 246 247 248 249 250
      return unsafe.toString()
           .replace(/&/g, "&amp;")
           .replace(/</g, "&lt;")
           .replace(/>/g, "&gt;")
           .replace(/"/g, "&quot;")
           .replace(/'/g, "&#039;");
    }
251

252 253 254 255 256 257 258 259 260
    function updateTable() {
      var select = document.getElementById("group-key");
      var key = select.options[select.selectedIndex].text;
      console.log(key);
      var tableBody = document.getElementById("table-body");
      removeAllChildren(tableBody);
      var groups = groupBy(entries, key, true);
      display(groups, tableBody);
    }
261

262 263 264
    function selecedOption(node) {
      return node.options[node.selectedIndex]
    }
265

266 267 268 269 270
    function removeAllChildren(node) {
      while (node.firstChild) {
        node.removeChild(node.firstChild);
      }
    }
271

272 273
    function display(entries, parent) {
      var fragment = document.createDocumentFragment();
274

275 276 277 278 279 280 281 282 283 284 285 286 287 288 289
      function td(tr, content, className) {
        var td = document.createElement("td");
        td.innerHTML = content;
        td.className = className
        tr.appendChild(td);
        return td
      }
      var max = Math.min(1000, entries.length)
      for (var i = 0; i < max; i++) {
        var entry = entries[i];
        var tr = document.createElement("tr");
        tr.entry = entry;
        td(tr, '<span onclick="toggleDetails(this)">details</a>', 'details');
        td(tr, entry.percentage + "%", 'percentage');
        td(tr, entry.count, 'count');
290
        td(tr, escapeHtml(entry.key), 'key');
291 292 293 294 295 296 297 298 299 300 301
        fragment.appendChild(tr);
      }
      var omitted = entries.length - max;
      if (omitted > 0) {
        var tr = document.createElement("tr");
        var td = td(tr, 'Omitted ' + omitted + " entries.");
        td.colSpan = 4;
        fragment.appendChild(tr);
      }
      parent.appendChild(fragment);
    }
302

303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
    function displayDrilldown(entry, previousSibling) {
      var tr = document.createElement('tr');
      tr.className = "entry-details";
      tr.style.display = "none";
      // indent by one td.
      tr.appendChild(document.createElement("td"));
      var td = document.createElement("td");
      td.colSpan = 3;
      for (var key in entry.groups) {
        td.appendChild(displayDrilldownGroup(entry, key));
      }
      tr.appendChild(td);
      // Append the new TR after previousSibling.
      previousSibling.parentNode.insertBefore(tr, previousSibling.nextSibling)
    }
318

319 320 321 322 323
    function displayDrilldownGroup(entry, key) {
      var max = 20;
      var group = entry.groups[key];
      var div = document.createElement("div")
      div.className = 'drilldown-group-title'
324
      div.innerHTML = key + ' [top ' + max + ' out of ' + group.length + ']';
325 326 327 328 329
      var table = document.createElement("table");
      display(group.slice(0, max), table, false)
      div.appendChild(table);
      return div;
    }
330

331 332 333
    function toggleDetails(node) {
      var tr = node.parentNode.parentNode;
      var entry = tr.entry;
334

335 336 337 338 339 340 341 342 343 344 345 346 347 348
      // Create subgroup in-place if the don't exist yet.
      if (entry.groups === undefined) {
        entry.createSubGroups();
        displayDrilldown(entry, tr);
      }
      var details = tr.nextSibling;
      var display = details.style.display;
      if (display != "none") {
        display = "none";
      } else {
        display = "table-row"
      };
      details.style.display = display;
    }
349

350 351 352 353 354 355 356 357
    function initGroupKeySelect() {
      var select = document.getElementById("group-key");
      for (var i in properties) {
        var option = document.createElement("option");
        option.text = properties[i];
        select.add(option);
      }
    }
358 359 360 361
    
    function handleOnLoad() {
      document.querySelector("#uploadInput").focus();
    }
362
  </script>
363 364
</head>

365
<body onload="handleOnLoad()">
366
  <h1>
367 368 369
      <span style="color: #00FF00">I</span>
      <span style="color: #FF00FF">C</span>
      <span style="color: #00FFFF">E</span>
370
    </h1> Your IC-Explorer.
371 372 373 374 375 376 377 378 379 380 381 382 383

  <div id="legend" style="padding-right: 200px">
    <div style="float:right;  border-style: solid; border-width: 1px; padding:20px">
    0 uninitialized<br>
    . premonomorphic<br>
    1 monomorphic<br>
    ^ recompute handler<br>
    P polymorphic<br>
    N megamorphic<br>
    G generic
    </div>
  </div>

384 385 386 387
  <h2>Usage</h2> Run your script with <code>--trace_ic</code> and upload on this page:<br/>
  <code>/path/to/d8 --trace_ic your_script.js > trace.txt</code>
  <h2>Data</h2>
  <form name="fileForm">
388
    <p>
389 390 391 392 393 394
      <input id="uploadInput" type="file" name="files" onchange="loadFile();"> trace
      entries: <span id="count">0</span>
    </p>
  </form>
  <h2>Result</h2>
  <p>
395 396
    Group-Key:
    <select id="group-key" onchange="updateTable()"></select>
397 398 399 400 401 402 403 404 405
  </p>
  <p>
    <table id="table" width="100%">
      <tbody id="table-body">
      </tbody>
    </table>
  </p>
</body>

406
</html>