在交换机的测试过程中发现一个问题,组播报文可以通过交换芯片发送到cpu,但是cpu的socket却不接收此组播报文,由于想要接收的组播报文来自不同的vlan,带有不同的vlan标签,所以一直排查不到问题。
1. 下面就是这个问题的原因:在socket加入组播的时候会做如下操作:
struct ip_mreq mreq;
unsigned long groupIpAddr = 0xefc20012;
unsigned long vlanIpAddr = 0x0a010001;
mreq.imr_multiaddr.s_addr = htonl(groupIpAddr);
mreq.imr_interface.s_addr = htonl(vlanIpAddr);
setsockopt(sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));
在上诉的代码中的groupIpAddr和vlanIpAddr是问题的根源,当我们调用下面的setsockopt的时候,那么在ip层会明白我们的接收的组播的来源应该是来源于vlanIpAddr所在的vlan,所以socket只能接收此vlan发送过来的组播报文,其他的vlan过来的组播报文,则会被socket丢掉;
修改为:
struct ip_mreq mreq;
unsigned long groupIpAddr = 0xefc20012;
unsigned long vlanIpAddr = 0x0a010001;
mreq.imr_multiaddr.s_addr = htonl(groupIpAddr);
mreq.imr_interface.s_addr = htonl(vlanIpAddr);
setsockopt(sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));
vlanIpAddr = 0x0a020001;
mreq.imr_interface.s_addr = htonl(vlanIpAddr);
setsockopt(sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));
这样就能将vlanip为10.1.0.1的vlan的组播收上来,也能将vlanip为10.2.0.1的vlan的组播收上来;IP_ADD_MEMBERSHIP这个选项的绑定关系是绑定了vlan和组播的关系,需要特别注意;