在Go语言中调用C语言共享库的方法
来源:stackoverflow
时间:2024-03-02 09:36:25 110浏览 收藏
珍惜时间,勤奋学习!今天给大家带来《在Go语言中调用C语言共享库的方法》,正文内容主要涉及到等等,如果你正在学习Golang,或者是对Golang有疑问,欢迎大家关注我!后面我会持续更新相关内容的,希望都能帮到正在学习的大家!
我使用 go 编写了 c 共享库。
package main import "c" import "log" //export runlib func runlib() { log.println("call runlib") } func init() { log.println("call init") } func main() { log.println("call main") }
我使用以下命令创建了库:
go build -buildmode=c-shared -o lib.so lib.go
为了使用该库,我编写了这段 golang 代码。
package main /* #include#include #include static void callfromlib() { void (*fn)(); void *h = dlopen("lib.so", rtld_lazy); if (!h) { fprintf(stderr, "error: %s\n", dlerror()); return; } *(void**)(&fn) = dlsym(h, "runlib"); if (!fn) { fprintf(stderr, "error: %s\n", dlerror()); dlclose(h); return; } fn(); dlclose(h); } */ import "c" func main() { c.callfromlib() }
如果我运行最后一个代码,它会抛出此错误(go run call.go
):
fatal error: bad sweepgen in refill goroutine 1 [running, locked to thread]: runtime.throw({0xb990782, 0xc000042a38}) /usr/local/go/src/runtime/panic.go:1198 +0x71 fp=0xc000042a18 sp=0xc0000429e8 pc=0xb930691 runtime.(*mcache).refill(0x41215b8, 0x2) /usr/local/go/src/runtime/mcache.go:156 +0x24e fp=0xc000042a68 sp=0xc000042a18 pc=0xb91434e runtime.(*mcache).nextfree(0x41215b8, 0x2) /usr/local/go/src/runtime/malloc.go:880 +0x85 fp=0xc000042ab0 sp=0xc000042a68 pc=0xb90ba85 runtime.mallocgc(0x8, 0xb9ba300, 0x1) /usr/local/go/src/runtime/malloc.go:1071 +0x4e8 fp=0xc000042b30 sp=0xc000042ab0 pc=0xb90c108 runtime.growslice(0xb9ba300, {0x0, 0x41a1910, 0x2}, 0xc00009c000) /usr/local/go/src/runtime/slice.go:267 +0x4ea fp=0xc000042b98 sp=0xc000042b30 pc=0xb94586a sync.(*pool).pinslow(0xba394e0) /usr/local/go/src/sync/pool.go:223 +0x105 fp=0xc000042c30 sp=0xc000042b98 pc=0xb960ec5 sync.(*pool).pin(0xba394e0) /usr/local/go/src/sync/pool.go:206 +0x4e fp=0xc000042c48 sp=0xc000042c30 pc=0xb960d8e sync.(*pool).get(0xba394e0) /usr/local/go/src/sync/pool.go:128 +0x25 fp=0xc000042c80 sp=0xc000042c48 pc=0xb960ac5 fmt.newprinter() /usr/local/go/src/fmt/print.go:137 +0x25 fp=0xc000042ca8 sp=0xc000042c80 pc=0xb985d45 fmt.sprintln({0xc000042d38, 0x1, 0x1}) /usr/local/go/src/fmt/print.go:280 +0x28 fp=0xc000042cf0 sp=0xc000042ca8 pc=0xb986008 log.println({0xc000042d38, 0x24, 0x0}) /usr/local/go/src/log/log.go:329 +0x1e fp=0xc000042d20 sp=0xc000042cf0 pc=0xb98cd5e main.runlib(...) /users/.../demo/lib.go:8 _cgoexp_6b951f94a90e_runlib(0xc000042d90) _cgo_gotypes.go:36 +0x45 fp=0xc000042d58 sp=0xc000042d20 pc=0xb98cf85 runtime.cgocallbackg1(0xb98cf40, 0xc000042e60, 0x0) /usr/local/go/src/runtime/cgocall.go:306 +0x29a fp=0xc000042e28 sp=0xc000042d58 pc=0xb903d1a runtime.cgocallbackg(0xc0000001a0, 0x300000002, 0xc0000001a0) /usr/local/go/src/runtime/cgocall.go:232 +0x109 fp=0xc000042eb8 sp=0xc000042e28 pc=0xb9039e9 runtime.cgocallbackg(0xb98cf40, 0x7ffeefbff737, 0x0):1 +0x2f fp=0xc000042ee0 sp=0xc000042eb8 pc=0xb95e32f runtime: unexpected return pc for runtime.cgocallback called from 0x4053e00 stack: frame={sp:0xc000042ee0, fp:0xc000042f08} stack=[0xc000042000,0xc000043000) 0x000000c000042de0: 0x000000c000042d9d 0x000000c000042e18 0x000000c000042df0: 0x000000000b95843b 0x000000c0000001a0 0x000000c000042e00: 0x000000c000042dd8 0x0000000000000000 0x000000c000042e10: 0x000000000b9c3dc0 0x000000c000042ea8 0x000000c000042e20: 0x000000000b9039e9 0x000000000b98cf40 <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> ... <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> 0x000000c000042ea0: 0x00007ffeefbff737 0x000000c000042ed0 0x000000c000042eb0: 0x000000000b95e32f 0x000000c0000001a0 ... 0x000000c000042ee0: <0x000000000b98cf40 <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> 0x00007ffeefbff737 ... runtime.cgocallback(0x4004165, 0x4058340, 0xc000042f70) /usr/local/go/src/runtime/asm_amd64.s:915 +0xb4 fp=0xc000042f08 sp=0xc000042ee0 pc=0xb95c134 goroutine 1 [runnable, locked to thread]: unicode.init() /usr/local/go/src/unicode/tables.go:9 +0x79 exit status 2
但是如果我使用 python 一切正常!
>>> import ctypes >>> lib = ctypes.cdll.loadlibrary("lib.so") >>> 2022/03/02 01:08:17 call init >>> lib.runlib() 2022/03/02 01:08:22 call runlib 0 >>>
信息
操作系统:macos big sur 11.6.3 (20g415)
>>> clang --version apple clang version 13.0.0 (clang-1300.0.29.30)
>>> nm lib.so| grep runlib 000000000008cfa0 t _runlib 000000000008cf40 t __cgoexp_6b951f94a90e_runlib 00000000000c4200 s __cgoexp_6b951f94a90e_runlib.stkobj
>>> go env GO111MODULE="on" GOARCH="amd64" GOBIN="" GOCACHE="~/Library/Caches/go-build" GOENV="~/Library/Application Support/go/env" GOEXE="" GOEXPERIMENT="" GOFLAGS="" GOHOSTARCH="amd64" GOHOSTOS="darwin" GOINSECURE="" GOMODCACHE="~/go/pkg/mod" GONOPROXY="" GONOSUMDB="" GOOS="darwin" GOPATH="~/go" GOPRIVATE="" GOPROXY="https://proxy.golang.org,direct" GOROOT="/usr/local/go" GOSUMDB="sum.golang.org" GOTMPDIR="" GOTOOLDIR="/usr/local/go/pkg/tool/darwin_amd64" GOVCS="" GOVERSION="go1.17" GCCGO="gccgo" AR="ar" CC="clang" CXX="clang++" CGO_ENABLED="1" GOMOD="~/projects/go/.../go.mod" CGO_CFLAGS="-g -O2" CGO_CPPFLAGS="" CGO_CXXFLAGS="-g -O2" CGO_FFLAGS="-g -O2" CGO_LDFLAGS="-g -O2" PKG_CONFIG="pkg-config" GOGCCFLAGS="-fPIC -arch x86_64 -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/wl/9rtfdx8x7jvgyn6t8hpq7yh00000gn/T/go-build401121298=/tmp/go-build -gno-record-gcc-switches -fno-common"
我用谷歌搜索并找到了这篇文章。但文章中并没有golang。
https://medium.com/learning-the-go-programming-language/calling-go-functions-from-other-languages-4c7d8bcc69bf
正确答案
一位 go 贡献者写道,一个镜像中不可能有两个 go 运行时。该评论指的是 windows,但似乎也适用于 macos,因为我可以可靠地重现 op 提到的崩溃:
参见https://github.com/golang/go/issues/50304#issuecomment-999302888
但是,由于 op 使用 macos,因此还有一种替代方法:使用插件,请参阅 https://pkg.go.dev/plugin:
创建库的命令将更改如下:
go build -buildmode=plugin -o lib.so lib.go
因此,您对库中定义的函数的调用在 go 中将如下所示:
package main import ( "plugin" ) func main() { plug, err := plugin.open("lib.so") if err != nil { panic(err) } runlib, err := plug.lookup("runlib") if err != nil { panic(err) } runlib.(func())() }
使用此程序进行的测试会生成以下日志输出:
stephan@mac golang-c-lib-in-go % go run call.go 2022/10/29 23:23:01 Call init 2022/10/29 23:23:01 Call RunLib
如果该库还用于从 c 和/或 python 调用,那么一种可能性是以两种变体创建该库:一种是从 c 和/或 python 调用,如问题所示,以及插件变体中的一个,然后可以由 go 程序使用。
在您的示例中,您不需要对 lib.go
进行任何更改,唯一的区别是创建库的命令行。
最后,来自 documentation 的重要提示:
今天带大家了解了的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习