我将 icmp echo 发送到 C 程序中的 250 个节点。套接字是这样创建的
sfd = socket(PF_INET, SOCK_RAW, IPPROTO_ICMP);
我有getsockopt SO_RCVBUF :262142 .ie 262KB 因为一个数据包需要84字节(包括IP,100字节在线),rcv缓冲区应该能够容纳262142/84 = 3120个数据包
sysctl parameters (defaults)
net.core.rmem_max = 131071
net.core.rmem_default = 113664
但是有10+滴。
我正在发送所有回显请求,然后使用 recvfrom() 获取数据包。很明显,replypacks 是在 socket 的 rcv 缓冲区中积累起来的,但是 rcv 缓冲区足够大,可以容纳 3120 个数据包。
数据包在接收主机上以 ehteral 正确显示。
当我设置:
sysctl -w net.core.rmem_max=1048576
and SO_RCVBUF to 2MB, the drops were 0.
为什么会这样?
很少有队列在行动。
- 网卡环形缓冲区。
- nic 到内核 que
- 每个套接字 rcv 缓冲区
我猜 net.core.rmem_max 只会改变每个套接字 rcv 缓冲区。
任何指向正确方向的链接。
Platform Linux suse10/x86
网卡:英特尔公司 PRO/无线 2200BG
-- 添加更多
我禁用了上述无线接口并开始使用有线接口以太网控制器:Broadcom Corporation NetXtreme BCM5705M_2 Gigabit Ethernet (rev 03) 情况发生了显着变化。
net.core.netdev_budget = 300
net.core.netdev_max_backlog = 1000
net.core.rmem_max = 131071
net.core.wmem_max = 131071
getsockopt SO_RCVBUF :262142
getsockopt SO_SNDBUF :262142
# ethtool -g eth1
Current hardware settings:
RX: 200
TX: 200
#
eth1 txqueuelen:1000
现在它就像每 250 个数据包 0 丢包和每 1000 个更改 rx,tx 大约 170 次,使用 ethtool -G 从默认值 200 但这没有效果。
然后我更改了每个套接字 rcv 缓冲区最大值
sysctl -w net.core.rmem_max=1024000
这允许每 6000 滴 0 滴和每 7000 滴 500 滴
进一步增加 per-socket-rcv-buffer-max
sysctl -w net.core.rmem_max=2048000
每 7000 个数据包 0 丢包
有线网络给出了更加一致和合作的结果。
但是当 262142 字节缓冲区可以容纳 3000 个大小为 84 的数据包(带有 IP 标头)时,问题仍然存在,为什么在 1000 个数据包时会发生丢包。即使在线数据包最大为 100 字节,262142 也可以容纳 2600 个数据包。
ethereal 大部分时间都能够获取数据包,并且 ifconfig 计数器显示没有丢弃,因此这是 nic 驱动程序将数据包提供给内核之后的事情。
ethereal 也错过了一些数据包,但这种情况较少发生。
通过仅更改上述这些
sysctl -w net.core.rmem_max=1024000
掉落 96/1000 掉落 0/6000
下降 500/7000
sysctl -w net.core.rmem_max=2048000
掉落 0/7000
sysctl -w net.core.rmem_max=512000
sysctl -w net.core.netdev_max_backlog=3000
掉落 0/3000
sysctl -w net.core.rmem_max=256000
sysctl -w net.core.netdev_max_backlog=3000
下降 1400/3000
hold = 2 * 1024 * 1024;
setsockopt(sfd, SOL_SOCKET, SO_RCVBUF, (char *)&hold, sizeof(hold));
for(;;)
...
if(recvfrom(sfd,(char *)packet, packlen, 0, (struct sockaddr *)&from, &fromlen)) < 0) {
rcvcount++;
process_packet(packet,packlen);
}