登录
首页 >  Golang >  Go问答

为什么在调用 os.File.Fd() 后,文件无法通过 SetDeadline、SetReadDeadline 和 SetWriteDeadline 设置超时?

来源:stackoverflow

时间:2024-02-08 18:48:24 329浏览 收藏

本篇文章向大家介绍《为什么在调用 os.File.Fd() 后,文件无法通过 SetDeadline、SetReadDeadline 和 SetWriteDeadline 设置超时?》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。

问题内容

我使用 os.File.SetReadDeadlineos.File.ReadFull 的组合。但即使使用 SetReadDeadline,我设置的截止日期也被完全忽略,并且 ReadFull 永远阻塞。这是为什么?

其他信息:我向文件触发了一些 IOCTLS,因此需要 os.File.Fd() 来获取文件描述符。


正确答案


tl;博士:

使用 os.file.fd() 后,在文件上使用 syscall.setnonblock(fd.fd(), true)

这是由于 read 在 golang unix 中:

func (fd *fd) read(p []byte) (int, error) {
    if err := fd.readlock(); err != nil {
        return 0, err
    }
    defer fd.readunlock()
    if len(p) == 0 {
        // if the caller wanted a zero byte read, return immediately
        // without trying (but after acquiring the readlock).
        // otherwise syscall.read returns 0, nil which looks like
        // io.eof.
        // todo(bradfitz): make it wait for readability? (issue 15735)
        return 0, nil
    }
    if err := fd.pd.prepareread(fd.isfile); err != nil {
        return 0, err
    }
    if fd.isstream && len(p) > maxrw {
        p = p[:maxrw]
    }
    for {
        n, err := ignoringeintrio(syscall.read, fd.sysfd, p)
        if err != nil {
            n = 0
            if err == syscall.eagain && fd.pd.pollable() {
                if err = fd.pd.waitread(fd.isfile); err == nil {
                    continue
                }
            }
        }
        err = fd.eoferror(n, err)
        return n, err
    }
}

如果文件设置为阻塞模式,第一个 n, err := ignoringeintrio(syscall.read, fd.sysfd, p) 将永远阻塞。 waitread 仅当文件以非阻塞模式打开时才会执行。但我确实以非阻塞模式打开了文件,那么发生了什么?

os.file.fd()的实现 破坏了它

func (f *File) Fd() uintptr {
    if f == nil {
        return ^(uintptr(0))
    }

    // If we put the file descriptor into nonblocking mode,
    // then set it to blocking mode before we return it,
    // because historically we have always returned a descriptor
    // opened in blocking mode. The File will continue to work,
    // but any blocking operation will tie up a thread.
    if f.nonblock {
        f.pfd.SetBlocking()
    }

    return uintptr(f.pfd.Sysfd)
}

fd() 始终将文件设置为阻塞。因此,我们必须在等待轮询读取之前撤消该操作。因此:

使用 os.file.fd() 后,在文件上使用 syscall.setnonblock(fd.fd(), true)

以上就是《为什么在调用 os.File.Fd() 后,文件无法通过 SetDeadline、SetReadDeadline 和 SetWriteDeadline 设置超时?》的详细内容,更多关于的资料请关注golang学习网公众号!

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