3

考虑下面的代码,

我正在尝试为多播绑定 UDP 套接字。
我已经将它绑定到一个特定的端口,并设置IP_ADD_MEMBERSHIP了要监听的地址。

我的问题:套接字会接收到该端口的单播 UDP 数据包吗?如果是这样,我该如何预防?我希望只接收多播。

int fd;
if ((fd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
{
    perror("socket");
    exit(1);
}

u_int yes = 1;
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) < 0)
{
    perror("Reusing ADDR failed");
    exit(1);
}

struct sockaddr_in addr;
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.s_addr = (source_iface.empty()
                          ? htonl(INADDR_ANY)
                          : inet_addr(source_iface.c_str()));

if (bind(fd,(struct sockaddr *)&addr, sizeof(addr)) < 0)
{
    perror("bind");
    exit(1);
}

struct ip_mreq mreq;
mreq.imr_multiaddr.s_addr = inet_addr(group.c_str());
mreq.imr_interface.s_addr = (source_iface.empty()
                               ? htonl(INADDR_ANY)
                               : inet_addr(source_iface.c_str()));

if (setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0)
{
    perror("setsockopt");
    exit(1);
}
4

2 回答 2

2

我相信您还需要bind在您想要收听的特定多播地址上,而不仅仅是在setsockopt通话中 - 后者也是确保网卡和 IGMP 也做正确的事情所必需的。

另请参阅绑定多播 (UDP) 套接字是什么意思?

于 2013-10-31T09:05:04.680 回答
0

好的,我让我的实例工作,所以我知道问题出在哪里。bind()必须针对多播 IP 地址发生:

struct sockaddr_in addr;
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.s_addr = (group.empty()
                          ? htonl(INADDR_ANY)   // <-- this will work here but not below
                          : inet_addr(group.c_str()));

if (bind(fd,(struct sockaddr *)&addr, sizeof(addr)) < 0)
{
    perror("bind");
    exit(1);
}

这里的区别是group而不是source_iface. 在我改变它之前,它根本不会接收广播数据包。

于 2020-10-17T22:08:37.777 回答