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, поэтому строка отображается в браузере.

Вывод

Вывод упомянут ниже -