0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

分析协议层注册进内核以及被socket的过程

B4Pb_gh_6fde77c 来源:Linux内核之旅 作者:陈莉君 2021-08-04 16:13 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

1. 前言

本文首先从宏观上概述了数据包发送的流程,接着分析了协议层注册进内核以及被socket的过程,最后介绍了通过 socket 发送网络数据的过程。

2. 数据包发送宏观视角

从宏观上看,一个数据包从用户程序到达硬件网卡的整个过程如下:

使用系统调用(如 sendto,sendmsg 等)写数据

数据穿过socket 子系统,进入socket 协议族(protocol family)系统

协议族处理:数据穿过协议层,这一过程(在许多情况下)会将数据(data)转换成数据包(packet)

数据穿过路由层,这会涉及路由缓存和 ARP 缓存的更新;如果目的 MAC 不在 ARP 缓存表中,将触发一次 ARP 广播来查找 MAC 地址

穿过协议层,packet 到达设备无关层(device agnostic layer)

使用 XPS(如果启用)或散列函数选择发送队列

调用网卡驱动的发送函数

数据传送到网卡的 qdisc(queue discipline,排队规则)

qdisc 会直接发送数据(如果可以),或者将其放到队列,下次触发NET_TX 类型软中断(softirq)的时候再发送

数据从 qdisc 传送给驱动程序

驱动程序创建所需的DMA 映射,以便网卡从 RAM 读取数据

驱动向网卡发送信号,通知数据可以发送了

网卡从 RAM 中获取数据并发送

发送完成后,设备触发一个硬中断(IRQ),表示发送完成

硬中断处理函数被唤醒执行。对许多设备来说,这会触发 NET_RX 类型的软中断,然后 NAPI poll 循环开始收包

poll 函数会调用驱动程序的相应函数,解除 DMA 映射,释放数据

3. 协议层注册

协议层分析我们将关注 IP 和 UDP 层,其他协议层可参考这个过程。我们首先来看协议族是如何注册到内核,并被 socket 子系统使用的。

当用户程序像下面这样创建 UDP socket 时会发生什么?

sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)

简单来说,内核会去查找由 UDP 协议栈导出的一组函数(其中包括用于发送和接收网络数据的函数),并赋给 socket 的相应字段。准确理解这个过程需要查看 AF_INET 地址族的代码。

内核初始化的很早阶段就执行了 inet_init 函数,这个函数会注册 AF_INET 协议族 ,以及该协议族内的各协议栈(TCP,UDP,ICMP 和 RAW),并调用初始化函数使协议栈准备好处理网络数据。inet_init 定义在net/ipv4/af_inet.c 。

AF_INET 协议族导出一个包含 create 方法的 struct net_proto_family 类型实例。当从用户程序创建 socket 时,内核会调用此方法:

static const struct net_proto_family inet_family_ops = {

.family = PF_INET,

.create = inet_create,

.owner = THIS_MODULE,

};

inet_create 根据传递的 socket 参数,在已注册的协议中查找对应的协议:

/* Look for the requested type/protocol pair. */

lookup_protocol:

err = -ESOCKTNOSUPPORT;

rcu_read_lock();

list_for_each_entry_rcu(answer, &inetsw[sock-》type], list) {

err = 0;

/* Check the non-wild match. */

if (protocol == answer-》protocol) {

if (protocol != IPPROTO_IP)

break;

} else {

/* Check for the two wild cases. */

if (IPPROTO_IP == protocol) {

protocol = answer-》protocol;

break;

}

if (IPPROTO_IP == answer-》protocol)

break;

}

err = -EPROTONOSUPPORT;

}

然后,将该协议的回调方法(集合)赋给这个新创建的 socket:

sock-》ops = answer-》ops;

可以在 af_inet.c 中看到所有协议的初始化参数。下面是TCP 和 UDP的初始化参数:

/* Upon startup we insert all the elements in inetsw_array[] into

* the linked list inetsw.

*/

static struct inet_protosw inetsw_array[] =

{

{

.type = SOCK_STREAM,

.protocol = IPPROTO_TCP,

.prot = &tcp_prot,

.ops = &inet_stream_ops,

.no_check = 0,

.flags = INET_PROTOSW_PERMANENT |

INET_PROTOSW_ICSK,

},

{

.type = SOCK_DGRAM,

.protocol = IPPROTO_UDP,

.prot = &udp_prot,

.ops = &inet_dgram_ops,

.no_check = UDP_CSUM_DEFAULT,

.flags = INET_PROTOSW_PERMANENT,

},

/* 。。。。 more protocols 。。。 */

IPPROTO_UDP 协议类型有一个 ops 变量,包含很多信息,包括用于发送和接收数据的回调函数:

const struct proto_ops inet_dgram_ops = {

.family = PF_INET,

.owner = THIS_MODULE,

/* 。。。 */

.sendmsg = inet_sendmsg,

.recvmsg = inet_recvmsg,

/* 。。。 */

};

EXPORT_SYMBOL(inet_dgram_ops);

prot 字段指向一个协议相关的变量(的地址),对于 UDP 协议,其中包含了 UDP 相关的回调函数。UDP 协议对应的 prot 变量为 udp_prot,定义在 net/ipv4/udp.c:

struct proto udp_prot = {

.name = “UDP”,

.owner = THIS_MODULE,

/* 。。。 */

.sendmsg = udp_sendmsg,

.recvmsg = udp_recvmsg,

/* 。。。 */

};

EXPORT_SYMBOL(udp_prot);

现在,让我们转向发送 UDP 数据的用户程序,看看 udp_sendmsg 是如何在内核中被调用的。

4. 通过 socket 发送网络数据

用户程序想发送 UDP 网络数据,因此它使用 sendto 系统调用:

ret = sendto(socket, buffer, buflen, 0, &dest, sizeof(dest));

该系统调用穿过Linux 系统调用(system call)层,最后到达net/socket.c中的这个函数:

/*

* Send a datagram to a given address. We move the address into kernel

* space and check the user space data area is readable before invoking

* the protocol.

*/

SYSCALL_DEFINE6(sendto, int, fd, void __user *, buff, size_t, len,

unsigned int, flags, struct sockaddr __user *, addr,

int, addr_len)

{

/* 。。。 code 。。。 */

err = sock_sendmsg(sock, &msg, len);

/* 。。。 code 。。。 */

}

SYSCALL_DEFINE6 宏会展开成一堆宏,后者经过一波复杂操作创建出一个带 6 个参数的系统调用(因此叫 DEFINE6)。作为结果之一,会看到内核中的所有系统调用都带 sys_前缀。

sendto 代码会先将数据整理成底层可以处理的格式,然后调用 sock_sendmsg。特别地, 它将传递给 sendto 的地址放到另一个变量(msg)中:

iov.iov_base = buff;

iov.iov_len = len;

msg.msg_name = NULL;

msg.msg_iov = &iov;

msg.msg_iovlen = 1;

msg.msg_control = NULL;

msg.msg_controllen = 0;

msg.msg_namelen = 0;

if (addr) {

err = move_addr_to_kernel(addr, addr_len, &address);

if (err 《 0)

goto out_put;

msg.msg_name = (struct sockaddr *)&address;

msg.msg_namelen = addr_len;

}

这段代码将用户程序传入到内核的(存放待发送数据的)地址,作为 msg_name 字段嵌入到 struct msghdr 类型变量中。这和用户程序直接调用 sendmsg 而不是 sendto 发送数据差不多,这之所以可行,是因为 sendto 和 sendmsg 底层都会调用 sock_sendmsg。

4.1 sock_sendmsg, __sock_sendmsg, __sock_sendmsg_nosec

sock_sendmsg 做一些错误检查,然后调用__sock_sendmsg;后者做一些自己的错误检查 ,然后调用__sock_sendmsg_nosec。__sock_sendmsg_nosec 将数据传递到 socket 子系统的更深处:

static inline int __sock_sendmsg_nosec(struct kiocb *iocb, struct socket *sock,

struct msghdr *msg, size_t size)

{

struct sock_iocb *si = 。。。。

/* other code 。。。 */

return sock-》ops-》sendmsg(iocb, sock, msg, size);

}

通过前面介绍的 socket 创建过程,可以知道注册到这里的 sendmsg 方法就是 inet_sendmsg。

4.2 inet_sendmsg

从名字可以猜到,这是 AF_INET 协议族提供的通用函数。此函数首先调用 sock_rps_record_flow 来记录最后一个处理该(数据所属的)flow 的 CPU; Receive Packet Steering 会用到这个信息。接下来,调用 socket 的协议类型(本例是 UDP)对应的 sendmsg 方法:

int inet_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *msg,

size_t size)

{

struct sock *sk = sock-》sk;

sock_rps_record_flow(sk);

/* We may need to bind the socket. */

if (!inet_sk(sk)-》inet_num && !sk-》sk_prot-》no_autobind && inet_autobind(sk))

return -EAGAIN;

return sk-》sk_prot-》sendmsg(iocb, sk, msg, size);

}

EXPORT_SYMBOL(inet_sendmsg);

本例是 UDP 协议,因此上面的 sk-》sk_prot-》sendmsg 指向的是之前看到的(通过 udp_prot 导出的)udp_sendmsg 函数。

sendmsg()函数作为分界点,处理逻辑从 AF_INET 协议族通用处理转移到具体的 UDP 协议的处理。

5. 总结

了解Linux内核网络数据包发送的详细过程,有助于我们进行网络监控和调优。本文只分析了协议层的注册和通过 socket 发送数据的过程,数据在传输层和网络层的详细发送过程将在下一篇文章中分析。

参考链接:

[1] https://blog.packagecloud.io/eng/2017/02/06/monitoring-tuning-linux-networking-stack-sending-data

[2] https://segmentfault.com/a/1190000008926093

本系列文章1-4,来源于陈莉君老师公众号“Linux内核之旅”

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内核
    +关注

    关注

    4

    文章

    1479

    浏览量

    43140

原文标题:Linux内核网络udp数据包发送(一)

文章出处:【微信号:gh_6fde77c41971,微信公众号:FPGA干货】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    DPE5500/W5500—SPI 接口以太网协议栈芯片

    DPE5500是一款SPI接口的以太网协议栈芯片,适用于单片机系统实现以太网通讯。DPE5500集成了全硬件TCP/IP协议栈,10/100M以太网数据链路层(MAC)和物理(PHY
    发表于 05-21 17:46

    [VirtualLab] SiO2膜的可变角椭圆偏振光谱(VASE)分析

    , ? = 0.00422??², ? = 1.89? - 05??4 基板材料:晶体硅 入射角度。75° **椭圆偏振分析仪 ** 椭圆偏振分析仪用于计算相位差?,以及反射光束的振幅分量Ψ。 有关该
    发表于 04-09 08:10

    TCP/IP(Socket协议深度剖析

    TCP/IP协议作为互联网通信的基础架构,其核心机制Socket编程承载着全球数据交换的使命。本文将深入剖析这一协议的七架构、三次握手与四次挥手的精妙设计、流量控制与拥塞控制的动态平
    的头像 发表于 03-03 17:06 855次阅读

    迭时空 Upstream | K3 获得 Linux 7.0 内核原生支持

    2026年2月22日,随着Linux内核正式发布v7.0-rc1版本,全球开源社区迎来了RISC-V生态的历史性跨越。迭时空(SpacemiT)研发的高性能RISC-VAICPU芯片K3作为全球首
    的头像 发表于 02-27 18:10 1.3w次阅读
    <b class='flag-5'>进</b>迭时空 Upstream | K3 获得 Linux 7.0 <b class='flag-5'>内核</b>原生支持

    socket是什么

    Socket(套接字)是计算机网络中的一个概念,它指示了一个可以进行网络通信的软件端点。通过使用Socket,计算机程序可以通过网络进行通信。Socket API提供了一套标准的网络通信接口,可用
    发表于 12-03 08:27

    Linux内核模块的加载机制

    内核模块是什么? 内核模块是动态加载到内核中的代码,可以在不重启系统的情况下扩展功能,比如设备驱动或者文件系统支持。这样用户不需要把所有功能都编译
    发表于 11-25 06:59

    蜂鸟内核ITCM模块以及取值过程

    的结构以及从ITCM中取值的过程。 模块介绍 首先,我们先得知道ITCM模块存储位置是在e203_CPU_top下。 而我们看ITCM的代码下只有一个子模块 该子模块是sram的通用模块,也就是说
    发表于 10-24 08:29

    FTTR-B主网关二透传配置过程

    透传(Layer 2 Transparent Transmission)指在数据链路层(OSI第二)上,数据帧在传输过程中保持原始的二信息(如MAC地址、VLAN标签等)不变,
    的头像 发表于 08-20 10:23 1850次阅读
    FTTR-B主网关二<b class='flag-5'>层</b>透传配置<b class='flag-5'>过程</b>

    GraniStudio : TCP/IP(Socket协议深度剖析

    在工业自动化与物联网领域,TCP/IP(Socket协议作为应用最广泛的网络通信标准,是实现设备间数据交互的核心技术。GraniStudio 软件作为工业级零代码开发平台,其内置的 TCP/IP
    的头像 发表于 08-03 22:20 1444次阅读
    GraniStudio : TCP/IP(<b class='flag-5'>Socket</b>)<b class='flag-5'>协议</b>深度剖析

    如何排除 USB 协议分析仪测试中的干扰源?

    5GHz),或增大主机PC的USB缓冲区(通过注册表修改)。 总结排除USB协议分析仪测试中的干扰需遵循“从物理到逻辑、从局部到全局”的原则: 优先检查物理(电缆、电源、EMI),确
    发表于 08-01 15:00

    第二十四章 W55MH32TCP_Client_Multi_Socket示例

    及处理连接关闭的完整过程。文章详细介绍了 TCP 协议的概念、特点、与 UDP 的区别、应用场景、数据交互流程、ACK 机制、重传机制和 Keepalive 机制,帮助读者理解其在可靠数据传输中的实际应用价值。
    的头像 发表于 07-24 16:08 1304次阅读
    第二十四章 W55MH32TCP_Client_Multi_<b class='flag-5'>Socket</b>示例

    协议分析仪能检测蓝牙设备的哪些潜在问题?

    协议分析仪能够检测蓝牙设备从物理到应用的全链路潜在问题,具体涵盖以下方面:一、物理(PHY Layer)问题 信号衰减与遮挡 RSS
    发表于 07-21 14:27

    协议分析仪需要支持哪些常见协议?

    设备:Keysight U4301B(支持Thunderbolt 3/4物理分析)。 调试重点:PCIe隧道协议、DisplayPort Alt Mode、热插拔时序。 HDMI
    发表于 07-17 15:40

    蓝牙协议分析仪能检测哪些问题?

    仪显示应用未处理特定按键的HID报告(Report ID=0x05未注册)。 车载蓝牙系统崩溃,捕获到应用发送非法指令导致协议栈溢出。 2. 性能瓶颈 检测内容: 吞吐量
    发表于 07-15 15:52

    VirtualLab Fusion应用:氧化硅膜的可变角椭圆偏振光谱(VASE)分析

    极化分量)的比率?,并输出相位差?,以及振幅分量Ψ,根据 在VirtualLab Fusion中,复数系数?p和?s是通过应用严格耦合波分析(RCWA),也被称为傅里叶模态法(FMM)来计算。因此,在
    发表于 06-05 08:46