WebAssembly - डायनामिक लिंकिंग

डायनेमिक लिंकिंग वह प्रक्रिया है जिसमें रनटाइम के दौरान दो या दो से अधिक मॉड्यूल को एक साथ जोड़ा जाएगा।

यह प्रदर्शित करने के लिए कि गतिशील लिंकिंग कैसे काम करती है, हम सी प्रोग्राम का उपयोग करेंगे और इसे एक्मास्क्रिप्टस्क्रिप्ट एसडीके का उपयोग करके संकलित करेंगे।

तो यहाँ हमारे पास है -

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 () और test2 () का उपयोग करता है, जो test1.c और test2.c के अंदर परिभाषित होते हैं। आइए देखें कि इन मॉड्यूल को 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 - इसका उपयोग स्टैक प्रबंधन के लिए किया जाता है, एक कार्यक्षमता जो ईम्स्क्रिप्टस्क्रिप्ट संकलित कोड द्वारा प्रदान की जाती है।

  • env.stackRestore - इसका उपयोग स्टैक प्रबंधन के लिए किया जाता है, एक कार्यक्षमता जो ईम्स्क्रिप्टस्क्रिप्ट संकलित कोड द्वारा प्रदान की जाती है।

  • env.__memory_base- यह एक अपरिवर्तनीय i32 वैश्विक ऑफसेट है, जिसका उपयोग env.memory में किया जाता है और इसे wasm मॉड्यूल के लिए आरक्षित किया जाता है। मॉड्यूल इस वैश्विक का उपयोग अपने डेटा सेगमेंट के इनिशियलाइज़र में कर सकता है, ताकि, वे सही पते पर लोड हो सकें।

  • env.__table_base- यह एक अपरिवर्तनीय i32 वैश्विक ऑफसेट है, जिसका उपयोग env.table में किया जाता है और इसे wasm मॉड्यूल के लिए आरक्षित किया जाता है। मॉड्यूल इस वैश्विक का उपयोग इसके टेबल एलिमेंट सेगमेंट के इनिशियलाइज़र में कर सकता है, ताकि, वे सही ऑफसेट पर लोड हो सकें।

  • 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>

उत्पादन

आउटपुट इस प्रकार है -