0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

什么是wasm組件?使用Rust開發(fā)wasm組件實戰(zhàn)

OSC開源社區(qū) ? 來源:凹凸實驗室 ? 2023-09-22 11:30 ? 次閱讀

什么是wasm組件?

wasm 全稱 WebAssembly,是通過虛擬機的方式,可以在服務端、客戶端如瀏覽器等環(huán)境執(zhí)行的二進制程序。它有速度快、效率高、可移植的特點。

對我們 Web 前端工程最大的好處就是可以在瀏覽器端使用二進制程序處理一些計算量大的處理,使用他比 javascript 快的特點優(yōu)化性能。

目前瀏覽器對wasm的兼容性如下:

cd675886-5871-11ee-939d-92fbcf53809c.jpghttps://img10.360buyimg.com/imagetools/jfs/t1/180904/35/36038/170761/64ded9bdF6f54c383/e85e037cdd4fa1fd.jpg

在移動端除了 android 4.4 和 ios 10 下不支持外,其他版本都能提供支持。還需要注意的是 wasm 有可能占用大量內(nèi)存,使用第三方包含 wasm 調(diào)用的組件需要注意內(nèi)存占用防止閃退。

為什么用Rust?

wasm模塊 可以用多種語言來編譯,包括 C/C++/C#、Rust、JAVA、Go。在這里使用 Rust 是因為他有嚴格的內(nèi)存管理機制,從語法上盡量避免內(nèi)存溢出,讓工程師寫出更安全的程序。

而且還有配套的工具 wasm-pack,讓使用 Rust 編寫的代碼,編譯包裝成 npm 包,讓使用這段程序的其他代碼可以像使用其他公共庫一樣調(diào)用,不需要額外學習成本。

工具安裝

安裝 rustup,它是 Rust 安裝器和版本管理工具。對于 web 前端來說相當于 nvm 這樣的工具。按照 Rust 官網(wǎng)的方法安裝:https://www.rust-lang.org/zh-CN/tools/install同時也會安裝 cargo,它是 Rust 的構建工具和包管理器。對于 web 前端來說相當于 npm 這樣的工具。

安裝 wasm-pack,他是上文提到的把 Rust 程序編譯包裝成 wasm 組件的工具。同樣按照 wasm-pack 官網(wǎng)的方法安裝:https://rustwasm.github.io/wasm-pack/installer/

使用 wasm 模板使用 wasm-pack 提供的模板可以快速生成 Rust 的 wasm 項目。

cargogenerate--githttps://github.com/rustwasm/wasm-pack-template

輸入希望的項目目錄名稱,將新建目錄并在其中生成項目。

在目錄下我們可以看到幾個文件,其中一個是 Cargo.toml ,這個是 Rust 項目的描述文件,對于 web 前端來說相當于 package.json 文件。

項目目錄下還有一個 src 目錄,里面有 lib.rs 和 utils.rs 兩個文件,其中 lib.rs 這個文件就是我們主要的邏輯入口,他引用了 wasm-bindgen 庫來輸出暴露給外部調(diào)用的接口,在函數(shù)之前加上#[wasm_bindgen]可以讓外部調(diào)用這個方法。

編譯項目

本來 Rust 的項目編譯用的是 cargo build 的命令,但是我們這里是希望編譯 wasm 組件,所以用的是 wasm-pack build 命令。

執(zhí)行后會在項目目錄下的 pkg 目錄下生成編譯后的產(chǎn)品,是一個 npm 包的結構。需要調(diào)用這個組件的邏輯只需要像其他公共包一樣 import 就可以使用了。

實戰(zhàn)

以上的就是 wasm-pack 官方的教程,還有其他組件測試、發(fā)布等的流程先不在這里介紹了。以下用一個實際開發(fā)中的模塊來說一下開發(fā) wasm 組件過程中遇到的問題和解決方法。

背景—

需要使用的 wasm 組件是一個優(yōu)化3D模型的方法,傳入一個模型的頂點信息和距離閾值,比較每個頂點位置之間的距離,如果沒達到閾值距離就合并這兩個頂點,以達到減少頂點的優(yōu)化目的。

原邏輯是使用 javascript 編寫的,在模型頂點數(shù)量比較多的時候執(zhí)行的時間比較長。這種大量計算的情況就很適合使用 wasm 來處理。

數(shù)據(jù)傳遞—

頂點信息是存儲在一個 Float32Array 的數(shù)組中的,而 wasm 設計上除了 int 和 float 類型(對應 javascript 就是 number 類型)可以直接傳遞外,其他的類型都通過地址來傳遞。這對我們的程序來說是好消息,因為頂點信息的數(shù)據(jù)非常多,如果以值傳遞,就需要做數(shù)據(jù)復制,這個過程消耗的時間可能比我們換成 wasm 處理減少的時間還要多。得益這個特點,我們的入?yún)⒖梢灾苯觽魅搿?/p>

/*---rust----*/

//rust獲取javasctipt數(shù)據(jù)
pubfnadd_attribute(&mutself,attribute:&Float32Array,item_size:u32){
self.attributes.push(BufferAttribute{
array:attribute.to_vec(),
item_size,
});
}
/*---javascript----*/

//javascript傳遞數(shù)據(jù)到rust
for(constnameofattributeNames){
constattr=attrArrays[name]
bg.add_attribute(attr.array,attr.itemSize)
}

而計算后的結果,wasm 也提供了返回數(shù)組的指針和數(shù)組長度的方法,javascript 可以讀取 wasm 的內(nèi)存空間,根據(jù)這兩個值構造新的頂點信息Float32Array。

/*---rust----*/

//返回指定數(shù)據(jù)的內(nèi)存指針位置
pubfnget_attribute_ptr(&self,index:usize)->*constf32{
self.attributes[index].array.as_ptr()
}

//返回指定數(shù)據(jù)的長度
pubfnget_attribute_length(&self,index:usize)->usize{
self.attributes[index].array.len()
}
/*---javascript----*/

//javascript或取rust內(nèi)存空間中的指定部分,構建Float32Array
constptr=bg.get_attribute_ptr(i)
constlength=bg.get_attribute_length(i)

constbuffer=newattr.array.constructor(wasm.getMemory().buffer,ptr,length)

數(shù)據(jù)類型—

合并頂點計算的邏輯中,有一段是這樣的:每個頂點的位置、UV等信息,經(jīng)過給定的精度計算后,生成一個特征值,之后比較每個頂點的特征值,如果是相同的話就表示這兩個頂點可以合并。

原 javascript 版本的代碼是逐個信息按順序,加上分隔號,拼成一個字符串。

Rust 版本的代碼如果也按同樣的方法處理,因為頂點的信息量是不定的,有可能只有位置信息,也有可能有UV、法線、顏色等信息,所以生成的特征值字符串長度也不確定。

Rust 對於可變長度的字符串使用 String 類型,每次對字符串使用push_str方法增加內(nèi)容。得到的結果 wasm 版本的執(zhí)行速度跟 javascript 版本相差不大,甚至在某些情況下耗時還更多,經(jīng)過逐個過程作排查,發(fā)現(xiàn)是在生成特征值和在表中查詢特征值這個過程中花費的時間比較多。

根據(jù)程序的意圖,特征值并不一定要是字符串,只需要在不同輸入值的時候能夠輸出相關的值就可以,這跟生成 hash 值的需求是一樣的,于是考慮將特征值生成替換成 hash 值計算。

因為在存儲特征值的表使用了std::hash_map類型,于是 hash 值也使用了其下的std::DefaultHasher類來計算

usestd::DefaultHasher;

...

letmuthasher=DefaultHasher::new();

forjin0..self.attributes.len(){
...

letvalue=(attr.array[i*attr.item_sizeasusize+indexasusize]
*self.shift_multiplier)
.trunc()asi32;

hasher.write_i32(value);

...
}

lethash=hasher.finish();

需要注意的是對寫入不同類型的內(nèi)容,需要調(diào)用不同的方法,頂點信息中的值是正負值都用,經(jīng)過精度計算后取整得到的值類型是i32,所以用write_i32來寫入內(nèi)容。

生成的 hash 值為u64,作為hash_map的key記錄對應頂點的序號。

替換特征值的類型之后,wasm 版本的耗時達到了 javascript 版本的 1/2,基本符合 wasm 設計的性能范圍。

適配打包工具—

wasm-pack 工具打包出來的 npm 包,可以直接在webpack下加載并調(diào)用運行。

我們原本的項目使用 vite 構建,vite 對import wasm 組件策略和 webpack 的不一樣,vite 加載會返回一個加載方法,調(diào)用加載方法會返回一個 Promise,resolve 后才會返回跟 webpack 加載一樣的 wasm 組件。

我們要對 wasm-pack 生成的產(chǎn)物作一些修改,假設我們的 wasm 組件命名為 merge_vertice_wasm,生成的主 js 文件應該會命名為merge_vertice_wasm.js,內(nèi)容如下:

import*aswasmfrom'./merge_vertice_wasm_bg.wasm'
import{__wbg_set_wasm}aswasm_bgfrom'./merge_vertice_wasm_bg.js'
__wbg_set_wasm(wasm);
export*from'./merge_vertice_wasm_bg.js'

為兼容 vite 的加載策略,修改成下面的內(nèi)容

import*aswasmfrom'./merge_vertice_wasm_bg.wasm'
import*aswasm_bgfrom'./merge_vertice_wasm_bg.js'

letmemory
if(wasm.default){
wasm.default({
'./merge_vertice_wasm_bg.js':wasm_bg,
}).then(_wasm=>{
memory=_wasm.memory
wasm_bg.__wbg_set_wasm(_wasm)
})
}else{
memory=_wasm.memory
wasm_bg.__wbg_set_wasm(wasm)
}
export*from'./merge_vertice_wasm_bg.js'

exportfunctiongetMemory(){
returnmemory
}

就可以在 webpack 和 vite 下都可以順利加載并運行了。

其中增加了getMemory的方法供外部獲取 wasm 組件的內(nèi)存空間。

wasm 調(diào)用 javascript 方法—

當我們在調(diào)試和測試性能表現(xiàn)時,需要打印日志,由于我們的 wasm 跑在瀏覽器環(huán)境中,我們需要調(diào)用 javascript 的方法,比如console.log和console.time。

wasm-bindgen 庫提供了 web-sys 的組件,讓 Rust 可以調(diào)用這些方法。

首先需要在cargo.toml中添加 web-sys 的依賴,并聲明需要用到的特性:

[dependencies]
wasm-bindgen="0.2.84"

[dependencies.web-sys]
version="0.3.64"
features=["console"]

這樣在下次編譯的時候,cargo 就會自動處理這些依賴,將會下載并構建。

然后在我們的 Rust 文件中,加入對 web-sys 的引用:

externcrateweb_sys;

就可以調(diào)用 javascript 的 console 下的方法了:

//調(diào)用console.log
web_sys::log_1(&JsValue::from(logContent));

//調(diào)用console.time(label)
web_sys::time_with_label(label);

//調(diào)用console.timeEnd(label)
web_sys::time_end_with_label(label);

原 javascript 版本優(yōu)化模型耗時:

cd788f3e-5871-11ee-939d-92fbcf53809c.jpg

https://img14.360buyimg.com/imagetools/jfs/t1/109410/21/37527/8537/64dedd1cFe4c8c5c4/596fc2d36cc9fe5c.jpg

wasm 版本優(yōu)化模型耗時:

cd87b464-5871-11ee-939d-92fbcf53809c.jpg

https://img12.360buyimg.com/imagetools/jfs/t1/188745/32/36809/10529/64dedd1cF49a8b5cc/8dea820d278ad577.jpg

總結

以上為根據(jù)官網(wǎng)文檔把模型合并頂點優(yōu)化方法遷移為 wasm 版本的開發(fā)經(jīng)歷,從安裝工具到發(fā)布、調(diào)試的整個過程。

中間因為對 Rust 數(shù)據(jù)類型的不熟悉和對不同前端構建工具對 wasm 組件處理的不同不夠清晰,在開發(fā)過程中遇到的問題和解決方法。

Rust 版本的代碼邏輯基本上是從 javascript 版本翻譯過來的,其中應該還有在 Rust 環(huán)境下的優(yōu)化手段,將在之后的學習中繼續(xù)迭代。

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • Web
    Web
    +關注

    關注

    2

    文章

    1254

    瀏覽量

    69215
  • C++
    C++
    +關注

    關注

    21

    文章

    2090

    瀏覽量

    73410
  • Rust
    +關注

    關注

    1

    文章

    228

    瀏覽量

    6528

原文標題:給Web前端工程師看的用Rust開發(fā)wasm組件實戰(zhàn)

文章出處:【微信號:OSC開源社區(qū),微信公眾號:OSC開源社區(qū)】歡迎添加關注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關推薦

    從WasmEdge運行環(huán)境讀寫Rust Wasm應用的時序數(shù)據(jù)

    WebAssembly (Wasm) 正在成為一個廣受歡迎的編譯目標,幫助開發(fā)者構建可遷移平臺的應用。最近 Greptime 和 WasmEdge 協(xié)作,支持了在 WasmEdge 平臺上的 Wasm 應用通過 MySQL 協(xié)議
    的頭像 發(fā)表于 12-22 11:03 ?913次閱讀

    鴻蒙實戰(zhàn)開發(fā):【FaultLoggerd組件】講解

    Faultloggerd部件是OpenHarmony中C/C++運行時崩潰臨時日志的生成及管理模塊。面向基于 Rust 開發(fā)的部件,F(xiàn)aultloggerd 提供了Rust Panic故障日志生成能力。系統(tǒng)
    的頭像 發(fā)表于 03-12 16:22 ?909次閱讀
    鴻蒙<b class='flag-5'>實戰(zhàn)</b><b class='flag-5'>開發(fā)</b>:【FaultLoggerd<b class='flag-5'>組件</b>】講解

    【RISC-V 生態(tài)軟件系列】Waft基礎教程三:模擬器的使用

    隆重宣布,waft應用現(xiàn)在可以脫離設備,運行在pc端上的模擬器了。waft應用開發(fā)又提效了,開發(fā)者可以在pc模擬器上調(diào)試好應用開發(fā),頁面調(diào)試,然后在真機上運行。下面介紹模擬器的使用。準備工作確保
    發(fā)表于 03-08 09:36

    Wasm軟件生態(tài)系統(tǒng)安全分析

    ,致?于宣傳、實施、推?Wasm與OpenHarmony的集成,在OpenHarmony終端設備上可以安全高效的運行第三方開發(fā)者用 C、C++、Rust 等語言編寫的 Wasm 程序,
    發(fā)表于 09-05 15:29

    微信小程序如何開發(fā)?微信小程序教程視頻常用組件API開發(fā)項目實戰(zhàn)

    本文檔的主要內(nèi)容介紹的是微信小程序如何開發(fā)?微信小程序教程視頻微信小程序入門與實戰(zhàn) 常用組件API開發(fā)技巧項目實戰(zhàn)詳細的介紹了
    發(fā)表于 08-28 14:41 ?88次下載

    迅雷鏈正式推出了WASM虛擬機模塊的源代碼和代碼庫

    隨著區(qū)塊鏈應用對虛擬機以及智能合約的需求增多,區(qū)塊鏈虛擬機技術也在逐漸完善。目前,基于WASM的虛擬機在速度和性能方面都有了顯著提升,且支持C、C++等多種編程語言,開發(fā)門檻更低。為了讓區(qū)塊鏈應用開發(fā)更便捷,迅雷鏈推出了雙虛擬機
    發(fā)表于 08-15 11:42 ?4402次閱讀
    迅雷鏈正式推出了<b class='flag-5'>WASM</b>虛擬機模塊的源代碼和代碼庫

    如何使用Rust進行Wasm合約開發(fā)

    Ontology Wasm 自從上線測試網(wǎng)以來,得到了社區(qū)開發(fā)人員的極大關注。因為這項技術使得業(yè)務邏輯復雜的 dApp 合約上鏈成本降低,極大豐富 dApp 生態(tài)。
    發(fā)表于 08-22 11:09 ?5246次閱讀

    如何使用C++進行Ontology Wasm合約開發(fā)

    Ontology Wasm 自從上線測試網(wǎng)以來便受到了社區(qū)開發(fā)人員的極大關注。Ontology Wasm 的上線將使得業(yè)務邏輯復雜的 dApp 合約上鏈成本降低,極大豐富 dApp 生態(tài)。在進行 Ontology
    發(fā)表于 08-29 17:30 ?880次閱讀

    重大性能更新:Wasm 后端將利用 SIMD指令和 XNNPACK多線程

    3 月,我們?yōu)?TensorFlow.js 推出了一個新的 WebAssembly(Wasm) 加速后端(繼續(xù)閱讀以進一步了解 Wasm 及其重要性)。今天,我們很高興宣布一項重大性能更新:自
    的頭像 發(fā)表于 09-30 15:15 ?9416次閱讀
    重大性能更新:<b class='flag-5'>Wasm</b> 后端將利用 SIMD指令和 XNNPACK多線程

    vim.wasm Vim-WebAssembly編譯器

    ./oschina_soft/vim.wasm.zip
    發(fā)表于 06-22 09:57 ?6次下載
    vim.<b class='flag-5'>wasm</b> Vim-WebAssembly編譯器

    Wasm container 如何運行服務WordPress的php.wasm鏡像

    該構建使用 WASI-SDK 工具集。它包括一個可以構建到 wasm32-wasi 目標的 clang 編譯器,以及在 WASI 之上實現(xiàn)基本 POSIX 系統(tǒng)調(diào)用接口的 wasi-libc。
    發(fā)表于 01-05 10:58 ?836次閱讀

    淺析Wasm-bpf架起Webassembly和eBPF內(nèi)核可編程的橋梁

    Wasm 最初是以瀏覽器安全沙盒為目的開發(fā)的,發(fā)展到目前為止,WebAssembly 已經(jīng)成為一個用于云原生軟件組件的高性能、跨平臺和多語言軟件沙箱環(huán)境
    的頭像 發(fā)表于 02-13 11:40 ?738次閱讀

    在WebAssembly中使用Rust編寫eBPF程序并發(fā)布OCI鏡像

    WebAssembly(Wasm)最初是以瀏覽器安全沙盒為目的開發(fā)的,發(fā)展到目前為止,WebAssembly 已經(jīng)成為一個用于云原生軟件組件的高性能、跨平臺和多語言軟件沙箱環(huán)境,Wasm
    的頭像 發(fā)表于 02-14 18:10 ?1069次閱讀

    鴻蒙實戰(zhàn)開發(fā)學習【FaultLoggerd組件

    Faultloggerd部件是OpenHarmony中C/C++運行時崩潰臨時日志的生成及管理模塊。面向基于 Rust 開發(fā)的部件,F(xiàn)aultloggerd 提供了Rust Panic故障日志生成能力。系統(tǒng)
    的頭像 發(fā)表于 03-17 20:39 ?413次閱讀
    鴻蒙<b class='flag-5'>實戰(zhàn)</b><b class='flag-5'>開發(fā)</b>學習【FaultLoggerd<b class='flag-5'>組件</b>】

    OpenHarmony實戰(zhàn)開發(fā)-如何實現(xiàn)組件動畫。

    ArkUI為組件提供了通用的屬性動畫和轉(zhuǎn)場動畫能力的同時,還為一些組件提供了默認的動畫效果。例如,List的滑動動效,Button的點擊動效,是組件自帶的默認動畫效果。在組件默認動畫效
    的頭像 發(fā)表于 04-28 15:49 ?515次閱讀
    OpenHarmony<b class='flag-5'>實戰(zhàn)</b><b class='flag-5'>開發(fā)</b>-如何實現(xiàn)<b class='flag-5'>組件</b>動畫。