Mengompilasi mkbitmap ke WebAssembly

Di Apa itu WebAssembly dan dari mana asalnya?, Saya menjelaskan bagaimana kami berkembang dengan WebAssembly hari ini. Dalam artikel ini, saya akan menunjukkan pendekatan saya untuk mengompilasi program C yang ada, mkbitmap, ke WebAssembly. Contoh ini lebih kompleks daripada contoh hello world, karena mencakup menangani file, berkomunikasi antara webAssembly dan JavaScript, dan menggambar ke kanvas, tetapi masih cukup mudah dikelola agar tidak membuat Anda kewalahan.

Artikel ini ditulis untuk developer web yang ingin mempelajari WebAssembly dan menunjukkan langkah demi langkah cara melanjutkan jika Anda ingin mengompilasi sesuatu seperti mkbitmap ke WebAssembly. Sebagai peringatan yang adil, tidak mendapatkan aplikasi atau library untuk dikompilasi saat pertama kali dijalankan adalah hal yang normal. Itulah sebabnya beberapa langkah yang dijelaskan di bawah ini tidak berhasil, jadi saya perlu melacaknya kembali dan mencoba lagi secara berbeda. Artikel ini tidak menunjukkan perintah kompilasi akhir ajaib seolah-olah telah turun dari langit, tetapi lebih menggambarkan progres saya yang sebenarnya, termasuk beberapa rasa frustrasi.

Tentang mkbitmap

Program mkbitmap C membaca gambar dan menerapkan satu atau beberapa operasi berikut ke gambar tersebut, dalam urutan sebagai berikut: inversi, pemfilteran highpass, penskalaan, dan volume minimum. Setiap operasi dapat dikontrol satu per satu serta diaktifkan atau dinonaktifkan. Penggunaan utama mkbitmap adalah untuk mengonversi gambar berwarna atau hitam putih menjadi format yang cocok sebagai input untuk program lain, terutama program pelacakan potrace yang membentuk dasar SVGcode. Sebagai alat prapemrosesan, mkbitmap sangat berguna untuk mengonversi gambar garis yang dipindai, seperti kartun atau teks tulisan tangan, menjadi gambar dua level beresolusi tinggi.

Anda menggunakan mkbitmap dengan meneruskan sejumlah opsi dan satu atau beberapa nama file. Untuk mengetahui detailnya, lihat halaman utama alat ini:

$ mkbitmap [options] [filename...]
Gambar kartun berwarna.
Gambar asli (Sumber).
Gambar kartun dikonversi ke hitam putih setelah pra-pemrosesan.
Pertama diskalakan, lalu ditetapkan ke nilai minimum: mkbitmap -f 2 -s 2 -t 0.48 (Sumber).

Mendapatkan kode

Langkah pertama adalah mendapatkan kode sumber mkbitmap. Anda dapat menemukannya di situs project. Pada saat penulisan ini, potrace-1.16.tar.gz adalah versi terbaru.

Mengompilasi dan menginstal secara lokal

Langkah berikutnya adalah mengompilasi dan menginstal alat secara lokal untuk membiasakan diri dengan perilakunya. File INSTALL berisi petunjuk berikut:

  1. cd ke direktori yang berisi kode sumber paket dan ketik ./configure untuk mengonfigurasi paket bagi sistem Anda.

    Menjalankan configure mungkin memerlukan waktu beberapa saat. Saat dijalankan, server akan mencetak beberapa pesan yang memberitahukan fitur mana yang diperiksa.

  2. Ketik make untuk mengompilasi paket.

  3. Secara opsional, ketik make check untuk menjalankan pengujian mandiri yang disertakan dengan paket, umumnya menggunakan biner yang di-uninstal yang baru saja dibuat.

  4. Ketik make install untuk menginstal program serta file data dan dokumentasi apa pun. Saat menginstal ke awalan yang dimiliki oleh root, sebaiknya paket dikonfigurasi dan dibangun sebagai pengguna biasa, dan hanya fase make install yang dijalankan dengan hak istimewa root.

Dengan mengikuti langkah-langkah ini, Anda akan memiliki dua file yang dapat dieksekusi, potrace dan mkbitmap, yang merupakan fokus artikel ini. Anda dapat memverifikasi bahwa kode tersebut berfungsi dengan benar dengan menjalankan mkbitmap --version. Berikut ini output dari keempat langkah dari mesin saya, yang dipangkas agar lebih singkat:

Langkah 1, ./configure:

 $ ./configure
checking for a BSD-compatible install... /usr/bin/install -c
checking whether build environment is sane... yes
checking for a thread-safe mkdir -p... ./install-sh -c -d
checking for gawk... no
checking for mawk... no
checking for nawk... no
checking for awk... awk
checking whether make sets $(MAKE)... yes
[…]
config.status: executing libtool commands

Langkah 2, make:

$ make
/Applications/Xcode.app/Contents/Developer/usr/bin/make  all-recursive
Making all in src
clang -DHAVE_CONFIG_H -I. -I..     -g -O2 -MT main.o -MD -MP -MF .deps/main.Tpo -c -o main.o main.c
mv -f .deps/main.Tpo .deps/main.Po
[…]
make[2]: Nothing to be done for `all-am'.

Langkah 3, make check:

$ make check
Making check in src
make[1]: Nothing to be done for `check'.
Making check in doc
make[1]: Nothing to be done for `check'.
[…]
============================================================================
Testsuite summary for potrace 1.16
============================================================================
# TOTAL: 8
# PASS:  8
# SKIP:  0
# XFAIL: 0
# FAIL:  0
# XPASS: 0
# ERROR: 0
============================================================================
make[1]: Nothing to be done for `check-am'.

Langkah 4, sudo make install:

$ sudo make install
Password:
Making install in src
 .././install-sh -c -d '/usr/local/bin'
  /bin/sh ../libtool   --mode=install /usr/bin/install -c potrace mkbitmap '/usr/local/bin'
[…]
make[2]: Nothing to be done for `install-data-am'.

Untuk memeriksa apakah kode tersebut berfungsi, jalankan mkbitmap --version:

$ mkbitmap --version
mkbitmap 1.16. Copyright (C) 2001-2019 Peter Selinger.

Jika Anda mendapatkan detail versi, berarti Anda telah berhasil mengompilasi dan menginstal mkbitmap. Selanjutnya, buat langkah-langkah yang setara dengan WebAssembly.

Mengompilasi mkbitmap ke WebAssembly

Emscripten adalah alat untuk mengompilasi program C/C++ ke WebAssembly. Dokumentasi Building Project Emscripten menyatakan hal berikut:

Membangun project besar dengan Emscripten sangat mudah. Emscripten menyediakan dua skrip sederhana yang mengonfigurasi makefile Anda untuk menggunakan emcc sebagai pengganti langsung untuk gcc—dalam sebagian besar kasus, sistem build project Anda saat ini tidak berubah.

Dokumentasi kemudian berlanjut (sedikit diedit):

Pertimbangkan kasus saat Anda biasanya mem-build dengan perintah berikut:

./configure
make

Untuk membuat aplikasi dengan Emscripten, sebaiknya gunakan perintah berikut:

emconfigure ./configure
emmake make

Jadi, pada dasarnya ./configure menjadi emconfigure ./configure dan make menjadi emmake make. Berikut ini cara melakukannya dengan mkbitmap.

Langkah 0, make clean:

$ make clean
Making clean in src
 rm -f potrace mkbitmap
test -z "" || rm -f
rm -rf .libs _libs
[…]
rm -f *.lo

Langkah 1, emconfigure ./configure:

$ emconfigure ./configure
configure: ./configure
checking for a BSD-compatible install... /usr/bin/install -c
checking whether build environment is sane... yes
checking for a thread-safe mkdir -p... ./install-sh -c -d
checking for gawk... no
checking for mawk... no
checking for nawk... no
checking for awk... awk
[…]
config.status: executing libtool commands

Langkah 2, emmake make:

$ emmake make
make: make
/Applications/Xcode.app/Contents/Developer/usr/bin/make  all-recursive
Making all in src
/opt/homebrew/Cellar/emscripten/3.1.36/libexec/emcc -DHAVE_CONFIG_H -I. -I..     -g -O2 -MT main.o -MD -MP -MF .deps/main.Tpo -c -o main.o main.c
mv -f .deps/main.Tpo .deps/main.Po
[…]
make[2]: Nothing to be done for `all'.

Jika semuanya berjalan dengan baik, seharusnya sudah ada .wasm file di suatu tempat dalam direktori. Anda dapat menemukannya dengan menjalankan find . -name "*.wasm":

$ find . -name "*.wasm"
./a.wasm
./src/mkbitmap.wasm
./src/potrace.wasm

Dua yang terakhir terlihat menjanjikan, jadi cd masuk ke direktori src/. Sekarang juga ada dua file baru yang sesuai, mkbitmap dan potrace. Untuk artikel ini, hanya mkbitmap yang relevan. Fakta bahwa mereka tidak memiliki ekstensi .js sedikit membingungkan, tetapi sebenarnya mereka adalah file JavaScript, dapat diverifikasi dengan panggilan head cepat:

$ cd src/
$ head -n 20 mkbitmap
// include: shell.js
// The Module object: Our interface to the outside world. We import
// and export values on it. There are various ways Module can be used:
// 1. Not defined. We create it here
// 2. A function parameter, function(Module) { ..generated code.. }
// 3. pre-run appended it, var Module = {}; ..generated code..
// 4. External script tag defines var Module.
// We need to check if Module already exists (e.g. case 3 above).
// Substitution will be replaced with actual code on later stage of the build,
// this way Closure Compiler will not mangle it (e.g. case 4. above).
// Note that if you want to run closure, and also to use Module
// after the generated code, you will need to define   var Module = {};
// before the code. Then that object will be used in the code, and you
// can continue to use Module afterwards as well.
var Module = typeof Module != 'undefined' ? Module : {};

// --pre-jses are emitted after the Module integration code, so that they can
// refer to Module (if they choose; they can also define Module)

Ganti nama file JavaScript menjadi mkbitmap.js dengan memanggil mv mkbitmap mkbitmap.js (dan mv potrace potrace.js masing-masing jika Anda mau). Sekarang saatnya melakukan pengujian pertama untuk melihat apakah kode tersebut berfungsi dengan menjalankan file menggunakan Node.js di command line dengan menjalankan node mkbitmap.js --version:

$ node mkbitmap.js --version
mkbitmap 1.16. Copyright (C) 2001-2019 Peter Selinger.

Anda telah berhasil mengompilasi mkbitmap ke WebAssembly. Sekarang langkah berikutnya adalah membuatnya berfungsi di browser.

mkbitmap dengan WebAssembly di browser

Salin file mkbitmap.js dan mkbitmap.wasm ke direktori baru yang bernama mkbitmap dan buat file boilerplate HTML index.html yang memuat file JavaScript mkbitmap.js.

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="utf-8" />
    <title>mkbitmap</title>
  </head>
  <body>
    <script src="mkbitmap.js"></script>
  </body>
</html>

Mulai server lokal yang melayani direktori mkbitmap dan buka di browser Anda. Anda akan melihat perintah yang meminta input. Hal ini sesuai dengan yang diharapkan, karena, sesuai dengan halaman utama alat, "[i]jika tidak ada argumen nama file yang diberikan, maka mkbitmap bertindak sebagai filter, membaca dari input standar", yang untuk Emscripten secara default adalah prompt().

Aplikasi mkbitmap menampilkan perintah yang meminta input.

Mencegah eksekusi otomatis

Agar mkbitmap segera berhenti dijalankan dan justru menunggu input pengguna, Anda harus memahami objek Module Emscripten. Module adalah objek JavaScript global dengan atribut yang dipanggil kode yang dihasilkan Emscripten di berbagai titik dalam eksekusinya. Anda dapat menyediakan implementasi Module untuk mengontrol eksekusi kode. Saat aplikasi Emscripten dimulai, aplikasi akan melihat nilai pada objek Module dan menerapkannya.

Dalam kasus mkbitmap, setel Module.noInitialRun ke true untuk mencegah operasi awal yang menyebabkan perintah muncul. Buat skrip yang disebut script.js, sertakan sebelum <script src="mkbitmap.js"></script> di index.html, lalu tambahkan kode berikut ke script.js. Saat Anda memuat ulang aplikasi sekarang, dialog akan hilang.

var Module = {
  // Don't run main() at page load
  noInitialRun: true,
};

Membuat build modular dengan beberapa flag build lainnya

Untuk memberikan input ke aplikasi, Anda dapat menggunakan dukungan sistem file Emscripten di Module.FS. Bagian Termasuk Dukungan Sistem File dalam dokumentasi menyatakan:

Emscripten memutuskan apakah akan menyertakan dukungan sistem file secara otomatis. Banyak program yang tidak memerlukan file, dan dukungan sistem file tidak dapat diabaikan ukurannya, jadi Emscripten menghindari menyertakannya jika tidak melihat alasannya. Artinya, jika kode C/C++ Anda tidak mengakses file, objek FS dan API sistem file lainnya tidak akan disertakan dalam output. Di sisi lain, jika kode C/C++ Anda menggunakan file, maka dukungan sistem file akan disertakan secara otomatis.

Sayangnya, mkbitmap adalah salah satu kasus saat Emscripten tidak otomatis menyertakan dukungan sistem file, sehingga Anda harus secara eksplisit memintanya untuk melakukannya. Ini berarti Anda harus mengikuti langkah-langkah emconfigure dan emmake yang dijelaskan sebelumnya, dengan beberapa flag lagi yang ditetapkan melalui argumen CFLAGS. Flag berikut juga mungkin berguna untuk project lain.

Selain itu, dalam kasus khusus ini, Anda perlu menyetel flag --host ke wasm32 untuk memberi tahu skrip configure yang Anda kompilasi untuk WebAssembly.

Perintah emconfigure akhir akan terlihat seperti ini:

$ emconfigure ./configure --host=wasm32 CFLAGS='-sFILESYSTEM=1 -sEXPORTED_RUNTIME_METHODS=FS,callMain -sMODULARIZE=1 -sEXPORT_ES6 -sINVOKE_RUN=0'

Jangan lupa untuk menjalankan emmake make lagi dan menyalin file yang baru dibuat ke folder mkbitmap.

Ubah index.html agar hanya memuat modul ES script.js, tempat Anda kemudian mengimpor modul mkbitmap.js.

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="utf-8" />
    <title>mkbitmap</title>
  </head>
  <body>
    <!-- No longer load `mkbitmap.js` here -->
    <script src="script.js" type="module"></script>
  </body>
</html>
// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  const Module = await loadWASM();
  console.log(Module);
};

run();

Saat membuka aplikasi sekarang di browser, Anda akan melihat objek Module dicatat ke konsol DevTools, dan prompt akan hilang, karena fungsi main() dari mkbitmap tidak lagi dipanggil saat awal.

Aplikasi mkbitmap dengan layar putih yang menampilkan objek Modul yang dicatat ke dalam log ke konsol DevTools.

Eksekusi fungsi utama secara manual

Langkah berikutnya adalah memanggil fungsi main() mkbitmap secara manual dengan menjalankan Module.callMain(). Fungsi callMain() menggunakan array argumen, yang cocok satu per satu dengan yang akan Anda teruskan di command line. Jika menjalankan mkbitmap -v di command line, Anda dapat memanggil Module.callMain(['-v']) di browser. Tindakan ini akan mencatat nomor versi mkbitmap ke konsol DevTools.

// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  const Module = await loadWASM();
  Module.callMain(['-v']);
};

run();

Aplikasi mkbitmap dengan layar putih, yang menunjukkan nomor versi mkbitmap yang dicatat ke konsol DevTools.

Mengalihkan output standar

Output standar (stdout) secara default adalah konsol. Namun, Anda dapat mengalihkannya ke hal lain, misalnya, fungsi yang menyimpan output ke variabel. Ini berarti Anda dapat menambahkan output ke HTML dengan menyetel properti Module.print.

// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  let consoleOutput = 'Powered by ';
  const Module = await loadWASM({
    print: (text) => (consoleOutput += text),
  });
  Module.callMain(['-v']);
  document.body.textContent = consoleOutput;
};

run();

Aplikasi mkbitmap menampilkan nomor versi mkbitmap.

Mendapatkan file input ke dalam sistem file memori

Untuk memasukkan file input ke sistem file memori, Anda memerlukan yang setara dengan mkbitmap filename pada command line. Untuk memahami cara pendekatan ini, pertama-tama, latar belakang tentang bagaimana mkbitmap mengharapkan inputnya dan membuat outputnya.

Format input mkbitmap yang didukung adalah PNM (PBM, PGM, PPM) dan BMP. Format {i>outputnya<i} adalah PBM untuk bitmap, dan PGM untuk peta abu-abu. Jika argumen filename diberikan, mkbitmap secara default akan membuat file output yang namanya diperoleh dari nama file input dengan mengubah akhirannya menjadi .pbm. Misalnya, untuk nama file input example.bmp, nama file outputnya adalah example.pbm.

Emscripten menyediakan sistem file virtual yang menyimulasikan sistem file lokal, sehingga kode native yang menggunakan API file sinkron dapat dikompilasi dan dijalankan dengan sedikit atau tanpa perubahan. Agar mkbitmap dapat membaca file input seolah-olah file tersebut diteruskan sebagai argumen command line filename, Anda harus menggunakan objek FS yang disediakan Emscripten.

Objek FS didukung oleh sistem file dalam memori (biasanya disebut sebagai MEMFS) dan memiliki fungsi writeFile() yang Anda gunakan untuk menulis file ke sistem file virtual. Anda menggunakan writeFile() seperti yang ditunjukkan dalam contoh kode berikut.

Untuk memverifikasi bahwa operasi tulis file berfungsi, jalankan fungsi readdir() objek FS dengan parameter '/'. Anda akan melihat example.bmp dan sejumlah file default yang selalu dibuat secara otomatis.

Perhatikan bahwa panggilan sebelumnya ke Module.callMain(['-v']) untuk mencetak nomor versi telah dihapus. Hal ini karena Module.callMain() adalah fungsi yang umumnya hanya mengharapkan dijalankan satu kali.

// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  const Module = await loadWASM();
  const buffer = await fetch('https://example.com/example.bmp').then((res) => res.arrayBuffer());
  Module.FS.writeFile('example.bmp', new Uint8Array(buffer));
  console.log(Module.FS.readdir('/'));
};

run();

Aplikasi mkbitmap menampilkan array file dalam sistem file memori, termasuk example.bmp.

Eksekusi pertama aktual

Setelah semuanya siap, jalankan mkbitmap dengan menjalankan Module.callMain(['example.bmp']). Catat konten folder '/' MEMFS, dan Anda akan melihat file output example.pbm yang baru dibuat di samping file input example.bmp.

// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  const Module = await loadWASM();
  const buffer = await fetch('https://example.com/example.bmp').then((res) => res.arrayBuffer());
  Module.FS.writeFile('example.bmp', new Uint8Array(buffer));
  Module.callMain(['example.bmp']);
  console.log(Module.FS.readdir('/'));
};

run();

Aplikasi mkbitmap menampilkan array file dalam sistem file memori, termasuk example.bmp dan example.pbm.

Mendapatkan file output dari sistem file memori

Fungsi readFile() objek FS memungkinkan Anda mengeluarkan example.pbm yang dibuat di langkah terakhir dari sistem file memori. Fungsi ini menampilkan Uint8Array yang Anda konversi ke objek File dan disimpan ke disk, karena browser umumnya tidak mendukung file PBM untuk melihat langsung di browser. (Ada cara yang lebih baik untuk menyimpan file, tetapi menggunakan <a download> yang dibuat secara dinamis adalah cara yang paling didukung secara luas.) Setelah file disimpan, Anda dapat membukanya di penampil gambar favorit.

// This is `script.js`.
import loadWASM from './mkbitmap.js';

const run = async () => {
  const Module = await loadWASM();
  const buffer = await fetch('https://example.com/example.bmp').then((res) => res.arrayBuffer());
  Module.FS.writeFile('example.bmp', new Uint8Array(buffer));
  Module.callMain(['example.bmp']);
  const output = Module.FS.readFile('example.pbm', { encoding: 'binary' });
  const file = new File([output], 'example.pbm', {
    type: 'image/x-portable-bitmap',
  });
  const a = document.createElement('a');
  a.href = URL.createObjectURL(file);
  a.download = file.name;
  a.click();
};

run();

macOS Finder dengan pratinjau file .bmp input dan file .pbm output.

Menambahkan UI interaktif

Sampai tahap ini, file input di-hardcode dan mkbitmap berjalan dengan parameter default. Langkah terakhir adalah memungkinkan pengguna memilih file input secara dinamis, menyesuaikan parameter mkbitmap, lalu menjalankan alat dengan opsi yang dipilih.

// Corresponds to `mkbitmap -o output.pbm input.bmp -s 8 -3 -f 4 -t 0.45`.
Module.callMain(['-o', 'output.pbm', 'input.bmp', '-s', '8', '-3', '-f', '4', '-t', '0.45']);

Format gambar PBM tidak terlalu sulit untuk diurai, sehingga dengan beberapa kode JavaScript, Anda bahkan dapat menampilkan pratinjau gambar output. Lihat kode sumber demo tersemat di bawah untuk mengetahui salah satu cara melakukannya.

Kesimpulan

Selamat, Anda telah berhasil mengompilasi mkbitmap ke WebAssembly dan membuatnya berfungsi di browser. Ada beberapa jalan buntu dan Anda harus mengompilasi alat lebih dari sekali sampai dapat berfungsi, tetapi seperti yang saya tulis di atas, itulah bagian dari pengalaman. Ingat juga tag webassembly StackOverflow jika Anda mengalami kesulitan. Selamat mengompilasi!

Ucapan terima kasih

Artikel ini diulas oleh Sam Clegg dan Rachel Andrew.