非阻塞 tun/tap 文件描述符上的 read() 出现 EAGAIN 错误

mis*_*yes 2 sockets file-descriptor openvpn tun

我想从非阻塞 tun/tap 文件描述符中读取 IP 数据包,tunfd 我将其设置tunfd为非阻塞并在 libevent 中为其注册 READ_EV 事件。

当事件被触发时,我首先读取前20个字节以获取IP头,然后读取其余部分。

nr_bytes = read(tunfd, buf, 20);
...
ip_len = .... // here I get the IP length
....
nr_bytes = read(tunfd, buf+20, ip_len-20);
Run Code Online (Sandbox Code Playgroud)

但是对于read(tunfd, buf+20, ip_len-20) 我收到 EAGAIN 错误,实际上应该有一个完整的数据包,所以应该有一些字节,为什么我会收到这样的错误?

tunfd 与非阻塞模式或 libevent 不兼容?

谢谢!

Cel*_*ada 5

使用 TUN/TAP 进行读取和写入,就像在数据报套接字上读取和写入一样,必须针对完整的数据包。如果读入的缓冲区太小而无法容纳一个完整的数据包,缓冲区将被填满,而数据包的其余部分将被丢弃。对于写入,如果您写入部分数据包,驱动程序会认为它是完整数据包并通过隧道设备传送截断的数据包。

因此,当您读取 TUN/TAP 设备时,您必须提供至少与tuntap接口上配置的 MTU 一样大的缓冲区。