登录
首页 >  Golang >  Go问答

调用 Syscall 函数并传递字符串

来源:stackoverflow

时间:2024-03-19 20:03:32 262浏览 收藏

在 Go 中调用 C DLL 时,传递字符串参数需要一定的技巧。由于 Go 字符串使用 UTF-8 编码,而 C 函数可能需要不同的编码,因此需要转换字符串。文章提供了几种常见的编码方案,包括 ASCII、UTF-16 和 Windows 特定的 UTF-16 转换,并提供了使用 mkwinsyscall 工具自动处理转换的示例。

问题内容

我需要将字符串作为参数传递给golang中的c dll,我想要得到这样的东西:

proc, e = syscall.getprocaddress(h, "jlinkarm_execcommand") //one of the functions

    vals := []string{"device = stm32f429zi"}
    start := uintptr(unsafe.pointer(&vals[0]))

    asd, _, _ = syscall.syscall6(uintptr(proc), 3, start, 0, 0, 0, 0, 0)

但它不起作用。系统调用将 uintptr 作为参数。 我无法使用 c 包,因为它无法在 windows 上构建。

在 c 语言中,它的工作方式如下:

strcpy(acIn, "device = STM32F407IE");

JLINKARM_ExecCommand(acIn, &acOut[0], sizeof(acOut));

那么 c dll 有可能获取字符串参数并正确使用它吗?


解决方案


要将字符串传递给系统调用,您需要传递指向字符串第一个字符的指针。第一个问题是 dll 函数需要什么字符串编码。 go 字符串被编码为 utf-8 unicode,因此如果您的 c 函数需要其他内容,您必须首先转换它们。以下是一些常见情况:

1) ascii 字符串

假设您的 c 函数需要一个以零结尾的 ascii 字符串,那么您可以执行以下操作:

s := "some string"
b := append([]byte(s), 0)
syscall.syscall(uintptr(proc), 1, uintptr(unsafe.pointer(&b[0])), 0, 0)

首先将字符串转换为字节数组,并在末尾添加 c 期望的零。然后将指针传递给第一个字节字符。

为了安全起见,您还应该确保实际上仅传入有效的 ascii 字符串,而不传入无效字符。通常只有 [0..127] 范围内的字符对于通用 ascii 有效。其余的取决于当前的代码页。

2) windows 上的 utf-16

如果您调用 windows dll,您通常希望使用该函数的 utf-16 版本,例如SendMessageW,因此需要将字符串转换为 utf-16。幸运的是,windows 下有一个包装函数,因此您只需执行以下操作:

s := "some string"
s16, err := syscall.utf16ptrfromstring(s)
if err == nil {
    syscall.syscall(uintptr(proc), 1, uintptr(unsafe.pointer(s16)), 0, 0)
}

这将转换为 utf-16 并在末尾附加预期的零。

处理此问题的一个好方法是使用 mkwinsyscall 工具。您可以创建 像这样的 go 文件:

package main
//go:generate mkwinsyscall -output zmsi.go msi.go
//sys msiinstallproduct(path string, command string) (e error) = msi.msiinstallproductw

然后运行gogenerate,你会得到一个像这样的结果文件(删除了一些部分 为简洁起见):

func MsiInstallProduct(path string, command string) (e error) {
    var _p0 *uint16
    _p0, e = syscall.UTF16PtrFromString(path)
    if e != nil {
        return
    }
    var _p1 *uint16
    _p1, e = syscall.UTF16PtrFromString(command)
    if e != nil {
        return
    }
    return _MsiInstallProduct(_p0, _p1)
}

func _MsiInstallProduct(path *uint16, command *uint16) (e error) {
    r0, _, _ := syscall.Syscall(procMsiInstallProductW.Addr(), 2, uintptr(unsafe.Pointer(path)), uintptr(unsafe.Pointer(command)), 0)
    if r0 != 0 {
        e = syscall.Errno(r0)
    }
    return
}

如您所见,它也会自动处理字符串转换 作为系统调用代码,甚至错误处理。

今天关于《调用 Syscall 函数并传递字符串》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

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