技术分享
🌐WebAssembly
00 min
Oct 30, 2023
Oct 31, 2023
type
status
date
slug
summary
tags
category
icon
password

JavaScript的限制

ChakraCore JavaScript引擎结构
ChakraCore JavaScript引擎结构
  • 字节码进入翻译器后,将字节码一行一行地翻译成效率十分高的Machine Code.
在项目运行的过程中,引擎会记住执行次数较多的function,引擎将其代码编译成Machine Code后打包送到顶部的Just-In-Time(JIT) Compiler,下次再执行这个function,就会直接执行编译好的Machine Code。但是由于JavaScript的动态变量,上一秒可能是Array,下一秒就变成了Object。那么上一次引擎所做的优化,就失去了作用,此时又要再一次进行编译。

由于JS缺失静态变量,JIT很难重用编译好的Machine Code,从而性能被限制
  1. console.log(sum(1,2,3))
  1. console.log(sum('1',2,3))
JIT 在遇到第一个 sum 时会编译成整数相加的机器码;但是在碰到第二个 sum 调用时,不得不重新编译一遍。这样一来,JIT 带来的效率提升便被抵消了。

什么是WebAssembly

WebAssembly 不是一种语言,而是一种编译目标
WebAssembly 由 asm.js 而来,asm.js 也是一种编译目标,但是 asm.js 仍然需要经过Parser 和 ByteCode Compiler,而WebAssembly不用经过这两步。
WebAssembly是经过编译器编译之后的代码,在语法上完全脱离JavaScript,同时具有沙盒化的执行环境。
  • 云原生:在服务端,Wasm 轻量级容器
WebAssembly 可以允许任何语言编译到它制定的 AST tree, 相当于使用其他高级语言写的代码可以直接在网页上运行
  • 这也使得很多应用能直接移植到Web上来,几乎不需要重构
notion image
Wasm省去了比较耗时的解析和编译的过程,是直接生成的二进制可执行机器码供浏览器进行执行。

与eBPF关联

目前大多数在浏览器外运行的 Wasm 轻量级容器需要使用 WASI(WebAssembly 系统接口)与其 host 操作系统交互,相对于传统的容器中可以使用几乎所有的系统调用,目前 WASI 所能提供的系统资源非常有限,目前仅仅在文件系统、socket 网络连接等方面提供了一些基本的支持,对于操作系统底层资源的访问、控制和管理能力仍然存在大量空白。这也使得大多数的 Wasm 轻量级容器在实际应用中还是主要集中于纯粹的计算密集型应用,而在网络、安全等方面,还是需要依赖于传统的容器技术。
eBPF 的兴起使得云原生开发人员能够构建安全的网络、和多种可观测性组件,并且提供强大的内核态可编程交互能力。
通过eBPF,能够让 Wasm 虚拟机或者 Wasm 轻量级容器中的应用,有能力下沉和拓展到内核态,获取内核态和用户态的几乎所有数据,在网络、安全等多个方面实现对整个操作系统层面的可编程控制,从而极大的拓展 WebAssembly 生态在非浏览器端的应用场景。
类似于在浏览器中运行的 Wasm 程序,它们通过 JavaScript 引擎接口访问浏览器提供的各种系统资源,那服务端的方案就是借助 eBPF 虚拟机访问操作系统的各类资源
让 Wasm 应用在沙箱中实现和用户态中 eBPF 应用一样的编程模型和执行逻辑。

Demo

C++:使用 Emscripten 来将它编译到 WebAssembly

环境准备

需要注意的是,source ./emsdk_env.sh 每次都要激活一下环境,才能使用 emcc
如下输出就代表成功
notion image
拉取代码:

样例编写

  1. 用C写一个斐波那契数列计算
  1. 编译生成wasm
-s SIDE_MODULE=1 表示这就是一个模块,-s EXPORTED_FUNCTIONS 表示导出的接口函数
notion image
  1. 如何加载 wasm 直接引用到页面中,官网是推荐两种,一个是 fetch,一个是 XMLHttpRequest,本文以 fetch 为例。
请注意,如果您尝试在本地运行代码跑fetch而不是通过本地服务器访问,则会收到“跨域”错误
浏览器通常不允许跨域加载本地文件
这里我把wasm文件放在服务器来下载使用(也会遇到跨域问题)
notion image
那我们用node起一个本地http服务器吧

最终测试结果

40的时候有点卡了
notion image
notion image
需要注意的是,两边fib的写发要保持一致。我一开始在js里写了迭代版本,导致测试结果比wasm快好多😹
notion image
迭代比递归快好多

与JS交互

  1. 用 EM_ASM() 之类的相关宏来内联JavaScript
notion image

Comments