0

我有一个在我的网络上发送多播数据的硬件设备。我编写了一个 python 脚本来接收数据并打印它。但是,我发现它只适用于我的 Windows XP PC,不适用于我的 Ubuntu Linux 10.04 PC。在 Linux 下,什么都没有收到。它只是在while循环中循环,从来没有收到任何数据。我的代码发布在下面。你能看出为什么这在 Linux 下不起作用吗?谢谢,拉布。

# Multicast client
# Adapted from: http://chaos.weblogs.us/archives/164
# on 05/03/2013

import socket

ANY = "0.0.0.0" 
MCAST_ADDR = "224.0.33.154"
MCAST_PORT = 31800

# Create a UDP socket
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)

# Allow multiple sockets to use the same PORT number
sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)

# Bind to the port that we know will receive multicast data
sock.bind((ANY,MCAST_PORT))

# Tell the kernel that we are a multicast socket
sock.setsockopt(socket.IPPROTO_IP, socket.IP_MULTICAST_TTL, 255)

# Tell the kernel that we want to add ourselves to a multicast group
# The address for the multicast group is the third param
status = sock.setsockopt(socket.IPPROTO_IP,
socket.IP_ADD_MEMBERSHIP,
socket.inet_aton(MCAST_ADDR) + socket.inet_aton(ANY));

# setblocking(0) is equiv to settimeout(0.0) which means we poll the socket.
# But this will raise an error if recv() or send() can't immediately find or send data. 
sock.setblocking(0)

while 1:
    try:
        data, addr = sock.recvfrom(1024)
    except socket.error as e:
        pass
    else:
        print "From: ", addr
        print "Data: ", data

这是我的 Windows PC 的一些示例输出:

From:  ('0.0.0.0', 31801)
Data:  EDCP

请注意,远程硬件设备没有 IP 地址,而是使用地址 0.0.0.0

编辑:我现在发现这在我的 Windows 笔记本电脑上也不起作用。因此,它看起来并不特定于操作系统。此外,我尝试运行另一个脚本将多播数据发送到相同的多播地址和端口。我可以从一台 PC 发送,所有其他人都可以使用我的接收脚本正确接收。但是,只有我的一台 Windows PC 能够从相关硬件设备接收数据。我想知道这是否与以太网适配器或其配置有关。这可能与硬件设备的 IP 地址为 0.0.0.0 并且需要告知这些以太网适配器和/或我的接收器脚本以接收具有此地址的消息的事实有关吗?在 Linux PC 上运行 Wireshark 可以看到来自硬件设备的数据。

4

2 回答 2

0

尝试绑定到多播组地址:

sock.bind((MCAST_ADDR,MCAST_PORT))

此外,您不需要在接收者上设置多播 TTL,这是针对发送者的,也是可选的。

于 2013-03-07T19:46:51.583 回答
0

我为同样的问题争论了两天。Wireshark 看到了数据包,但我的代码没有。来自各种来源的所谓“确定”答案都没有对我有用。密钥来自https://serverfault.com/questions/163244/linux-kernel-not-passing-through-multicast-udp-packets

运行“ip maddr”显示与您的代码类似的代码没有将多播地址添加到任何接口。我强迫它用 smcroute 添加(见上面的链接)。仍然没有喜悦。数据包的源 IP 为 172.22...我的接口是 172.17...我向该 NIC 添加了 172.22 地址。答对了!现在我的代码收到了数据包。

现在如何在没有 smcroute 的情况下使程序工作?我注释掉了 setsockopt() 调用。仍然有效。使用 smcroute 取消链接多播地址 -- 失败。取消注释 setsockopt() 调用并用我的 172.22 地址替换“ANY”。成功!

概括:

  1. 确保您的 IP 与传入的数据包位于同一网段。
  2. 在 IP_ADD_MEMBERSHIP 调用中使用该地址而不是 INADDR_ANY。

如果您只有一个 NIC,您可能不需要执行 2)。我有三个,必须这样做。

以防万一,我使用的是 Ubuntu 12.04。我不需要像其他人描述的那样更改任何默认的 /etc/sysctl.conf 设置。我试过了。他们没有帮助,所以我将它们重置为安装默认值。

于 2013-03-08T16:58:00.287 回答