我需要创建两个侦听同一IP:端口但在不同接口上的套接字:
直到我意识到Linux将所有这些合并到相同的流量中之前,它似乎非常直接。例如,假设eth1上只有流量,而且eth0没有活动。当我第一次创建socket0时,它不会接收任何数据,但是一旦我创建socket1(并加入多播组),socket0也将开始接收相同的数据。我发现this link解释了这一点。
现在这对我来说实际上是有意义的,因为我指定网络接口的唯一时刻是加入多播组setsockopt(socket,IPPROTO_IP,IP_ADD_MEMBERSHIP,...)
和ip_mreq.imr_interface.s_addr
。我相信这指定哪个接口加入组,但与套接字将从哪个接口接收无关。
到目前为止我尝试将套接字绑定到多播地址和端口,其行为如上所述。我已经尝试绑定到接口地址,但这在Linux上不起作用(虽然它似乎在Windows上这样做),你在套接字上没有收到任何流量。最后,我尝试绑定到INADDR_ANY
,但这不是我想要的,因为我将收到发送到端口的任何其他数据而不管目标IP,例如单播,并且它仍然不会停止来自其他接口的组播数据。
我无法使用SO_BINDTODEVICE
,因为它需要root权限。
所以我想知道的是,这是否可行。如果不能做到那就没关系,我会把它作为一个答案并继续前进,我只是无法找到任何方法。哦,我已将问题标记为C
,因为这是我们正在使用的内容,但我认为它可能并不特定于该语言。
我没有包含此代码,因为我认为它更像是一个理论问题,而不是源代码的问题。我们一直在使用套接字(多播或其他)一段时间没有任何问题,这只是我们第一次必须处理多个接口。但如果您认为这可能有所帮助,我可以写一些最小的工作示例。
关于possible duplicate:的修改
我认为我在这里尝试实现的用例是不同的。套接字应该从相同的组播组和端口(上例中的224.2.2.2:5000)接收数据,但只能从一个特定接口接收数据。换句话说,两个接口都从同一个组播组接收数据(但不同的网络,因此数据不同),我需要每个套接字只能在一个接口上监听。
我认为这个问题是关于同一端口上的多个组,而不是来自不同接口的相同组。除非我没有看到那些可能真正帮助我的东西。
答案 0 :(得分:3)
是的,你可以在Linux上做你想要的,没有root权限:
绑定到INADDR_ANY
并设置IP_PKTINFO
套接字选项。然后,您必须使用recvmsg()
接收多播UDP数据包并扫描IP_PKTINFO
控制消息。这为您提供了收到的UDP数据包的一些边带信息:
struct in_pktinfo {
unsigned int ipi_ifindex; /* Interface index */
struct in_addr ipi_spec_dst; /* Local address */
struct in_addr ipi_addr; /* Header Destination address */
};
ipi_ifindex
是接收数据包的接口索引。 (您可以使用if_indextoname()
将其转换为接口名称,或使用if_nametoindex()
转换为接口名称。
正如你在Windows上所说的那样,相同的网络功能具有不同的语义,特别是对于UDP,甚至更多用于组播。
UDP套接字的IP地址的Linux bind()
语义几乎没用。它本质上只是一个目标地址过滤器。您几乎总是希望绑定到INADDR_ANY
以获取UDP套接字,因为您不关心数据包的发送地址,也不想接收多个地址的数据包(例如,接收单播和多播)。