问题 如何判断套接字从哪个接口收到消息?


如果绑定了套接字 IN6ADDR_ANY 要么 INADDR_ANY 你使用的电话如 recvfrom() 在套接字上接收消息。有没有办法找出消息来自哪个接口?

在IPv6链接范围消息的情况下,我希望from的参数 recvfrom() 会有的 scope_id 字段初始化为接口Id。不幸的是它被设定为 0 在我的测试程序中。

谁知道找到这些信息的方法?


8250
2018-03-02 19:14


起源



答案:


除了绑定到每个接口之外,我本身并不知道使用IPv4的方法。

IPv6已添加IPV6_PKTINFO套接字选项以解决此缺点。有效的选项,a struct in6_pktinfo 将作为辅助数据返回。


3
2018-03-03 17:20



谢谢,这就是我想要的。太糟糕了,它还没有得到Python套接字模块的支持 - 尽管我在这个问题中确实指定了libc。 :) - Readonly


答案:


除了绑定到每个接口之外,我本身并不知道使用IPv4的方法。

IPv6已添加IPV6_PKTINFO套接字选项以解决此缺点。有效的选项,a struct in6_pktinfo 将作为辅助数据返回。


3
2018-03-03 17:20



谢谢,这就是我想要的。太糟糕了,它还没有得到Python套接字模块的支持 - 尽管我在这个问题中确实指定了libc。 :) - Readonly


dwc是对的,IPV6_PKTINFO适用于Linux上的IPv6。

此外,IP_PKTINFO将适用于IPv4 - 您可以在联机帮助页ip(7)中查看详细信息


8
2018-03-03 20:15



Linux并不是整个世界。了解可移植性问题是件好事。 - dwc
@dwc:如果你没有Linux,你应该模仿它:P - Matt Joiner


我构建了一个提取源,目标和接口地址的示例。为简洁起见,未提供错误检查。看到这个副本: 获取收到的UDP数据包的目标地址

// 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
}

2
2018-03-15 08:33





自从我一直在进行C / C ++ TCP / IP编码以来已经有一段时间了,但就我记忆中的每条消息(或派生套接字)而言,您可以进入IP头信息。这些标头应包含接收地址,该地址将是您询问的接口的IP。


-1
2018-03-02 22:16





除了在Glomek建议的每个接口上打开一个单独的套接字之外,我知道在Windows上明确执行此操作的唯一方法是使用原始套接字,例如,

  SOCKET s = socket(AF_INET, SOCK_RAW, IPPROTO_IP);

每个从这个套接字接收将是一个 IP包,包含源地址和目标地址。我工作的程序要求我使用SIO_RCVALL选项将套接字置于混杂模式。这样做意味着我可以获得网络上“看到”的每个IP数据包。要明确地为我的应用程序提取数据包,需要我使用IP和TCP / UDP标头中的地址和端口过滤数据。显然,这可能比你感兴趣的更多开销。我只提到它说 - 我从来没有使用原始套接字而没有将它放在混杂模式中。所以我不确定你是否可以将它绑定到INADDR_ANY并从那时起将其用作常规套接字。在我看来,你可以;我从来没有尝试过。

编辑:  读这个 文章 有关Windows上的原始套接字的限制。我在项目中面临的最大障碍是,必须成为Administrators组的成员才能在Windows 2000及更高版本上打开原始套接字。


-2
2018-03-03 05:34