从 Rust 到 Go 的回调函数
我正在尝试创建从 go 调用 rust 函数的可能性,然后所述 rust 函数将函数回调到 go。我使用 cgo 作为 go 和 rust 之间的 ffi 接口。 以下是我的 go 代码(src/main.go):
package main import ( "c" "fmt" "unsafe" ) /* #cgo cflags: -i./../lib #cgo ldflags: -l./../bin -lgo_move -wl,-rpath=./bin #include "move.h" */ //export cosmoscallbackwrapper func cosmoscallbackwrapper(data *c.uchar, datalen c.int) { // convert data to go slice godata := c.gobytes(unsafe.pointer(data), datalen) // call the actual callback function cosmoscallback(godata) } // setcosmoscallback sets the callback function to be called by the move vm. func setcosmoscallback(callback func([]byte)) { cosmoscallback = callback c.set_cosmos_callback((c.cosmos_callback)(unsafe.pointer(c.cosmoscallbackwrapper))) } var cosmoscallback func([]byte) func main() { // create a new move interpreter // set the callback function setcosmoscallback(func(data []byte) { fmt.println("received data from move vm:", data) // handle data and call cosmos sdk functions as needed }) }
这是我的 rust 代码 (src/lib.rs)
use std::os::raw::{c_char, c_int}; use std::ffi::cstring; use std::sync::mutex; #[macro_use] extern crate lazy_static; pub fn main() { } pub type cosmoscallback = extern "c" fn(*const c_char, c_int); lazy_static! { static ref callback: mutex<option<cosmoscallback>> = mutex::new(none); } #[no_mangle] pub extern "c" fn set_cosmos_callback(callback: cosmoscallback) { let mut cb = callback.lock().unwrap(); *cb = some(callback); } #[no_mangle] pub extern "c" fn cosmoscallbackwrapper(data: *const c_char, data_len: c_int) { let cb = callback.lock().unwrap(); if let some(callback) = &*cb { callback(data, data_len); } }
这是我的 cargo.toml 文件:
[package] name = "go-move" version = "0.1.0" edition = "2021" [lib] crate-type = ["cdylib"] [dependencies] libc = "0.2" lazy_static = "1.4.0" [workspace] members = [ ".",]
这是 lib/move.h(共享库的头文件):
#ifndef move_vm_lib_h #define move_vm_lib_h #include <stdint.h> //typedef void (*cosmos_callback)(const unsigned char *data, int data_len); //void set_cosmos_callback(cosmos_callback callback); typedef void (*cosmos_callback)(const uint8_t* data, int32_t data_len); void set_cosmos_callback(cosmos_callback callback); #endif
这是我的 makefile:
shell := /bin/bash .phony: build os = $(shell uname) clean: rm -rf bin rm -rf target build: @echo the os is $(os) mkdir bin cargo build --release ifeq ($(os),linux) cp target/release/libgo_move.so bin/ else cp target/release/libgo_move.dylib bin/ endif cp -a lib/. bin/ go build --ldflags="-l./bin -lgo_move" -o bin/main src/main.go run: export ld_library_path=./bin && ./main
文件结构如下:
src main.go main.rs bin libgo_move.so (after cargo build) lib move.h cargo.toml makefile
运行 make clean build
后,我得到以下输出:
cp target/release/libgo_move.so bin/ cp -a lib/. bin/ go build -o bin/main src/main.go # command-line-arguments src/main.go:27:59: could not determine kind of name for C.cosmosCallbackWrapper src/main.go:27:25: could not determine kind of name for C.cosmos_callback src/main.go:27:2: could not determine kind of name for C.set_cosmos_callback
由于某种原因,它找不到 ffi 函数。
正确答案
这是一个愚蠢的错误:
/* #cgo CFLAGS: -I./../lib #cgo LDFLAGS: -L./../bin -lgo_move -Wl,-rpath=./bin #include "move.h" */
必须在导入之前进行。仅此而已。
以上是从 Rust 到 Go 的回调函数的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题











typedef struct 在 C 语言中用于创建结构体类型别名,简化结构体使用。它通过指定结构体别名将一个新的数据类型作为现有结构体的别名。优点包括增强可读性、代码重用和类型检查。注意:在使用别名前必须定义结构体,别名在程序中必须唯一并且仅在其声明的作用域内有效。

在C++中使用函数指针时,必须谨慎考虑内存管理以避免陷阱。这些陷阱包括悬浮指针(指向超出其范围的函数)和野指针(从未初始化或设置为nullptr的函数指针)。为了避免这些陷阱,请遵循以下最佳实践:始终初始化函数指针,谨慎管理内存,并使用智能指针。这样,您就可以安全地使用函数指针,并避免陷入指针陷阱。

typedef的用法是为已经存在的数据类型创建新的别名。使用typedef可以增加代码的可读性和可维护性,特别是在处理复杂的数据类型时。对于简单的数据类型,如整数、浮点数或字符,使用别名的好处并不明显。然而,对于指针、结构体、数组和函数等复杂的数据类型,使用别名的优势就显而易见了。typedef不能用于变量或函数定义之前,通常在程序文件的顶部或结构体定义之后创建。

Linux内核定时器与延迟工作是两种常用的实现定时任务和延后执行任务的机制,它们可以让驱动程序在合适的时间点执行特定的函数,以适应硬件设备的需求和特性。但是,如何正确地使用Linux内核定时器与延迟工作呢?本文将从理论和实践两方面,介绍Linux内核定时器与延迟工作驱动开发的基本知识和技巧,以及一些常见的问题和解决方法。内核定时器软件上的定时器最终要依靠硬件时钟来实现,简单的说,内核会在时钟中断发生后检测各个注册到内核的定时器是否到期,如果到期,就回调相应的注册函数,将其作为中断底半部来执行。实

要编写一个简单的 C 语言烟花代码,需要遵循以下步骤:包含头文件和库。定义常量和宏。创建粒子数据结构。声明全局变量。在 main() 函数中初始化烟花粒子。在游戏循环中更新粒子的位置和速度,并绘制它们。检查和销毁已达到寿命的粒子。

typedef struct和struct的区别:typedef struct创建结构体类型的别名,而struct定义新的结构体类型。typedef struct创建的别名在声明之后即可使用,而struct定义的结构体在定义之后才可使用。typedef struct和struct都不会创建额外的存储空间。

函数指针技术可提升代码效率和可复用性,具体表现为:提升效率:使用函数指针可减少重复代码,优化调用过程。提高可复用性:函数指针允许使用通用函数处理不同数据,提高程序的可复用性。

我正在尝试创建从go调用rust函数的可能性,然后所述rust函数将函数回调到go。我使用cgo作为go和rust之间的ffi接口。以下是我的go代码(src/main.go):packagemainimport("c""fmt""unsafe")/*#cgocflags:-i./../lib#cgoldflags:-l./../bin-lgo_move-wl,-rpath=./bin#include"m
