在过去两天中遇到以下问题:如果发件人订阅了多播组,环回数据包是否会通过接入点返回发件人?即使不是这种情况,是否可以通过AP强制环回?
另外,为什么loopback不能使用以下代码?
char * server_addr_name = "239.255.0.1"; // multicast group
int port_number = 8888;
int enable_loopback = 1;
int udp_socket;
struct message msg; // random message
char buffer[BUFFER_SIZE];
/* create socket */
struct sockaddr_in server_addr, rcv_addr;
socklen_t server_addr_size = sizeof(struct sockaddr_in);
socklen_t rcv_addr_size = sizeof(struct sockaddr_in);
/* initialize socket */
memset(& server_addr, 0, sizeof(struct sockaddr_in));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = inet_addr(server_addr_name);
server_addr.sin_port = htons(port_number);
if ((udp_socket = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP)) == -1) {
fprintf(stderr, "Error initializing UDP socket.\n");
exit(EXIT_FAILURE);
}
struct ip_mreq mreq;
mreq.imr_multiaddr.s_addr = inet_addr(server_addr_name);
mreq.imr_interface.s_addr = htonl(INADDR_ANY);
if (setsockopt(udp_socket, IPPROTO_IP, IP_ADD_MEMBERSHIP, & mreq, sizeof(mreq)) < 0) {
fprintf(stderr, "Error on setting multicast membership on socket.\n");
exit(EXIT_FAILURE);
}
unsigned char do_enable = (unsigned char) enable_loopback;
if (setsockopt(udp_socket, IPPROTO_IP, IP_MULTICAST_LOOP,
& do_enable, sizeof(do_enable)) < 0) {
fprintf(stderr, "Error on setting multicast loopback on socket.\n");
exit(EXIT_FAILURE);
}
int read_size = 0;
while (1) {
/* send the packet */
if (sendto(udp_socket, &msg, sizeof(struct message), 0,
(struct sockaddr *) & server_addr, server_addr_size) == -1) {
fprintf(stderr, "Error on sending UDP packet.\n");
}
else
printf("Sent my message.\n");
/* get response from the server/multicast address */
read_size = recvfrom(udp_socket, buffer, BUFFER_SIZE, 0,
(struct sockaddr *) & rcv_addr, & rcv_addr_size);
if (read_size < 1)
break;
else
printf("Got my packet!\n");
}
任何帮助表示感谢。
答案 0 :(得分:1)
setsockopt()的参数应该是int,而不是unsigned char。
答案 1 :(得分:1)
刚刚遇到同样的问题。即使Linux与地址:端口匹配,Linux也不会在同一个套接字上镜像该数据包。 您应该以与udp_socket和recvfrom相同的方式创建另一个套接字udp_socket_receiver。看起来IP_MULTICAST_LOOP意味着在除发送方套接字之外的所有侦听器上本地环回多播数据包。
答案 2 :(得分:0)
如果发件人订阅了多播组,那么环回数据包会通过接入点返回发件人吗?
我确实不确定,但乍一看答案是胖的否。如果网关将数据包弹回到它所来自的位置,那么你将会复制流量,这会破坏多播点。
但是,发件人仍会收到该邮件,因为订阅该组的本地网络中的任何人都将收到该邮件,并且发件人已订阅。
即使不是这样,是否有可能通过AP强制回送?
我不知道,但你可能不需要那样。见下文。
为什么环回使用以下代码?
因为您忘记了bind
套接字。你没有告诉套接字它应该在哪里监听,所以即使多播可能正常弹跳,UDP也会丢弃数据包,因为它连接到一个随机端口,这很可能不是8888。
在socket()
和setsockopt(IP_ADD_MEMBERSHIP)
之间添加以下块。适合我:
struct sockaddr_in src;
memset(&src, 0, sizeof(src));
src.sin_family = AF_INET;
src.sin_port = htons(port_number);
src.sin_addr.s_addr = INADDR_ANY;
if (bind(udp_socket, (struct sockaddr*)&src, sizeof(src))) {
perror("bind() failed");
close(udp_socket);
return EXIT_FAILURE;
}
当你完成套接字时,你也应该close()
插入套接字。