WebAssembly - «Привет, мир»
В этой главе мы собираемся написать простую программу на C, преобразовать ее в .wasm и выполнить то же самое в браузере, чтобы получить текст «Hello World».
Будет использоваться инструмент wasm explorer, который преобразует программу C в .wasm и будет использовать .wasm внутри нашего .html файла.
Инструмент Wasm explorer, доступный по адресу https://mbebenita.github.io/WasmExplorer/ looks as follows −
Код C, который мы собираемся использовать, выглядит следующим образом:
#include <stdio.h>
char *c_hello() {
return "Hello World";
}
Обновите первый блок в wasm explorer с помощью кода C, как показано ниже -
Нажмите кнопку COMPILE, чтобы скомпилировать WASM и WAT и веб-сборку Firefox x86, как показано ниже -
Используйте ЗАГРУЗИТЬ, чтобы получить файл .wasm и сохранить его как firstprog.wasm.
Создайте файл .html с именем firstprog.html, как показано ниже -
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>WebAssembly Hello World</title>
</head>
<body>
<div id="textcontent"></div>
<script type="text/javascript">
//Your code from webassembly here
</script>
</body>
</html>
Давайте теперь используем firstprog.wasm для чтения helloworld из функции C c_hello ().
Шаг 1
Используйте fetch () api, чтобы прочитать код firstprog.wasm.
Шаг 2
Код .wasm необходимо преобразовать в буфер массива с помощью ArrayBuffer. Объект ArrayBuffer вернет вам буфер двоичных данных фиксированной длины.
Код пока будет следующим -
<script type="text/javascript">
fetch("firstprog.wasm") .then(bytes => bytes.arrayBuffer())
</script>
Шаг 3
Байты из ArrayBuffer должны быть скомпилированы в модуль с помощью WebAssembly.compile(buffer) функция.
Код будет выглядеть следующим образом -
<script type="text/javascript">
fetch("firstprog.wasm")
.then(bytes => bytes.arrayBuffer())
.then(mod => WebAssembly.compile(mod))
</script>
Шаг 4
Чтобы получить модуль, мы должны вызвать конструктор webassembly.instance, как показано ниже -
<script type="text/javascript">
fetch("firstprog.wasm")
.then(bytes => bytes.arrayBuffer())
.then(mod => WebAssembly.compile(mod))
.then(module => {return new WebAssembly.Instance(module) })
</script>
Шаг 5
Давайте теперь консольируем экземпляр, чтобы увидеть подробности в браузере.
<script type="text/javascript">
fetch("firstprog.wasm") .then(bytes => bytes.arrayBuffer())
.then(mod => WebAssembly.compile(mod)) .then(module => {
return new WebAssembly.Instance(module)
})
.then(instance => {
console.log(instance);
});
</script>
Подробности console.log показаны ниже -
Чтобы получить строку «Hello World» из функции c_hello (), нам нужно добавить код на javascript.
Сначала получите сведения о буфере памяти, как показано ниже -
let buffer = instance.exports.memory.buffer;;
Значение буфера необходимо преобразовать в типизированный массив, чтобы мы могли считывать из него значения. В буфере есть строка Hello World.
Чтобы преобразовать в типизированный, вызовите конструктор Uint8Array, как показано ниже -
let buffer = new Uint8Array(instance.exports.memory.buffer);
Теперь мы можем прочитать значение из буфера в цикле for.
Давайте теперь получим начальную точку для чтения буфера, вызвав функцию, которую мы написали, как показано ниже -
let test = instance.exports.c_hello();
Теперь у тестовой переменной есть начальная точка для чтения нашей строки. В WebAssembly нет строковых значений, все хранится как целые числа.
Поэтому, когда мы читаем значение из буфера, оно будет целым числом, и нам нужно преобразовать его в строку, используя fromCharCode () в javascript.
Код выглядит следующим образом -
let mytext = "";
for (let i=test; buffer[i]; i++){
mytext += String.fromCharCode(buffer[i]);
}
Теперь, когда вы консолей mytext, вы должны увидеть строку «Hello World».
пример
Полный код выглядит следующим образом -
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>WebAssembly Add Function</title>
<style>
div {
font-size : 30px; text-align : center; color:orange;
}
</style>
</head>
<body>
<div id="textcontent"></div>
<script>
fetch("firstprog.wasm")
.then(bytes => bytes.arrayBuffer())
.then(mod => WebAssembly.compile(mod))
.then(module => {return new WebAssembly.Instance(module)})
.then(instance => {
console.log(instance);
let buffer = new Uint8Array(instance.exports.memory.buffer);
let test = instance.exports.c_hello();
let mytext = "";
for (let i=test; buffer[i]; i++) {
mytext += String.fromCharCode(buffer[i]);
}
console.log(mytext); document.getElementById("textcontent").innerHTML = mytext;
});
</script>
</body>
</html>
Мы добавили div, и содержимое добавлено в div, поэтому строка отображается в браузере.
Вывод
Вывод упомянут ниже -