2天时间,聊今年最热的 Agent、上下文工程、AI 产品创新等话题。2025 年最后一场~ 了解详情
写点什么

基础为零?如何将 C++ 编译成 WebAssembly(一)

  • 2019-12-16
  • 本文字数:2148 字

    阅读完需:约 7 分钟

基础为零?如何将 C++ 编译成 WebAssembly(一)


本文知识点提炼:

1、如何使用 Emscripten 把 C++ 编译成 wasm。

2、如何使用 wasi-sdk 把 C++ 编译成 wasm。

3、如何运行编译好的 wasm 包。


WebAssembly 是一个 W3C 推出的二进制指令格式,近日它的 1.0 版本也正式定稿成为了规范,关于它的基本概念这里不再展开介绍了,网上已经有很多文章了,大家可以自行了解,推荐阅读官方文档①、spec 仓库②、MDN 的教程③、以及 Lin Clark 的文章④,其他随意。


能编译成 wasm 的语言有很多,C++ 和 Rust 是其中两个比较成熟而且大量被使用的,本文以 C++ 为例,一步一步介绍如何把 C++ 代码编译成 wasm 并且运行起来。例子很简单,相信不了解 C/C++ 开发的同学也能看懂。


说是 C++ 其实本文用到的代码都是纯 C 的。


文章用到的源码和编译脚本都在: https://github.com/Hanks10100/cpp2wasm


Hello World!


首先,我们来编译一个 C 语言的 Hello World,创建一个 hello.c 文件:


#include <stdio.h>
int main() { printf("Hello World!\n"); return 0;}
复制代码

▐ 编译成可执行文件

代码就是输出了一句 Hello World! ,使用 clang 或 gcc 或很多工具都可以把这段代码编译成可执行的二进制,找不到命令的话,可以在网上找教程配置一下。以 clang 为例:


clang hello.c -O3 -o out/hello
复制代码


-O3 表示了优化级别, 生成的可执行文件是 hello ,但是这个文件只能在特定平台上执行,在 windows 上编译出来的文件没办法跑在 mac 上(不绝对),在 32 位系统编译出来的文件无法跑在 64 位系统上。


然而如果把它编译成 wasm 就可以跨平台分发了,这也是 wasm 的一大优势。只需要编译一次,同一个 wasm 包,可以运行在浏览器中、Node.js 中、各种独立的 runtime 里,但是要求目标平台具备执行 wasm 包的能力,而且符合规范。

▐ WebAssembly 的编译和运行流程

在编译 WebAssembly 之前先了解一下它基本的编译和运行流程,想要以何种方式运行 wasm 的包,决定了以何种方式来编译它。


目前来看,大部分使用 WebAssembly 的例子都是运行在浏览器中的,有一部分运行在 Node.js 里,和 JS 的渊源很深,因为在标准里定义了一套 JS API 来编译、实例化 wasm 文件,这部分 API 已经被 JS 引擎实现了,功能已经稳定可用。因此,wasm 最常见的是搭配 js 一起使用,这种场景下用 Emscripten ⑤ 可以搞定,它在编译 wasm 包的同时也会生成一份 js “glue” 代码,把 wasm 包的初始化接口导入导出都封装在 js 里了,使用时引入这个 js 文件即可。


Emscripten 也支持编译成独立的 wasm 包(不含 JS),但是想要运行这个 wasm 包需要宿主环境给它注入很多基础的 API,而且这些  API 是非标准的。如果想在 JS 环境里运行独立 wasm 包的话,要用 JS 实现这些 API。
复制代码


其实 WebAssembly 本质上和 JS 无关,完全可以运行在独立的沙箱环境里,通过标准化的 API (wasi ⑥) 来调用系统能力。现在已经有不少 wasm 的独立运行时了,如 Wasmtime ⑦ 和 wasm-micro-runtime ⑧,它们都可以加载并独立执行 wasm 文件,并且实现了一致的 wasi 接口。


关于 wasi,推荐阅读《Standardizing WASI: A system interface to run WebAssembly outside the web》


https://hacks.mozilla.org/2019/03/standardizing-wasi-a-webassembly-system-interface/



如上图所示,面对自己的 C/C++ 代码,想要把它运行在浏览器或 Node.js 中,就使用 Emscripten 把它编译成 wasm + js 文件;想要把它运行在独立的运行时里,就使用 wasi-sdk ⑨ 进行编译,生成单独的 wasm 包。(此结论简单粗暴,为了方便理解,并不严谨)

▐ 使用 Emscripten 编译

首先安装官方文档安装 Emscripten (https://emscripten.org/) ,安装完成后命令行环境里会有 emcc 命令,使用方式和 gcc 差不多,执行如下代码就可以生成 wasm 的包:


emcc hello.c -O3 -o out/hello-emcc.wasm
复制代码


但是,上面这个命令隐含了 -s STANDALONE_WASM 的配置 ,实际上触发的是 WebAssembly Standalone build ⑩,只生成了一个 wasm 的包,需要自己写 loader 加载和执行。如果不想费这个劲,就可以使用如下命令直接生成 wasm + js 文件:


emcc hello.c -O3 -o out/hello-emcc.js
复制代码


该命令除了生成 js 文件以外,还会生成同名的 hello-emcc.wasm 文件,可以使用 WABT ⑪ (WebAssembly Binary Toolkit) 提供的小工具把 wasm 文件转成对等的文本格式,方便阅读。


wasm2wat out/hello-emcc.wasm -o out/hello-emcc.wat
复制代码


代码比较短,但是生成出来的 wasm 文件有 2.1KB,js 文件 16KB,主要是因为 stdio.h 头文件里有很多依赖,在运行时是由 js 代码来实现的。用 wasm 做 io 本身也不是个好的用法。


最后,直接在 Node.js 环境里执行这个 js 文件就行了,可以看到控制台输出了 Hello World! 。


node out/hello-emcc.js

▐ 使用 wasi-sdk 编译

首先根据自己的系统下载相应的 wasi-sdk ,配置好环境变量之后,就可以调用其中自带的 clang 工具编译生成 wasm 文件:


clang hello.c -O3 -o out/hello-wasi.wasm
复制代码


大概率跑不通…… 因为要配各种环境变量还要指定 sysroot 才行。假如你下载的是 8.0 版本,放到了个人目录之下,可以用下面这个命令编译代码,不需要配置环境变量:


~/wasi-sdk-8.0/bin/clang --sysroot ~/wasi-sdk-8.0/share/wasi-sysroot
复制代码


本文转载自淘系技术公众号。


原文链接:https://mp.weixin.qq.com/s/XrOHuoJB4vwkozBDI4t1yA


2019-12-16 18:291823

评论

发布
暂无评论
发现更多内容

涵盖了Java基础+JVM+多线程并发编程+spring全家桶+Linux+数据结构+数据库+nginx+分布式,这份Java技术成长笔记太强了

Java架构之路

Java 程序员 架构 面试 编程语言

Golang 实现 RTP

天黑黑

音视频 rtp Go 语言

所谓软件测试工作能力强,其实就是这5点

程序员阿沐

软件测试 自动化测试 测试工程师 黑盒测试 白盒测试

前端项目配置ts,axios,router,vuex

Vue js ts vuex VueRouter

要不要去创业?

石云升

创业 5月日更

如何高效率的度过一天?

程序员海军

效率 方法论

spring boot项目TPS压测性能优化

李日盛

Spring Boot 性能调优

第八大洲环游记(三):人间胜境新西兰,AI孤岛or方舟?

脑极体

Golang中runtime包的基本使用方式

liuzhen007

Go 语言 5月日更

硬核资源!清华博士的Spring Boot中AOP与SpEL笔记,码农:膜拜

牛哄哄的java大师

Java

深入了解 JavaScript 对象

程序员海军

JavaScript 大前端 对象

2021,国产数据库人的最好时代

BinTools图尔兹

数据安全 数据库管理 国产数据库

Map在Java 8中增加非常实用哪些函数接口?

xcbeyond

Java java8 5月日更 内容合集

谈谈测试环境管理与实践

大卡尔

测试环境 工程效能

网络攻防学习笔记 Day6

穿过生命散发芬芳

5月日更 网络攻防

区块链是什么意思?源中瑞开发BaaS平台促进企业数字转型升级

源中瑞-龙先生

企业数字化转型 #区块链# 源中瑞 Baas

架构实战营详细架构设计文档模板

Geek_e0c25c

Nginx基础配置-基础模块配置

梁龙先森

nginx 大前端

京东丨阿里丨携程面试总结,已成功拿到京东offer

Java架构师迁哥

北大学霸!手抄万字Java数组笔记,2小时吃透,你确定不拿走?

牛哄哄的java大师

Java 后端

编程规范的意义

顿晓

5月日更 编程规范

Vue 组件通信的 8 种方式

程序员海军

Vue 大前端 组件通信 引航计划

2021金三银四(拿下5个offer)面试经历,附阿里4面+京东4面【面经分享】

Java 编程 程序员 面试 计算机

一击必杀!内网渗透——对不出网目标的精准打击

Thrash

安全

Redis - 哈希表

旺仔大菜包

redis

吴凡 ベ莫离: 网友都说MyBatis多表查询太难了,小白:就这?我都学会了

牛哄哄的java大师

模板格式不统一?百度AI产品经理为你讲解如何高效构建定制化OCR模型

百度大脑

百度 AI OCR

谷歌大佬的LeetCode算法刷题笔记,详细讲解了刷 LeetCode 时常用的技巧。

Java架构之路

Java 程序员 架构 面试 编程语言

如有神助!阿里P7大牛把Spring Boot讲解得如此透彻,送你上岸

飞飞JAva

Dubbo 服务分组与多版本

青年IT男

从操作系统底层的IO原理入手讲解,同时提供高性能开发的实战案例!美团大佬最新总结的1053页Java高并发核心编程笔记!

Java架构之路

Java 程序员 架构 面试 编程语言

基础为零?如何将 C++ 编译成 WebAssembly(一)_文化 & 方法_张翰(门柳)_InfoQ精选文章