Bug 1620642 - reftest-analyzer improvements for analyzing test failures r=jgilbert
authorBert Peers <bpeers@mozilla.com>
Mon, 09 Mar 2020 19:45:11 +0000
changeset 517711 263ce25c220aeab11843445e3634f29b004f44dc
parent 517710 cc06b305d9e44a019fbed3cafbb66fa7e35efe4f
child 517712 13f39411b04da3765b9f68824882f0024440b945
push id109572
push userbpeers@mozilla.com
push dateMon, 09 Mar 2020 23:41:47 +0000
treeherderautoland@263ce25c220a [default view] [failures only]
perfherder[talos] [build metrics] [platform microbench] (compared to previous push)
reviewersjgilbert
bugs1620642
milestone76.0a1
first release with
nightly linux32
nightly linux64
nightly mac
nightly win32
nightly win64
last release without
nightly linux32
nightly linux64
nightly mac
nightly win32
nightly win64
Bug 1620642 - reftest-analyzer improvements for analyzing test failures r=jgilbert Turn the difference checkbox into a radio that adds "heatmap"; it uses WebGL to show both images, their absolute difference, and a color-coded max difference. The quadrants split following the mouse. This helps to separate large variations (red) from small variations (green) and helps to compare the images without losing track of where they are. Differential Revision: https://phabricator.services.mozilla.com/D65841
gfx/wr/wrench/script/reftest-analyzer.xhtml
--- a/gfx/wr/wrench/script/reftest-analyzer.xhtml
+++ b/gfx/wr/wrench/script/reftest-analyzer.xhtml
@@ -62,16 +62,171 @@ Features to add:
   #pixelhint:hover { color: #000; }
   #pixelhint:hover > * { display: block; }
   #pixelhint p { margin: 0; }
   #pixelhint p + p { margin-top: 1em; }
 
   ]]></style>
   <script type="text/javascript"><![CDATA[
 
+let heatmapCanvas = null;
+let heatmapUMouse;
+let gl = null;
+
+function heatmap_render_setup(canvas) {
+  gl = canvas.getContext('webgl', {antialias: false, depth: false, preserveDrawingBuffer:false});
+
+  const vertices = [
+    0, 0,
+    1, 0,
+    0, 1,
+    1, 1,
+  ];
+
+  const vertexBuffer = gl.createBuffer();
+  gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
+  gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);
+
+  const vsCode =
+    `
+    attribute vec2 a_vertCoord;
+    varying   vec2 v_texCoord;
+    void main(void) {
+      gl_Position = vec4(2.0 * a_vertCoord - 1.0, 0.0, 1.0);
+      v_texCoord = a_vertCoord;
+    }`;
+
+  const VS = gl.createShader(gl.VERTEX_SHADER);
+  gl.shaderSource(VS, vsCode);
+  gl.compileShader(VS);
+
+  const psCode =
+    `
+    precision mediump float;
+    uniform vec2 heatmapUMouse;
+    varying vec2 v_texCoord;
+    uniform sampler2D u_image1, u_image2;
+    void main(void) {
+      vec2 dxy = abs(heatmapUMouse - gl_FragCoord.xy);
+      if(dxy.x < 1.0 || dxy.y < 1.0) {  // crosshair
+        gl_FragColor = vec4( 1.0, 1.0, 0.5, 1.0 );
+        return;
+      }
+
+      vec3 img1 = texture2D(u_image1, v_texCoord).rgb;
+      vec3 img2 = texture2D(u_image2, v_texCoord).rgb;
+
+      bool is_top  = gl_FragCoord.y > float(heatmapUMouse.y);
+      bool is_left = gl_FragCoord.x < float(heatmapUMouse.x);
+
+      vec3 rgb;
+      if(is_top) {
+        if(is_left) {
+          rgb = img1;
+        } else {
+          rgb = img2;
+        }
+      } else {
+        vec3 diff = abs(img1 - img2);
+        if(is_left) {
+          rgb = diff;
+        } else {
+          float max_diff = max(diff.r, max(diff.g, diff.b));
+          if(max_diff == 0.0) {
+            rgb = vec3(0.0, 0.0, 0.2);
+          } else {
+            // some arbitrary colorization -- transition from green to red
+            // with some contrast tweaks to make red stand out a bit more
+            // at about 0.5'ish
+            rgb = vec3( pow(max_diff, 0.5), pow(1.0 - max_diff, 3.0), 0.0 );
+          }
+        }
+      }
+
+      gl_FragColor = vec4( rgb, 1.0 );
+    }`;
+
+  const FS = gl.createShader(gl.FRAGMENT_SHADER);
+  gl.shaderSource(FS, psCode);
+  gl.compileShader(FS);
+
+  const program = gl.createProgram();
+  gl.attachShader(program, VS);
+  gl.attachShader(program, FS);
+  gl.linkProgram(program);
+  if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
+    console.error('Link failed: ' + gl.getProgramInfoLog(program));
+    console.error('vs info-log: ' + gl.getShaderInfoLog(VS));
+    console.error('fs info-log: ' + gl.getShaderInfoLog(FS));
+    return;  // don't assign heatmapCanvas
+  }
+  gl.useProgram(program);
+
+  gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
+
+  const coord = gl.getAttribLocation(program, "a_vertCoord");
+  gl.vertexAttribPointer(coord, 2, gl.FLOAT, false, 0, 0);
+  gl.enableVertexAttribArray(coord);
+
+  heatmapUMouse = gl.getUniformLocation(program, "heatmapUMouse");
+
+  gl.uniform1i(gl.getUniformLocation(program, 'u_image1'), 0);
+  gl.uniform1i(gl.getUniformLocation(program, 'u_image2'), 1);
+
+  gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, true);
+  heatmapCanvas = canvas;
+}
+
+function heatmap_change_image(index, image) {
+  if (heatmapCanvas === null) {
+    return;
+  }
+  const texture = gl.createTexture();
+  gl.activeTexture(gl.TEXTURE0 + index);
+  gl.bindTexture  (gl.TEXTURE_2D, texture);
+  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
+  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
+  gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
+  gl.texImage2D   (gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
+}
+
+function heatmap_render(mouse_x, mouse_y) {
+  if (heatmapCanvas === null) {
+    return;
+  }
+
+  gl.uniform2f(heatmapUMouse, mouse_x, mouse_y);
+
+  // the canvas resizes as user selects different reftests
+  gl.viewport(0, 0, heatmapCanvas.width, heatmapCanvas.height);
+
+  gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);
+}
+
+function heatmap_on_mousemove(mousemove_event) {
+  if (heatmapCanvas === null) {
+    return;
+  }
+  const rect = heatmapCanvas.getBoundingClientRect();
+  let x = mousemove_event.clientX - rect.left;
+  let y = mousemove_event.clientY - rect.top;
+  x = x * heatmapCanvas.width  / heatmapCanvas.clientWidth;
+  y = y * heatmapCanvas.height / heatmapCanvas.clientHeight;
+
+  // mouse has Y == 0 at the top, GL has it at the bottom:
+  const flip_y = heatmapCanvas.height-1 - y;
+  heatmap_render(x, flip_y);
+
+  return { x:x, y:y };
+}
+
+  ]]></script>
+
+  <script type="text/javascript"><![CDATA[
+
 var XLINK_NS = "http://www.w3.org/1999/xlink";
 var SVG_NS = "http://www.w3.org/2000/svg";
 var IMAGE_NOT_AVAILABLE = "";
 
 var gPhases = null;
 
 var gIDCache = {};
 
@@ -337,31 +492,37 @@ function get_image_data(src, whenReady) 
 function sync_svg_size(imageData) {
   // We need the size of the 'svg' and its 'image' elements to match the size
   // of the ImageData objects that we're going to read pixels from or else our
   // magnify() function will be very broken.
   ID("svg").setAttribute("width", imageData.width);
   ID("svg").setAttribute("height", imageData.height);
 }
 
+function sync_heatmap_size(imageData) {
+  ID("heat_canvas").setAttribute("width" , imageData.width);
+  ID("heat_canvas").setAttribute("height", imageData.height);
+}
+
 function show_images(i) {
   var item = gTestItems[i];
   var cell = ID("images");
 
   // Remove activeitem class from any existing elements
   var activeItems = document.querySelectorAll(".activeitem");
   for (var activeItemIdx = activeItems.length; activeItemIdx-- != 0;) {
     activeItems[activeItemIdx].classList.remove("activeitem");
   }
 
   ID("item" + i).classList.add("activeitem");
   ID("image1").style.display = "";
   ID("image2").style.display = "none";
-  ID("diffrect").style.display = "none";
+  show_diff_none();
   ID("imgcontrols").reset();
+  ID("diffcontrols").reset();
 
   ID("image1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]);
   // Making the href be #image1 doesn't seem to work
   ID("feimage1").setAttributeNS(XLINK_NS, "xlink:href", item.images[0]);
   if (item.images.length == 1) {
     ID("imgcontrols").style.display = "none";
   } else {
     ID("imgcontrols").style.display = "";
@@ -371,18 +532,18 @@ function show_images(i) {
     ID("feimage2").setAttributeNS(XLINK_NS, "xlink:href", item.images[1]);
 
     ID("label1").textContent = 'Image ' + item.imageLabels[0];
     ID("label2").textContent = 'Image ' + item.imageLabels[1];
   }
 
   cell.style.display = "";
 
-  get_image_data(item.images[0], function(data) { gImage1Data = data; sync_svg_size(gImage1Data); });
-  get_image_data(item.images[1], function(data) { gImage2Data = data });
+  get_image_data(item.images[0], function(data) { gImage1Data = data; sync_svg_size(gImage1Data); sync_heatmap_size(gImage1Data); heatmap_change_image(0, gImage1Data); });
+  get_image_data(item.images[1], function(data) { gImage2Data = data; heatmap_change_image(1, gImage2Data); });
 }
 
 function show_image(i) {
   if (i == 1) {
     ID("image1").style.display = "";
     ID("image2").style.display = "none";
   } else {
     ID("image1").style.display = "none";
@@ -396,17 +557,20 @@ function handle_keyboard_shortcut(event)
     document.getElementById("radio1").checked = true;
     show_image(1);
     break;
   case 50: // "2" key
     document.getElementById("radio2").checked = true;
     show_image(2);
     break;
   case 100: // "d" key
-    document.getElementById("differences").click();
+    document.getElementById("radio_diff_circle").click();
+    break;
+  case 104: // "h" key
+    document.getElementById("radio_diff_heatmap").click();
     break;
   case 112: // "p" key
     shift_images(-1);
     break;
   case 110: // "n" key
     shift_images(1);
     break;
   }
@@ -424,18 +588,45 @@ function shift_images(dir) {
     elm = dir > 0 ? elm.nextElementSibling : elm.previousElementSibling;
     if (elm) {
       elm.getElementsByTagName("a")[0].click();
     }
     return;
   }
 }
 
-function show_differences(cb) {
-  ID("diffrect").style.display = cb.checked ? "" : "none";
+function show_diff_none() {
+  ID("svg")        .style.display = "";
+  ID("diffrect")   .style.display = "none";
+  ID("heat_canvas").style.display = "none";
+}
+
+function show_diff_circle() {
+  ID("svg")        .style.display = "";
+  ID("diffrect")   .style.display = "";
+  ID("heat_canvas").style.display = "none";
+}
+
+function show_diff_heatmap() {
+  ID("svg")        .style.display = "none";
+  ID("diffrect")   .style.display = "none";
+  ID("heat_canvas").style.display = "";
+
+  if (heatmapCanvas === null) {
+    canvas = document.getElementById('heat_canvas');
+    heatmap_render_setup(canvas);
+    heatmap_change_image(0, gImage1Data);
+    heatmap_change_image(1, gImage2Data);
+    heatmap_render(0, 0);
+
+    window.addEventListener('mousemove', e => {
+      var { x: x, y: y } = heatmap_on_mousemove(e);
+      magnify_around(Math.floor(x), Math.floor(y));
+    });
+  }
 }
 
 function flash_pixels(on) {
   var stroke = on ? "red" : "black";
   var strokeWidth = on ? "2px" : "1px";
   for (var i = 0; i < gFlashingPixels.length; i++) {
     gFlashingPixels[i].setAttribute("stroke", stroke);
     gFlashingPixels[i].setAttribute("stroke-width", strokeWidth);
@@ -464,16 +655,23 @@ function canvas_pixel_as_hex(data, x, y)
 }
 
 function hex_as_rgb(hex) {
   return "rgb(" + [parseInt(hex.substring(1, 3), 16), parseInt(hex.substring(3, 5), 16), parseInt(hex.substring(5, 7), 16)] + ")";
 }
 
 function magnify(evt) {
   var { x: x, y: y } = cursor_point(evt);
+  magnify_around(x, y);
+}
+
+function magnify_around(x, y) {
+  if (x < 0 || y < 0 || x >= gImage1Data.width || y >= gImage1Data.height) {
+    return;
+  }
   var centerPixelColor1, centerPixelColor2;
 
   var dx_lo = -Math.floor(gMagWidth / 2);
   var dx_hi = Math.floor(gMagWidth / 2);
   var dy_lo = -Math.floor(gMagHeight / 2);
   var dy_hi = Math.floor(gMagHeight / 2);
 
   flash_pixels(false);
@@ -567,18 +765,30 @@ function show_pixelinfo(x, y, pix1rgb, p
       </svg>
     </div>
   </div>
   <div id="itemlist"></div>
   <div id="images" style="display:none">
     <form id="imgcontrols">
     <input id="radio1" type="radio" name="which" value="0" onchange="show_image(1)" checked="checked" /><label id="label1" title="1" for="radio1">Image 1</label>
     <input id="radio2" type="radio" name="which" value="1" onchange="show_image(2)"                   /><label id="label2" title="2" for="radio2">Image 2</label>
-    <label><input id="differences" type="checkbox" onchange="show_differences(this)" />Circle differences</label>
     </form>
+
+    <form id="diffcontrols">
+      Differences:
+      <input id="radio_diff_none"    name="diff" value="0" type="radio" onchange="show_diff_none()" checked="checked"/>
+        <label for="radio_diff_none">None</label>
+      <input id="radio_diff_circle"  name="diff" value="1" type="radio" onchange="show_diff_circle()" />
+        <label for="radio_diff_circle">Circle</label>
+      <input id="radio_diff_heatmap" name="diff" value="2" type="radio" onchange="show_diff_heatmap()" />
+        <label for="radio_diff_heatmap">Heatmap</label>
+    </form>
+
+    <canvas width="800" height="1000" id="heat_canvas" style="display:none;"></canvas>
+
     <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="800" height="1000" id="svg">
       <defs>
         <!-- use sRGB to avoid loss of data -->
         <filter id="showDifferences" x="0%" y="0%" width="100%" height="100%"
                 style="color-interpolation-filters: sRGB">
           <feImage id="feimage1" result="img1" xlink:href="#image1" />
           <feImage id="feimage2" result="img2" xlink:href="#image2" />
           <!-- inv1 and inv2 are the images with RGB inverted -->