UDP内核发包流程

yizhihongxing

背景

工作中遇到客户反馈,上层应用UDP固定间隔100ms发包,但本地tcpdump抓包存在波动,有的数据包之间间隔107ms甚至更多,以此重新梳理了下udp的发送流程。

udp发包流程

udp发包流程

udp_sendmsg

UDP corking 是一项优化技术,允许内核将多次数据累积成单个数据报发送。在用户程序中有两种方法可以启用此选项:

使用 setsockopt 系统调用设置 socket 的 UDP_CORK 选项
程序调用 send,sendto 或 sendmsg 时,带 MSG_MORE 参数

如果没设置UDP_CORK,直接发送到ip层,根据客户只是偶尔出现时延过高的情况,可以确定UDP_CORK并没有被设置,udp这里应该是直接下发的,udp_send_skb之后直接到ip层,排除udp_sendmsg导致时延波动问题

int udp_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)
{
... ...
/* Lockless fast path for the non-corking case. */
	if (!corkreq) {
		struct inet_cork cork;

		skb = ip_make_skb(sk, fl4, getfrag, msg, ulen,
				  sizeof(struct udphdr), &ipc, &rt,
				  &cork, msg->msg_flags);
		err = PTR_ERR(skb);
		if (!IS_ERR_OR_NULL(skb))
			err = udp_send_skb(skb, fl4, &cork);
		goto out;
	}
... ...
}

qdisc发包流程

qdisc
当配额quota < 0 || need_resched时将触发软中断,否则将直接进行发包。
quota 对应 net.core.dev_weight,可通过sysctl进行更改。

网卡及驱动

如果网卡慢,会导致网卡队列满返回BUSY,数据包重新入队,tcpdump将抓到重复的数据包。客户并未反馈该现象,排除网卡及网卡驱动。

总结

未配置UDP_CORK的情况下,上层udp应用send调用包含两个路径,一个是send直接到网卡驱动进行发送,另一个是send到网络设备子系统__dev_queue_skb,然后由软中断调用继续发送。
决定直接发还是由软中断发的条件是,net.core.dev_weight和 need_resched(需要强制调度),可以通过sysctl -a | grep weight查看其默认值。

对于__qdisc_run来讲,dev_weight代表循环次数,可尝试适当增大这个值,但可能会导致上层应用发送时延增加。
sysctl net.core.dev_weight=4096

need_resched则与中断\异常相关。

时延大的问题可能是软中断调度问题。

原文链接:https://www.cnblogs.com/forwards/p/17381458.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:UDP内核发包流程 - Python技术站

(0)
上一篇 2023年5月7日
下一篇 2023年5月8日

相关文章

  • [转]linux shell 多线程实现

    shell脚本的执行效率虽高,但当任务量巨大时仍然需要较长的时间,尤其是需要执行一大批的命令时。因为默认情况下,shell脚本中的命令是串行执行的。如果这些命令相互之间是独立的,则可以使用“并发”的方式执行这些命令,这样可以更好地利用系统资源,提升运行效率,缩短脚本执行的时间。如果命令相互之间存在交互,则情况就复杂了,那么不建议使用shell脚本来完成多线程…

    Linux 2023年4月12日
    00
  • 安装redhat 8.0红帽系统的图文教程(小白必备)

    以下是“安装redhat 8.0红帽系统的图文教程(小白必备)”的完整攻略: 安装redhat 8.0红帽系统的图文教程(小白必备) 1. 下载redhat 8.0镜像文件 首先,我们需要从官网下载redhat 8.0的镜像文件。在网站的下载页面,我们选择合适的版本并下载对应的ISO文件。 2. 制作启动盘 接着,我们需要将ISO文件制作成可引导的启动盘。我…

    Linux 2023年5月24日
    00
  • 如何在Linux上安装libvirt虚拟化工具

    这里是在Linux上安装libvirt虚拟化工具的完整攻略。 1. 安装libvirt 在命令行终端中执行以下命令以安装libvirt: sudo apt-get install libvirt-daemon-system libvirt-clients 2. 安装虚拟化工具 你可以选择KVM、VirtualBox、QEMU等虚拟化工具。在这里,我们以KVM…

    Linux 2023年5月24日
    00
  • Centos7下yum安装Ceph分布式存储教程

    CentOS7下yum安装Ceph分布式存储教程 简介 Ceph是一种被广泛使用的分布式存储系统,采用了对象存储、块存储、文件系统等多种存储方式。本教程将介绍如何使用CentOS7下yum安装Ceph分布式存储系统。 步骤 安装EPEL仓库 首先,我们需要安装EPEL仓库,以便使用yum命令来安装Ceph: $ sudo yum install epel-r…

    Linux 2023年5月14日
    00
  • Linux系统MySQL8.0.19快速安装配置教程图解

    Linux系统MySQL8.0.19快速安装配置教程图解 简介 MySQL是一款开源的关系型数据库管理系统,广泛应用于Web应用程序的开发和管理。本教程将介绍在Linux系统下安装MySQL 8.0.19的详细步骤,并且给出图解说明。 步骤 安装MySQL依赖 MySQL依赖于一些其他的程序,因此我们需要先安装这些程序,运行以下命令: sudo apt up…

    Linux 2023年5月14日
    00
  • Linux文件的复制、删除和移动命令使用说明

    下面是关于Linux文件的复制、删除和移动命令使用说明的完整攻略: 复制命令cp cp命令用于将一个或多个文件或目录复制到指定位置,同时也可以给目标文件或目录重命名。 命令格式 cp [选项] 源文件 目标文件 常用选项 -r:递归复制目录及其中的所有内容; -p:保留文件的权限、所有者、时间戳和链接信息; -f:当目标文件已经存在时,强制覆盖目标文件; -…

    Linux 2023年5月14日
    00
  • linux下把 python 程序运行的输出结果记录到 log 文件中

    Linux中将 python 程序运行结果记录到文件中的方法 https://www.cnblogs.com/shineriver/p/10922970.html 仅转向不显示(1)ls > test.txt 把输出转向到指定的文件,如文件已存在的话也会重新写入,文件原内容不会保留(2)ls >> test.txt 是把输出附向到文件的后面…

    Linux 2023年4月16日
    00
  • Linux系统上创建.NET6项目(通过命令行(CTL)方式)

    前言 平时大家创建项目基本上都是借助开发工具创建,比如visual studio,visual studio code,今天我们在Linux系统上,通过命令行的形式创建.NET6项目。 版本介绍 系统版本:Ubuntu22.04 SDK版本:.NET6 安装.NET环境 # 升级系统 sudo apt-get update # 找到.NET6 sudo ap…

    Linux 2023年5月11日
    00
合作推广
合作推广
分享本页
返回顶部