登录
首页 >  Golang >  Go问答

如何从 golang 正确进行 IOCTL

来源:stackoverflow

时间:2024-04-12 15:36:37 241浏览 收藏

小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《如何从 golang 正确进行 IOCTL》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!

问题内容

我正在尝试将 raspberrypi 的用户空间代码从 c 移植到 golang,并且遇到了一个涉及 ioctl() 的程序。

我在执行 c 代码时遇到了麻烦

#define major_num 100
#define ioctl_mbox_property _iowr(major_num, 0, char *)
static int mbox_property(int file_desc, void *buf){

   int ret_val = ioctl(file_desc, ioctl_mbox_property, buf);
   return ret_val;
}

我的 go 等效项是

func mBoxProperty(f *os.File, buf [256]int64) {
        err := Ioctl(f.Fd(), IOWR(100, 0, 8), uintptr(unsafe.Pointer(&buf[0])))

        if err != nil {
                log.Fatalln("mBoxProperty() : ", err)
        }

}

func Ioctl(fd, op, arg uintptr) error {
        _, _, ep := syscall.Syscall(syscall.SYS_IOCTL, fd, op, arg)
        if ep != 0 {
                return syscall.Errno(ep)
        }
        return nil
}

func IOWR(t, nr, size uintptr) uintptr {
        return IOC(IocRead|IocWrite, t, nr, size)
}
func IOC(dir, t, nr, size uintptr) uintptr {
        return (dir << IocDirshift) | (t << IocTypeshift) | (nr << IocNrshift) | (size << IocSizeshift)
}

但是每当我运行这个时,我都会收到 invalid argument 错误,我认为这可能是由于我如何调用 ioctl() 但我不确定,我该如何解决这个问题?


解决方案


"golang.org/x/sys/unix"中有ioctl(2)包装器。unix.IoctlSetInt可能会满足您的需求。

看起来您还将对一个小内存缓冲区的控制权交给了内核。这样做需要小心:Go 垃圾收集器会释放它认为没有使用的内存对象,即使某些东西正在使用,它也可以移动它。内核不会知道这一点,并将继续使用旧指针。 unsafe.Pointer 文档对这个主题有相当多的论述,即使是关于不太奇特的系统调用。我不知道有什么可以“锁定”内存中的 Go 对象以防止其被移动或释放(例如,runtime 包中没有任何内容跳出)。

您可能会考虑使用 cgo 编写一个非常小的扩展,malloc()ed 是一个适当的缓冲区并将其传递给 ioctl。 malloc 的内存不会被垃圾收集,因此它不会移动或从您的下面释放;一些低级工具可能认为这看起来像内存泄漏(保留指针的旧值以便以后能够释放它并避免这种情况不是一个坏主意)。

您可能还会遇到 uintpr(unsafe.Pointer(...)) 需要在对 syscall.Syscall 的调用中发生的细节。

以下是详细信息,来自 https://golang.org/pkg/unsafe/#Pointer

使用 unsafe 的技巧可以让你“锁定”上面 Dave Maze 正在寻找的 Go 对象。

终于介绍完啦!小伙伴们,这篇关于《如何从 golang 正确进行 IOCTL》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

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