c#如何使用WASM跨语言调用?

c#,如何,使用,wasm,语言,调用 · 浏览次数 : 551

小编点评

**Wasm(WebAssembly)**是一种用于基于堆栈的二进制指令格式。 **wasmtime** 是一个与 WebAssembly 紧密相关的运行时,它支持 WASI (WebAssembly System Interface)。 **wasmtime** 提供了以下主要功能: * 在非浏览器环境中执行 WebAssembly 代码。 * 支持 WASI 接口,可以运行使用 WASI 接口的 WebAssembly 程序。 * 提供安全和跨平台支持。 **wasmtime 和 WASM 的关系** wasmtime 是 WASM 的一个主要实现,它支持 WASI。这使得 wasmtime 可以运行那些使用 WASI 接口的 WebAssembly 程序。 **wasmtime 和 WASM 的区别** * WASM 是一个二进制格式。 * WAT 是一个文本格式。 **使用 wasmtime 执行 WebAssembly 代码** 1. 安装 Wasmtime SDK:`dotnet add Wasi.Sdk` 2. 创建一个控制台项目并添加 NuGet 包。 3. 生成 .wasm 文件:右键项目,选择生成 > WASM 文件。 4. 在控制台中运行 wasm 文件:使用 `wasmtime` 命令。 **示例** ```csharp using Wasi.Sdk; public class MyClass { public int Add(int a, int b) { return a + b; } } // 使用 Wasmtime 编译代码 var engine = new Engine(); var module = Module.FromTextFile(engine, "my.wasm"); var linker = new Linker(engine); var store = new Store(engine); var table = new Table(store, TableKind.FuncRef, null, 2); table.SetElement(0, Function.FromCallback(store, (int a, int b) => a + b)); // 创建实例并执行 wasm 代码 var instance = linker.Instantiate(store, module); var callIndirect = instance.GetFunction("callIndirect"); Console.WriteLine(callIndirect(100, 25)); ```

正文

介绍Wasm(WebAssembly)

WebAssembly(简称Wasm)是一种用于基于堆栈的虚拟机的二进制指令格式。Wasm被设计为编程语言的可移植编译目标,支持在web上部署客户端和服务器应用程序。

什么是wasmtime (WebAssembly Time)?它和WASM(WebAssembly)是什么关系?

wasmtime 是一个独立的、轻量级的 WebAssembly (WASM) 运行时,它支持 WASI (WebAssembly System Interface)。wasmtime 由 Bytecode Alliance 开发,该联盟致力于创建新的软件基础设施,使得模块化、可组合、安全和高效的软件成为可能。

wasmtime 和 WASM (WebAssembly) 的关系如下:

  1. WebAssembly 运行时: wasmtime 是一个运行时,它允许你在本地环境中执行 WebAssembly 代码,而不需要浏览器。这意味着你可以使用 wasmtime 运行任何编译为 WASM 的代码,无论是从 C、Rust、Go 还是其他语言编译的。

  2. 支持 WASI: wasmtime 是 WASI 的一个主要实现,这意味着它可以运行那些使用 WASI 接口的 WebAssembly 程序,从而让这些程序可以访问文件、网络和其他系统资源。

  3. 安全性: 与 WebAssembly 一样,wasmtime 也提供了一个沙盒环境,确保 WASM 代码在受限制的环境中运行,从而提供了一定的安全性。

  4. 跨平台: wasmtime 可以在多种操作系统和平台上运行,包括 Windows、Linux 和 macOS。

  5. 高效: wasmtime 使用了先进的即时编译 (JIT) 技术,确保 WebAssembly 代码能够高效地执行。

总之,wasmtime 是一个与 WebAssembly 紧密相关的运行时,它允许开发者在非浏览器环境中执行 WASM 代码,并提供了对 WASI 的支持,从而扩展了 WebAssembly 的能力和应用范围。

准备工作

环境

请先安装wasihttps://github.com/bytecodealliance/wasmtime/releases中找到适合的操作系统下载wsmi

安装完成以后在cmd中执行即可查看是否安装成功

wasmtime

效果如图。

创建一个控制台项目

创建一个ConsoleApp2的控制台项目

添加NuGet包。

<ItemGroup>
      <PackageReference Include="Wasi.Sdk" Version="0.1.4-preview.10020" />
</ItemGroup>

Wasi.Sdk是用于生成.wasm文件的sdk,仓库地址:https://github.com/dotnet/dotnet-wasi-sdk

当我们右键项目的适合点击生成则会在当前项目的bin/Debug|Release文件夹下面生成一个{项目名称}.wasm的文件,当然还包括了.dll文件。

生成.wasm文件

选中我们的项目,右键重新生成

然后右键项目,在文件资源管理器中打开文件夹。

依次打开bin=>Release|Debug=>net7.0(看选择的SDK)

在当前路径打开控制台。然后使用wasmtim执行wasm文件。

wasmtime ConsoleApp2.wasm

就这样完成了简单的wasm使用。使用c#编译成wasm的格式,然后执行。

执行wat

什么是wat

WAT (WebAssembly Text Format) 是 WebAssembly 的文本表示形式。当我们谈论 WebAssembly (WASM),我们通常指的是其二进制格式,这是一种为浏览器和其他宿主环境设计的低级虚拟机代码。然而,为了方便人类阅读和编写,WASM 也有一个等效的文本格式,即 WAT。

以下是一些关于 WAT 的关键点:

  1. 可读性: 虽然 WASM 二进制格式是为机器设计的,但 WAT 格式是为人类设计的。它提供了一种更加可读和可编辑的方式来表示 WebAssembly 代码。

  2. 结构: WAT 代码通常包含一系列的指令、函数定义和其他模块级声明。它的语法是 S-expression,这是一种用于表示嵌套结构的简单文本格式。

  3. 转换: 你可以使用工具,如 wasm2watwat2wasm,来在 WAT 和 WASM 之间进行转换。这意味着你可以手动编写或修改 WAT 代码,然后将其编译为 WASM 二进制格式,或者从现有的 WASM 代码反编译为 WAT 格式。

  4. 示例: 下面是一个简单的 WAT 示例,该示例定义了一个函数,该函数接受两个整数参数并返回它们的和:

    (module
      (func $add (param $a i32) (param $b i32) (result i32)
        get_local $a
        get_local $b
        i32.add)
      (export "add" (func $add))
    )
    

总的来说,WAT 是 WebAssembly 的文本表示形式,它为开发者提供了一种更加直观和可读的方式来查看、编写或修改 WebAssembly 代码。

在当前解决方案中在创建一个项目

创建一个ConsoleApp1的控制台项目

在项目中添加一下nuget包

  <ItemGroup>
    <PackageReference Include="wasmtime" Version="11.0.1" />
  </ItemGroup>

官方仓库:https://github.com/bytecodealliance/wasmtime-dotnet

然后新增test.wat文件,写入一下代码,以下代码则是WAT代码。

(module
  (import "" "table" (table $t 4 funcref))
  (func (export "call_indirect") (param i32 i32 i32) (result i32)
    (call_indirect $t (param i32 i32) (result i32) (local.get 1) (local.get 2) (local.get 0))
  )
)

设置test.wat文件属性

打开Program.cs文件,修改代码

using Wasmtime;

using var engine = new Engine();
using var module = Module.FromTextFile(engine, "test.wat");
using var linker = new Linker(engine);
using var store = new Store(engine);

var table = new Table(store, TableKind.FuncRef, null, 4);

table.SetElement(0, Function.FromCallback(store, (int a, int b) => a + b));
table.SetElement(1, Function.FromCallback(store, (int a, int b) => a - b));
table.SetElement(2, Function.FromCallback(store, (int a, int b) => a * b));
table.SetElement(3, Function.FromCallback(store, (int a, int b) => a / b));

linker.Define("", "table", table);

var instance = linker.Instantiate(store, module);

var call_indirect = instance.GetFunction<int, int, int, int>("call_indirect");
if (call_indirect is null)
{
    Console.WriteLine("error: `call_indirect` export is missing");
    return;
}

Console.WriteLine($"100 + 25 = {call_indirect(0, 100, 25)}");
Console.WriteLine($"100 - 25 = {call_indirect(1, 100, 25)}");
Console.WriteLine($"100 * 25 = {call_indirect(2, 100, 25)}");
Console.WriteLine($"100 / 25 = {call_indirect(3, 100, 25)}");

这里提供了一个使用c#调用test.wat的案例。

执行项目

如何执行其他语言的wasm?

Rust 编译 WebAssembly 指南 - 知乎 (zhihu.com)

可以参考这个博客,使用将rust编译成wasm,然后在编译成wat格式。

技术交流

qq技术交流群:737776595

更多技术分享关注token的技术分享公众号

img

与c#如何使用WASM跨语言调用?相似的内容:

c#如何使用WASM跨语言调用?

## 介绍Wasm(WebAssembly) WebAssembly(简称Wasm)是一种用于基于堆栈的虚拟机的二进制指令格式。Wasm被设计为编程语言的可移植编译目标,支持在web上部署客户端和服务器应用程序。 ### 什么是wasmtime (WebAssembly Time)?它和WASM(W

Higress 基于自定义插件访问 Redis

本文介绍了Higress,一个支持基于WebAssembly (WASM) 的边缘计算网关,它允许用户使用Go、C++或Rust编写插件来扩展其功能。文章特别讨论了如何利用Redis插件实现限流、缓存和会话管理等高级功能。

C#使用MX Component实现三菱PLC软元件数据采集的完整步骤(仿真)

前言 本文介绍了如何使用三菱提供的MX Component插件实现对三菱PLC软元件数据的读写,记录了使用计算机仿真,模拟PLC,直至完成测试的详细流程,并重点介绍了在这个过程中的易错点,供参考。 用到的软件: 1. PLC开发编程环境GX Works2,GX Works2下载链接 https://

如何使用csproj构建C#源代码组件NuGet包?

一般我们构建传统的NuGet包,都是打包和分发dll程序集文件。 至于打包和分发C#源代码文件的做法,比较少见。 那么这种打包源代码文件的做法,有什么优点和缺点呢? 优点: 方便阅读源代码。 方便断点调试。 减少 Assembly 程序集模块加载个数。 更利于发布期间的剪裁(PublishTrimm

C#S7.NET实现西门子PLCDB块数据采集的完整步骤

前言 本文介绍了如何使用S7.NET库实现对西门子PLC DB块数据的读写,记录了使用计算机仿真,模拟PLC,自至完成测试的详细流程,并重点介绍了在这个过程中的易错点,供参考。 用到的软件: 1.Windows环境下链路层网络访问的行业标准工具(WinPcap_4_1_3.exe)下载链接:http

如何不使用图形来创建ACFS文件系统

客户需求,提供在19c环境下,ACFS的命令行操作的具体步骤,便于在图形界面不可用场景使用。 当然,如果有图形可操作,还是推荐首选图形,避免复杂度以及不必要的错误。 其实之前有测试过11g环境下的ACFS命令创建,如下: - [通过命令行创建ACFS文件系统](https://www.cnblogs

C++如何在main函数开始之前(或结束之后)执行一段逻辑?

1. 问题 2. 考察的要点 3. 解决策略 3.1. 方案一:使用GCC的拓展功能 3.2. 方案二:使用全局变量 3.3. 方案三:atexit 4. Demo测试 4.1. 测试代码 4.2. 执行结果 5. 程序异常退出场景 5.1. 存在的问题 5.2. 解决方案 5.2.1. 原理 5.

设计模式-C#实现简单工厂模式

前言 上一篇文章写了如何使用RabbitMQ做个简单的发送邮件项目,然后评论也是比较多,也是准备去学习一下如何确保RabbitMQ的消息可靠性,但是由于时间原因,先来说说设计模式中的简单工厂模式吧! 在了解简单工厂模式之前,我们要知道C#是一款面向对象的高级程序语言。它有3大特性,封装、继承、多态。

C/C++如何写调试宏

1. 调试宏以及测试 在写代码时,不可避免需要打印提示、警告、错误等信息,且要灵活控制打印信息的级别。另外,还有可能需要使用宏来控制代码段(主要是调试代码段)是否执行。为此,本文提供一种调试宏定义方案,包括打印字符串信息LOG1宏和格式化打印LOG2宏,且能通过宏控制代码段执行。完整代码如下: #i

c# 如何将程序加密隐藏?

下面将介绍如何通过`LiteDB`将自己的程序进行加密,首先介绍一下`LiteDB`。 ## LiteDB LiteDB是一个轻量级的嵌入式数据库,它是用C#编写的,适用于.NET平台。它的设计目标是提供一个简单易用的数据库解决方案,可以在各种应用程序中使用。 LiteDB使用单个文件作为数据库存储