如何使用 Emscripten 和 Embind 发出的 .js 和 .wasm 工件?

如何解决如何使用 Emscripten 和 Embind 发出的 .js 和 .wasm 工件?

我有两个相当简单的 C++ 类定义及其接口,uuid.{hpp,cpp}uuid_util.{hpp,cpp},我还有一个文件 uuid_bind.cpp#include <emscripten/bind.h> 来绑定 C++ 类, JavaScript 的函数和静态函数定义。

首先将这两个类构建为静态库uuid_lib.a,然后将其链接到后面的 C++ 源文件并使用 em++ --bind -o uuid_module.js uuid_bind.cpp uuid_lib.a(由 CMake 生成)构建以生成 uuid_module.jsuuid_module.wasm。现在,我该怎么处理这些?

documentation on Embind 有点稀疏,只说

生成quick_example.js 文件可以作为节点模块或通过 <script> 标签加载: ...

我发现 this Google tutorial 结合 Emscripten/embind 和 node.js,我已经尽可能多地复制了它(不包括 Docker 上的部分,因为我的 Linux 发行版直接为 Emscripten 服务)。我有 index.htmlpackage.json 文件npm test 启动 http-server,我从 Chrome 运行。

我的印象是 Emscripten/embind 只是作为任何绑定类、函数(静态或其他)、变量的翻译层,并且可以直接从 JavaScript 调用,但事实证明这是不是这样。我在这里错过了什么吗?我对JS不是很熟悉。我想要在 index.js 中做的事情是这样的:

index.js

import uuid_module from './uuid_lib.js';

console.log(uuid_module.uuid_util.generate_type1_uuid("BLA"));
// ... other calls

并使用 node index.js 运行它,它将在控制台上打印一个 UUID 字符串。


作为背景,我在下面提供了我的 CMakeLists.txtuuid_bind.cpp

CMakeLists.txt

cmake_minimum_required(VERSION 3.16)

# Set project file
project(
    uuid_generator
    VERSION 1.0
    DESCRIPTION "A UUID generator"
    LANGUAGES CXX)

# Export compile commands for clangd
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)

# Add Emscripten CMake toolchain file
if(EMSCRIPTEN)
    message("Using Emscripten.")
    set(CMAKE_TOOLCHAIN_FILE
        ${PROJECT_SOURCE_DIR}/deps/emsdk/upstream/emscripten/cmake/Modules/Platform/Emscripten.cmake)
else()
    message("Using system compilers.")
endif(EMSCRIPTEN)

# C++20 guaranteed,no extensions eg. GNU/MSVC
set(CMAKE_CXX_STANDARD 20)
set(CMAKE_CXX_STANDARD_required ON)
set(CMAKE_CXX_EXTENSIONS OFF)

# Strict compilation for MSVC or GCC/Clang
if(MSVC)
    add_compile_options(/W4 /WX)
else()
    add_compile_options(-Wall -Wextra -pedantic)
endif()

# Add uuid_generator as a shared library
add_library(uuid_lib STATIC src/uuid.cpp src/uuid_util.cpp)

# Set C++ library properties: output name,location,etc
set_target_properties(
    uuid_lib
    PROPERTIES OUTPUT_NAME uuid_lib
               PREFIX ""
               ARCHIVE_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/dist)

# Set JS binding application properties: set bind,output type,etc. Runs only if Emscripten is enabled.
if(DEFINED ENV{EMSDK})
    add_executable(uuid_module src/uuid_bind.cpp)
    set_target_properties(
        uuid_module
        PROPERTIES OUTPUT_NAME uuid_module
                   SUFFIX ".js"
                   RUNTIME_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/dist)
    target_include_directories(uuid_module
                               PRIVATE ${PROJECT_SOURCE_DIR}/deps/emsdk/upstream/emscripten/system/include)
    target_link_libraries(uuid_module uuid_lib)
    target_link_options(
        uuid_module
        PUBLIC
        $<$<CONfig:DEBUG>:
        -v
        --bind
        -sEXCEPTION_DEBUG=1
        -sDEMANGLE_SUPPORT=1
        -sdisABLE_EXCEPTION_CATCHING=1
        -sWASM_BIGINT=1
        -sMODULARIZE=1
        -sEXPORT_ES6=1
        -sEXPORT_NAME=uuid_module
        >)
endif(DEFINED ENV{EMSDK})

uuid_bind.cpp

#include "uuid.hpp"
#include "uuid_util.hpp"

#include <cstdint>
#include <emscripten.h>
#include <emscripten/bind.h>
#include <string>
#include <iostream>

EMSCRIPTEN_BINDINGS(uuid_generator)
{
    emscripten::constant("UINT29_MAX",uuid_gen::uuid::UINT29_MAX);
    emscripten::constant("ICAO_ADDRESS_MAX",uuid_gen::uuid_util::ICAO_ADDRESS_MAX);
    emscripten::constant("MAC_ADDRESS_MAX",uuid_gen::uuid_util::MAC_ADDRESS_MAX);
    emscripten::constant("RANDOM_NAMESPACE_MAX",uuid_gen::uuid_util::RANDOM_NAMESPACE_MAX);

    emscripten::class_<uuid_gen::uuid>("uuid")
        .property("namespace_type",&uuid_gen::uuid::get_namespace_type)
        .property("namespace_value",&uuid_gen::uuid::get_namespace_value)
        .property("namespace_string",&uuid_gen::uuid::get_namespace_string)
        .property("to_string",&uuid_gen::uuid::to_string)
        .class_function("uniform_int_distr",&uuid_gen::uuid::uniform_int_dist);

    emscripten::class_<uuid_gen::uuid_util>("uuid_util")
        .class_function("generate_type1_uuid",&uuid_gen::uuid_util::generate_type1_uuid)
        .class_function("generate_type2_uuid",&uuid_gen::uuid_util::generate_type2_uuid)
        .class_function("generate_type3_uuid",&uuid_gen::uuid_util::generate_type3_uuid)
        .class_function("generate_type4_uuid",&uuid_gen::uuid_util::generate_type4_uuid)
        .class_function("generate_type5_uuid_str",emscripten::select_overload<uuid_gen::uuid(const std::string &)>(
                                                       &uuid_gen::uuid_util::generate_type5_uuid))
        .class_function("generate_type5_uuid_int",emscripten::select_overload<uuid_gen::uuid(std::uint32_t)>(
                                                       &uuid_gen::uuid_util::generate_type5_uuid))
        .class_function("generate_type6_uuid_str",emscripten::select_overload<uuid_gen::uuid(const std::string &)>(
                                                       &uuid_gen::uuid_util::generate_type6_uuid))
        .class_function("generate_type6_uuid_int",emscripten::select_overload<uuid_gen::uuid(std::uint64_t)>(
                                                       &uuid_gen::uuid_util::generate_type6_uuid))
        .class_function("generate_type7_uuid",&uuid_gen::uuid_util::generate_type7_uuid)
        .class_function("from_string",&uuid_gen::uuid_util::from_string)
        .class_function("is_valid_oper_agency",&uuid_gen::uuid_util::is_valid_oper_agency)
        .class_function("is_valid_loc_atm_id",&uuid_gen::uuid_util::is_valid_loc_atm_id)
        .class_function("is_valid_mac_address_string",&uuid_gen::uuid_util::is_valid_mac_address_string)
        .class_function("namespace_char_encode",&uuid_gen::uuid_util::namespace_char_encode)
        .class_function("namespace_char_decode",&uuid_gen::uuid_util::namespace_char_decode)
        .class_function("namespace_encode",&uuid_gen::uuid_util::namespace_encode)
        .class_function("namespace_decode",&uuid_gen::uuid_util::namespace_decode)
        .class_function("mac_address_encode",&uuid_gen::uuid_util::mac_address_encode)
        .class_function("mac_address_decode",&uuid_gen::uuid_util::mac_address_decode);
};

解决方法

我在这里错过了什么吗?

您缺少的是在 -s MODULARIZE=1 模式下,默认导出是工厂,而不是模块对象本身。

您需要先创建一个模块,然后才能访问公开的属性:

import uuidModuleFactory from './uuid_lib.js';

const uuid_module = uuidModuleFactory(/* optional Emscripten config goes here */);

console.log(uuid_module.uuid_util.generate_type1_uuid("BLA"));
// ... other calls

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?