最近在学习组播时,遇到一个报错,setsockopt: No such device,根据网上的说法,原因是这样的:
1、机器没有可用的组播功能
2、没有为主机添加路由,有些系统不添加默认情况。你需要在终端下输入:route add -net 224.0.0.0 netmask 224.0.0.0 eth0(or similar).
但是这两个答案我都不怎么满意,因为之前已经有包含组播的程序在运行了。而报错的代码是这样的一段:
struct ip_mreq ipmr;ipmr.imr_interface.s_addr = htonl(INADDR_ANY);inet_pton(AF_INET, argv[3], &ipmr.imr_multiaddr);if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipmr, sizeof(ipmr)) < 0){perror("setsockopt");exit(EXIT_FAILURE);}
即加入某一个组播,我在设置本地接口时使用的是INADDR_ANY。我使用ifconfig查看本地网卡,可以看到网卡是存在MULTICAST功能的,而且《UNIX网络编程》中也说可以指定通配地址,会由内核选择一个本地接口。但是我想也许这部分与操作系统有关,所以我决定试一下指定本地端口,于是将代码改为了这样
inet_pton(AF_INET, argv[1], &ipmr.imr_interface);inet_pton(AF_INET, argv[3], &ipmr.imr_multiaddr);if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipmr, sizeof(ipmr)) < 0){perror("setsockopt");exit(EXIT_FAILURE);}
并在参数1中指定了本地网卡IP。这样修改之后,问题就解决了。虽然不是很清楚具体原因。。。。。。而且我发现在绑定端口时,使用通配地址,在加入组播时,使用确定的接口ip这样比较好。大体代码如下:
int sockfd = socket(AF_INET, SOCK_DGRAM, 0);struct sockaddr_in servaddr;memset(&servaddr, 0, sizeof(servaddr));servaddr.sin_family = AF_INET;servaddr.sin_addr.s_addr = htonl(INADDR_ANY);servaddr.sin_port = htons(atoi(argv[2]));bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));struct ip_mreq ipmr;inet_pton(AF_INET, argv[1], &ipmr.imr_interface);inet_pton(AF_INET, argv[3], &ipmr.imr_multiaddr);if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipmr, sizeof(ipmr)) < 0){perror("setsockopt");exit(EXIT_FAILURE);}
其中参数1是本地ip,参数2是端口号,参数3是组播ip
program 11.11.11.11 6789 224.1.1.1