登录
首页 >  Golang >  Go问答

将使用 *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学习网公众号,带你了解更多关于的知识点!

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