gpt4 book ai didi

c++ - 尽管有 IP_MULTICAST_LOOP(Linux、C++、UDP),IP_ADD_MEMBERSHIP 结果为 "No device found"

转载 作者:太空狗 更新时间:2023-10-29 12:43:16 24 4
gpt4 key购买 nike

首先,我知道这里有很多关于多播主题的问题。我还看到大多数人的问题是没有启用 IP_MULTICAST_LOOP。我已经解决了这个问题,但仍然以“找不到设备”错误告终。

我正在编写一个程序,它可以与环回接口(interface)或连接的网络设备一起使用,具体取决于用户在硬件方面的操作(插入或拔出电缆)。我正在使用 UDP 进行数据传输。我在 Linux 上使用 C++ 编写代码。

当插入网络电缆并且说 eth0 启动并运行时,一切都很好(我的多播设置运行良好)。如果只有 lo 可用(并且启动并运行,根据 ifconfig),调用

setsockopt(this->socket(), IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipmReq, sizeof(ipmReq))

-1 失败,导致 errno 报告“找不到设备”。

这是我的 MulticastEndpoint 类中的相关代码片段:

MulticastEndpoint::MulticastEndpoint(std::string strMulticastGroup, unsigned short usPort) : m_strMulticastGroup(strMulticastGroup), m_usPort(usPort) {
this->setSocket(::socket(AF_INET, SOCK_DGRAM, 0));

memset(&m_saAddrGroup, 0, sizeof(m_saAddrGroup));
m_saAddrGroup.sin_family = AF_INET;
m_saAddrGroup.sin_addr.s_addr = inet_addr(m_strMulticastGroup.c_str());
m_saAddrGroup.sin_port = htons(m_usPort);

unsigned int unYes = 1;
bool bOK = false;

if(setsockopt(this->socket(), SOL_SOCKET, SO_REUSEADDR, &unYes, sizeof(unYes)) >= 0) {
if(setsockopt(this->socket(), IPPROTO_IP, IP_MULTICAST_LOOP, &unYes, sizeof(unYes)) >= 0) {
memset(&m_saAddrAny, 0, sizeof(m_saAddrAny));
m_saAddrAny.sin_family = AF_INET;
m_saAddrAny.sin_addr.s_addr = htonl(INADDR_ANY);
m_saAddrAny.sin_port = htons(m_usPort);

if(bind(this->socket(), (struct sockaddr*)&m_saAddrAny, sizeof(m_saAddrAny)) >= 0) {
struct ip_mreq ipmReq;
ipmReq.imr_multiaddr.s_addr = inet_addr(m_strMulticastGroup.c_str());
ipmReq.imr_interface.s_addr = htonl(INADDR_ANY);

if(setsockopt(this->socket(), IPPROTO_IP, IP_ADD_MEMBERSHIP, &ipmReq, sizeof(ipmReq)) >= 0) {
bOK = true;
}
}
}
}

if(!bOK) {
std::cerr << strerror(errno) << std::endl;
::close(this->socket());
this->setSocket(-1);
}

std::cout << this->socket() << std::endl;
}

该类使用多播地址 224.0.0.1 和端口 2077 进行实例化。 setSocket(int)socket() 方法是一个简单的 setter 和 getter 函数,将套接字存储为实例中的 int

有人知道这件事吗?从理论上讲,多播应该只在 lo 上工作,不是吗?另外,也许我用错了通用多播地址?就我对 linux 套接字的理解而言,INADDR_ANY 也应该涵盖环回接口(interface)。如果在这种情况下有误,请纠正我。

我非常感谢任何启发。

最佳答案

环回接口(interface)通常不支持多播。

如果在 eth0 启动并运行时运行 ifconfig -a,您会看到 eth0 有一个 标志>MULTICAST,而 lo 没有。

这是我的一台机器显示的内容:

eth0      Link encap:Ethernet  HWaddr 00:11:22:33:44:55
inet addr:10.112.161.84 Bcast:10.112.161.127 Mask:255.255.255.192
inet6 addr: fe80::211:22ff:fe33:4455/64 Scope:Link
UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1
RX packets:20001770 errors:0 dropped:0 overruns:0 frame:0
TX packets:10074436 errors:0 dropped:0 overruns:0 carrier:0
collisions:0 txqueuelen:1000
RX bytes:4589702477 (4.2 GiB) TX bytes:2896096295 (2.6 GiB)

lo Link encap:Local Loopback
inet addr:127.0.0.1 Mask:255.0.0.0
inet6 addr: ::1/128 Scope:Host
UP LOOPBACK RUNNING MTU:16436 Metric:1
RX packets:1212663 errors:0 dropped:0 overruns:0 frame:0
TX packets:1212663 errors:0 dropped:0 overruns:0 carrier:0
collisions:0 txqueuelen:0
RX bytes:60939041 (58.1 MiB) TX bytes:60939041 (58.1 MiB)

编辑:

为了让它工作,您需要手动打开环回多播:

sudo ifconfig lo multicast

一旦你这样做了,你应该能够加入环回的多播组,但是你需要为 ipmReq.imr_interface 指定 127.0.0.1 而不是 INADDR_ANY .

关于c++ - 尽管有 IP_MULTICAST_LOOP(Linux、C++、UDP),IP_ADD_MEMBERSHIP 结果为 "No device found",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34717299/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com