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

标签 c++ linux sockets udp multicast

首先,我知道这里有很多关于多播主题的问题。我还看到大多数人的问题是没有启用 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/

相关文章:

php - 将一个文件从一台 Web 服务器同步到另一台

linux - 汇编程序段失败

linux - Polymer serve 拒绝外部连接

c - WSAStringToAddress 错误 10022/从控制台参数读取 IPv6

c++ - 使用 qSort 对字符串进行排序

c++ - 有没有一种方法可以在不更改递归函数中的数组的情况下忽略数组中的元素?

c++ - 迭代二维网格子集的最佳方法

c++ - S.E.M.F.物理C++代码

java - 如何解决 SSL "bad record MAC"异常

c - EV3 上的蓝牙