我有一个硬件设备,可以在我的网络上发送多播数据。我编写了一个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可以看到来自硬件设备的数据。答案 0 :(得分:0)
尝试绑定到多播组地址:
sock.bind((MCAST_ADDR,MCAST_PORT))
此外,您不需要在接收器上设置多播TTL,也可以在发送器上设置,也可以选择。
答案 1 :(得分:0)
我在同样的问题上打了两天。 Wireshark看到了数据包,但我的代码却没有。各种来源的所谓“确定性”答案都不适合我。密钥来自https://serverfault.com/questions/163244/linux-kernel-not-passing-through-multicast-udp-packets。
运行“ip maddr”显示类似于您的代码没有将多播地址添加到任何接口。我强迫它添加smcroute(参见上面的链接)。仍然没有快乐。数据包的源IP为172.22 ...我的接口是172.17 ...我在该网卡上添加了一个172.22地址。答对了!现在我的代码收到了数据包。
现在如何让程序在没有smcroute的情况下工作?我注释掉了setsockopt()调用。仍然有效。将多播地址与smcroute取消链接 - 失败。取消注释setsockopt()调用并用我的172.22地址替换“ANY”。成功了!
要点:
如果你只有一个网卡,你可能不需要做2)。我有三个,必须这样做。
如果它有密切关系,我使用的是Ubuntu 12.04。我不需要像其他人描述的那样更改任何默认的/etc/sysctl.conf设置。我试过了。他们没有帮助,所以我将它们重置为安装默认值。