13

如果一个套接字被绑定到IN6ADDR_ANYorINADDR_ANY并且您使用诸如recvfrom()在套接字上接收消息的调用。有没有办法找出消息来自哪个接口?

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

有人知道找到这些信息的方法吗?

4

5 回答 5

10

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

此外,IP_PKTINFO 适用于 IPv4 — 您可以在手册页 ip(7) 中查看详细信息

于 2009-03-03T20:15:12.010 回答
5

我构建了一个提取源、目标和接口地址的示例。为简洁起见,不提供错误检查。请参阅此副本:获取接收到的 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
}
于 2011-03-15T08:33:24.317 回答
4

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

IPv6 添加了 IPV6_PKTINFO 套接字选项来解决这个缺点。使用该选项,astruct in6_pktinfo将作为辅助数据返回。

于 2009-03-03T17:20:41.367 回答
-1

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

于 2009-03-02T22:16:28.887 回答
-2

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

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

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

编辑: 阅读这篇文章以了解有关 Windows 上原始套接字的限制。我在项目中遇到的最大障碍是必须是管理员组的成员才能在 Windows 2000 及更高版本上打开原始套接字。

于 2009-03-03T05:34:15.357 回答