登录
首页 >  Golang >  Go问答

如何调用作为参数接收的 C 函数

来源:stackoverflow

时间:2024-04-03 14:27:37 170浏览 收藏

golang学习网今天将给大家带来《如何调用作为参数接收的 C 函数》,感兴趣的朋友请继续看下去吧!以下内容将会涉及到等等知识点,如果你是正在学习Golang或者已经是大佬级别了,都非常欢迎也希望大家都能给我建议评论哈~希望能帮助到大家!

问题内容

我想使用 go 库,并在 c 中进行一些调整。 我制作了 goadder go 函数,它有 3 个参数 int x、y 和函数类型 f。

goadder 函数将调用 f 参数。

adder.go

package main

import "fmt"

import "c" 

//export ftest
type ftest func(c.int);

//export goadder
func goadder(x, y int, f ftest) int {
    fmt.printf("go says: adding %v and %v\n", x, y) 
    f(10); 

    return x + y
}

func main() {} // required but ignored

我在上面将 go 包构建为名为 libadder.a 的静态库,如下所示:

go build -buildmode=c-archive -o libadder.a adder.go

然后我编写了下面的 c++ 代码。

ma​​in.c

#include 
#include "adder/libadder.h"

void a( goint a ){
  printf("hello %d", a);
}

int main() {
  printf("c says: about to call go...\n");
  int total = goadder(1, 7, &a);
  printf("c says: go calculated our total as %i\n", total);
  return 0;
}

我已经编译了这样的来源:

gcc -pthread -o static_go_lib main.c adder/libadder.a

执行上面的代码时出现错误

unexpected fault address 0x0
fatal error: fault
[signal SIGSEGV: segmentation violation code=0x80 addr=0x0 pc=0x563c99b74244]

goroutine 17 [running, locked to thread]:
...

如何在go函数goadder中获取正确的c函数地址a

我引用了https://github.com/draffensperger/go-interlang/tree/master/c_to_go/static_go_lib


解决方案


c 函数只是跳转指针,而 golang 的回调是复杂的结构体,并且无法转换它们。 只有一种(安全)方法来调用 c 函数指针: 1)在某处声明:

//go:linkname cgocall runtime.cgocall
//go:nosplit
func cgocall(fn, arg unsafe.pointer /* may be uintptr */) int32

2)另外,请注意类型安全:

func goadder(x, y c.int, f unsafe.pointer /*  don't sure if this available, mb c.uintptr_t */) c.int

3) c 函数应该将指针(指向任何内容)作为参数

void a(goint *a)

(我会使用本机类型)

4)

ten := 10
cgocall(f, unsafe.Pointer(&ten))

(如果你想传递多个参数,它应该是结构体)

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>