WASM Canvas

We have build foundation to write binary WASM code with JavaScript and inspect memory as hexdump in the browser. Now let's visualize the memory via Canvas API.

Blank canvas

chapter_wasm_canvas/blank_canvas.js

(async () => { const canvas = document.getElementById('blank_canvas'); canvas.width = 512; canvas.height = 512; canvas.style.border = '1px solid black'; })();

ImageData of a blank canvas

getImageData

chapter_wasm_canvas/image_data_blank_canvas.js

(async () => { const canvas = document.getElementById('image_data_blank_canvas'); canvas.width = 512; canvas.height = 512; canvas.style.border = '1px solid black'; const ctx = canvas.getContext('2d'); ctx.fillRect(0, 0, canvas.width, canvas.height); const { data } = ctx.getImageData(0, 0, canvas.width, canvas.height); const canvas_data = document.getElementById('image_data_blank_canvas_data'); canvas_data.innerHTML = ` First ${0x100} bytes of the canvas data hexdump({data, length: 0x100 }) ${hexdump({ data, length: 0x100 })} ` })();

First 256 bytes of the canvas data

hexdump({data, length: 0x100 })
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff
00 00 00 ff 00 00 00 ff 00 00 00 ff 00 00 00 ff

ImageData of a green canvas

Now let's make a green canvas.

chapter_wasm_canvas/image_data_green_canvas.js

(async () => { const canvas = document.getElementById('image_data_green_canvas'); const width = 512; const height = 512; canvas.width = width; canvas.height = height; canvas.style.border = '1px solid black'; const ctx = canvas.getContext('2d'); ctx.fillRect(0, 0, width, height); let imageData = ctx.getImageData(0, 0, width, height); for (let i = 0; i < imageData.data.length / 4; i++) { // RRGGBBAA (big-endian) imageData.data[4 * i + 1] = 0xff; imageData.data[4 * i + 3] = 0xff; } ctx.putImageData(imageData, 0, 0); const { data } = imageData; const canvas_data = document.getElementById('image_data_green_canvas_data'); canvas_data.innerHTML = ` First ${0x100} bytes of the canvas data hexdump({data, length: 0x100 }) ${hexdump({ data, length: 0x100 })} ` })();

First 256 bytes of the canvas data

hexdump({data, length: 0x100 })
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff

WASM green canvas

Now let's do it again with WASM.

chapter_wasm_canvas/wasm_image_data_green_canvas.js

(async () => { const canvas = document.getElementById('wasm_image_data_green_canvas'); const width = 512; const height = 512; canvas.width = width; canvas.height = height; canvas.style.border = '1px solid black'; const ctx = canvas.getContext('2d'); ctx.fillRect(0, 0, width, height); let imageData = ctx.getImageData(0, 0, width, height); const magic = [0x00, 0x61, 0x73, 0x6d]; const version = [0x01, 0x00, 0x00, 0x00]; const section_01 = [ 0x01, // type section 0x07, // 7 bytes 0x01, // number of functions 0x60, // function type 0x02, // takes two params 0x7f, 0x7f, // i32 0x01, // return 1 result 0x7f // i32 ]; const section_03 = [ 0x03, // func section 0x02, // 2 bytes 0x01, // number of functions 0x00 // type of the function ]; const section_05 = [ 0x05, // memory section 0x03, // 3 bytes 0x01, // number of memory 0x00, // min 0x10 // max: 16 pages of data ]; const section_07 = [ 0x07, // export section 0x09, // 9 bytes 0x02, // number of exports 0x01, 0x6d, // "m" 0x02, 0x00, 0x01, 0x66, // "f" 0x00, 0x00 ]; const section_0a = [ 0x0a, // code section 0x0d, // 13 bytes 0x01, // number of function bodies. 0x0b, // 11 bytes 0x00, // number of local variables 0x20, // local.get 0x00, // 0 0x20, // local.get 0x01, // 1 0x36, // i32.store 0x02, // align 0x00, // offset 0x20, // local.get 0x01, // 1 (return the stored i32) 0x0b // opcode for end ]; const wasm = new Uint8Array( magic.concat(version) .concat(section_01) .concat(section_03) .concat(section_05) .concat(section_07) .concat(section_0a) ); const importObject = {}; const module = await WebAssembly.compile(wasm.buffer); const instance = await WebAssembly.instantiate(module, importObject); const { exports } = instance; const { f, m } = exports; const memory_data = new Uint8Array(m.buffer); console.log({ imageData, memory_data }) for (let i = 0; i < imageData.data.length / 4; i++) { f(i * 4, 0xff00ff00); // green AAGGBBRR } for (let i = 0; i < imageData.data.length; i++) { imageData.data[i] = memory_data[i]; } ctx.putImageData(imageData, 0, 0); const { data } = imageData; const canvas_data = document.getElementById('wasm_image_data_green_canvas_data'); canvas_data.innerHTML = ` Canva data data.length = ${data.length} = ${data.length / 64 / 1024} pages First ${0x200} bytes of the canvas data hexdump({data, length: 0x200 }) ${hexdump({ data, length: 0x200 })} ` })();


Canva data

data.length = 1048576 = 16 pages 

First 512 bytes of the canvas data

hexdump({data, length: 0x200 })
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff
00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 ff



Setting pixels in imageData via a loop calling an exported WASM function is definitely not an efficient way to draw on a canvas. We just showed that it is possible to pass WASM memory back to canvas.

In the next step, we want to do computation in the WASM and "read" the memory out on Canvas.