canvas drawImage vs. WebGL drawArrays

JavaScript performance comparison

Test case created by Andor Salga

Info

I wanted to experiment how much faster WebGL can draw sprites compared to canvas.

Preparation code

<canvas id="cvs2d" width="1024" height="768"></canvas>
<canvas id="cvs3d" width="1024" height="768"></canvas>
<script>
Benchmark.prototype.setup = function() {
    var img, tex, vloc, tloc,vertexBuff, texBuff;
   
    var cvs2d = document.getElementById('cvs2d');
    var cvs3d = document.getElementById('cvs3d');
    var ctx2d = cvs2d.getContext('2d');
    var ctx3d = cvs3d.getContext('experimental-webgl');
   
   
    // create shaders
    var vertexShaderSrc =
    "attribute vec2 aVertex;" +
    "attribute vec2 aUV;" +
    "varying vec2 vTex;" +
    "void main(void) {" +
    "  gl_Position = vec4(aVertex, 0.0, 1.0);" +
    "  vTex = aUV;" +
    "}";
   
    var fragmentShaderSrc =
    "precision highp float;" +
    "varying vec2 vTex;" +
    "uniform sampler2D sampler0;" +
    "void main(void){" +
    "  gl_FragColor = texture2D(sampler0, vTex);"+
    "}";
   
    var vertShaderObj = ctx3d.createShader(ctx3d.VERTEX_SHADER);
    var fragShaderObj = ctx3d.createShader(ctx3d.FRAGMENT_SHADER);
    ctx3d.shaderSource(vertShaderObj, vertexShaderSrc);
    ctx3d.shaderSource(fragShaderObj, fragmentShaderSrc);
    ctx3d.compileShader(vertShaderObj);
    ctx3d.compileShader(fragShaderObj);
   
    var progObj = ctx3d.createProgram();
    ctx3d.attachShader(progObj, vertShaderObj);
    ctx3d.attachShader(progObj, fragShaderObj);
   
    ctx3d.linkProgram(progObj);
    ctx3d.useProgram(progObj);
   
    ctx3d.viewport(0, 0, 1024, 768);
   
    vertexBuff = ctx3d.createBuffer();
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
    ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([-1/8, 1/6, -1/8, -1/6, 1/8, -1/6, 1/8, 1/6]), ctx3d.STATIC_DRAW);
   
    texBuff = ctx3d.createBuffer();
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
    ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([0, 1, 0, 0, 1, 0, 1, 1]), ctx3d.STATIC_DRAW);
   
    vloc = ctx3d.getAttribLocation(progObj, "aVertex");
    tloc = ctx3d.getAttribLocation(progObj, "aUV");
   
    img = new Image();
    img.src = '%3D';
   
    img.onload = function(){
      tex = ctx3d.createTexture();
      ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
      ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MIN_FILTER, ctx3d.NEAREST);
      ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MAG_FILTER, ctx3d.NEAREST);
      ctx3d.generateMipmap(ctx3d.TEXTURE_2D);
      ctx3d.texImage2D(ctx3d.TEXTURE_2D, 0,  ctx3d.RGBA,  ctx3d.RGBA, ctx3d.UNSIGNED_BYTE, this);
   
    ctx3d.enableVertexAttribArray(vloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
    ctx3d.vertexAttribPointer(vloc, 2, ctx3d.FLOAT, false, 0, 0);
   
    ctx3d.enableVertexAttribArray(tloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
    ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
    ctx3d.vertexAttribPointer(tloc, 2, ctx3d.FLOAT, false, 0, 0);
    };
};
</script>

Preparation code output

Test runner

Warning! For accurate results, please disable Firebug before running the tests. (Why?)

Java applet disabled.

Testing in unknown unknown
Test Ops/sec
canvas drawImage
ctx2d.drawImage(img, 0, 0);
pending…
WebGL drawArrays
ctx3d.drawArrays(ctx3d.TRIANGLE_FAN, 0, 4);
pending…

Compare results of other browsers

Revisions

You can edit these tests or add even more tests to this page by appending /edit to the URL. Here’s a list of current revisions for this page:

1 comment

joshua commented :

Came across this post, it interesting... what do you think renders better quality, using canvas i seem to get more so jagged edges when objects not in motion?

Anyways cool

Add a comment