余晖落尽暮晚霞,黄昏迟暮远山寻
本站
当前位置:网站首页 > 编程知识 > 正文

WebAssembly:用emscripten创建一个光秃秃的模块

xiyangw 2022-11-25 12:08 14 浏览 0 评论

在上一篇文章“ WebAssembly简介”中,我介绍了使用Emscripten处理WebAssembly模块的一些基本知识。

本文是一些研究和实验的结果,以查看是否可以使用Emscripten构建WebAssembly模块,但没有任何管道代码。

例如,如果我们使用下面的命令行构建下面的C文件,结果将是一个101 KB的HTML文件,一个80.2 KB的JS文件和一个9.4 KB的wasm文件!

#include <stdio.h>

#include "../emscripten/emscripten.h"

int main() { return 0; }

int EMSCRIPTEN_KEEPALIVE add(int x, int y) { return x + y; }

拥有所有的管道是非常方便的,因为它可以让你立刻开始使用WebAssembly模块,但是如果我们只想要最低限度的,并且会自己处理HTML和JavaScript呢?

幸运的是,有一种方法可以告诉Emscripten只输出裸骨文件。

我们首先将C文件剥离到最低限度。在这种情况下,我们只需要添加方法:

int add(int x,int y){ return x + y ; }

如果我们使用下面的命令行,我们将得到只是wasm文件,它只有202个字节!

emcc test.c -s WASM=1 -s SIDE_MODULE=1 -O1 -o test.wasm

SIDE_MODULE标志告诉Emscripten编译只有我们的方法,并没有别的意思,你也无法获得的东西像printf或者malloc的。

如果未指定,则使用的默认优化标志是-O0 (大写字母o和数字0),但在尝试加载模块时会导致以下错误:

LinkError: import object field 'DYNAMICTOP_PTR' is not a Number

添加任何优化标志-O0 将解决问题,所以我们用-O1 (大写字母o和数字1)来代替这个例子。

但是有一点需要注意,O似乎区分大小写。可用的各种优化标志可以在这里找到。

我们还需要在命令行中指定输出文件的名称,因为如果我们不这样做,Emscripten将输出名称为a.out.wasm的文件。

因为我们决定不使用Emscripten的管道代码,所以我们需要编写自己的HTML和JavaScript。以下是在模块中加载的一些HTML和JavaScript示例:

<!DOCTYPE html>

<html>

<head>

<meta charset="utf-8"/>

</head>

<body>

<input type="button" value="test" onclick="javascript:OnClickTest();" />

<script type="text/javascript">

var gModule = null;

var importObject = {

'env': {

'memoryBase': 0,

'tableBase': 0,

'memory': new WebAssembly.Memory({initial: 256}),

'table': new WebAssembly.Table({initial: 0, element: 'anyfunc'})

}

};

fetch('test.wasm').then(response =>

response.arrayBuffer()

).then(bytes =>

WebAssembly.instantiate(bytes, importObject)

).then(results => {

gModule = results.instance; // Hold onto the module's instance so that we can reuse it

});

function OnClickTest(){

alert(gModule.exports._add(1, 2));

}

</script>

</body>

</html>

有一件事你可能已经注意到了,与我们之前的博客文章中所做的相比,我们对C方法的调用不同,是因为我们没有使用Module.ccall或Module.cwrap。这些是Emscripten帮助方法。在这里,我们直接调用C方法。

还有一点需要注意的是,你的JavaScript需要在方法名前包含一个下划线字符。例如,在我们的例子中,我们的方法叫做add。在JavaScript中调用add方法时,可以使用_add(1,2); 而不是加(1,2);

虽然这种方法可能不是一个可以在任何情况下都能正常工作的解决方案,但是如果你不能访问像malloc这样的东西,那么如果你正在进行数字运算,不需要Emscripten的所有管理费用。

相关推荐

spring利用spring.handlers解析自定义配置(spring validation 自定义)

一、问题我们在spring的xml配置文件里经常定义各种各样的配置(tx、bean、mvc、bean等等)。以及集成第三方框架时,也会看到一些spring之外的配置,例如dubbo的配置、securi...

「Spring源码分析」AOP源码解析(上篇)(spring源码深度解析(第2版))

前言前面写了六篇文章详细地分析了SpringBean加载流程,这部分完了之后就要进入一个比较困难的部分了,就是AOP的实现原理分析。为了探究AOP实现原理,首先定义几个类,一个Dao接口:1&nbs...

Spring 解析注册BeanDefinition这一篇就Over
Spring 解析注册BeanDefinition这一篇就Over

一、简介:学习过Spring框架的人一定都会听过Spring的IoC(控制反转)、DI(依赖注入)这两个概念,对于初学Spring的人来说,总觉得IoC、...

2023-03-20 14:53 xiyangw

域、模块、空间、闭包,你真的懂了吗?(模块控制域与作用域的关系)

Javascript有一个特性叫做域。尽管对于初学者来说理解域是有难度的,但我会尽力用最简单的方式让你理解域。理解域能让你的代码更优秀,减少错误,及有助于你做出更强大的模式设计。什么是域域是在运行时,...

这一次搞懂Spring自定义标签以及注解解析原理
这一次搞懂Spring自定义标签以及注解解析原理

前言在上一篇文章中分析了Spring是如何解析默认标签的,并封装为BeanDefinition注册到缓存中,这一篇就来看看对于像context这种自定义标签是如...

2023-03-20 14:53 xiyangw

前端基础进阶(七)-前端工程师最容易出错的问题-this关键字
前端基础进阶(七)-前端工程师最容易出错的问题-this关键字

我们在学习JavaScript的时候,因为对一些概念不是很清楚,但是又会通过一些简洁的方式把它给记下来,那么这样自己记下来的概念和真正的概念产生了很强的偏差.当...

2023-03-20 14:52 xiyangw

深入K8s:守护进程DaemonSet及其源码分析(k8s 进程)
深入K8s:守护进程DaemonSet及其源码分析(k8s 进程)

建议学习:膜拜!阿里内部都在强推的K8S(kubernetes)学习指南,不能再详细了最近也一直在加班,处理项目中的事情,发现问题越多越是感觉自己的能力不足,...

2023-03-20 14:52 xiyangw

Spring 是如何解析 bean 标签的?(spring beans标签)
Spring 是如何解析 bean 标签的?(spring beans标签)

前情回顾上回「SpringIoC容器初始化(2)」说到了Spring如何解析我们定义的<bean>标签,代码跟进了一层又一层,跋山涉水,...

2023-03-20 14:52 xiyangw

快速了解JavaScript文本框操作(javascript文本框代码)
快速了解JavaScript文本框操作(javascript文本框代码)

HTML中使用<input>元素表示单行输入框和<textarea>元素表示多行文本框。HTML中使用的<input&...

2023-03-20 14:51 xiyangw

荐读|30道JavaOOP面试题,可以和面试官扯皮了
荐读|30道JavaOOP面试题,可以和面试官扯皮了

面试是我们每个人都要经历的事情,大部分人且不止一次,今天给大家准备了30道JavaOOP面试题,希望能够帮助到对Java感兴趣的同学,让大家在找工作的时候能够...

2023-03-20 14:51 xiyangw

源码系列——mybatis源码刨析总结,下(mybatis源码分析)
源码系列——mybatis源码刨析总结,下(mybatis源码分析)

接上文简答题一.1.Mybatis动态sql是做什么的?1.动态sql就是根据条件标签动态的拼接sql,包括判空,循环,拼接等2.哪些动态sql?动态sql大...

2023-03-20 14:50 xiyangw

Java面试题(第二弹)(java面试题及答案整理)
Java面试题(第二弹)(java面试题及答案整理)

1.抽象类和接口的区别?接口可以被多重implements,抽象类只能被单一extends接口只有定义,抽象类可以有定义和实现接口的字段定义默认为:public...

2023-03-20 14:50 xiyangw

mybatis3 源码深度解析-动态 sql 实现原理(sql数据库基础知识)
mybatis3 源码深度解析-动态 sql 实现原理(sql数据库基础知识)

大纲动态sql使用示例SqlSource和BoundSql以及实现类LanguageDriver以及实现类SqlNode以及实现类动态sql解...

2023-03-20 14:50 xiyangw

第43节 Text、Comment及CDATASection(第43节 Text、Comment及CDATASection)
第43节 Text、Comment及CDATASection(第43节 Text、Comment及CDATASection)

本内容是《Web前端开发之Javascript视频》的课件,请配合大师哥《Javascript》视频课程学习。文本节点用Text类型表示,包含的是可以按字面解释...

2023-03-20 14:49 xiyangw

Qt读写三种文件(qt读取文件数据并赋值给变量)

第一种INI配置文件.ini文件是InitializationFile的缩写,即初始化文件。除了windows现在很多其他操作系统下面的应用软件也有.ini文件,用来配置应用软件以实现不同用户的要...

取消回复欢迎 发表评论: