当前位置: 首页 > news >正文

网卡中的Ring buffer -- 解决 rx_resource_errors 丢包

1、软硬件环境

        硬件: 飞腾E2000Q 平台

        软件: linux 4.19.246

2、问题现象

网卡在高速收包的过程中,出现 rx error , 细查是 rx_resource_errors  如下:

root@E2000-Ubuntu:~# ifconfig eth1
eth1: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500inet 10.100.1.2  netmask 255.255.255.0  broadcast 10.100.1.255inet6 fe80::5ed2:bff:fe13:817d  prefixlen 64  scopeid 0x20<link>ether 5c:d2:0b:13:81:7d  txqueuelen 1000  (Ethernet)RX packets 28043321  bytes 41384388153 (41.3 GB)RX errors 17434  dropped 0  overruns 1305  frame 16129TX packets 26633002  bytes 39782515051 (39.7 GB)TX errors 0  dropped 0 overruns 0  carrier 0  collisions 0device interrupt 93  root@E2000-Ubuntu:~# 
root@E2000-Ubuntu:~# ethtool -S eth1 | grep errortx_carrier_sense_errors: 0rx_frame_check_sequence_errors: 0rx_length_field_frame_errors: 0rx_symbol_errors: 0rx_alignment_errors: 0rx_resource_errors: 16129rx_ip_header_checksum_errors: 0rx_tcp_checksum_errors: 0rx_udp_checksum_errors: 0

问题复现过程如下:

Server端 (问题设备):
ifconfig eth3 192.168.1.11 netmask 255.255.255.0
iperf3 -s  -B 192.168.1.11 -p 10002Client 端 (正常设备):
ifconfig eth3 192.168.1.10 netmask 255.255.255.0
iperf3 -c 192.168.1.11 -p 10002 -t500 -u -b 0 

3、问题分析

rx error 有很多种类,具体ethtool 就列出了这几类,有些是硬件原因,有些是软件可调整的。

rx_frame_check_sequence_errors
rx_length_field_frame_errors
rx_symbol_errors
rx_alignment_errors
rx_resource_errors
rx_ip_header_checksum_errors
rx_tcp_checksum_errors
rx_udp_checksum_errors

MAC在收发包的同时,如果出现有CRC的错包,或者来不及缓存被溢出包,都会被统计到相应的寄存器中,这些数值一般都可以在MAC 的寄存器中读出的,以E2000Q为例

drivers/net/ethernet/phytium/macb.h

 /* GEM register offsets. */
.......
#define GEM_RXUNDRCNT           0x0184 /* Undersize Frames Received Counter */
#define GEM_RXOVRCNT            0x0188 /* Oversize Frames Received Counter */
#define GEM_RXJABCNT            0x018c /* Jabbers Received Counter */
#define GEM_RXFCSCNT            0x0190 /* Frame Check Sequence Error Counter */
#define GEM_RXLENGTHCNT         0x0194 /* Length Field Error Counter */
#define GEM_RXSYMBCNT           0x0198 /* Symbol Error Counter */
#define GEM_RXALIGNCNT          0x019c /* Alignment Error Counter */
#define GEM_RXRESERRCNT         0x01a0 /* Receive Resource Error Counter */
#define GEM_RXORCNT             0x01a4 /* Receive Overrun Counter */
#define GEM_RXIPCCNT            0x01a8 /* IP header Checksum Error Counter */
#define GEM_RXTCPCCNT           0x01ac /* TCP Checksum Error Counter */
#define GEM_RXUDPCCNT           0x01b0 /* UDP Checksum Error Counter */.......

我们通过 ethtools -S eth1 查到我们具体错误的类型 Receive Resource Error,查了一下该寄存器的说明如下:

GEM: Receive Resource Error Counter

the register counting the number of frames that were successfully received by the MAC (correct address matched frame and adequate slot time) but could not be copied to memory because no receive buffer was available. This occurs when the GEM reads a buffer descriptor with its ownership (or used) bit set.

refer: rx_resource_errorshttps://docs.xilinx.com/r/en-US/ug1087-zynq-ultrascale-registers/rx_resource_errors-GEM-Register

看来是收包的时候 receive buffer不足造成的。那该如何调整接收buffer 呢?

看了 iperf3 的有个参数选项 可以调整

-l, --length    #[KMG]    length of buffer to read or write(default 128 KB for TCP, dynamic or 1460 for UDP)

加上参数 iperf3 -c 192.168.1.11 -p 10002 -t500 -u -b 0 -l 65500 同样存在问题, 看来不是应用层receive buffer的问题。

因为网卡在收包的时候,会涉及到多个buffer, 驱动层,应用层的,我们先来研究一下。

Receive ring buffers are shared between the device driver and network interface controller (NIC). The card assigns a transmit (TX) and receive (RX) ring buffer. As the name implies, the ring buffer is a circular buffer where an overflow overwrites existing data. There are two ways to move data from the NIC to the kernel, hardware interrupts and software interrupts, also called SoftIRQs.

The kernel uses the RX ring buffer to store incoming packets until they can be processed by the device driver. The device driver drains the RX ring, typically using SoftIRQs, which puts the incoming packets into a kernel data structure called an sk_buff or skb to begin its journey through the kernel and up to the application which owns the relevant socket.

The kernel uses the TX ring buffer to hold outgoing packets which are destined for the wire. These ring buffers reside at the bottom of the stack and are a crucial point at which packet drop can occur, which in turn will adversely affect network performance.

Increase the size of an Ethernet device’s ring buffers if the packet drop rate causes applications to report a loss of data, timeouts, or other issues.

refer: Chapter 32. Increasing the ring buffers to reduce a high packet drop rate Red Hat Enterprise Linux 9 | Red Hat Customer Portal

上述文章描述的意思大概就是内核会创建两个环形的缓冲区,RX/TX ring buffer ,  RX ring buffer的存在 就是当硬件中断来的时候,内核会先将数据放到一个叫 RX ring buffer的环形缓冲区,然后触发一个软中断,等待网卡驱动去消费 RX ring buffer的数据,因为是环形缓冲区,如果缓冲区太小,而收包的速度很快,就很容易溢出,导致丢包。

 那问题可能就会在这里了。

4 完美解决

那应该如何设置  RX/TX ring buffer的大小呢?其实有两种方法,其中一种是通过ethtool ,一种是通过setsockopt(PACKET_RX_RING/PACKET_TX_RING)设置环形buffer参数。这里选择用ethtool , 首先我们看下我们的网卡支持的最大缓冲区是多少

root@E2000-Ubuntu:~# ethtool -g eth1
Ring parameters for eth1:
Pre-set maximums:
RX:		8192
RX Mini:	0
RX Jumbo:	0
TX:		4096
Current hardware settings:
RX:		512
RX Mini:	0
RX Jumbo:	0
TX:		512

Pre-set maximums 中的 RX/TX 值为该网卡的 Buffer size 最大值;
Current hardware settings 中 RX/TX 值代表该网卡当前的 Buffer size 大小。
所以,设置的 Current hardware settings 的 RX/TX 值必须在 Pre-set maximums 的限制之内

ethtool -G eth1 rx 4096 tx 512

设置之后,重新测试,问题完美解决!

 5 拓展

注意:我们之前调整的 rmem_max 与 wmem_max  也是接收缓存区大小,当然这个缓冲区与Ring buffer 无关, rmem_max 与 wmem_max 只针对 TCP , 我们一般的查看或者调整方式如下:


root@E2000-Ubuntu:~# sysctl -a | grep rmem
net.core.rmem_default = 212992
net.core.rmem_max = 212992
net.ipv4.tcp_rmem = 4096	131072	6291456
net.ipv4.udp_rmem_min = 4096root@E2000-Ubuntu:~# sysctl -a | grep wmem
net.core.wmem_default = 212992
net.core.wmem_max = 212992
net.ipv4.tcp_wmem = 4096	16384	4194304
net.ipv4.udp_wmem_min = 4096root@E2000-Ubuntu:~# cat /proc/sys/net/core/wmem_max
212992
root@E2000-Ubuntu:~# cat /proc/sys/net/core/rmem_max
212992
root@E2000-Ubuntu:~# cat /proc/sys/net/core/rmem_default
212992setsockopt( sock, SOL_SOCKET, SO_RCVBUF,.....)
setsockopt( sock, SOL_SOCKET, SO_SNDBUF,.....)

上面主要针对 TCP 的接收和发送缓冲区,在收包时,都发生在网卡驱动从rx ring buffer 
拿到数据之后,在发包时,发生在 tx ring buffer之前。

那具体 SO_RCVBUF 与 tcp_rmem 有什么关系呢?


tcp连接建立时,SO_RCVBUF初始化为tcp_rmem。随着tcp握手及通信,SO_RCVBUF是会动态调整的,调整的范围不受rmem_max限制,只受tcp_rmem的限制。但是如果手动通过setsockopt设置接收缓冲区大小,则自动调整接收缓冲区大小的机制失效,而且setsockopt是否成功会受到rmem_max的限制。


参考链接:https://www.jianshu.com/p/c93727fa8c2e
 

http://www.lryc.cn/news/92633.html

相关文章:

  • 六月九号补题日记:Codeforces Round 877 (Div. 2)
  • python基础选择题,高中适用
  • Linux 面试题-(腾讯,百度,美团,滴滴)
  • DDD--战略设计步骤
  • Web Scoket简述
  • “Docker 技术在企业中的应用及挑战解决方案“
  • vue中开发包、生产包、全局包的区别以及安装语法
  • list的模拟实现
  • ChatGLM简介和SSE聊天接口测试效果
  • darknet yolo标注、训练详细说明
  • chatgpt赋能python:Python如何产生随机整数?
  • 大话Stable-Diffusion-Webui-客制化主题(四)
  • Excel函数VLOOKUP常用方法
  • systemV的工作原理+原理代码
  • Kubeflow--TFJob实现机制学习
  • 百度出品,Nature重磅 -- 优化的mRNA设计算法可改善mRNA的稳定性和免疫原性
  • CKA 01_docker部署Kubernetes 部署docker 使用kubeadm引导集群 安装Pod网络
  • Redis的使用规范小建议
  • 操作受限的线性表——栈
  • C++基类指针或引用指向或引用派生类对象(实现动态多态四种手段)父类指针访问子类成员变量(需要dynamic_cast)
  • WTM框架运行报错0308010C:digital envelope routines::unsupported
  • (二)CSharp-索引器
  • 配合AI刷leetcode 实现1170
  • English Learning - L3 作业打卡 Lesson5 Day36 2023.6.9 周五
  • 前端框架笔记
  • 详细设计文档
  • Java011——Java数据类型转换(基本数据类型)
  • mybatis-plus用法(二)
  • SQL笔记-存储过程+循环
  • HNU-操作系统OS-作业1(4-9章)