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

时间:2013-06-16 23:16:27

标签: sockets file-descriptor openvpn tun

我想从非阻塞的tun / tap文件描述符tunfd中读取IP数据包 我将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);

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

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

谢谢!

1 个答案:

答案 0 :(得分:4)

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

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