我们现有的软件定期将UDP数据包广播到本地子网(x.x.x.255)上的特定端口(7125) . 我们在HP-UX(11.11)上运行的监控软件能够接收这些数据包没有问题 . 但是,在将监控软件移植到Linux(RHEL 6.1)后,我们发现它没有收到广播数据包 . tcpdump显示到达Linux主机的数据包,但内核不会将它们发送到我们的软件 .
我一直在使用一些python 2.x脚本来模仿监控软件用来测试不同场景的套接字API调用 . 如果发送方使用单播(10.1.0.5),而不是广播(10.1.0.255),Linux内核会将数据包传递给接收方软件 . 我已经在网上搜索了几天,但没有找到任何有同样问题的人 . 有任何想法吗?
receiver.py
from __future__ import print_function
import socket
localHost = ''
localPort = 7125
remoteHost = '10.1.0.5'
remotePort = 19100
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
s.bind((localHost, localPort))
s.connect((remoteHost, remotePort))
print('Listening on {0}:{1} for traffic from {2}:{3}'.format(localHost, localPort, remoteHost, remotePort))
data = s.recv(1024)
print('Received: {0}'.format(data))
s.close()
sender.py
from __future__ import print_function
import socket
import time
localHost = ''
localPort = 19100
remoteHost = '10.1.0.255'
remotePort = 7125
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
s.bind((localHost, localPort))
s.connect((remoteHost, remotePort))
data = 'sending this from {0}:{1} to {2}:{3}'.format(localHost, localPort, remoteHost, remotePort)
print(data)
print('2')
time.sleep(1)
print('1')
time.sleep(1)
s.send(data)
print('sent at {0}'.format(time.ctime()))
s.close()
1 回答
好吧,我在评论中提出了这个答案,并且在实践中证明是正确的 . 我想用我自己的代码进一步调查周围的细微差别,但这是规范的案例 - 更接近 .
除了在两侧设置
SO_BROADCAST
套接字选项(正如您已正确执行的那样)之外,还必须将接收器绑定到广播地址(例如,INADDR_BROADCAST
,其为255.255.255.255,并且基本上与INADDR_ANY
用于单播的角色相同) .显然,在原始海报的HP-UX配置中,绑定到单播地址(或
INADDR_ANY
,特别是)但设置了SO_BROADCAST
套接字选项的UDP套接字仍将接收发往本地广播地址以及单播的所有UDP数据报针对主机的流量 .在Linux下,情况并非如此 . 将UDP套接字(即使
SO_BROADCAST
-enabled)绑定到INADDR_ANY
不足以在绑定端口上接收单播和广播数据报 . 可以为广播流量使用单独的INADDR_BROADCAST
-boundSO_BROADCAST
套接字 .