获取接收到的UDP数据包的目的地址

Posted

技术标签:

【中文标题】获取接收到的UDP数据包的目的地址【英文标题】:Get destination address of a received UDP packet 【发布时间】:2011-07-13 23:26:34 【问题描述】:

收到 UDP 数据包后,我需要用他用来发送我要回复的数据包的地址来回复发件人。

recvfrom 调用让我获得了发送者的地址,但我如何获得接收到的数据包的目标地址,该地址应该与本地主机接口之一的地址匹配?

【问题讨论】:

@sarnold: getsockname 在绑定到0.0.0.0:0(或[::]:0)的侦听套接字上没有那么有用。使用 TCP,您在 accept 之后有一个本地地址,但使用 UDP……我不知道如何回答 OP 的问题。 @ephemient,当马特第一次问这个问题时,我建议连接他的 UDP 套接字,这样他就可以使用getsockname(2)。听起来它会奏效:) 现在我个人的利益就是要为他找到解决方案。 :) 顺便说一下,另一种方法是为每个接口创建一个单独的 UDP 套接字,绑定到每个接口。然后你接收数据的套接字直接与接收数据的接口相关联。 How to tell which interface the socket received the message from? 的可能重复项 @JasonC 这不是重复的。目标地址和目标接口是两个不同的东西。一个接口可以有并且经常有多个地址。至少在 Linux 上,一个地址可以分配给两个接口,尽管我不知道有什么好的用例。 【参考方案1】:

我构建了一个提取源地址、目标地址和接口地址的示例。为简洁起见,不提供错误检查。

// sock is bound AF_INET socket, usually SOCK_DGRAM
// include struct in_pktinfo in the message "ancilliary" control data
setsockopt(sock, IPPROTO_IP, IP_PKTINFO, &opt, sizeof(opt));
// the control data is dumped here
char cmbuf[0x100];
// the remote/source sockaddr is put here
struct sockaddr_in peeraddr;
// if you want access to the data you need to init the msg_iovec fields
struct msghdr mh = 
    .msg_name = &peeraddr,
    .msg_namelen = sizeof(peeraddr),
    .msg_control = cmbuf,
    .msg_controllen = sizeof(cmbuf),
;
recvmsg(sock, &mh, 0);
for ( // iterate through all the control headers
    struct cmsghdr *cmsg = CMSG_FIRSTHDR(&mh);
    cmsg != NULL;
    cmsg = CMSG_NXTHDR(&mh, cmsg))

    // ignore the control headers that don't match what we want
    if (cmsg->cmsg_level != IPPROTO_IP ||
        cmsg->cmsg_type != IP_PKTINFO)
    
        continue;
    
    struct in_pktinfo *pi = CMSG_DATA(cmsg);
    // at this point, peeraddr is the source sockaddr
    // pi->ipi_spec_dst is the destination in_addr
    // pi->ipi_addr is the receiving interface in_addr

【讨论】:

我很好奇是否有一种方法可以在没有 for 循环的情况下获取目标地址...寻找 O(1) 操作。 很好奇将目标地址命名为“对等地址”。对等地址实际上是地址。 @EJP: 接收数据包时只有来源。 @Hei:不。为了提供 O(1) 查找,您需要将辅助控制结构打包到提供 O(1) 查找的数据容器中。填充该结构至少是 O(n) 辅助控制消息的数量。您不必担心查找所需控制消息的复杂性:总是只有几个,而且只有您请求的那些。还需要代表您进行零内存管理。它的开销可以忽略不计。 接收数据包是本题的主题。【参考方案2】:

您使用 setsockopt 设置 IP_PKTINFO 选项,然后使用 recvmsg 并在 struct msghdr 的 msg_control 成员中获取 in_pktinfo 结构。 in_pktinfo 有一个包含数据包目标地址的字段。

请参阅:http://www.linuxquestions.org/questions/programming-9/how-to-get-destination-address-of-udp-packet-600103/,我在其中找到了更多详细信息。

【讨论】:

我在this answer 中提供了一个完整的工作示例。

以上是关于获取接收到的UDP数据包的目的地址的主要内容,如果未能解决你的问题,请参考以下文章

包过滤防火墙无法处理UDP、RPC的协议

ACL——访问控制列表

发送 UDP 数据包的长时间延迟

如何从多个 IP 数据包重建 TCP 流?

使用 Qt 获取 UDP 数据并显示成图片

C# UDP Socket:获取接收者地址