如何在Linux上的C / C ++中使用ipv6 udp套接字进行多播?

(英语不是我的母语,不用担心某些句子是否很奇怪;)。

我正在开发 并且通过创建一些类来帮助我管理窗口,事件…和 因为我在游戏中添加了

,但是当前您必须输入您想与之联系的人的地址。一起玩。解决此问题的方法是

。对于ipv4来说这很容易,只需使用地址255.255.255.255,但是我们在2017年就提供了仅适用于ipv4的功能。

然后,我寻找一种使用ipv6进行广播的方法,并且了解了多播,但是这部分让我迷失了方向。=(

,发现了几个不适用于我的多播示例。我目前所做的最好的工作是将udp数据包从该程序的一个实例发送到同一台计算机上的另一个实例。

在Internet上找到的最佳代码(我对其进行了重新排列)几乎可以正常工作(客户端和服务全部合而为一,可以通过在argv上加上1或0来选择):

int main(int argc, char const *argv[]) {

struct sockaddr_in6 groupSock;

int sd = -1;

char databuf[10];

int datalen = sizeof databuf;

/* Create a datagram socket on which to send/receive. */

if((sd = socket(AF_INET6, SOCK_DGRAM, 0)) < 0) {

perror("Opening datagram socket error");

return 1;

} else {

cout << "Opening the datagram socket...OK." << endl;;

}

/* Enable SO_REUSEADDR to allow multiple instances of this */

/* application to receive copies of the multicast datagrams. */

int reuse = 1;

if(setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, (char *)&reuse, sizeof reuse) < 0) {

perror("Setting SO_REUSEADDR error");

close(sd);

return 1;

} else {

cout << "Setting SO_REUSEADDR...OK." << endl;

}

/* Initialize the group sockaddr structure with a */

memset((char *) &groupSock, 0, sizeof groupSock);

groupSock.sin6_family = AF_INET6;

// address of the group

inet_pton(AF_INET6, "ff0e::/16", &groupSock.sin6_addr);

groupSock.sin6_port = htons(4321);

/* Set local interface for outbound multicast datagrams. */

/* The IP address specified must be associated with a local, */

/* multicast capable interface. */

int ifindex = if_nametoindex ("enp3s0");

cout << "ifindex is " << ifindex << endl;

if(setsockopt(sd, IPPROTO_IPV6, IPV6_MULTICAST_IF, &ifindex, sizeof ifindex)) {

perror("Setting local interface error");

return 1;

} else {

cout << "Setting the local interface...OK" << endl;

}

// choice is 0 for sending and 1 for receiving

int choice;

if (argc < 2) {

cout << "missing argv[1]" << endl;

return 1;

}

sscanf (argv[1], "%d", &choice);

// if sending

if (choice == 0) {

memset(databuf, 'a', datalen);

databuf[sizeof databuf - 1] = '\0';

if (sendto(sd, databuf, datalen, 0, (sockaddr*)&groupSock, sizeof groupSock) < 0) {

cout << "Error in send" << endl;

} else {

cout << "Send okay!" << endl;

}

}

// if receiving

else if (choice == 1) {

groupSock.sin6_addr = in6addr_any;

if(bind(sd, (sockaddr*)&groupSock, sizeof groupSock)) {

perror("Binding datagram socket error");

close(sd);

return 1;

} else {

cout << "Binding datagram socket...OK." << endl;

}

/* Join the multicast group ff0e::/16 on the local */

/* interface. Note that this IP_ADD_MEMBERSHIP option must be */

/* called for each local interface over which the multicast */

/* datagrams are to be received. */

struct ipv6_mreq group;

inet_pton (AF_INET6, "ff0e::", &group.ipv6mr_multiaddr.s6_addr);

group.ipv6mr_interface = ifindex;

if(setsockopt(sd, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, (char *)&group, sizeof group) < 0) {

perror("Adding multicast group error");

close(sd);

return 1;

} else {

cout << "Adding multicast group...OK." << endl;

}

if (read(sd, databuf, datalen) < 0) {

perror("Error in read");

} else {

databuf[sizeof databuf - 1] = '\0';// just for safety

cout << "Read Okay" << endl;

cout << "Message is : " << databuf << endl;

}

}

return 0;

}

这里的地址是ff0e ::,但是我尝试使用ff01 ::和ff02 ::。

我需要帮助,但还没有找到任何简单的文档。预先感谢您的任何答复。

编辑:感谢Ron Maupin和Jeremy Friesner的这些评论,它对我有帮助。

编辑:谢谢杰里米!您的建议是使用ff12 :: blah:blah(…)代替ff0e ::可以!我应该为问题写答案以关闭线程吗?

回答:

下面的代码是正确的:

唯一错误的是用于多播的地址。

就像Je​​remy所说的那样, 。

使用if_nameindex()可以获得可用接口的名称和索引。

更新:我试图删除一些代码,看看没有它是否可以工作,我设法做到了:

服务器:

// OPEN

int fd = socket(AF_INET6, SOCK_DGRAM, 0);

// BIND

struct sockaddr_in6 address = {AF_INET6, htons(4321)};

bind(fd, (struct sockaddr*)&address, sizeof address);

// JOIN MEMBERSHIP

struct ipv6_mreq group;

group.ipv6mr_interface = 0;

inet_pton(AF_INET6, "ff12::1234", &group.ipv6mr_multiaddr);

setsockopt(fd, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, &group, sizeof group);

// READ

char buffer[128];

read(fd, buffer, sizeof buffer);

客户:

// OPEN

int fd = socket(AF_INET6, SOCK_DGRAM, 0);

// ADDRESS

struct sockaddr_in6 address = {AF_INET6, htons(4321)};

inet_pton(AF_INET6, "ff12::1234", &address.sin6_addr);

// SEND TO

char buffer[128];

strcpy(buffer, "hello world!");

sendto(fd, buffer, sizeof buffer, 0, (struct sockaddr*)&address, sizeof address);

以上是 如何在Linux上的C / C ++中使用ipv6 udp套接字进行多播? 的全部内容, 来源链接: utcz.com/qa/410979.html

回到顶部