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

Mat*_*ner 19 linux networking udp ipv4

在收到UDP数据包后,我需要使用他用来发送我正在回复的数据包的地址来响应发送方.

recvfrom调用允许我获取发送方的地址,但是如何获取接收数据包的目标地址,该地址应该与本地主机接口之一的地址匹配?

Mat*_*ner 20

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

// 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
}
Run Code Online (Sandbox Code Playgroud)

  • @Hei:不。为了提供 O(1) 查找,您需要将辅助控制结构打包到提供 O(1) 查找的数据容器中。填充该结构至少需要 O(n) 数量的辅助控制消息。您不必担心查找所需控制消息的复杂性:总是只有少数,而且只有您请求的那些。您还需要零内存管理。这是可以忽略不计的开销。 (2认同)

gby*_*gby 13

使用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/我在哪里找到了答案以获取更多详细信息.