From: stevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Date: Fri, 18 Dec 2009 08:03:50 +0000 (+0000)
Subject: Move the timings-specific pieces of the buildbot infrastructure into
X-Git-Url: http://git.tremily.us/?a=commitdiff_plain;h=159815ad9df69136ff8be2c0cf6bda8794151fc5;p=scons.git

Move the timings-specific pieces of the buildbot infrastructure into
the trunk/timings directory.  We'll map them into the buildbot directory
using svn:externals.  This will let us keep all the pieces of a timing
configuration, including its buildbot pieces, in one place, and will
let us simplify the Master initialization (since it will be able to look
on-disk for the configurations for which it should set up buildbot steps,
instead of querying the SVN server).


git-svn-id: http://scons.tigris.org/svn/scons/trunk@4562 fdb21ef1-2011-0410-befe-b5e4ea1792b1
---

diff --git a/timings/changelog.html b/timings/changelog.html
new file mode 100644
index 00000000..586ebadc
--- /dev/null
+++ b/timings/changelog.html
@@ -0,0 +1,204 @@
+<html>
+<head>
+<script src="js/common.js"></script>
+<style type="text/css">
+table {
+  border-collapse: collapse;
+}
+thead {
+  border-top: solid 1px gray;
+  border-left: solid 1px gray;
+}
+tbody {
+  border-top: solid 1px gray;
+  border-bottom: solid 1px gray;
+  border-left: solid 1px gray;
+}
+th {
+  text-align: center;
+  border-right: solid 1px gray;
+}
+td {
+  padding-left: 0.5em;
+  padding-top: 0.3em;
+  padding-bottom: 0.3em;
+  padding-right: 1.4em;
+  border-top: solid 1px gray;
+  vertical-align: top;
+  font-family: monospace;
+}
+form {
+  background-color: lightblue;
+  border: 1px solid gray;
+  padding: 2px;
+}
+iframe#content {
+  border: none;
+  width: 0px;
+  height: 0px;
+}
+/*
+form {
+  position: fixed;
+  left: 0px;
+  top: 0px;
+  width: 100%;
+}
+*/
+</style>
+</head>
+<body>
+<form name="ui">
+  SVN path: <input id="url" type="text" name="url" value="">
+  SVN revision range: <input id="range" type="text" name="range" value="">
+  <input id="mode_text" type="radio" name="mode" value="text">text
+  <input id="mode_html" type="radio" name="mode" value="html">html
+  <input type="submit" value="Show Changelog">
+</form>
+
+<script>
+params = ParseParams();
+
+function fix_text(str, n) {
+  if (str.length > n)
+    return str.substring(0, n);
+
+  for (var i = str.length; i < n; ++i)
+    str = str + ' ';
+  return str;
+}
+
+function get_entries() {
+  return content.contentDocument.getElementsByTagName("logentry");
+}
+
+function get_info(entry) {
+  var r = new Object;
+  r.rev = entry.getAttribute("revision");
+  r.author = entry.getElementsByTagName("author")[0].textContent;
+  r.msg = entry.getElementsByTagName("msg")[0].textContent;
+  r.paths = [];
+  var paths = entry.getElementsByTagName("path")
+  for (var i = 0; i < paths.length; ++i) {
+    r.paths.push({"action" : paths[i].getAttribute("action"),
+                  "value"  : paths[i].textContent});
+  }
+  return r;
+}
+
+function render_log_callback() {
+  if ("mode" in params && params.mode == "text") {
+    var out = document.createElement("PRE");
+    document.body.appendChild(out);
+
+    var entries = get_entries();
+    for (var i = 0; i < entries.length; ++i) {
+      var info = get_info(entries[i]);
+
+      var msg = info.msg;
+      msg = msg.replace(/\n/g, ' ' );
+      msg = msg.replace(/\t/g, ' ' );
+      while (msg.charAt(0) == ' ')
+        msg = msg.substring(1);
+      
+      var msg_clipped = msg.substring(0, 66);
+      if (msg_clipped.length < msg.length)
+        msg_clipped = msg_clipped + "...";
+
+      out.appendChild(document.createTextNode(
+          fix_text(info.rev, 6) + " " +
+          fix_text(info.author, 8) + " " +
+          msg_clipped + "\n"));
+    }
+  } else {
+    var table = document.createElement("TABLE");
+    table.setAttribute("class", "log");
+    document.body.appendChild(table);
+
+    var entries = get_entries();
+    for (var i = 0; i < entries.length; ++i) {
+      var info = get_info(entries[i]);
+
+      var tr = document.createElement("TR");
+      table.appendChild(tr);
+
+      var td, a;
+
+      // revision:
+      td = document.createElement("TD");
+      tr.appendChild(td);
+
+      a = document.createElement("A");
+      a.setAttribute("href", "http://scons.tigris.org/source/browse/scons?view=rev&revision=" + info.rev);
+      a.appendChild(document.createTextNode(info.rev));
+
+      td.appendChild(a);
+
+      // author:
+      td = document.createElement("TD");
+      tr.appendChild(td);
+
+      a = document.createElement("A");
+      a.setAttribute("href", "mailto:" + info.author);
+      a.appendChild(document.createTextNode(info.author));
+
+      td.appendChild(a);
+
+      // details:
+      td = document.createElement("TD");
+      tr.appendChild(td);
+
+      var p = document.createElement("PRE");
+      td.appendChild(p);
+
+      var s = info.msg;
+      p.appendChild(document.createTextNode(s));
+
+      for (var j = 0; j < info.paths.length; ++j) {
+        td.appendChild(document.createTextNode(info.paths[j]["action"] + " - "))
+        var a = document.createElement("A");
+        a.setAttribute("href", "http://scons.tigris.org/source/browse/scons" + info.paths[j]["value"] + "?r1=" + info.rev + "&r2=" + (info.rev - 1) + "&pathrev=" + info.rev);
+        a.appendChild(document.createTextNode(info.paths[j]["value"]));
+        td.appendChild(a);
+        td.appendChild(document.createElement("BR"));
+      }
+    }
+  }
+}
+
+function render_log() {
+  var svn_url = params["url"];
+  var svn_range = params["range"];
+  if (svn_url == undefined || svn_range == undefined)
+    return;
+
+  var url = "http://" + location.host + "/cgi-bin/svn-log?url=http://codf21.jail/svn/" +
+      unescape(svn_url) + "&range=" + unescape(svn_range);
+
+  // global 'content' variable: a hidden iframe used to fetch svn data.
+  content = document.createElement("IFRAME");
+  content.setAttribute("id", "content");
+  content.setAttribute("onload", "render_log_callback()");
+  content.setAttribute("src", url);
+  document.body.appendChild(content);
+
+  var el;
+  if ("mode" in params && params["mode"] == "text") {
+    el = document.getElementById("mode_text");
+  } else {
+    el = document.getElementById("mode_html");
+  }
+  el.setAttribute("checked", "1");
+
+  el = document.getElementById("url");
+  el.setAttribute("value", unescape(svn_url));
+
+  el = document.getElementById("range");
+  el.setAttribute("value", unescape(svn_range));
+}
+
+render_log()
+</script>
+</body>
+</html>
+
diff --git a/timings/graph.html b/timings/graph.html
new file mode 100644
index 00000000..e418069d
--- /dev/null
+++ b/timings/graph.html
@@ -0,0 +1,411 @@
+<html>
+
+<!--
+  Copyright (c) 2006-2009 The Chromium Authors. All rights reserved.
+  Use of this source code is governed by a BSD-style license that can be
+  found in the LICENSE file.
+-->
+
+<!--
+  A brief note on terminology as used here: a "graph" is a plotted screenful
+  of data, showing the results of one type of test: for example, the
+  page-load-time graph.  A "trace" is a single line on a graph, showing one
+  one for the test: for example, the reference build trace on the
+  page-load-time graph.
+
+  This page plots arbitrary numerical data loaded from files in a specific
+  format.  It uses two or more data files, all JSON-encoded:
+
+    graphs.dat: a list of objects, each with these properties: name (the name
+        of a graph) and units (the units for the data to be read by humans).
+        Schematically:
+          [{"name": <graph_name>, "units": <units>}, ...]
+
+    <graphname>-summary.dat: for each of the graphs listed in graphs.dat, the
+        corresponding summary file holds rows of data. Each row of data is an
+        object with several properties:
+          "rev": the revision number for this row of data
+          "traces": an object with several properties of its own. The name of
+              the property corresponds to a trace name, used only as an
+              internal identifier, and the property's value is an array of
+              its measurement and that measurement's standard deviation (or
+              other measurement error).
+        Schematically:
+          {"rev": <rev>,
+           "traces": {<trace_name1>: [<value1>, <stddev1>],
+                      <trace_name2>: [<value2>, <stddev2>], ...}
+          }
+-->
+<head>
+<style>
+body {
+  font-family: sans-serif;
+}
+div#output {
+  cursor: pointer;
+}
+div#switcher {
+  cursor: pointer;
+}
+div#switcher a {
+  border-top: 1px solid black;
+  border-left: 1px solid black;
+  padding-left: 0.5em;
+  padding-right: 0.5em;
+}
+canvas.plot {
+  border: 1px solid black;
+}
+div.plot-coordinates {
+  font-family: monospace;
+}
+iframe {
+  display: none;
+  width: 100%;
+  height: 100%;
+  border: none;
+}
+div.selector {
+  border: solid 1px black;
+  cursor: pointer;
+  padding-left: 0.3em;
+  background-color: white;
+}
+div.selector:hover {
+  background-color: rgb(200,200,250);
+}
+div.selected {
+  border-left: none;
+}
+div#selectors {
+  width: 80px;
+  display: none;
+}
+#explain {
+  font-size: 0.75em;
+  font-style: italic;
+  color: rgb(100,100,100);
+}
+</style>
+
+<script src="js/common.js"></script>
+<script src="js/plotter.js"></script>
+<script src="js/coordinates.js"></script>
+<!-- <script src="config.js"></script> -->
+<script>
+var Config = {
+  'title': "TODO title",
+  'source': "http://scons.tigris.org/svn/scons/trunk",
+  'changeLinkPrefix': "changelog.html?mode=html&range=",
+  'builder': "TODO ",
+  'builderLink': "http://buildbot.scons.org:8010/",
+  'detailTabs': ['view-change'],
+};
+document.title = Config.title + ' - ' + Config.buildslave;
+
+var did_position_details = false;
+var units = 'thing-a-ma-bobs';
+var graph_list = [];
+var first_trace = '';
+
+var params = ParseParams();
+if (!('history' in params)) {
+  params.history = 150;
+  // make this option somewhat user discoverable :-/
+  window.location.href = MakeURL(params);
+}
+
+function jsonToJs(data) {
+  return eval('(' + data + ')')
+}
+
+function report_error(error) {
+  document.getElementById("output").innerHTML = "<p>" + error + "</p>";
+}
+
+function received_graph_list(data, error) {
+  if (error) {
+    report_error(error);
+    return;
+  }
+  graph_list = jsonToJs(data);
+
+  if (!('graph' in params) || params.graph == '') {
+    if (graph_list.length > 0)
+      params.graph = graph_list[0].name
+  }
+
+  // Add a selection tab for each graph, and find the units for the selected
+  // one while we're at it.
+  tabs = [];
+  for (var index = 0; index < graph_list.length; ++index) {
+    var graph = graph_list[index];
+    tabs.push(graph.name);
+    if (graph.name == params.graph)
+      units = graph.units;
+  }
+  initPlotSwitcher(tabs);
+
+  // Fetch the data for the selected graph.
+  fetch_summary();
+}
+
+function go_to(graph) {
+  params.graph = graph;
+  if (params.graph == '')
+    delete params.graph;
+  window.location.href = MakeURL(params);
+}
+
+function get_url() {
+  new_url = window.location.href;
+  new_url = new_url.replace(/50/, "150");
+  new_url = new_url.replace(/\&lookout/, "");
+  return new_url;
+}
+
+function on_clicked_plot(prev_cl, cl) {
+  if ('lookout' in params) {
+    window.open(get_url());
+    return;
+  }
+
+  // Define sources for detail tabs
+  if ('view-change' in Config.detailTabs) {
+    document.getElementById('view-change').
+        setAttribute('src', Config.changeLinkPrefix + prev_cl + ':' + cl);
+  }
+  if ('view-pages' in Config.detailTabs) {
+    document.getElementById('view-pages').
+        setAttribute('src', 'details.html?cl=' + cl + '&trace=' + first_trace);
+  }
+  if ('view-coverage' in Config.detailTabs) {
+    document.getElementById('view-coverage').
+        setAttribute('src', Config.coverageLinkPrefix + cl);
+  }
+
+  if (!did_position_details) {
+    position_details();
+    did_position_details = true;
+  }
+}
+
+function received_summary(data, error) {
+  if (error) {
+    report_error(error);
+    return;
+  }
+  // Parse the summary data file.
+  var rows = data.split('\n');
+  var max_rows = rows.length;
+  if (max_rows > params.history)
+    max_rows = params.history;
+
+  var allTraces = {};
+
+  // graphData[rev] = {trace1:[value, stddev], trace2:[value, stddev], ...}
+  var graphData = {};
+  for (var i = 0; i < max_rows; ++i) {
+    if (!rows[i].length)
+      continue;
+    var row = jsonToJs(rows[i]);
+    var traces = row['traces'];
+    var revision = parseInt(row['rev']);
+    graphData[revision] = traces;
+
+    // Collect unique trace names.
+    for (var traceName in traces)
+      allTraces[traceName] = 1;
+  }
+
+  // Build a list of all the trace names we've seen, in the order in which
+  // they appear in the data file. Although JS objects are not required by
+  // the spec to iterate their properties in order, in practice they do,
+  // because it causes compatibility problems otherwise.
+  var traceNames = [];
+  for (var traceName in allTraces)
+    traceNames.push(traceName);
+
+  first_trace = traceNames[0];
+
+  // Build and numerically sort a list of revision numbers.
+  var revisionNumbers = [];
+  for (var rev in graphData)
+    revisionNumbers.push(rev);
+  revisionNumbers.sort(
+      function(a, b) { return parseInt(a, 10) - parseInt(b, 10) });
+
+  // Build separate ordered lists of trace data.
+  var traceData = {};
+  for (var revIndex = 0; revIndex < revisionNumbers.length; ++revIndex) {
+    var rev = revisionNumbers[revIndex];
+    var revisionData = graphData[rev];
+    for (var nameIndex = 0; nameIndex < traceNames.length; ++nameIndex) {
+      var traceName = traceNames[nameIndex];
+      if (!traceData[traceName])
+        traceData[traceName] = [];
+      if (!revisionData[traceName])
+        traceData[traceName].push([NaN, NaN]);
+      else
+        traceData[traceName].push(revisionData[traceName]);
+    }
+  }
+  var plotData = [];
+  for (var traceName in traceData)
+    plotData.push(traceData[traceName]);
+
+  var plotter = new Plotter(revisionNumbers, plotData, traceNames, units,
+    document.getElementById("output"), true);
+  plotter.onclick = on_clicked_plot;
+  plotter.plot();
+}
+
+function fetch_summary() {
+  if ('graph' in params)
+    file = escape(params.graph) + ".dat"
+  else
+    file = "summary.dat"
+  Fetch(file, received_summary);
+}
+
+function fetch_graph_list() {
+  Fetch("graphs.dat", received_graph_list);
+}
+
+function initPlotSwitcher(tabs) {
+  var switcher = document.getElementById("switcher");
+  for(var i = 0; i < tabs.length; i++) {
+    var anchor = document.createElement("a");
+    anchor.appendChild(document.createTextNode(tabs[i] + " "));
+    anchor.addEventListener("click", goToClosure(tabs[i]), false);
+    switcher.appendChild(anchor);
+  }
+}
+
+function goToClosure(graph) {
+  return function(){go_to(graph)};
+}
+
+function position_details() {
+  var output = document.getElementById("output");
+
+  var win_height = window.innerHeight;
+
+  var details = document.getElementById("views");
+
+  var views = document.getElementById("views");
+  var selectors = document.getElementById("selectors");
+  selectors.style.display = "block";
+
+  var views_width = output.offsetWidth - selectors.offsetWidth;
+
+  views.style.border = "1px solid black";
+  views.style.width = views_width + "px";
+  views.style.height = (win_height - output.offsetHeight - output.offsetTop -
+                        30) + "px";
+
+  selectors.style.position = "absolute";
+  selectors.style.left = (views.offsetLeft + views_width + 1) + "px";
+  selectors.style.top = views.offsetTop + "px";
+
+  // Change to the first detail tab
+  for (var tab in Config.detailTabs) {
+    change_view(tab);
+    break; 
+  }
+}
+
+function change_view(target) {
+  for (var tab in Config.detailTabs) {
+    document.getElementById(tab).style.display = 
+        (tab == target ? "block" : "none");
+  }
+}
+
+function init() {
+  // We need to fill the graph list before parsing the params or fetching the
+  // data, so we have a default graph in case none was specified.
+  fetch_graph_list();
+}
+
+window.addEventListener("load", init, false);
+</script>
+</head>
+
+
+<body>
+<div id="header_lookout" align="center">
+  <font style='color: #0066FF; font-family: Arial, serif;
+               font-size: 20pt; font-weight: bold;'>
+    <script>
+      document.write("<a target=\"_blank\" href=\"");
+      document.write(get_url());
+      document.write("\">");
+      if ('header' in params && params.header != '') {
+        document.write(escape(params.header));
+      } else {
+        document.write(Config.title);
+      }
+      document.write("</a>");
+    </script>
+  </font>
+</div>
+
+<div id="header_text">
+Builds generated by the <a href="http://buildbot.chromium.org/">buildbot</a>
+are run through <b>
+<script>
+document.write(Config.title);
+</script>
+</b>and the results of that test are charted here.
+</div>
+
+<div id="explain">
+The vertical axis is measured values, and the horizontal
+axis is the revision number for the build being tested.
+</div>
+<p></p>
+<div id="switcher">
+
+</div>
+<div id="output"></div>
+<div id="details">
+  <div id="views">
+    <script>
+      for (var tab in Config.detailTabs) {
+        document.write("<iframe id=\"" + tab + "\"></iframe>");
+      }
+    </script>
+  </div>
+  <div id="selectors">
+    <script>
+      var firstTab = true;
+      for (var tab in Config.detailTabs) {
+        document.write("<div ");
+	if (firstTab) {
+	  firstTab = false;
+	} else {
+	  document.write("style=\"border-top: none\" ");
+	}
+	document.write("class=\"selector\" onclick=\"change_view('" 
+	    + tab + "')\">" + Config.detailTabs[tab] + "</div>");
+      }
+    </script>
+  </div>
+</div>
+<pre id="log"></pre>
+<script>
+if ('lookout' in params) {
+  document.getElementById("switcher").style.display = "none";
+  document.getElementById("details").style.display = "none";
+  document.getElementById("header_text").style.display = "none";
+  document.getElementById("explain").style.display = "none";
+  if ('thumbnail' in params) {
+    document.getElementById("header_lookout").style.display = "none";
+  }
+} else {
+  document.getElementById("header_lookout").style.display = "none";
+}
+</script>
+</body>
+</html>
diff --git a/timings/index.html b/timings/index.html
new file mode 100644
index 00000000..de102ead
--- /dev/null
+++ b/timings/index.html
@@ -0,0 +1,196 @@
+<html>
+  <head>
+    <title>SCons Timings</title>
+    <script src="js/common.js"></script>
+    <script language="javascript">
+      var perf_url = DirName(window.location.href);
+      function DisplayGraph(name, heighth, width, thumbnail, graph) {
+        var params = ParseParams();
+        var url = perf_url + '/' + name + '/graph.html' +
+                  '?history=50' +
+                  '&lookout';
+        //var debug = url;
+        var debug = '';
+        if (typeof graph == 'undefined') {
+          if ('graph' in params) {
+            graph = params.graph;
+          }
+          url += '&header=';
+        } else if (graph != null) {
+          url += '&header=' + graph;
+        }
+        if (graph != null) {
+          url += '&graph=' + graph;
+        }
+        if (thumbnail) {
+          url += '&thumbnail';
+        }
+        document.write('<td>' + debug + '<iframe' +
+                       //' scrolling="no"' +
+                       ' height="' + heighth + '"' +
+                       ' width="' + width + '"' +
+                       ' src="' + url + '">' +
+                       '</iframe></td>');
+      }
+    </script>
+  </head>
+  <body>
+    <center>
+      <script language="javascript">
+        var params = ParseParams();
+        // TODO:  parameterize these in a .dat file
+        var builder_map = {
+            'Ubuntu': 'ubuntu-timings',
+        };
+        var graphs_map = {
+            'Performance': 'TimeSCons-elapsed',
+            'Memory': 'memory-final',
+        };
+        var test_map = {
+            'CPPPATH': 'CPPPATH',
+            'JTimer': 'JTimer',
+            'hundred': 'hundred',
+        };
+        if ('builder' in params) {
+          builder_list = params.builder.split(',');
+        } else {
+          builder_list = Keys(builder_map);
+        }
+        if ('graph' in params) {
+          graphs_list = param.graph.split(',');
+        } else {
+          graphs_list = Keys(graphs_map);
+        }
+        if ('test' in params) {
+          test_list = params.test.split(',');
+        } else {
+          test_list = Keys(test_map);
+        }
+        function write_builder_table() {
+          document.write('<table><tr>');
+          for (var tindex=0; tindex < test_list.length; tindex++) {
+            test = test_list[tindex];
+            test_name = test_map[test];
+            DisplayGraph(builder_id + '/' + test, 250, 400, false);
+            DisplayGraph(builder_id + '/' + test, 250, 400, false);
+            if ((tindex % 3) == 2) {
+              document.write('</tr><tr>');
+            }
+          }
+          document.write('</tr></table>');
+        }
+        function write_test_table() {
+          document.write('<table>');
+          row = [];
+          function write_row(row) {
+            document.write('<tr>');
+            for (var bidx in row) {
+              builder = row[bidx];
+              document.write('<th align="center"><p><br>' +
+                             builder + '</p></th>');
+            }
+            document.write('</tr><tr>');
+            for (var bidx in row) {
+              builder_id = builder_map[row[bidx]];
+              DisplayGraph(builder_id + '/' + test, 250, 400, true);
+            }
+            document.write('</tr>');
+          }
+          for (var bindex=0; bindex < builder_list.length; bindex++) {
+            builder = builder_list[bindex];
+            row.push(builder)
+            if ((bindex % 3) == 2) {
+              write_row(row);
+              row = [];
+            }
+          }
+          if (row.length > 0) {
+            write_row(row);
+          }
+          document.write('</table>');
+        }
+        function write_builders_header_row() {
+          document.write('<tr><th>&nbsp;</th>');
+          for (var bindex=0; bindex < builder_list.length; bindex++) {
+            builder = builder_list[bindex];
+            url = MakeURL({'builder':escape(builder)});
+            var s ='<th align="center" colspan="' +
+                   graphs_list.length +
+                   '">' +
+                   '<a href="' + url + '">' +
+                   builder +
+                   '</a></th>';
+            document.write(s);
+          }
+          document.write('</tr>');
+        }
+        function write_graphs_header_row() {
+          document.write('<tr><th>&nbsp;</th>');
+          for (var bindex=0; bindex < builder_list.length; bindex++) {
+            for (var gindex=0; gindex < graphs_list.length; gindex++) {
+              graph = graphs_list[gindex];
+              document.write('<th align="center">' + graph + '</th>');
+            }
+          }
+        }
+        function write_overview_table() {
+          document.write('<table>');
+          var width = document.body.scrollWidth / (builder_list.length + 2);
+          write_builders_header_row();
+          for (var tindex=0; tindex < test_list.length; tindex++) {
+            // Write a graphs header every four rows for navigability.
+            // TODO:  Use more sophisticated freezing of the header rows?
+            if ((tindex % 4) == 0) {
+              write_graphs_header_row();
+            }
+            test = test_list[tindex];
+            test_name = test_map[test];
+            document.write('<tr>');
+            url = MakeURL({'test':escape(test)});
+            document.write('<th valign="center"><a href="' + url + '">'
+                           + test_name + '</a></th>');
+            for (var bindex=0; bindex < builder_list.length; bindex++) {
+              builder = builder_list[bindex];
+              builder_id = builder_map[builder];
+              DisplayGraph(builder_id + '/' + test, 250, width, true, "TimeSCons-elapsed")
+              DisplayGraph(builder_id + '/' + test, 250, width, true, "memory-final")
+            }
+            document.write('<th valign="center"><a href="' + url + '">'
+                           + test_name + '</a></th>');
+            document.write('</tr>');
+          }
+          write_graphs_header_row();
+          write_builders_header_row();
+          document.write('</table>');
+        }
+        function write_header(header_string, url_string) {
+          document.write('<h1><a href="' + window.location.pathname + '">'
+                         + 'SCons Timings</a>');
+          if (header_string) {
+            document.write(':  ' + header_string);
+          }
+          if ('graph' in params) {
+            document.write(':  ' + escape(params.graph))
+          } else {
+            document.write(':  overview');
+          }
+          document.write('</h1>');
+        }
+        if (builder_list.length == 0) {
+          builder = builder_list[0];
+          builder_id = builder_map[builder];
+          write_header(builder, '&builder=' + escape(builder));
+          write_builder_table()
+        } else if (test_list.length == 1) {
+          test = test_list[0];
+          test_name = test_map[test];
+          write_header(test_name, '&test=' + escape(test));
+          write_test_table()
+        } else {
+          write_header('', '');
+          write_overview_table();
+        }
+      </script>
+    </center>
+  </body>
+</html>
diff --git a/timings/js/common.js b/timings/js/common.js
new file mode 100644
index 00000000..80510b32
--- /dev/null
+++ b/timings/js/common.js
@@ -0,0 +1,96 @@
+/*
+  Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
+  Use of this source code is governed by a BSD-style license that can be
+  found in the LICENSE file.
+*/
+
+/*
+  Common methods for performance-plotting JS.
+*/
+
+function Fetch(url, callback) {
+  var r = new XMLHttpRequest();
+  r.open("GET", url, true);
+  r.setRequestHeader("pragma", "no-cache");
+  r.setRequestHeader("cache-control", "no-cache");
+  r.onreadystatechange = function() {
+    if (r.readyState == 4) {
+      var error;
+      var text = r.responseText;
+      if (r.status != 200) {
+        error = url + ": " + r.status + ": " + r.statusText;
+      } else if (! text) {
+        error = url + ": null response";
+      }
+      callback(text, error);
+    }
+  }
+
+  r.send(null);
+}
+
+// Returns the keys of an object.
+function Keys(obj) {
+  result = [];
+  for (key in obj) {
+    result.push(key)
+  }
+  return result
+}
+
+// Returns the "directory name" portion of the string (URL),
+// stripping the last element.
+function DirName(s) {
+  elements = s.split('/')
+  elements.pop()
+  return elements.join('/')
+}
+
+// Returns an Object with properties given by the parameters specified in the
+// URL's query string.
+function ParseParams() {
+  var result = new Object();
+  var s = window.location.search.substring(1).split('&');
+  for (i = 0; i < s.length; ++i) {
+    var v = s[i].split('=');
+    result[v[0]] = unescape(v[1]);
+  }
+  return result;
+}
+
+// Creates the URL constructed from the current pathname and the given params.
+function MakeURL(params) {
+  var url = window.location.pathname;
+  var sep = '?';
+  for (p in params) {
+    if (!p)
+      continue;
+    url = url + sep + p + '=' + params[p];
+    sep = '&';
+  }
+  return url;
+}
+
+// Returns a string describing an object, recursively.  On the initial call,
+// |name| is optionally the name of the object and |indent| is not needed.
+function DebugDump(obj, opt_name, opt_indent) {
+  var name = opt_name || '';
+  var indent = opt_indent || '';
+  if (typeof obj == "object") {
+    var child = null;
+    var output = indent + name + "\n";
+
+    for (var item in obj) {
+      try {
+        child = obj[item];
+      } catch (e) {
+        child = "<Unable to Evaluate>";
+      }
+      output += DebugDump(child, item, indent + "  ");
+    }
+
+    return output;
+  } else {
+    return indent + name + ": " + obj + "\n";
+  }
+}
diff --git a/timings/js/coordinates.js b/timings/js/coordinates.js
new file mode 100644
index 00000000..69cb4c22
--- /dev/null
+++ b/timings/js/coordinates.js
@@ -0,0 +1,125 @@
+/*
+  Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
+  Use of this source code is governed by a BSD-style license that can be
+  found in the LICENSE file.
+*/
+
+/**
+ * 'Understands' plot data positioning.
+ *  @constructor
+ *
+ * @param {Array} plotData data that will be displayed
+ */
+function Coordinates(plotData) {
+  this.plotData = plotData;
+  
+  height = window.innerHeight - 16;
+  width = window.innerWidth - 16;
+
+  this.widthMax = width;
+  this.heightMax = Math.min(400, height - 85);
+
+  this.xMinValue = -0.5;
+  this.xMaxValue = (this.plotData[0].length - 1)+ 0.5;
+  this.processYValues_();
+}
+
+Coordinates.prototype.processYValues_ = function () {
+  var merged = [];
+  for (var i = 0; i < this.plotData.length; i++)
+    for (var j = 0; j < this.plotData[i].length; j++)
+      merged.push(this.plotData[i][j][0]);
+  var max = Math.max.apply( Math, merged );
+  var min = Math.min.apply( Math, merged );
+
+  // If we have a missing value, find the real max and min the hard way.
+  if (isNaN(min)) {
+    for (var i = 0; i < merged.length; ++i) {
+      if (isNaN(min) || merged[i] < min)
+        min = merged[i];
+      if (isNaN(max) || merged[i] > max)
+        max = merged[i];
+    }
+  }
+  var yd = (max - min) / 10.0;
+  if (yd == 0)
+    yd = max / 10;
+  this.yMinValue = min - yd;
+  this.yMaxValue = max + yd;
+};
+
+/**
+ * Difference between horizontal max min values.
+ */
+Coordinates.prototype.xValueRange = function() {
+  return this.xMaxValue - this.xMinValue;
+};
+
+/**
+ * Difference between vertical max min values.
+ */
+Coordinates.prototype.yValueRange = function() {
+  return this.yMaxValue - this.yMinValue
+};
+
+/**
+ * Converts horizontal data value to pixel value on canvas.
+ * @param {number} value horizontal data value
+ */
+Coordinates.prototype.xPoints = function(value) {
+  return this.widthMax * ((value - this.xMinValue) / this.xValueRange());
+};
+
+/**
+ * Converts vertical data value to pixel value on canvas.
+ * @param {number} value vertical data value
+ */
+Coordinates.prototype.yPoints = function(value) {
+  /* Converts value to canvas Y position in pixels. */
+  return this.heightMax  - this.heightMax * (value - this.yMinValue) / 
+    this.yValueRange();
+};
+
+/**
+ * Converts X point on canvas to value it represents.
+ * @param {number} position horizontal point on canvas.
+ */
+Coordinates.prototype.xValue = function(position) {
+  /* Converts canvas X pixels to value. */
+  return position / this.widthMax * (this.xValueRange()) + this.xMinValue;
+};
+
+/**
+ * Converts Y point on canvas to value it represents.
+ * @param {number} position vertical point on canvas.
+ */
+Coordinates.prototype.yValue = function(position) {
+  /* Converts canvas Y pixels to value. 
+  position is point value is from top.
+  */
+  var position = this.heightMax - position;
+  var ratio = parseFloat(this.heightMax / position);
+  return  this.yMinValue + this.yValueRange() / ratio;
+};
+
+/**
+ * Converts canvas X pixel to data index.
+ * @param {number} xPosition horizontal point on canvas
+ */
+Coordinates.prototype.dataSampleIndex = function(xPosition) {
+  var xValue = this.xValue(xPosition);
+  var index;
+  if (xValue < 0) {
+    index = 0;
+  } else if (xValue > this.plotData[0].length - 1) {
+    index = this.plotData[0].length - 1;
+  } else {
+    index = xValue.toFixed(0);
+  }
+  return index;
+};
+
+Coordinates.prototype.log = function(val) {
+  document.getElementById('log').appendChild(
+    document.createTextNode(val + '\n'));
+};
diff --git a/timings/js/plotter.js b/timings/js/plotter.js
new file mode 100644
index 00000000..86fb2304
--- /dev/null
+++ b/timings/js/plotter.js
@@ -0,0 +1,336 @@
+/*
+  Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
+  Use of this source code is governed by a BSD-style license that can be
+  found in the LICENSE file.
+*/
+
+// Collection of classes used to plot data in a <canvas>.  Create a Plotter()
+// to generate a plot.
+
+// vertical marker for columns
+function Marker(color) {
+  var m = document.createElement("DIV");
+  m.setAttribute("class", "plot-cursor");
+  m.style.backgroundColor = color;
+  m.style.opacity = "0.3";
+  m.style.position = "absolute";
+  m.style.left = "-2px";
+  m.style.top = "-2px";
+  m.style.width = "0px";
+  m.style.height = "0px";
+  return m;
+}
+
+/**
+ * HorizontalMarker class
+ * Create a horizontal marker at the indicated mouse location.
+ * @constructor
+ *
+ * @param canvasRect {Object} The canvas bounds (in client coords).
+ * @param clientY {Number} The vertical mouse click location that spawned
+ *    the marker, in the client coordinate space.
+ * @param yValue {Number} The plotted value corresponding to the clientY
+ *     click location.
+ */
+function HorizontalMarker(canvasRect, clientY, yValue) {
+  // Add a horizontal line to the graph.
+  var m = document.createElement("DIV");
+  m.setAttribute("class", "plot-baseline");
+  m.style.backgroundColor = HorizontalMarker.COLOR;
+  m.style.opacity = "0.3";
+  m.style.position = "absolute";
+  m.style.left = canvasRect.offsetLeft;
+  var h = HorizontalMarker.HEIGHT;
+  m.style.top = (clientY - h/2).toFixed(0) + "px";
+  m.style.width = canvasRect.width + "px";
+  m.style.height = h + "px";
+  this.markerDiv_ = m;
+
+  this.value = yValue;
+}
+
+HorizontalMarker.HEIGHT = 5;
+HorizontalMarker.COLOR = "rgb(0,100,100)";
+
+// Remove the horizontal line from the graph.
+HorizontalMarker.prototype.remove_ = function() {
+  this.markerDiv_.parentNode.removeChild(this.markerDiv_);
+}
+
+/**
+ * Plotter class
+ * @constructor
+ *
+ * Draws a chart using CANVAS element. Takes array of lines to draw with
+ * deviations values for each data sample.
+ *
+ * @param {Array} clNumbers list of clNumbers for each data sample.
+ * @param {Array} plotData list of arrays that represent individual lines.
+ *                         The line itself is an Array of value and stdd.
+ * @param {Array} dataDescription list of data description for each line
+ *                         in plotData.
+ * @units {string} units name of measurement used to describe plotted data.
+ *
+ * Example of the plotData:
+ *  [
+ *    [line 1 data],
+ *    [line 2 data]
+ *  ].
+ *  Line data looks like  [[point one], [point two]].
+ *  And individual points are [value, deviation value]
+ */
+function Plotter(clNumbers, plotData, dataDescription, units, resultNode) {
+  this.clNumbers_ = clNumbers;
+  this.plotData_ = plotData;
+  this.dataDescription_ = dataDescription;
+  this.resultNode_ = resultNode;
+  this.units_ = units;
+  this.coordinates = new Coordinates(plotData);
+
+  // A color palette that's unambigous for normal and color-deficient viewers.
+  // Values are (red, green, blue) on a scale of 255.
+  // Taken from http://jfly.iam.u-tokyo.ac.jp/html/manuals/pdf/color_blind.pdf
+  this.colors = [[0, 114, 178],   // blue
+                 [230, 159, 0],   // orange
+                 [0, 158, 115],   // green
+                 [204, 121, 167], // purplish pink
+                 [86, 180, 233],  // sky blue
+                 [213, 94, 0],    // dark orange
+                 [0, 0, 0],       // black
+                 [240, 228, 66]   // yellow
+                ];
+}
+
+/**
+ * Does the actual plotting.
+ */
+Plotter.prototype.plot = function() {
+  var canvas = this.canvas();
+  this.coordinates_div_ = this.coordinates_();
+  this.ruler_div_ = this.ruler();
+  // marker for the result-point that the mouse is currently over
+  this.cursor_div_ = new Marker("rgb(100,80,240)");
+  // marker for the result-point for which details are shown
+  this.marker_div_ = new Marker("rgb(100,100,100)");
+  var ctx = canvas.getContext("2d");
+  for (var i = 0; i < this.plotData_.length; i++)
+    this.plotLine_(ctx, this.nextColor(i), this.plotData_[i]);
+
+  this.resultNode_.appendChild(canvas);
+  this.resultNode_.appendChild(this.coordinates_div_);
+
+  this.resultNode_.appendChild(this.ruler_div_);
+  this.resultNode_.appendChild(this.cursor_div_);
+  this.resultNode_.appendChild(this.marker_div_);
+  this.attachEventListeners(canvas);
+  this.canvasRectangle = {
+    "offsetLeft": canvas.offsetLeft,
+    "offsetTop":  canvas.offsetTop,
+    "width":      canvas.offsetWidth,
+    "height":     canvas.offsetHeight
+  };
+};
+
+Plotter.prototype.drawDeviationBar_ = function(context, strokeStyles, x, y,
+                                                deviationValue) {
+  context.strokeStyle = strokeStyles;
+  context.lineWidth = 1.0;
+  context.beginPath();
+  context.moveTo(x, (y + deviationValue));
+  context.lineTo(x, (y - deviationValue));
+  context.moveTo(x, (y - deviationValue));
+  context.closePath();
+  context.stroke();
+};
+
+Plotter.prototype.plotLine_ = function(ctx, strokeStyles, data) {
+  ctx.strokeStyle = strokeStyles;
+  ctx.lineWidth = 2.0;
+  ctx.beginPath();
+  var initial = true;
+  var deviationData = [];
+  for (var i = 0; i < data.length; i++) {
+    var x = this.coordinates.xPoints(i);
+    var value = data[i][0];
+    var stdd = data[i][1];
+    var y = 0.0;
+    var err = 0.0;
+    if (isNaN(value)) {
+      // Re-set 'initial' if we're at a gap in the data.
+      initial = true;
+    } else {
+      y = this.coordinates.yPoints(value);
+      // We assume that the stdd will only be NaN (missing) when the value is.
+      if (parseFloat(value) != 0.0)
+        err = y * parseFloat(stdd) / parseFloat(value);
+      if (initial)
+        initial = false;
+      else
+        ctx.lineTo(x, y);
+    }
+
+    ctx.moveTo(x, y);
+    deviationData.push([x, y, err])
+  }
+  ctx.closePath();
+  ctx.stroke();
+
+  for (var i = 0; i < deviationData.length; i++) {
+    this.drawDeviationBar_(ctx, strokeStyles, deviationData[i][0],
+                            deviationData[i][1], deviationData[i][2]);
+  }
+};
+
+Plotter.prototype.attachEventListeners = function(canvas) {
+  var self = this;
+  canvas.parentNode.addEventListener(
+    "mousemove", function(evt) { self.onMouseMove_(evt); }, false);
+  this.cursor_div_.addEventListener(
+    "click", function(evt) { self.onMouseClick_(evt); }, false);
+};
+
+Plotter.prototype.updateRuler_ = function(evt) {
+  var r = this.ruler_div_;
+  r.style.left = this.canvasRectangle.offsetLeft + "px";
+
+  r.style.top = this.canvasRectangle.offsetTop + "px";
+  r.style.width = this.canvasRectangle.width + "px";
+  var h = evt.clientY - this.canvasRectangle.offsetTop;
+  if (h > this.canvasRectangle.height)
+    h = this.canvasRectangle.height;
+  r.style.height = h + "px";
+};
+
+Plotter.prototype.updateCursor_ = function() {
+  var c = this.cursor_div_;
+  c.style.top = this.canvasRectangle.offsetTop + "px";
+  c.style.height = this.canvasRectangle.height + "px";
+  var w = this.canvasRectangle.width / this.clNumbers_.length;
+  var x = (this.canvasRectangle.offsetLeft +
+            w * this.current_index_).toFixed(0);
+  c.style.left = x + "px";
+  c.style.width = w + "px";
+};
+
+
+Plotter.prototype.onMouseMove_ = function(evt) {
+  var canvas = evt.currentTarget.firstChild;
+  var positionX = evt.clientX - this.canvasRectangle.offsetLeft;
+  var positionY = evt.clientY - this.canvasRectangle.offsetTop;
+
+  this.current_index_ = this.coordinates.dataSampleIndex(positionX);
+  var yValue = this.coordinates.yValue(positionY);
+
+  this.coordinates_td_.innerHTML =
+      "r" + this.clNumbers_[this.current_index_] + ": " +
+      this.plotData_[0][this.current_index_][0].toFixed(2) + " " +
+      this.units_ + " +/- " +
+      this.plotData_[0][this.current_index_][1].toFixed(2) + " " +
+      yValue.toFixed(2) + " " + this.units_;
+
+  // If there is a horizontal marker, also display deltas relative to it.
+  if (this.horizontal_marker_) {
+    var baseline = this.horizontal_marker_.value;
+    var delta = yValue - baseline
+    var fraction = delta / baseline; // allow division by 0
+
+    var deltaStr = (delta >= 0 ? "+" : "") + delta.toFixed(0) + " " +
+        this.units_;
+    var percentStr = (fraction >= 0 ? "+" : "") +
+        (fraction * 100).toFixed(3) + "%";
+
+    this.baseline_deltas_td_.innerHTML = deltaStr + ": " + percentStr;
+  }
+
+  this.updateRuler_(evt);
+  this.updateCursor_();
+};
+
+Plotter.prototype.onMouseClick_ = function(evt) {
+  // Shift-click controls the horizontal reference line.
+  if (evt.shiftKey) {
+    if (this.horizontal_marker_) {
+      this.horizontal_marker_.remove_();
+    }
+
+    var canvasY = evt.clientY - this.canvasRectangle.offsetTop;
+    this.horizontal_marker_ = new HorizontalMarker(this.canvasRectangle,
+        evt.clientY, this.coordinates.yValue(canvasY));
+
+    // Insert before cursor node, otherwise it catches clicks.
+    this.cursor_div_.parentNode.insertBefore(
+        this.horizontal_marker_.markerDiv_, this.cursor_div_);
+  } else {
+    var index = this.current_index_;
+    var m = this.marker_div_;
+    var c = this.cursor_div_;
+    m.style.top = c.style.top;
+    m.style.left = c.style.left;
+    m.style.width = c.style.width;
+    m.style.height = c.style.height;
+    if ("onclick" in this) {
+      var this_x = this.clNumbers_[index];
+      var prev_x = index > 0 ? (parseInt(this.clNumbers_[index-1]) + 1) :
+                                this_x;
+      this.onclick(prev_x, this_x);
+    }
+  }
+};
+
+Plotter.prototype.canvas = function() {
+  var canvas = document.createElement("CANVAS");
+  canvas.setAttribute("id", "_canvas");
+  canvas.setAttribute("class", "plot");
+  canvas.setAttribute("width", this.coordinates.widthMax);
+  canvas.setAttribute("height", this.coordinates.heightMax);
+  return canvas;
+};
+
+Plotter.prototype.ruler = function() {
+  ruler = document.createElement("DIV");
+  ruler.setAttribute("class", "plot-ruler");
+  ruler.style.borderBottom = "1px dotted black";
+  ruler.style.position = "absolute";
+  ruler.style.left = "-2px";
+  ruler.style.top = "-2px";
+  ruler.style.width = "0px";
+  ruler.style.height = "0px";
+  return ruler;
+};
+
+Plotter.prototype.coordinates_ = function() {
+  var coordinatesDiv = document.createElement("DIV");
+  var table_html =
+     "<table border=0 width='100%'><tbody><tr>" +
+     "<td colspan=2 class='legend'>Legend: ";
+  for (var i = 0; i < this.dataDescription_.length; i++) {
+    if (i > 0)
+      table_html += ", ";
+    table_html += "<span class='legend_item' style='color:" +
+      this.nextColor(i) + "'>" + this.dataDescription_[i] + "</span>";
+  }
+  table_html += "</td></tr><tr>" +
+     "<td class='plot-coordinates'><i>move mouse over graph</i></td>" +
+     "<td align=right style='color: " + HorizontalMarker.COLOR +
+     "'><i>Shift-click to place baseline</i></td>" +
+     "</tr></tbody></table>";
+  coordinatesDiv.innerHTML = table_html;
+
+  var tr = coordinatesDiv.firstChild.firstChild.childNodes[1];
+  this.coordinates_td_ = tr.childNodes[0];
+  this.baseline_deltas_td_ = tr.childNodes[1];
+
+  return coordinatesDiv;
+};
+
+Plotter.prototype.nextColor = function(i) {
+  var index = i % this.colors.length;
+  return "rgb(" + this.colors[index][0] + "," +
+                  this.colors[index][1] + "," +
+                  this.colors[index][2] + ")";
+};
+
+Plotter.prototype.log = function(val) {
+  document.getElementById('log').appendChild(
+    document.createTextNode(val + '\n'));
+};