0

0

解决 Emscripten Asyncify 多异步操作冲突的实战方案

霞舞

霞舞

发布时间:2026-02-26 10:56:13

|

581人浏览过

|

来源于php中文网

原创

解决 Emscripten Asyncify 多异步操作冲突的实战方案

本文详解 emscripten 中因 asyncify 机制限制导致“cannot have multiple async operations in flight at once”崩溃的根本原因,并提供禁用 embind 自动异步包装、手动控制执行流等安全可靠的解决方案。

本文详解 emscripten 中因 asyncify 机制限制导致“cannot have multiple async operations in flight at once”崩溃的根本原因,并提供禁用 embind 自动异步包装、手动控制执行流等安全可靠的解决方案。

在使用 Emscripten 将 C++ 代码编译为 WebAssembly 并与 JavaScript 交互时,若启用了 ASYNCIFY(例如通过 -sASYNCIFY 编译选项),Emscripten 会自动注入 Asyncify 运行时以支持 C++ 中的“假同步阻塞调用”(如 emscripten_sleep、malloc 在某些场景下的挂起等)。然而,该机制会对所有 Embind 绑定函数无差别地添加异步包装层——即使 C++ 函数本身是纯同步的(如示例中的 sync_func()),它也可能被 Asyncify 视为潜在可挂起点,从而在 JS 回调中引发并发异步操作冲突。

典型崩溃复现逻辑如下:

  • JS 传入一个 async 回调给 C++;
  • C++ 异步触发该回调(如 func().await());
  • 回调内部同时发起一个原生异步操作(如 fetch())和一个同步 C++ 调用(如 Module.sync_func());
  • 此时 Asyncify 检测到 fetch().then() 和 sync_func() 的底层 Asyncify 唤醒路径并行激活,触发断言失败:
    RuntimeError: Aborted(Assertion failed: Cannot have multiple async operations in flight at once)。

⚠️ 关键认知:该错误并非源于 JS 代码逻辑错误,而是 Asyncify 运行时对“异步飞行中操作数”(in-flight async operations)的硬性单例限制 —— 它设计上只允许一个 Asyncify 挂起/恢复周期处于活跃状态。

✅ 推荐解决方案:显式解除 Embind 的 Asyncify 包装

最稳定、可控的方式是避免让 Asyncify 干预 Embind 函数调用流。可通过以下 EM_ASM 注入方式,在运行时劫持 Asyncify 的调度钩子:

Descript
Descript

一个多功能的音频和视频编辑引擎

下载
// 在 Module 初始化前或 onRuntimeInitialized 中执行
EM_ASM(
  if (Asyncify && Asyncify.whenDone) {
    // 替换默认的 whenDone 行为:不再等待 Asyncify 恢复,直接 resolve
    Asyncify.whenDone = () => Promise.resolve();
  }
);

? 原理说明:Asyncify.whenDone 是 Asyncify 内部用于协调异步边界的关键 Promise 钩子。将其重写为立即 resolve,可有效“短路”EmbBind 函数的自动异步封装,使 sync_func() 等调用回归纯同步语义,从而与 fetch() 等原生异步操作和平共存。

完整修复后的 JS 示例:

<html>
  <body>
    <script src="./out/build/em-x64-debug/async_test.js"></script>
    <script>
      // 关键:在模块加载前或初始化时禁用 Embind 的 Asyncify 包装
      Module['onRuntimeInitialized'] = () => {
        // 注入运行时补丁
        Module._emscripten_asm_const_int('Asyncify.whenDone = () => Promise.resolve();');

        // 或更兼容写法(适用于较新 Emscripten)
        Module.addOnPreRun(() => {
          Module._emscripten_asm_const_int('if (Asyncify && Asyncify.whenDone) Asyncify.whenDone = () => Promise.resolve();');
        });

        // 现在可安全混合调用
        async function async_cb() {
          await fetch('./test.txt');     // 原生异步
          Module.sync_func();           // 同步 C++ 调用(已脱离 Asyncify 干预)
          console.log('✅ Both calls succeeded.');
        }

        Module.async_func(async_cb);
      };
    </script>
  </body>
</html>

⚠️ 注意事项与替代建议

  • 慎用 ASYNCIFY 全局启用:除非项目明确依赖 C++ 层的同步阻塞语义(如移植旧有 pthread 程序),否则建议优先通过 Asyncify 的细粒度白名单(-sASYNCIFY_IMPORTS=['func1','func2'])控制仅包装必要函数,而非全局开启。
  • 避免 await + 同步调用的嵌套陷阱:即使打补丁后 sync_func() 变为真正同步,也请确保其内部不隐式触发任何 Asyncify 操作(如调用其他被 Asyncify 包装的函数),否则仍可能触发竞争。
  • 长期演进方向:迁移到 Wasm GC / WASI-threads(未来):Asyncify 是过渡性技术,性能开销大且行为隐蔽。Emscripten 2.0.30+ 已支持 --no-asyncify 默认关闭,配合 emrun --no-asyncify 可彻底规避该问题;对于新项目,推荐采用 Promise-based 主动异步接口设计,由 JS 控制流程,C++ 仅暴露非阻塞 API。

通过精准干预 Asyncify 的调度行为,开发者可在保留现有代码结构的前提下,安全实现 JS 异步逻辑与 C++ 同步函数的协同调用——这既是底层机制理解的体现,也是 WebAssembly 工程实践中关键的调试能力。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1657

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

506

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2310

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

40

2026.01.19

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

526

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

494

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

638

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5782

2023.08.17

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

331

2026.02.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号