将使用 *byte 类型的结构体传递给 Syscall 并在执行后读取其数据
来源:stackoverflow
时间:2024-03-24 19:24:33 241浏览 收藏
在调用 DLL 中的 C 方法时,需要将一个包含 char* 字段的 C 结构体传递给 Syscall。为了在 Go 中读取结构体数据的错误字符串,可以使用 *byte 类型来表示 char* 字段,并使用 gostringn 或 GoString 函数来获取字符串的内容。同时,需要确保 C 结构体的 int 字段与 Go 中的 int 类型大小一致,否则可能会导致错误的偏移量。
我正在使用 syscall.syscall(...)
调用 dll 中的 c 方法。
这是 c 方法签名:
sensei_api hsensei sensei_open(const char* sensigrafo, const char* options, sensei_err* se);
这是 sensei_err
结构:
typedef struct { int code; char* error_string; } sensei_err;
在我的 go 程序中,我声明了一个结构:
type senseierr struct { code int error_string *byte }
并尝试调用该方法:
var nargs uintptr = 3 var err senseiErr ret, _, callErr := syscall.Syscall(uintptr(senseiOpen), nargs, uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr("en"))), uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(""))), uintptr(unsafe.Pointer(&err)), )
正如您可能已经猜到的,sensei_open
方法使用错误的代码和文本填充 sensei_err
参数。
现在我需要阅读该错误的内容。
err.code
实际上具有正确的值。
关于 err.error_string
我不知道。我是 go 新手,有一些问题:
- 由于 c 结构体具有字段
char* error_string
,所以我的 go 结构体中的error_string *byte
是否正确?- 我应该使用
[]byte
还是其他东西?
- 我应该使用
- 如何读取
error_string
字段的内容?fmt.println(err.error_string)
打印内存地址fmt.println(*err.error_string)
始终打印“101”
解决方案
1) 我怀疑 cost char*
是否应该采用 utf16 编码。因此,您所需要的只是获取原始数据:
sensigrafo := "en\000" // \000 = 0 = null termination, \0 does not valid options := "\000" ... uintptr(*(*unsafe.pointer)(unsafe.pointer(&sensigrafo)) uintptr(*(*unsafe.pointer)(unsafe.pointer(&options)) // *(*unsafe.pointer) are accessing the first field of string header: type string struct { data *byte len int } // same with slices // but for them there's less ugly way: sensigrafo := []byte("en\000") options := []byte("\000") uintptr(unsafe.pointer(&sensigrafo[0])) uintptr(unsafe.pointer(&options[0]))
2) c 的 int
和 golang 的 int
可能有不同的 sizeof,所以这需要 cgo 声明(c.int
)或手动随机选择匹配(如果不想使用 cgo,也可以尝试 int32、int64)
type senseierr struct { code c.int /* golang's int32/int64 */ error_string *byte // pointer types are same as c's void* or golang's unsafe.pointer }
错误的偏移量可能会导致 error_string 为空或指向随机地址。
3) 要读取内容,您必须使用与 c 相同的方法(读取数据直到 null 终止字节,考虑到 *byte 指向字符串的第一个元素),但我建议使用已经实现的运行时函数:
//go:linkname gostringn runtime.gostringn func gostringn(p *byte, l int) string //go:linkname findnull runtime.findnull //go:nosplit func findnull(s *byte) int ... error_string := gostringn(err.error_string, findnull(err.error_string)) // or cgo one: type senseiErr struct { code C.int error_string *C.char } ... error_string := C.GoString(err.error_string)
到这里,我们也就讲完了《将使用 *byte 类型的结构体传递给 Syscall 并在执行后读取其数据》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注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次学习