我可以从套接字中读取一个UDP数据包吗?

时间:2010-03-30 18:21:30

标签: c udp

在Linux上使用UNIX套接字API,有什么方法可以保证我只读一个UDP数据包,只读一个UDP数据包?我正在使用recvmsg从非阻塞套接字读取数据包,缓冲区大小比我们内部网络的MTU略大。这应该确保我总能收到完整的UDP数据包,但我不确定如果数据包很小,我可以保证每次recvmsg调用永远不会收到多个数据包。

recvmsg手册页引用了MSG_WAITALL选项,该选项尝试等待缓冲区填满。我们没有使用它,这是否意味着recvmsg将在读取一个数据报后始终返回?有什么方法可以保证吗?

理想情况下,我想要一个跨UNIX的解决方案,但如果不存在,那么Linux是否具体?

3 个答案:

答案 0 :(得分:13)

recvmsg将返回一个数据包,它将是整个数据包(只要你提供的缓冲区足够大)。

来自the POSIX documentation

  

recvmsg()函数将从连接模式或无连接模式套接字接收消息。

“消息”仅指一个消息(或数据包),以及

  

对于基于消息的套接字,例如SOCK_DGRAM和SOCK_SEQPACKET,整个消息应该在一次操作中读取。

答案 1 :(得分:0)

一个选项(我说选项)是使用libpcap使用pcap_next并将其拆分以查看它是否是udp数据包。你可以这样做:

/* jump pass the ethernet header */
ipdata = (struct ip*)(packet + sizeof(struct ether_header));
length -= sizeof(struct ether_header);

(借用tcpdump)

然后通过执行以下操作来测试ip结构以查看它是否是udp数据包:

if ( ipdata->ip_p == IPPROTO_UDP )

如果失败,请继续循环(调用pcap_next)直到获得udp数据包。当然,udp数据报的提取更难以这种方式,但它确实让你很好地进入数据包内部。请参阅tcpdump源以了解如何删除信息以及出现的信息。

答案 2 :(得分:0)

接受的答案没有明确回答OP的问题,因为它只是在传递时提到了缓冲区大小。

  

我正在使用recvmsg从非阻塞套接字读取数据包,   缓冲区大小比我们内部的MTU略大   网络

重要的是你的缓冲区足够大以适应整个数据报。数据报最多可达65,536字节。当一个大数据报由于MTU而被分段,它将被堆栈重新组装,你将不会知道这一点,在收到所有片段并将它们放回原始数据报之前,你将不会收到任何信息。如果你的缓冲区略大于一个MTU,例如1600字节,并且你在一个40K字节的传入数据报上调用recv(),你将只获得前1600字节。