WebAssembly-빠른 가이드

WebAssembly는 웹을위한 새로운 컴퓨터 프로그래밍 언어입니다. WebAssembly 코드는 웹과 호환되며 최신 웹 브라우저에서 쉽게 실행할 수있는 저수준 바이너리 형식입니다. 생성 된 파일 크기는 작고로드 및 실행 속도가 더 빠릅니다. 이제 C, C ++, Rust 등과 같은 언어를 바이너리 형식으로 컴파일 할 수 있으며 자바 스크립트처럼 웹에서 실행할 수 있습니다.

WebAssembly의 정의

WebAssembly의 공식 웹 사이트에 따라 https://webassembly.org/, WebAssembly (약칭 Wasm)는 스택 기반 가상 머신에 대한 이진 명령어 형식으로 정의됩니다. Wasm은 C / C ++ / Rust와 같은 고급 언어를 컴파일하기위한 이식 가능한 타겟으로 설계되어 클라이언트 및 서버 애플리케이션을 웹에 배포 할 수 있습니다.

웹 어셈블리는 개발자가 작성해야하는 것이 아니지만 코드는 C, C ++, Rust와 같은 언어로 작성되며 WebAssembly (wasm)로 컴파일 할 수 있습니다. 웹 브라우저 내에서 동일한 코드를 실행할 수 있습니다.

웹 어셈블리는 새로운 언어이고 코드는 저수준 어셈블리 언어이지만 텍스트 형식 기능을 사용하면 코드를 읽을 수 있고 필요한 경우 디버깅이 가능합니다.

WebAssembly의 목표

WebAssembly의 개방형 표준은 모든 주요 브라우저의 대표와 W3C 워킹 그룹을 포함하는 W3C 커뮤니티 그룹에서 개발되었습니다.

WebAssembly의 주요 목표는 다음과 같습니다.

  • Faster, Efficient and Portable − WebAssembly 코드는 사용 가능한 하드웨어를 활용하여 다른 플랫폼에서 더 빠르게 실행되도록합니다.

  • Easy to read and debug − 낮은 수준의 어셈블리 언어 인 WebAssembly는 텍스트 형식을 지원하므로 문제에 대한 코드를 디버깅하고 필요한 경우 코드를 다시 작성할 수 있습니다.

  • Security − WebAssembly는 권한 및 동일 출처 정책을 관리하므로 웹 브라우저에서 안전하게 실행할 수 있습니다.

WebAssembly의 장점

다음은 WebAssembly의 장점입니다-

  • Run is Modern Browsers − WebAssembly는 사용 가능한 최신 웹 브라우저에서 문제없이 실행할 수 있습니다.

  • Multiple Language support− C, C ++, Rust, Go와 같은 언어는 이제 코드를 WebAssembly로 컴파일하고 웹 브라우저에서 동일하게 실행할 수 있습니다. 따라서 브라우저에서 실행할 수 없었던 언어도 이제 그렇게 할 수 있습니다.

  • Faster, Efficient and Portable − 코드 크기가 작기 때문에로드 및 실행 속도가 더 빠릅니다.

  • Easy to understand− 개발자는 WebAssembly에서 코드를 작성할 필요가 없기 때문에 WebAssembly 코딩을 이해하는 데 많은 스트레스를받을 필요가 없습니다. 대신 WebAssembly에서 코드를 컴파일하고 웹에서 동일하게 실행하십시오.

  • Easy to Debug − 최종 코드는 저수준 어셈블리 언어로되어 있지만 읽고 디버그하기 쉬운 텍스트 형식으로도 얻을 수 있습니다.

WebAssembly의 단점

다음은 WebAssembly의 단점입니다-

  • WebAssembly는 아직 작업 중이며 미래를 결정하기에는 너무 이르다.

  • WebAssembly는 DOM (문서 개체 모델)과 상호 작용하기 위해 javascript에 의존합니다.

WebAssembly는 2017 년에 처음 소개 된 WASM이라고도합니다. WebAssembly의 기원 뒤에있는 큰 기술 회사는 Google, Apple, Microsoft, Mozilla 및 W3C입니다.

소문은 WebAssembly가 더 빠른 실행으로 인해 Javascript를 대체 할 것이라는 점입니다.하지만 그렇지 않습니다. WebAssembly와 Javascript는 복잡한 문제를 해결하기 위해 함께 작동합니다.

WebAssembly의 필요성

지금까지 브라우저 내에서 성공적으로 작동 할 수있는 Javascript 만 있습니다. 자바 스크립트를 사용하는 브라우저에서 수행하기 어려운 매우 무거운 작업이 있습니다.

예를 들어 이미지 인식, CAD (Computer-Aided Design) 애플리케이션, 라이브 비디오 증강, VR 및 증강 현실, 음악 애플리케이션, 과학적 시각화 및 시뮬레이션, 게임, 이미지 / 비디오 편집 등이 있습니다.

WebAssembly는 더 빨리로드하고 실행할 수있는 바이너리 명령어가있는 새로운 언어입니다. 위에서 언급 한 작업은 C, C ++, Rust 등과 같은 고급 언어로 쉽게 수행 할 수 있습니다. C, C ++, Rust에있는 코드를 컴파일하여 웹 브라우저에서 사용할 수있는 방법이 필요합니다. WebAssembly를 사용하여 동일한 작업을 수행 할 수 있습니다.

WebAssembly 코드가 브라우저 내부에로드 될 때. 그런 다음 브라우저는 프로세서가 이해할 수있는 시스템 형식으로 변환하는 작업을 처리합니다.

자바 스크립트의 경우 코드를 다운로드하고 구문 분석 한 후 컴퓨터 형식으로 변환해야합니다. 많은 시간이 소요되며 앞서 언급 한 것과 같은 무거운 작업의 경우 매우 느릴 수 있습니다.

WebAssembly 작업

C, C ++ 및 Rust와 같은 고급 언어는 바이너리 형식으로 컴파일됩니다. .wasm 및 텍스트 형식 .wat.

C, C ++ 및 Rust로 작성된 소스 코드는 다음과 같이 컴파일됩니다. .wasm컴파일러 사용. Emscripten SDK를 사용하여 C / C ++를 다음과 같이 컴파일 할 수 있습니다..wasm.

흐름은 다음과 같습니다-

C / C ++ 코드는 다음으로 컴파일 될 수 있습니다. .wasmEmscripten SDK 사용. 나중에.wasm 코드는 html 파일의 javascript를 사용하여 출력을 표시하는 데 사용할 수 있습니다.

WebAssembly의 주요 개념

주요 개념은 다음과 같습니다.

기준 치수

모듈은 브라우저에서 실행 가능한 기계어 코드로 컴파일되는 개체입니다. 모듈은 상태 비 저장이라고하며 창과 웹 작업자간에 공유 할 수 있습니다.

기억

WebAssembly의 메모리는 arraybuffer데이터를 보유하고 있습니다. Javascript api WebAssembly.memory ()를 사용하여 메모리를 할당 할 수 있습니다.

WebAssembly의 테이블은 WebAssembly 메모리 외부에 있으며 대부분 함수에 대한 참조를 포함하는 형식화 된 배열입니다. 함수의 메모리 주소를 저장합니다.

인스턴스는 브라우저 내에서 실행하기 위해 자바 스크립트에서 호출 할 수있는 내 보낸 모든 함수를 포함 할 개체입니다.

WebAssembly는 자바 스크립트를 개선 한 wasm이라고도합니다. 자바 스크립트와 마찬가지로 nodejs와 마찬가지로 브라우저 내에서 실행되도록 설계되었습니다. C, C ++, Rust와 같은 고급 언어가 컴파일 될 때 wasm 출력을 얻습니다.

다음 C 프로그램을 고려하십시오-

int factorial(int n) {
   if (n == 0) 
      return 1; 
   else 
      return n * factorial(n-1); 
}

WasmExplorer를 사용하십시오.https://mbebenita.github.io/WasmExplorer/ 아래와 같이 컴파일 된 코드를 얻으려면-

팩토리얼 프로그램의 웹 어셈블리 텍스트 형식은 다음과 같습니다.

(module 
   (table 0 anyfunc) 
   (memory $0 1) (export "memory" (memory $0)) (export "factorial" (func $factorial)) (func $factorial (; 0 ;) (param $0 i32) (result i32) (local $1 i32) 
      (local $2 i32) (block $label$0 (br_if $label$0 (i32.eqz (get_local $0) 
            )
         )
         (set_local $2 (i32.const 1) ) (loop $label$1 (set_local $2 
               (i32.mul 
                  (get_local $0) (get_local $2) 
               ) 
            ) 
            (set_local $0 (tee_local $1        (i32.add 
                  (get_local $0) (i32.const -1) ) ) ) (br_if $label$1 (get_local $1) 
            ) 
         ) 
         (return 
            (get_local $2)
         ) 
      ) 
      (i32.const 1) 
   )
)

Wat2Wasm 도구를 사용하면 아래에 언급 된 것처럼 WASM 코드를 볼 수 있습니다.

개발자는 wasm으로 코드를 작성하거나 그 안에서 코딩하는 법을 배워서는 안됩니다. 대부분은 고급 언어를 컴파일 할 때 생성되기 때문입니다.

스택 머신 모델

WASM에서는 모든 명령이 스택으로 푸시됩니다. 인수가 팝되고 결과가 스택으로 다시 푸시됩니다.

2 개의 숫자를 추가하는 다음 WebAssembly 텍스트 형식을 고려하십시오.

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

함수의 이름은 $add, 2 개의 매개 변수가 필요합니다. $a and $비. 결과는 32 비트 정수 유형입니다. 로컬 변수는 get_local을 사용하여 액세스하고 추가 작업은 i32.add를 사용하여 수행합니다.

실행하는 동안 2 개의 숫자를 추가하는 스택 표현은 다음과 같습니다.

step 1 − get_local 실행 $a instruction, the first parameters i.e., $a가 스택에 푸시됩니다.

step 2 − get_local 실행 중 $b instruction, the second parameters i.e., $b가 스택에 푸시됩니다.

step 3− i32.add를 실행하면 스택에서 요소가 팝되고 결과가 스택으로 다시 푸시됩니다. 스택 내부의 끝에 남아있는 값은 $ add 함수의 결과입니다.

이 장에서는 C / C ++를 컴파일하기 위해 Emscripten SDK를 설치하는 방법을 배웁니다. Emscripten은 C / C ++에서 생성 된 바이트 코드를 가져와 브라우저 내에서 쉽게 실행할 수있는 JavaScript로 컴파일하는 LLVM (Low level virtual machine)입니다.

C / C ++를 WebAssembly로 컴파일하려면 먼저 Emscripten sdk를 설치해야합니다.

Emscripten SDK 설치

Emscripten sdk를 설치하는 단계는 다음과 같습니다.

Step 1 − emsdk 저장소 복제 : git clone https://github.com/emscripten-core/emsdk.git.

E:\wa>git clone https://github.com/emscripten-core/emsdk.git 
Cloning into 'emsdk'... 
remote: Enumerating objects: 14, done. 
remote: Counting objects: 100% (14/14), done. 
remote: Compressing objects: 100% (12/12), done. 
remote: Total 1823 (delta 4), reused 4 (delta 2), pack-reused 1809 receiving obje 
cts: 99% (1819/1823), 924.01 KiB | 257.00 KiB/s 
Receiving objects: 100% (1823/1823), 1.01 MiB | 257.00 KiB/s, done. 
Resolving deltas: 100% (1152/1152), done.

Step 2 − emsdk 디렉토리에 입력합니다.

cd emsdk

Step 3 − Windows의 경우 : 다음 명령을 실행합니다.

emsdk install latest

For linux,이 명령은 java, python 등과 같은 필요한 도구를 설치하는 데 약간의 시간이 소요됩니다.

./emsdk install latest

Step 4 − 최신 SDK를 활성화하려면 터미널에서 다음 명령을 실행하십시오.

For windows, 다음 명령을 실행하십시오-

emsdk activate latest

For linux, 아래 언급 된 명령을 실행하십시오-

./emsdk activate latest

Step 5 − PATH 및 기타 환경 변수를 활성화하려면 터미널에서 다음 명령을 실행하십시오.

For windows, 명령을 실행하십시오-

emsdk_env.bat

For linux, 다음 명령을 실행하십시오-

source ./emsdk_env.sh

emsdk 설치가 완료되었으며 이제 C 또는 C ++ 코드를 컴파일 할 수 있습니다. C / C ++의 컴파일은 다음 장에서 수행됩니다.

다음은 C 또는 C ++ 코드를 컴파일하려면 다음 명령입니다.

emcc source.c or source.cpp -s WASM=1 -o source.html

출력은 source.html 파일, source.js 및 source.wasm 파일을 제공합니다. js는 source.wasm을 가져올 api를 갖게되며 브라우저에서 source.html을 누르면 출력을 볼 수 있습니다.

wasm 파일을 얻으려면 다음 명령을 사용할 수 있습니다. 이 명령은 source.wasm 파일 만 제공합니다.

emcc source.c or source.cpp -s STANDALONE_WASM

이 장에서는 WebAssembly로 작업하는 동안 매우 유용한 몇 가지 사용하기 쉬운 도구에 대해 설명합니다. WebAssembly.studio 도구에 대해 배우는 것으로 시작하겠습니다.

WebAssembly.studio

이 도구를 사용하면 C, Rust, Wat를 Wasm 등으로 컴파일 할 수 있습니다.

시작하려면 Empty C Project, Empty Rust Project, Empty Wat Project를 클릭하여 C를 컴파일하고 Rust를 WASM으로 컴파일 할 수 있습니다. 5.

빌드, 실행을 통해 코드를 빌드하고 출력을 확인합니다. 다운로드 버튼을 사용하면.wasm브라우저 내부에서 테스트하는 데 사용할 수있는 파일입니다. 이 도구는 C 및 Rust 코드를 컴파일하고 출력을 확인하는 데 매우 유용합니다.

웹 어셈블리 탐색기

WebAssembly Explorer를 사용하면 C 및 C ++ 코드를 컴파일 할 수 있습니다. 링크 참조https://mbebenita.github.io/WasmExplorer/상세 사항은. 링크를 클릭하면 나타나는 화면은 다음과 같습니다.

C 및 C ++ 버전을 선택할 수 있습니다. C 또는 C ++의 소스 코드가 여기에 작성되었습니다.

컴파일 버튼을 클릭하면 아래 블록에 WebAssembly 텍스트 형식 (WAT) 및 Firefox x86 어셈블리 코드가 제공됩니다.

당신은 다운로드 할 수 있습니다 .wasm 브라우저 내에서 테스트하는 코드입니다.

WASMF 중간

Wasmfiddle 은 C 코드를 WebAssembly로 컴파일하고 출력을 테스트하는 데 도움이됩니다. 링크를 클릭 한 후https://wasdk.github.io/WasmFiddle/, 다음 페이지가 표시됩니다-

빌드를 클릭하여 코드를 컴파일하십시오. Wat 및 Wasm을 클릭하여 Wat 및 Wasm 코드를 다운로드 할 수 있습니다. 출력을 테스트하려면 실행 버튼을 클릭합니다.

WASM에서 WAT

도구 wat2wasmWebAssembly 텍스트 형식을 입력하면 wasm 코드를 제공합니다. 링크를 클릭 할 수 있습니다.https://webassembly.github.io/wabt/demo/wat2wasm/ 데모 및 나타날 화면은 다음과 같습니다.

업로드 버튼을 사용하여 .wasm을 업로드하면 텍스트 영역에 텍스트 형식이 표시됩니다.

WAT에서 WASM

wat2wasm 도구는 WebAssembly 텍스트 형식을 입력 할 때 wasm 코드를 제공합니다. 링크를 클릭 할 수 있습니다.https://webassembly.github.io/wabt/demo/wat2wasm/ 데모 및 나타날 화면은 다음과 같습니다.

이 도구는 출력도 테스트하는 데 도움이되므로 매우 유용합니다. WAT 코드를 입력하고 .wasm 코드를 살펴보고 코드를 실행하여 출력을 볼 수도 있습니다.

WASM이라고도하는 WebAssembly는 가장 효율적인 방식으로 브라우저 내에서 실행되도록 개발 된 바이너리 형식의 저수준 코드입니다. WebAssembly 코드는 다음 개념으로 구성됩니다.

  • Values
  • Types
  • Instructions

지금 자세히 알아 보겠습니다.

가치

WebAssembly의 값은 텍스트, 문자열 및 벡터와 같은 복잡한 데이터를 저장하기위한 것입니다. WebAssembly는 다음을 지원합니다-

  • Bytes
  • Integers
  • 부동 소수점
  • Names

바이트

바이트는 WebAssembly에서 지원되는 가장 간단한 형식의 값입니다. 값은 16 진수 형식입니다.

For example

b로 표시되는 바이트는 자연수 n도 취할 수 있습니다. 여기서 n <256입니다.

byte ::= 0x00| .... |0xFF

정수

WebAssembly에서 지원되는 정수는 다음과 같습니다.

  • i32 : 32 비트 정수
  • i64 : 64 비트 정수

부동 소수점

WebAssembly에서 지원되는 부동 소수점 숫자는 다음과 같습니다.

  • f32 : 32 비트 부동 소수점
  • f64 : 64 비트 부동 소수점

이름

이름은 링크에서 사용할 수있는 유니 코드로 정의 된 스칼라 값이있는 문자 시퀀스입니다. http://www.unicode.org/versions/Unicode12.1.0/ 여기에 주어진.

종류

WebAssembly의 엔티티는 유형으로 분류됩니다. 지원되는 유형은 다음과 같습니다.

  • 값 유형
  • 결과 유형
  • 기능 유형
  • Limits
  • 메모리 유형
  • 테이블 유형
  • 글로벌 유형
  • 외부 유형

하나씩 공부합시다.

값 유형

WebAssembly에서 지원하는 값 유형은 다음과 같습니다.

  • i32 : 32 비트 정수
  • i64 : 64 비트 정수
  • f32 : 32 비트 부동 소수점
  • f64 : 64 비트 부동 소수점
valtype ::= i32|i64|f32|f64

결과 유형

대괄호 안에 기록 된 값은 결과 유형 내에 실행되고 저장됩니다. 결과 유형은 값으로 구성된 코드 블록 실행의 출력입니다.

resulttype::=[valtype?]

기능 유형

함수 유형은 매개 변수 벡터를 받아 결과 벡터를 반환합니다.

functype::=[vec(valtype)]--> [vec(valtype)]

제한

한계는 메모리 및 테이블 유형과 연결된 스토리지 범위입니다.

limits ::= {min u32, max u32}

메모리 유형

메모리 유형은 선형 메모리와 크기 범위를 다룹니다.

memtype ::= limits

테이블 유형

테이블 유형은 할당 된 요소 유형에 따라 분류됩니다.

tabletype ::= limits elemtype
elemtype ::= funcref

테이블 유형은 할당 된 최소 및 최대 크기 제한에 따라 다릅니다.

글로벌 유형

Global Type은 값이 있고 변경되거나 동일하게 유지 될 수있는 전역 변수를 보유합니다.

globaltype ::= mut valtype
mut ::= const|var

외부 유형

외부 유형은 가져 오기 및 외부 값을 다룹니다.

externtype ::= func functype | table tabletype | mem memtype | global globaltype

명령

WebAssembly 코드는 스택 머신 모델을 따르는 일련의 명령어입니다. WebAssembly가 스택 머신 모델을 따르기 때문에 명령어가 스택에 푸시됩니다.

예를 들어 함수의 인수 값은 스택에서 팝되고 결과는 스택으로 푸시됩니다. 결국 스택에는 하나의 값만 있으며 그 결과입니다.

일반적으로 사용되는 지침 중 일부는 다음과 같습니다.

  • 숫자 지침
  • 가변 지침

숫자 지침

숫자 명령어는 숫자 값에 대해 수행되는 연산입니다.

For example
nn, mm ::= 32|64
ibinop ::= add|sub|mul|div_sx|rem_sx|and|or|xor
irelop ::= eq | ne | lt_sx | gt_sx | le_sx | ge_sx
frelop ::= eq | ne | lt | gt | le | ge

가변 지침

변수 명령어는 로컬 및 전역 변수에 액세스하는 것입니다.

For example

지역 변수에 액세스하려면-

get_local $a
get_local $b

set 지역 변수-

set_local $a
set_local $b

access 전역 변수-

get_global $a
get_global $b

set 전역 변수-

set_global $a
set_global $b

이 장에서는 WebAssembly와 Javascript 간의 비교를 나열합니다.

Javascript는 브라우저에서 많이 사용하는 언어입니다. 이제 WebAssembly 릴리스를 통해 브라우저 내에서 WebAssembly를 사용할 수도 있습니다.

WebAssembly가 등장한 이유는 자바 스크립트를 대체하기 위해서가 아니라 자바 스크립트로 처리하기 어려운 특정 사항을 처리하기 위해서입니다.

For example

이미지 인식, CAD 애플리케이션, 라이브 비디오 증강, VR 및 증강 현실, 음악 애플리케이션, 과학적 시각화 및 시뮬레이션, 게임, 이미지 / 비디오 편집 등의 작업을 자바 스크립트로 수행하기가 어렵습니다.

이제 WebAssembly로 컴파일 할 수있는 C / C ++, Rust와 같은 고급 언어를 사용하면 위에서 언급 한 작업을 쉽게 수행 할 수 있습니다. WebAssembly는 브라우저 내에서 실행하기 쉬운 바이너리 코드를 생성합니다.

여기에 Javascript와 WebAssembly를 비교 한 목록이 있습니다.

매개 변수 자바 스크립트 웹 어셈블리

코딩

자바 스크립트로 쉽게 코드를 작성할 수 있습니다. 작성된 코드는 사람이 읽을 수 있으며 .js로 저장됩니다. 브라우저 내에서 사용할 때는 <script> 태그를 사용해야합니다.

코드는 WebAssembly에서 텍스트 형식으로 작성할 수 있으며 .wat로 저장됩니다. .wat 형식으로 코드를 작성하는 것은 어렵습니다. .wat에서 처음부터 작성하는 대신 다른 고급 언어에서 코드를 컴파일하는 것이 가장 좋습니다.

브라우저 내에서 .wat 파일을 실행할 수 없으며 사용 가능한 컴파일러 또는 온라인 도구를 사용하여 .wasm으로 변환해야합니다.

실행

브라우저 내에서 사용할 때 자바 스크립트로 작성된 코드를 다운로드, 구문 분석, 컴파일 및 최적화해야합니다.

.wasm의 WebAssembly 코드는 이미 컴파일되어 있고 바이너리 형식입니다.

메모리 관리

자바 스크립트는 변수가 생성 될 때 메모리를 할당하고 사용하지 않을 때 메모리를 해제하고 가비지 컬렉션에 추가합니다.

WebAssembly의 메모리는 데이터를 보유하는 배열 버퍼입니다. Javascript API WebAssembly.memory ()를 사용하여 메모리를 할당 할 수 있습니다.

WebAssembly 메모리는 이해하기 쉽고 실행을 수행하기 쉬운 플랫 메모리 모델과 같은 배열 형식으로 저장됩니다.

WebAssembly에서 메모리 모델의 단점은-

  • 복잡한 계산에는 시간이 걸립니다.

  • Webassembly는 메모리 재사용을 허용하지 않는 가비지 수집을 지원하지 않으며 메모리가 낭비됩니다.

로드 시간 및 성능

자바 스크립트의 경우 브라우저 내에서 호출되면 자바 스크립트 파일을 다운로드하고 파싱해야합니다. 나중에 파서는 소스 코드를 javascript 엔진이 브라우저에서 코드를 실행하는 바이트 코드로 변환합니다.

Javascript 엔진은 매우 강력하므로 WebAssembly에 비해 javascript의로드 시간과 성능이 매우 빠릅니다.

WebAssembly의 가장 중요한 목표는 JavaScript보다 빠르다는 것입니다. 고급 언어에서 생성 된 Wasm 코드는 크기가 작아서로드 시간이 더 빠릅니다.

그러나 GO와 같은 언어는 wasm으로 컴파일 될 때 작은 코드 조각에 대해 큰 파일 크기를 생성합니다.

WebAssembly는 컴파일 속도가 더 빠르며 모든 주요 브라우저에서 실행될 수 있도록 설계되었습니다. WebAssembly는 여전히 javascript와 비교하여 성능 측면에서 많은 개선 사항을 추가해야합니다.

디버깅

자바 스크립트는 사람이 읽을 수 있으며 쉽게 디버깅 할 수 있습니다. 브라우저 내에서 자바 스크립트 코드에 중단 점을 추가하면 코드를 쉽게 디버그 할 수 있습니다.

WebAssembly는 읽을 수있는 텍스트 형식으로 코드를 제공하지만 여전히 디버깅하기는 매우 어렵습니다. Firefox에서는 브라우저 내에서 wasm 코드를 .wat 형식으로 볼 수 있습니다.

.wat에 중단 점을 추가 할 수 없으며 이는 향후 사용 가능할 것입니다.

브라우저 지원

Javascript는 모든 브라우저에서 잘 작동합니다.

모든 주요 웹 브라우저는 WebAssembly를 지원합니다.

이 장에서는 javascript webassembly API를 사용하여 wasm 코드를로드하고 브라우저에서 실행하는 방법을 이해합니다.

다음은 몇 가지 중요한 API입니다. 자습서 전체에서 wasm 코드를 실행하는 데 사용할 것입니다.

  • fetch () 브라우저 API
  • WebAssembly.compile
  • WebAssembly.instance
  • WebAssembly.instantiate
  • WebAssembly.instantiateStreaming

WebAssembly javascript API에 대해 논의하기 전에 API와 출력을 테스트하기 위해 다음 C 프로그램과 wasm 탐색기를 사용하여 c 프로그램에서 생성 된 .wasm 코드를 사용할 것입니다.

C 프로그램의 예는 다음과 같습니다.

#include<stdio.h>
int square(int n) { 
   return n*n; 
}

WASM 탐색기를 사용하여 wasm 코드를 얻습니다.

WASM 코드를 다운로드하고이를 사용하여 API를 테스트하십시오.

fetch () 브라우저 API

fetch () API는 .wasm 네트워크 리소스를로드하기위한 것입니다.

<script>
   var result = fetch("findsquare.wasm");
   console.log(result);
</script>

아래와 같이 약속을 반환합니다.

XMLHttpRequest 메서드를 사용하여 wasm 네트워크 리소스를 가져올 수도 있습니다.

WebAssembly.compile ()

API 책임은 .wasm에서 가져온 모듈 세부 사항을 컴파일하는 것입니다.

통사론

구문은 다음과 같습니다.

WebAssembly.compile(buffer);

매개 변수

Buffer − .wasm의이 코드는 컴파일을위한 입력으로 제공하기 전에 형식화 된 배열 또는 배열 버퍼로 변환해야합니다.

반환 값

컴파일 된 모듈을 가질 약속을 반환합니다.

webAssembly.compile ()을 사용하여 컴파일 된 모듈로 출력을 제공하는 한 가지 예를 살펴 보겠습니다.

<script> 
   fetch("findsquare.wasm") .then(bytes => bytes.arrayBuffer()) 
   .then(mod => {
      var compiledmod = WebAssembly.compile(mod);
      compiledmod.then(test=> {
         console.log(test); 
      })
   })
</script>

산출

console.log는 브라우저에서 확인하면 컴파일 된 모듈 세부 정보를 제공합니다.

모듈에는 가져 오기, 내보내기 및 customSections가있는 생성자 객체가 있습니다. 컴파일 된 모듈에 대한 자세한 정보를 얻기 위해 다음 API를 살펴 보겠습니다.

WebAssembly.instance

WebAssembly.instance를 사용하여 API는 출력을 얻기 위해 추가로 실행할 수있는 컴파일 된 모듈의 실행 가능한 인스턴스를 제공합니다.

통사론

구문은 다음과 같습니다.

new WebAssembly.Instance(compiled module)

반환 값

반환 값은 실행할 수있는 내보내기 함수 배열이있는 개체입니다.

<script> 
   fetch("findsquare.wasm") 
      .then(bytes => bytes.arrayBuffer())
      .then(mod => WebAssembly.compile(mod)).then(module => {
         let instance = new WebAssembly.Instance(module); 
         console.log(instance); 
      })
</script>

산출

출력은 아래와 같이 내보내기 기능의 배열을 제공합니다.

컴파일 된 C 코드에서 얻은 square 함수를 볼 수 있습니다.

square 함수를 실행하려면 다음을 수행하십시오.

<script>
   fetch("findsquare.wasm") 
   .then(bytes => bytes.arrayBuffer()) 
   .then(mod => WebAssembly.compile(mod)) 
   .then(module => { 
      let instance = new WebAssembly.Instance(module);
      console.log(instance.exports.square(15));
   })
</script>

출력은-

225

WebAssembly.instantiate

이 API는 모듈을 함께 컴파일하고 인스턴스화합니다.

통사론

구문은 다음과 같습니다-

WebAssembly.instantiate(arraybuffer, importObject)

매개 변수

arraybuffer − .wasm의 코드는 인스턴스화를위한 입력으로 제공하기 전에 형식화 된 배열 또는 배열 버퍼로 변환되어야합니다.

importObject− 가져 오기 개체는 모듈 내부에서 사용할 메모리, 가져온 함수에 대한 세부 정보가 있어야합니다. 공유 할 것이없는 경우 빈 모듈 객체 일 수 있습니다.

반환 값

모듈 및 인스턴스 세부 정보를 포함하는 promise를 반환합니다.

<script type="text/javascript">
   const importObj = {
      module: {}
   };
   fetch("findsquare.wasm")
      .then(bytes => bytes.arrayBuffer())
      .then(module => WebAssembly.instantiate(module, importObj)) 
      .then(finalcode => { 
         console.log(finalcode); console.log(finalcode.instance.exports.square(25)); 
      }); 
</script>

산출

코드를 실행하면 아래에 언급 된 출력이 표시됩니다.

WebAssembly.instantiateStreaming

이 API는 주어진 .wasm 코드에서 WebAssembly 모듈을 컴파일하고 인스턴스화합니다.

통사론

구문은 다음과 같습니다.

WebAssembly.instantiateStreaming(wasmcode, importObject);

매개 변수

wasmcode − wasm 코드를 제공하고 promise를 반환하는 fetch 또는 기타 API의 응답.

importObject− 가져 오기 개체는 모듈 내부에서 사용할 메모리, 가져온 함수에 대한 세부 정보가 있어야합니다. 공유 할 것이없는 경우 빈 모듈 객체가 될 수 있습니다.

반환 값

모듈 및 인스턴스 세부 정보를 포함하는 promise를 반환합니다.

예는 아래에 설명되어 있습니다.

<script type="text/javascript">     
   const importObj = { 
      module: {} 
   };
   WebAssembly.instantiateStreaming(fetch("findsquare.wasm"), importObj).then(obj => {
      console.log(obj); 
   }); 
</script>

브라우저에서 테스트하면 오류가 표시됩니다.

서버 측에서 작동하게하려면 mime 유형 application / wasm을 추가하거나 WebAssembly.instantiate (arraybuffer, importObject)를 사용해야합니다.

WebAssembly 지원은 Chrome, Firefox와 같이 현재 사용 가능한 모든 최신 브라우저에 추가됩니다. Firefox 버전 54 이상은 wasm 코드를 디버깅하는 특별한 기능을 제공합니다.

이를 위해 wasm을 호출하는 Firefox 브라우저 내에서 코드를 실행하십시오. 예를 들어 숫자의 제곱을 찾는 다음 C 코드를 고려하십시오.

C 프로그램의 예는 다음과 같습니다.

#include<stdio.h>
int square(int n) {
   return n*n;
}

WASM 탐색기를 사용하여 wasm 코드를 얻습니다.

WASM 코드를 다운로드하고이를 사용하여 브라우저에서 출력을 확인합니다.

wasm을로드하는 html 파일은 다음과 같습니다.

!doctype html> 
<html>
   <head>
      <meta charset="utf-8"> 
      <title>WebAssembly Square function</title> 
      <style> 
         div { 
            font-size : 30px; text-align : center; color:orange; 
         } 
      </style> 
   </head> 
   <body> 
      <div id="textcontent"></div> 
      <script> 
         let square; 
         fetch("findsquare.wasm").then(bytes => bytes.arrayBuffer()) 
            .then(mod => WebAssembly.compile(mod)) 
            .then(module => {return new WebAssembly.Instance(module) }) 
            .then(instance => {  
            square = instance.exports.square(13);
            console.log("The square of 13 = " +square);           
            document.getElementById("textcontent").innerHTML = "The square of 13 = " +square; 
         }); 
      </script> 
   </body> 
</html>

Firefox 브라우저를 열고 위의 html 파일을로드하고 디버거 도구를 엽니 다.

디버거 도구에 wasm : // 항목이 표시되어야합니다. wasm : //을 클릭하면 위와 같이 .wat 형식으로 변환 된 wasm 코드가 표시됩니다.

내 보낸 함수의 코드를 살펴보고 문제가 발생하면 코드를 디버그 할 수 있습니다. Firefox는 또한 중단 점을 추가하여 코드를 디버그하고 실행 흐름을 확인할 수 있습니다.

이 장에서는 C로 간단한 프로그램을 작성하고 .wasm으로 변환하고 브라우저에서 동일한 작업을 실행하여 "Hello World"라는 텍스트를 얻을 것입니다.

C 프로그램을 .wasm으로 변환하고 .html 파일 내에서 .wasm을 사용하는 wasm 탐색기 도구를 사용합니다.

사용할 수있는 Wasm 탐색기 도구 https://mbebenita.github.io/WasmExplorer/ looks as follows −

우리가 사용할 C 코드는 다음과 같습니다.

#include <stdio.h>
char *c_hello() {
   return "Hello World"; 
}

wasm 탐색기의 첫 번째 블록을 아래와 같이 C 코드로 업데이트하십시오.

WASM 및 WAT 및 Firefox x86 웹 어셈블리로 컴파일하려면 COMPILE 버튼을 클릭하십시오.

.wasm 파일을 가져 와서 저장하려면 다운로드를 사용하십시오. firstprog.wasm.

아래와 같이 firstprog.html이라는 .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을 사용하여 C 함수 c_hello ()에서 helloworld를 읽어 보겠습니다.

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 세부 정보는 다음과 같습니다.

c_hello () 함수에서 "Hello World"문자열을 얻으려면 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 ()를 사용하여 문자열로 변환해야합니다.

코드는 다음과 같습니다-

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에 추가되었으므로 문자열이 브라우저에 표시됩니다.

산출

출력은 아래에 언급되어 있습니다.

c / c ++ 코드에서 .wasm 파일을 얻는 방법을 살펴 보았습니다. 이 장에서는 wasm을 WebAssembly 모듈로 변환하고 브라우저에서 동일하게 실행합니다.

아래와 같이 C ++ 팩토리얼 코드를 사용하겠습니다.

int fact(int n) {
   if ((n==0)||(n==1))
      return 1;
   else
      return n*fact(n-1);
}

사용할 수있는 Wasm Explorer를 엽니 다. https://mbebenita.github.io/WasmExplorer/ as shown below −

첫 번째 열에는 C ++ 팩토리얼 함수가 있고 두 번째 열에는 WebAssembly 텍스트 형식이 있고 마지막 열에는 x86 어셈블리 코드가 있습니다.

WebAssembly 텍스트 형식-

(module
   (table 0 anyfunc)
   (memory $0 1)
   (export "memory" (memory $0)) (export "_Z4facti" (func $_Z4facti))
   (func $_Z4facti (; 0 ;) (param $0 i32) (result i32)
      (local $1 i32) (set_local $1
         (i32.const 1)
      )
      (block $label$0
         (br_if $label$0
            (i32.eq
               (i32.or
                  (get_local $0) (i32.const 1) ) (i32.const 1) ) ) (set_local $1
            (i32.const 1)
         )
         (loop $label$1
            (set_local $1 (i32.mul (get_local $0)
                  (get_local $1) ) ) (br_if $label$1 (i32.ne (i32.or (tee_local $0
                        (i32.add
                           (get_local $0) (i32.const -1) ) ) (i32.const 1) ) (i32.const 1) ) ) ) ) (get_local $1)
   )
)

C ++ 함수 fact "_Z4facti”을 WebAssembly Text 형식으로합니다.

다운로드 버튼을 클릭하여 wasm 코드를 다운로드하고 파일을 factorial.wasm으로 저장하십시오.

이제 .wasm 코드를 모듈로 변환하려면 다음을 수행해야합니다.

1 단계

다음을 사용하여 .wasm을 배열 버퍼로 변환하십시오. ArrayBuffer. ArrayBuffer 객체는 고정 길이 이진 데이터 버퍼를 반환합니다.

2 단계

ArrayBuffer의 바이트는 다음을 사용하여 모듈로 컴파일해야합니다. WebAssembly.compile(buffer) 함수.

그만큼 WebAssembly.compile() 함수는 주어진 바이트에서 WebAssembly.Module을 컴파일하고 반환합니다.

다음은 1 단계와 2 단계에서 설명하는 Javascript 코드입니다.

<script type="text/javascript">
   let factorial;
   fetch("factorial.wasm")
      .then(bytes => bytes.arrayBuffer())
      .then(mod => WebAssembly.compile(mod))
      .then(module => {return new WebAssembly.Instance(module) })
      .then(instance => {
      
      factorial = instance.exports._Z4facti;
      console.log('Test the output in Brower Console by using factorial(n)');
   });
</script>

코드 설명

  • Javascript 브라우저 API 가져 오기는 factorial.wasm의 내용을 가져 오는 데 사용됩니다.

  • 내용은 arrayBuffer ()를 사용하여 바이트로 변환됩니다.

  • 모듈은 WebAssembly.compile (mod)를 호출하여 바이트에서 생성됩니다.

  • 모듈의 인스턴스는 new를 사용하여 생성됩니다.

    WebAssembly.Instance(module)

  • 계승 함수 export _Z4facti는 WebAssembly.Module.exports ()를 사용하여 변수 계승에 할당됩니다.

여기, javascript 코드와 함께 module.html이 있습니다-

module.html

<!doctype html>
<html>
   <head>
      <meta charset="utf-8">
      <title>WebAssembly Module</title>
   </head>
   <body>
      <script>
      let factorial;
      fetch("factorial.wasm")
      .then(bytes => bytes.arrayBuffer())
      .then(mod => WebAssembly.compile(mod))
      .then(module => {return new WebAssembly.Instance(module) })
      .then(instance => {
         factorial = instance.exports._Z4facti;
         console.log('Test the output in Browser Console by using factorial(n)');
      });
      </script>
   </body>
</html>

산출

브라우저에서 module.html을 실행하여 출력을 확인하십시오.

이 장에서는 .wasm 출력을 검증하는 webassembly.validate () 함수에 대해 설명합니다. .wasm은 C, C ++ 또는 rust 코드를 컴파일 할 때 사용할 수 있습니다.

다음 도구를 사용하여 wasm 코드를 얻을 수 있습니다.

  • Wasm Fiddler는 https://wasdk.github.io/WasmFiddle/

  • WebAssembly Explorer는 https://mbebenita.github.io/WasmExplorer/.

통사론

구문은 다음과 같습니다.

WebAssembly.validate(bufferSource);

매개 변수

bufferSource− bufferSource에는 C, C ++ 또는 Rust 프로그램에서 가져온 바이너리 코드가 있습니다. typedarray 또는 ArrayBuffer의 형태입니다.

반환 값

이 함수는 .wasm 코드가 유효하면 true를 반환하고 그렇지 않으면 false를 반환합니다.

한 가지 예를 들어 보겠습니다. Wasm fiddler로 이동하십시오 .https://wasdk.github.io/WasmFiddle/, 선택한 C 코드와 wasm 코드를 입력하십시오.

빨간색으로 표시된 블록은 C 코드입니다. 중앙의 빌드 버튼을 클릭하여 코드를 실행합니다.

Wasm, 버튼을 클릭하여 .wasm 코드를 다운로드하십시오. 마지막에 .wasm을 저장하고 유효성 검사를 위해 .wasm을 사용하십시오.

예를 들면 : validate.html

<!doctype html>
<html>
   <head> 
      <meta charset="utf-8">
      <title>Testing WASM validate()</title>
   </head>
   <body>
      <script> 
         fetch('program.wasm').then(res => res.arrayBuffer() ).then(function(testbytes) {
         var valid = WebAssembly.validate(testbytes); 
            if (valid) {
               console.log("Valid Wasm Bytes!"); 
            } else {
               console.log("Invalid Wasm Code!"); 
            }
         }); 
      </script> 
   </body>
</html>

위의 .html 파일을 다운로드 .wasm 파일과 함께 wamp 서버에 호스팅했습니다. 다음은 브라우저에서 테스트 할 때의 출력입니다.

산출

출력은 다음과 같습니다.

WebAssembly에는 WASM이라는 바이너리 형식의 코드가 있습니다. WebAssembly에서 텍스트 형식을 가져올 수도 있으며 WAT (WebAssembly Text 형식)라고합니다. 개발자는 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_localset_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.

이전 장에서 우리는 코드를 작성하는 방법을 보았습니다. .wat즉, WebAssembly 텍스트 형식입니다. WebAssembly 텍스트 형식은 브라우저 내에서 직접 작동하지 않으며 브라우저 내에서 작동하려면 WASM과 같은 바이너리 형식으로 변환해야합니다.

WAT에서 WASM

.WAT를 .WASM으로 변환 해 보겠습니다.

우리가 사용할 코드는 다음과 같습니다.

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

이제 이동 WebAssembly Studio에서 사용할 수 있습니다. https://webassembly.studio/.

링크를 누르면 다음과 같은 내용이 표시됩니다.

Empty Wat 프로젝트를 클릭하고 하단의 Create 버튼을 클릭합니다.

아래와 같이 빈 프로젝트로 이동합니다.

main.wat를 클릭하고 기존 코드를 자신의 것으로 바꾸고 저장 버튼을 클릭합니다.

저장되면 빌드를 클릭하여 .wasm-

빌드가 성공하면 아래와 같이 생성 된 .wasm 파일을 볼 수 있습니다.

main.wasm 파일 아래로 내려가 .html 파일 내에서 사용하여 아래와 같이 출력을 확인하십시오.

예를 들어- add.html

<!doctype html>
<html>
   <head>
      <meta charset="utf-8">
      <title>WebAssembly Add Function</title>
   </head>
   <body>
      <script> 
         let sum; 
         fetch("main.wasm")
            .then(bytes => bytes.arrayBuffer()) 
            .then(mod => WebAssembly.compile(mod)) .then(module => {
            
            return new WebAssembly.Instance(module) 
         })
         .then(instance => {
            sum = instance.exports.add(10,40); 
            console.log("The sum of 10 and 40 = " +sum); 
         }); 
      </script>
   </body>
</html>

함수 add는 코드에 표시된대로 내보내집니다. 전달 된 매개 변수는 2 개의 정수 값 10과 40이며 그 합계를 반환합니다.

산출

출력이 브라우저에 표시됩니다.

동적 연결은 런타임 중에 두 개 이상의 모듈이 함께 연결되는 프로세스입니다.

동적 연결이 어떻게 작동하는지 보여주기 위해 C 프로그램을 사용하고 Ecmascript sdk를 사용하여 wasm으로 컴파일합니다.

그래서 여기에 우리는-

test1.c

int test1(){ 
   return 100; 
}

test2.c

int test2(){ 
   return 200; 
}

main.c

#include <stdio.h>

int test1(); 
int test2();
int main() { 
   int result = test1() + test2(); 
   return result; 
}

main.c 코드에서는 test1.c 및 test2.c 내부에 정의 된 test1 () 및 test2 ()를 사용합니다. WebAssembly에서 이러한 모듈을 연결하는 방법을 확인해 보겠습니다.

위 코드를 컴파일하는 명령은 다음과 같습니다. 명령과 같이 동적 연결을 위해 SIDE_MODULE = 1을 사용합니다.

emcc test1.c test2.c main.c -s SIDE_MODULE=1 -o maintest.wasm

사용 가능한 WasmtoWat 사용 https://webassembly.github.io/wabt/demo/wasm2wat/, maintest.wasm의 WebAssembly 텍스트 형식을 가져옵니다.

(module 
   (type $t0 (func (result i32))) (type $t1 (func)) 
   (type $t2 (func (param i32))) (type $t3 (func (param i32 i32) (result i32))) 
   (import "env" "stackSave" (func $env.stackSave (type $t0))) 
   (import "env" "stackRestore" (func $env.stackRestore (type $t2))) 
   (import "env" "__memory_base" (global $env.__memory_base i32)) (import "env" "__table_base" (global $env.__table_base i32)) 
   (import "env" "memory" (memory $env.memory 0)) (import "env" "table" (table $env.table 0 funcref)) 
   (func $f2 (type $t1) 
      (call $__wasm_apply_relocs) ) (func $__wasm_apply_relocs (export "__wasm_apply_relocs") (type $t1)) (func $test1 (export "test1") (type $t0) (result i32) (local $l0 i32) 
      (local.set $l0 (i32.const 100) ) (return (local.get $l0)
      )
   )
   (func $test2 (export "test2") (type $t0) (result i32) 
      (local $l0 i32) (local.set $l0 
         (i32.const 200)) 
      (return 
         (local.get $l0) ) ) (func $__original_main 
      (export "__original_main") 
      (type $t0) (result i32) (local $l0 i32) 
      (local $l1 i32) (local $l2 i32) 
      (local $l3 i32) (local $l4 i32) 
      (local $l5 i32) (local $l6 i32) 
      (local $l7 i32) (local $l8 i32) 
      (local $l9 i32) (local.set $l0(call $env.stackSave)) (local.set $l1 (i32.const 16))
      (local.set $l2 (i32.sub (local.get $l0) (local.get $l1))) (call $env.stackRestore (local.get $l2) ) (local.set $l3(i32.const 0)) 
      (i32.store offset=12 (local.get $l2) (local.get $l3)) 
      (local.set $l4 (call $test1)) 
      (local.set $l5 (call $test2)) 
      (local.set $l6 (i32.add (local.get $l4) (local.get $l5))) (i32.store offset=8 (local.get $l2) (local.get $l6)) (local.set $l7 (i32.load offset=8 (local.get $l2))) (local.set $l8 (i32.const 16)) 
      (local.set $l9 (i32.add (local.get $l2) (local.get $l8))) (call $env.stackRestore (local.get $l9)) (return(local.get $l7))
   )
   (func $main (export "main") (type $t3) 
      (param $p0 i32) (param $p1 i32) 
      (result i32) 
      (local $l2 i32) (local.set $l2 
      (call $__original_main)) (return (local.get $l2))
   ) 
   (func $__post_instantiate (export "__post_instantiate") (type $t1) (call $f2)) (global $__dso_handle (export "__dso_handle") i32 (i32.const 0))
)

WebAssembly 텍스트 형식에는 아래와 같이 정의 된 일부 가져 오기가 있습니다.

(import "env" "stackSave" (func $env.stackSave (type $t0)))       
(import "env" "stackRestore" (func $env.stackRestore (type $t2))) 
(import "env" "__memory_base" (global $env.__memory_base i32)) (import "env" "__table_base" (global $env.__table_base i32)) 
(import "env" "memory" (memory $env.memory 0)) (import "env" "table" (table $env.table 0 funcref))

이것은 emcc (emscripten sdk)로 코드를 컴파일 할 때 추가되며 WebAssembly에서 메모리 관리를 다룹니다.

가져 오기 및 내보내기 작업

이제 출력을 보려면 .wat 코드에서 볼 수있는 가져 오기를 정의해야합니다.

(import "env" "stackSave" (func $env.stackSave (type $t0)))       
(import "env" "stackRestore" (func $env.stackRestore (type $t2))) 
(import "env" "__memory_base" (global $env.__memory_base i32)) (import "env" "__table_base" (global $env.__table_base i32)) 
(import "env" "memory" (memory $env.memory 0)) (import "env" "table" (table $env.table 0 funcref))

위의 용어는 다음과 같이 설명됩니다.

  • env.stackSave − emscripten 컴파일 된 코드에서 제공하는 기능인 스택 관리에 사용됩니다.

  • env.stackRestore − emscripten 컴파일 된 코드에서 제공하는 기능인 스택 관리에 사용됩니다.

  • env.__memory_base− env.memory에서 사용되며 wasm 모듈 용으로 예약 된 변경 불가능한 i32 전역 오프셋입니다. 모듈은 데이터 세그먼트의 이니셜 라이저에서이 전역을 사용할 수 있으므로 올바른 주소에로드됩니다.

  • env.__table_base− env.table에서 사용되며 wasm 모듈 용으로 예약 된 변경 불가능한 i32 전역 오프셋입니다. 모듈은 테이블 요소 세그먼트의 이니셜 라이저에서이 전역을 사용할 수 있으므로 올바른 오프셋에서로드됩니다.

  • env.memory − wasm 모듈간에 공유해야하는 메모리 세부 정보가 있습니다.

  • env.table − wasm 모듈간에 공유해야하는 테이블 세부 정보가 있습니다.

수입은 다음과 같이 자바 스크립트로 정의되어야합니다-

var wasmMemory = new WebAssembly.Memory({'initial': 256,'maximum': 65536}); 
const importObj = { 
   env: {
      stackSave: n => 2, stackRestore: n => 3, //abortStackOverflow: () => {
         throw new Error('overflow'); 
      }, 
      table: new WebAssembly.Table({ 
         initial: 0, maximum: 65536, element: 'anyfunc' 
      }), __table_base: 0,
      memory: wasmMemory, __memory_base: 256 
   } 
};

다음은 WebAssembly.instantiate 내에서 importObj를 사용하는 자바 스크립트 코드입니다.

<!DOCTYPE html> 
<html>
   <head>
      <meta charset="UTF-8">
   </head>
   <body>
      <script>
         var wasmMemory = new WebAssembly.Memory({'initial': 256,'maximum': 65536}); 
         const importObj = {
            env: {
               stackSave: n => 2, stackRestore: n => 3, //abortStackOverflow: () => {
                  throw new Error('overflow'); 
               }, 
               table: new WebAssembly.Table({ 
                  initial: 0, maximum: 65536, element: 'anyfunc' 
               }), __table_base: 0,
               memory: wasmMemory, __memory_base: 256 
            } 
         };
         fetch("maintest.wasm") .then(bytes => bytes.arrayBuffer()) .then(
            module => WebAssembly.instantiate(module, importObj)
         )
         .then(finalcode => {        
            console.log(finalcode);     
            console.log(WebAssembly.Module.imports(finalcode.module)); 
            console.log(finalcode.instance.exports.test1());    
            console.log(finalcode.instance.exports.test2());   
            console.log(finalcode.instance.exports.main()); 
         });
      </script>
   </body>
</html>

산출

출력은 다음과 같습니다.

WebAssembly.org의 공식 웹 사이트에 따라 https://webassembly.org/docs/security/ 보안 측면에서 WebAssembly의 주요 목표는 다음과 같습니다.

WebAssembly의 보안 모델에는 두 가지 중요한 목표가 있습니다.

  • 버그가 있거나 악성 모듈로부터 사용자를 보호합니다.

  • 개발자에게 (1)의 제약 내에서 안전한 응용 프로그램을 개발하기위한 유용한 기본 요소 및 완화 방법을 제공합니다.

컴파일 된 코드, 즉 C / C ++ / Rust의 WASM은 브라우저 내에서 직접 실행되지 않으며 Javascript API를 사용합니다. WASM 코드는 샌드 박스 처리됩니다. 즉, Javascript API 래퍼를 통해 실행되고 브라우저는 API를 사용하여 WASM과 통신합니다.

다음은 브라우저 내에서 .wasm 파일을 사용하는 예입니다.

예- C Program

#include<stdio.h> 
int square(int n) { 
   return n*n; 
}

WASM 탐색기를 사용하여 wasm 코드를 얻습니다.

WASM 코드를 다운로드하고이를 사용하여 API를 테스트하십시오.

<script type="text/javascript"> 
   const importObj = {
      module: {}
   }; 
   fetch("findsquare.wasm")      
      .then(bytes => bytes.arrayBuffer())          
      .then(module => WebAssembly.instantiate(module,importObj))                 
      .then(finalcode => {
         
      console.log(finalcode); console.log(finalcode.instance.exports.square(25)); 
   }); 
</script>

산출

다음과 같은 출력이 표시됩니다.

내보내기 개체에는 호출 할 함수에 대한 참조가 있습니다. 함수 square를 호출하려면 다음과 같이해야합니다.

console.log(finalcode.instance.exports.square(25));

WASM 컴파일 된 코드의 문제

다음은 WASM 컴파일 코드의 문제입니다.

  • wasm으로 컴파일하면서 악성 코드가 삽입되어 있는지 확인하기가 어렵습니다. 현재 코드를 검증 할 수있는 도구가 없습니다.

  • Wasm은 분석하기 어렵고 버그 / 악성 코드는 브라우저 내에서 쉽게 실행할 수 있습니다.

이 장에서는 간단한 C 프로그램을 javascript로 컴파일하고 브라우저에서 동일한 작업을 실행합니다.

예를 들어- C Program

#include<stdio.h> 
int square(int n) { 
   return n*n; 
}

wa / 폴더에 emsdk를 설치했습니다. 같은 폴더에 다른 폴더 cprog /를 만들고 위 코드를 square.c로 저장합니다.

이전 장에서 이미 emsdk를 설치했습니다. 여기서는 위의 c 코드를 컴파일하기 위해 emsdk를 사용할 것입니다.

아래와 같이 명령 프롬프트에서 test.c를 컴파일하십시오.

emcc square.c -s STANDALONE_WASM –o findsquare.wasm

emcc 명령은 코드 컴파일을 처리하고 .wasm 코드를 제공합니다. .wasm 파일 만 제공하는 STANDALONE_WASM 옵션을 사용했습니다.

예- findsquare.html

<!doctype html> 
<html>
   <head>
      <meta charset="utf-8">
      <title>WebAssembly Square function</title>
      <style>
         div { 
            font-size : 30px; text-align : center; color:orange; 
         } 
      </style>
   </head> 
   <body>
      <div id="textcontent"></div>
      <script> 
      let square; fetch("findsquare.wasm").then(bytes => bytes.arrayBuffer()) 
      .then(mod => WebAssembly.compile(mod)) .then(module => {
         return new WebAssembly.Instance(module) 
      }) 
      .then(instance => {
         square = instance.exports.square(13); 
         console.log("The square of 13 = " +square);         
         document.getElementById("textcontent").innerHTML = "The square of 13 = " +square; 
      }); 
      </script>
   </body>
</html>

산출

출력은 다음과 같습니다.

이 장에서는 간단한 C ++ 프로그램을 자바 스크립트로 컴파일하고 브라우저에서 동일한 작업을 실행합니다.

C ++ 프로그램-주어진 숫자 반전.

#include <iostream> 
int reversenumber(int n) { 
   int reverse=0, rem; 
   while(n!=0) { 
      rem=n%10; reverse=reverse*10+rem; n/=10; 
   } 
   return reverse; 
}

wa / 폴더에 emsdk를 설치했습니다. 같은 폴더에 다른 폴더 cprog /를 만들고 위의 코드를 reverse.cpp로 저장합니다.

이전 장에서 이미 emsdk를 설치했습니다. 여기서는 위의 c 코드를 컴파일하기 위해 emsdk를 사용할 것입니다.

아래와 같이 명령 프롬프트에서 test.c를 컴파일하십시오.

emcc reverse.cpp -s STANDALONE_WASM –o reverse.wasm

emcc 명령은 코드 컴파일을 처리하고 .wasm 코드를 제공합니다.

예- reversenumber.html

<!doctype html> 
<html>
   <head> 
      <meta charset="utf-8">
      <title>WebAssembly Reverse Number</title>
      <style>
         div { 
            font-size : 30px; text-align : center; color:orange; 
         } 
      </style>
   </head>
   <body>
      <div id="textcontent"></div>
      <script> 
         let reverse; 
         fetch("reverse.wasm")  
            .then(bytes => bytes.arrayBuffer()) 
            .then(mod => WebAssembly.compile(mod)) 
            .then(module => {return new WebAssembly.Instance(module) })
            .then(instance => { 
            
            console.log(instance); 
            reverse = instance.exports._Z13reversenumberi(1439898); 
            console.log("The reverse of 1439898 = " +reverse); 
            document.getElementById("textcontent")
               .innerHTML = "The reverse of 1439898 = " +reverse; 
         }); 
      </script>
   </body>
</html>

산출

출력은 다음과 같습니다.

RUST 컴파일 코드를 얻기 위해 WebAssembly.studio 도구를 사용합니다.

Go to 에서 사용할 수있는 WebAssembly.studio 로 이동합니다.https://webassembly.studio/ 그러면 아래와 같이 화면이 표시됩니다.

Empty Rust Project를 클릭합니다. 완료되면 src / 폴더에 3 개의 파일이 있습니다.

main.rs 파일을 열고 원하는 코드를 변경하십시오.

두 개의 주어진 숫자를 추가하는 다음 함수를 추가하고 있습니다.

fn add_ints(lhs: i32, rhs: i32) -> i32 {
   lhs+rhs
}

main.rs에서 사용 가능한 코드는 다음과 같습니다.

#[no_mangle]
pub extern "C" fn add_one(x: i32) -> i32 {
   x + 1
}

아래와 같이 fn add_one을 귀하의 것으로 교체하십시오-

#[no_mangle]
pub extern "C" fn add_ints(lhs: i32, rhs: i32) -> i32 {
   lhs+rhs
}

main.js에서 함수 이름을 add_one에서 add_ints로 변경합니다.

fetch('../out/main.wasm').then(
   response =>
   response.arrayBuffer()
).then(bytes => WebAssembly.instantiate(bytes)).then(results => {
   instance = results.instance;
   document.getElementById("container").textContent = instance.exports.add_one(41);
}).catch(console.error);

instance.exports.add_one을 instance.exports.add_ints (100,100)로 바꿉니다.

fetch('../out/main.wasm').then(
   response =>
   response.arrayBuffer()
).then(bytes => WebAssembly.instantiate(bytes)).then(results => {
   instance = results.instance;
   document.getElementById("container").textContent = instance.exports.add_ints(100,100)
}).catch(console.error);

webassembly.studio UI에서 제공되는 빌드 버튼을 클릭하여 코드를 빌드합니다.

빌드가 완료되면 UI에서 사용할 수있는 실행 버튼을 클릭하여 출력을 확인합니다.

instance.exports.add_ints (100,100)를 전달 했으므로 200으로 출력을 얻습니다.

마찬가지로 rust에 대해 다른 프로그램을 작성하고 webassembly.studio에서 컴파일 할 수 있습니다.

Go는 버전 1.1부터 WebAssembly에 대한 지원을 추가했습니다. 먼저 다운로드하여 테스트하려면 Go.

golang 사이트로 이동하십시오. https://golang.org/dl/다운로드 Go를 클릭하십시오. 운영 체제에 따라 Go를 다운로드하여 설치하십시오.

완료되면 두 개의 숫자를 추가하는 간단한 프로그램을 작성하십시오.

testnum.go

package main
import "fmt"
func main() { 
   var a int = 100 
   var b int = 200 
   var ret int 
   ret = sum(a, b) 
   fmt.Printf( "Sum is : %d\n", ret ) 
}
 
/* function returning the max between two numbers */ 
func sum(num1, num2 int) int { 
   return num1+num2 
}

위의 코드를 wasm으로 컴파일하려면 먼저 Go에서 환경 변수를 설정하십시오.

다음 명령을 실행해야합니다-

Set GOOS=js
GOARCH=wasm

완료되면 아래 명령을 실행하십시오-

go build -o testnum.wasm testnum.go

명령이 실행되면 testnum.wasm 파일을 가져와야합니다.

이제 브라우저에서 코드를 테스트 해 보겠습니다. 이를 위해 go와 함께 설치된 wasm_exec.js를 가져와야합니다.

wasm_exec.js 파일은 go의 misc / wasm / 폴더에서 사용할 수 있습니다.

다음은 wasm_exec.js 및 testnum.wasm을 사용하는 testgo.html 용 코드입니다.

<html> 
   <head> 
      <meta charset="utf-8"/>
      <script src="wasm_exec.js"></script>
   </head>
   <body>
      <script type="text/javascript"> 
         const importObj = {
            module: {} 
         };
         const go = new Go(); 
         async function fetchAndInstantiate() { 
            const response = await fetch("testnum.wasm"); 
            const buffer = await response.arrayBuffer(); 
            const obj = await WebAssembly.instantiate(buffer, go.importObject); 
            console.log(obj); 
            go.run(obj.instance); 
         } 
         fetchAndInstantiate(); 
      </script>
   </body>
</html>

산출

출력은 다음과 같습니다.

Javascript에는 wasm 코드와 함께 작동 할 수있는 API가 많이 있습니다. API는 nodejs에서도 지원됩니다.

시스템에 NODEJS를 설치하십시오. Factorialtest.js 파일을 만듭니다.

아래와 같이 C ++ 팩토리얼 코드를 사용하겠습니다.

int fact(int n) {
   if ((n==0)||(n==1))
      return 1;
   else
      return n*fact(n-1);
}

다음 위치에서 사용할 수있는 Wasm Explorer를 엽니 다. https://mbebenita.github.io/WasmExplorer/ 아래와 같이-

첫 번째 열에는 C ++ 팩토리얼 함수가 있고 두 번째 열에는 WebAssembly 텍스트 형식이 있고 마지막 열에는 x86 어셈블리 코드가 있습니다.

WebAssembly 텍스트 형식은 다음과 같습니다.

(module
   (table 0 anyfunc)
   (memory $0 1) (export "memory" (memory $0))
   (export "_Z4facti" (func $_Z4facti)) (func $_Z4facti (; 0 ;) (param $0 i32) (result i32) (local $1 i32)
      (set_local $1(i32.const 1)) (block $label$0 (br_if $label$0 (i32.eq (i32.or (get_local $0)
                  (i32.const 1)
               )
               (i32.const 1)
            )
         )
         (set_local $1 (i32.const 1) ) (loop $label$1 (set_local $1
               (i32.mul
                  (get_local $0) (get_local $1)
               )
            )
            (br_if $label$1
               (i32.ne
                  (i32.or
                     (tee_local $0 (i32.add (get_local $0)
                           (i32.const -1)
                        )
                     )
                     (i32.const 1)
                  )
                  (i32.const 1)
               )
            )
         )
      )
      (get_local $1)
   )
)

C ++ 함수 팩트는 "_Z4facti”을 WebAssembly Text 형식으로합니다.

Factorialtest.js

const fs = require('fs');
const buf = fs.readFileSync('./factorial.wasm');
const lib = WebAssembly.instantiate(new Uint8Array(buf)).
   then(res => {
      for (var i=1;i<=10;i++) {
         console.log("The factorial of "+i+" = "+res.instance.exports._Z4facti(i))
      }
   }
);

명령 줄에서 factorialtest.js 명령 노드를 실행하면 출력은 다음과 같습니다.

C:\wasmnode>node factorialtest.js
The factorial of 1 = 1
The factorial of 2 = 2
The factorial of 3 = 6
The factorial of 4 = 24
The factorial of 5 = 120
The factorial of 6 = 720
The factorial of 7 = 5040
The factorial of 8 = 40320
The factorial of 9 = 362880
The factorial of 10 = 3628800

이 장에서는 WebAssembly와 관련된 예제를 설명합니다.

예 1

다음은 최대 요소를 얻는 C 프로그램의 예입니다-

void displaylog(int n);
/* function returning the max between two numbers */ 
int max(int num1, int num2) {
   /* local variable declaration */ int result; 
   if (num1 > num2) 
      result = num1; 
   else result = num2;
      displaylog(result);
   return result; 
}

wasm fiddle에서 코드를 컴파일하고 .wasm 및 .wat 코드를 다운로드하십시오.

Wat code

Wat 코드는 다음과 같습니다.

(module 
   (type $FUNCSIG$vi (func (param i32))) (import "env" "displaylog" (func $displaylog (param i32))) 
   (table 0 anyfunc) 
   (memory $0 1) (export "memory" (memory $0)) 
   (export "max" (func $max)) (func $max (; 1 ;) (param $0 i32) (param $1 i32) (result i32) 
      (call $displaylog (tee_local $0 
            (select 
               (get_local $0) (get_local $1) 
               (i32.gt_s (get_local $0) (get_local $1)) 
            )
         )
      )
      (get_local $0) 
   )
)

.wasm 코드를 다운로드하고 아래와 같이 .html 파일에서 사용합니다.

<!DOCTYPE html> 
<html>
   <head>
      <meta charset="UTF-8">
   </head>
   <body>
      <script>
         const importObj = {
            env: { 
               displaylog: n => alert("The max of (400, 130) is " +n) 
            } 
         };
         fetch("testmax.wasm") .then(bytes => bytes.arrayBuffer()) 
            .then(module => WebAssembly.instantiate(module, importObj)) 
            .then(finalcode => { 
            console.log(finalcode); 
            console.log(finalcode.instance.exports.max(400,130)); 
         }); 
      </script> 
   </body>
</html>

산출

출력은 다음과 같습니다.

예 2

다음은 주어진 숫자의 피보나치 시리즈를 가져 오는 C ++ 코드입니다.

#include <iostream>>
void displaylog(int n); 
int fibonacciSeries(int number) {
   int n1=0,n2=1,n3,i; 
   for(i=2;i<number;++i) { 
      n3=n1+n2; displaylog(n); n1=n2; n2=n3;
   }
   return 0; 
}

wasm 탐색기를 사용하여 코드를 컴파일하고 있습니다. Wat와 Wasm을 다운로드하고 브라우저에서 동일하게 테스트하십시오.

아래에 언급 된 코드를 사용할 수 있습니다.

<!DOCTYPE html> 
<html>
   <head> 
      <meta charset="UTF-8">
   </head>
   <body>
      <script> 
         const importObj = { 
            env: { _Z10displaylogi: n => console.log(n) } 
         };
         fetch("fib.wasm") 
            .then(bytes => bytes.arrayBuffer()) 
            .then(module => WebAssembly.instantiate(module, importObj)) 
            .then(finalcode => { 
            console.log(finalcode); 
            console.log(finalcode.instance.exports._Z15fibonacciSeriesi(10)); 
         });
      </script> 
   </body>
</html>

산출

출력은 다음과 같습니다.

예제 3

다음은 주어진 배열에 요소를 추가하는 Rust 코드입니다.

fn add_array(x: i32) -> i32 { 
   let mut sum = 0; 
   let mut numbers = [10,20,30]; for i in 0..3 { 
      sum += numbers[i]; 
   } 
   sum 
}

RUST를 wasm으로 컴파일하기 위해 WebAssembly Studio를 사용할 것입니다.

코드를 빌드하고 wasm 파일을 다운로드하고 브라우저에서 동일하게 실행하십시오.

<!DOCTYPE html> 
<html>
   <head> 
      <meta charset="UTF-8">
   </head>
      <body>
      <script> 
         const importObj = { 
            env: {
            } 
         };
         fetch("add_array.wasm") .then(bytes => bytes.arrayBuffer())
            .then(module => WebAssembly.instantiate(module, importObj)) 
            .then(finalcode => { 
            console.log(finalcode); 
            console.log(finalcode.instance.exports.add_array());
         }); 
      </script> 
   </body> 
</html>

산출

출력은 다음과 같습니다.