WebAssembly - текстовый формат

WebAssembly имеет код в двоичном формате под названием WASM. Вы также можете получить текстовый формат в WebAssembly, и он называется WAT (текстовый формат WebAssembly). Как разработчик вы не должны писать код на WebAssembly, вместо этого вам нужно скомпилировать языки высокого уровня, такие как C, C ++ и Rust, в WebAssembly.

Код WAT

Напишем WAT-код пошагово.

Step 1 - Отправной точкой в ​​WAT является объявление модуля.

(module)

Step 2 - Давайте теперь добавим к нему некоторые функции в виде функции.

Функция объявлена, как показано ниже -

(func <parameters/result> <local variables> <function body>)

Функция начинается с ключевого слова func, за которым следуют параметры или результат.

Параметры / Результат

Параметры и возвращаемое значение в результате.

Параметры могут иметь следующий тип, поддерживаемый wasm -

  • i32: 32-битное целое число
  • i64: 64-битное целое число
  • f32: 32-битное число с плавающей запятой
  • f64: 64-битное число с плавающей запятой

Параметры для функций написаны, как указано ниже -

  • (параметр i32)
  • (параметр i64)
  • (параметр f32)
  • (параметр f64)

Результат будет записан следующим образом -

  • (результат i32)
  • (результат i64)
  • (результат f32)
  • (результат f64)

Функция с параметрами и возвращаемым значением будет определена следующим образом:

(func (param i32) (param i32) (result i64) <function body>)

Локальные переменные

Локальные переменные - это те, которые вам нужны в вашей функции. Локальное значение функции будет определено следующим образом:

(func (param i32) (param i32) (local i32) (result i64) <function body>)

Тело функции

Тело функции - это логика, которую нужно выполнить. Окончательная программа будет выглядеть так -

(module (func (param i32) (param i32) (local i32) (result i64) <function body>) )

Step 3 - Для чтения и установки параметров и локальных переменных.

Чтобы прочитать параметры и локальные переменные, используйте get_local и set_local команда.

Example

(module 
   (func (param i32) (param i32) (local i32) (result i64) get_local 0 
      get_local 1 
      get_local 2 
   ) 
)

Согласно сигнатуре функции,

  • get_local 0 даст param i32

  • get_local 1 даст следующий параметр param i32

  • get_local 2 даст local value i32

Вместо того, чтобы ссылаться на параметры и локальные переменные с использованием числовых значений, таких как 0,1,2, вы также можете использовать имя перед параметрами, добавив к имени префикс со знаком доллара.

В следующем примере показано, как использовать имя с параметрами и локальными переменными.

Example

(module 
   (func 
      (param $a i32) 
      (param $b i32) 
      (local $c i32) 
      (result i64) get_local $a get_local $b get_local $c 
   ) 
)

Step 4 - Инструкция по телу и исполнению функции.

Выполнение в wasm следует стратегии стека. Выполненные инструкции отправляются одна за другой в стек. Например, инструкция get_local $ a протолкнет значение, которое она считывает в стеке.

Инструкция вроде i32.add который добавит выталкивание элементов из стека.

(func (param $a i32) (param $b i32) 
   get_local $a 
   get_local $b 
   i32.add
)

Инструкция для i32.add является ($a+$b). Последнее значение i32.add будет помещено в стек и присвоено результату.

Если в сигнатуре функции объявлен результат, в конце выполнения в стеке должно быть одно значение. Если параметра результата нет, стек в конце должен быть пустым.

Итак, окончательный код вместе с телом функции будет следующим:

(module 
   (func (param $a i32) (param $b i32) (result i32) 
      get_local $a
      get_local $b 
      i32.add
   )
)

Step 5 - Выполнение вызова функции.

Окончательный код с телом функции показан на шаге 4. Теперь, чтобы вызвать функцию, нам нужно ее экспортировать.

Чтобы экспортировать функцию, это можно сделать со значениями индекса, например 0,1, но мы также можем дать имена. Имя будет иметь префикс $ и будет добавлено после ключевого слова func.

Example

(module 
   (func $add (param $a i32) (param $b i32) (result i32) 
      get_local $a 
      get_local $b i32.add
   ) 
)

Функция $ add должна быть экспортирована с использованием ключевого слова export, как показано ниже -

(module 
   (func $add 
      (param $a i32) 
      (param $b i32) 
      (result i32) 
      get_local $a get_local $b i32.add
   ) 
   (export "add" (func $add))
)

Чтобы протестировать приведенный выше код в браузере, вам нужно будет преобразовать его в двоичную форму (.wasm). Обратитесь к следующей главе, в которой показано, как преобразовать.WAT to .WASM.