网卡接收数据的关键过程

2024-04-17 04:52

本文主要是介绍网卡接收数据的关键过程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

网卡接收数据的关键过程

    • 网卡中断处理
    • 网络软中断处理
    • 协议栈处理
    • 传输层处理

Linux内核tracers的实现原理与应用
前年ftrace for io /去年ftrace for mm/今年ftrace for network.今年ftrace也被深度定制加强。

在这篇文章中,我们将深入探讨网卡接收数据的完整过程,了解数据是如何从网卡到达应用程序的。我们将使用Linux内核源代码来分析这一过程:

网卡中断处理

当网卡接收到数据时,会触发一个中断,内核将调用相应的中断处理函数。对于virtio网卡,中断处理函数为vring_interrupt():

vring_interrupt()skb_recv_done()napi_schedule_prep() virtqueue_disable_cb() // disable virtnet interrupt__napi_schedule(); // start network softirq

napi_schedule_prep()函数中,会先禁用virtnet中断,然后通过__napi_schedule()函数启动网络软中断。

bool napi_schedule_prep(struct napi_struct *n)
{unsigned long new, val = READ_ONCE(n->state);do {if (unlikely(val & NAPIF_STATE_DISABLE))return false;new = val | NAPIF_STATE_SCHED;new |= (val & NAPIF_STATE_SCHED) / NAPIF_STATE_SCHED * // 如果state还在sched状态,也设置missNAPIF_STATE_MISSED;} while (!try_cmpxchg(&n->state, &val, new)); // state至少也设置为sched状态return !(val & NAPIF_STATE_SCHED);        // 驱动会判断如果state不是sched状态,则开启napi模式
}

网络软中断处理

网络软中断处理函数为net_rx_action(),它会调用__napi_poll()函数来轮询virtnet设备:

__do_softirq()net_rx_action()__napi_poll()virtnet_poll() // virtnet poll__napi_alloc_skb() // allocate skbnapi_gro_receive() // GRO 

virtnet_poll()函数中,会分配skb缓冲区,并通过napi_gro_receive()函数进行GRO(Generic Receive Offload)处理。

当轮询完成后,会调用virtqueue_napi_complete()函数完成napi过程:

virtnet_poll() // virtnet poll prepare to completevirtqueue_napi_complete()virtqueue_enable_cb_prepare() // 开启virtnet interrupt

协议栈处理

napi_complete_done()函数中,会调用gro_normal_list()函数将接收到的数据包交给协议栈处理:

napi_complete_done()gro_normal_list()netif_receive_skb_list_internal()ip_rcv()

对于IP数据包,会调用ip_rcv()函数进行处理:

static struct packet_type ip_packet_type __read_mostly = {.type = cpu_to_be16(ETH_P_IP),.func = ip_rcv,.list_func = ip_list_rcv,
};
#define ETH_P_IP        0x0800          /* Internet Protocol packet */

ip_rcv()函数中,会根据路由表查找结果,确定是将数据包交给本地接收还是进行转发:

ip_rcv()fib_table_lookup()struct rtable *rth = rt_dst_alloc()rth->dst.input = ip_local_deliver/ip_forwardrth->dst->dev = nhc->nhc_dev; // 从路由表确定接收或者转发网络设备dst->ops = ops;

传输层处理

对于本地接收的数据包,会根据传输层协议(TCP或UDP)进行相应的处理:

static const struct net_protocol tcp_protocol = {.handler        =       tcp_v4_rcv,.err_handler    =       tcp_v4_err,.no_policy      =       1,   .icmp_strict_tag_validation = 1, 
};static const struct net_protocol udp_protocol = {.handler =      udp_rcv,.err_handler =  udp_err,.no_policy =    1,   
};ip_local_deliver() // ip_forwardip_local_deliver_finiship_hdr(skb)->protocol   // <<< tcp/udptcp_v4_rcv() / udp_rcv() // by tcp/udp protocol

至此,数据包就从网卡接收,经过协议栈处理,最终到达应用程序。(以上函数流程都来自ftrace工具)

Linux内核tracers的实现原理与应用

–JeffXie

这篇关于网卡接收数据的关键过程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/910826

相关文章

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

redis群集简单部署过程

《redis群集简单部署过程》文章介绍了Redis,一个高性能的键值存储系统,其支持多种数据结构和命令,它还讨论了Redis的服务器端架构、数据存储和获取、协议和命令、高可用性方案、缓存机制以及监控和... 目录Redis介绍1. 基本概念2. 服务器端3. 存储和获取数据4. 协议和命令5. 高可用性6.

PLsql Oracle 下载安装图文过程详解

《PLsqlOracle下载安装图文过程详解》PL/SQLDeveloper是一款用于开发Oracle数据库的集成开发环境,可以通过官网下载安装配置,并通过配置tnsnames.ora文件及环境变... 目录一、PL/SQL Developer 简介二、PL/SQL Developer 安装及配置详解1.下

在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程

《在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程》本文介绍了在Java中使用ModelMapper库简化Shapefile属性转JavaBean的过程,对比... 目录前言一、原始的处理办法1、使用Set方法来转换2、使用构造方法转换二、基于ModelMapper

springboot启动流程过程

《springboot启动流程过程》SpringBoot简化了Spring框架的使用,通过创建`SpringApplication`对象,判断应用类型并设置初始化器和监听器,在`run`方法中,读取配... 目录springboot启动流程springboot程序启动入口1.创建SpringApplicat

本地搭建DeepSeek-R1、WebUI的完整过程及访问

《本地搭建DeepSeek-R1、WebUI的完整过程及访问》:本文主要介绍本地搭建DeepSeek-R1、WebUI的完整过程及访问的相关资料,DeepSeek-R1是一个开源的人工智能平台,主... 目录背景       搭建准备基础概念搭建过程访问对话测试总结背景       最近几年,人工智能技术

Linux部署jar包过程

《Linux部署jar包过程》文章介绍了在Linux系统上部署Java(jar)包时需要注意的几个关键点,包括统一JDK版本、添加打包插件、修改数据库密码以及正确执行jar包的方法... 目录linux部署jar包1.统一jdk版本2.打包插件依赖3.修改密码4.执行jar包总结Linux部署jar包部署

C/C++通过IP获取局域网网卡MAC地址

《C/C++通过IP获取局域网网卡MAC地址》这篇文章主要为大家详细介绍了C++如何通过Win32API函数SendARP从IP地址获取局域网内网卡的MAC地址,感兴趣的小伙伴可以跟随小编一起学习一下... C/C++通过IP获取局域网网卡MAC地址通过win32 SendARP获取MAC地址代码#i

SpringBoot 整合 Grizzly的过程

《SpringBoot整合Grizzly的过程》Grizzly是一个高性能的、异步的、非阻塞的HTTP服务器框架,它可以与SpringBoot一起提供比传统的Tomcat或Jet... 目录为什么选择 Grizzly?Spring Boot + Grizzly 整合的优势添加依赖自定义 Grizzly 作为

mysql-8.0.30压缩包版安装和配置MySQL环境过程

《mysql-8.0.30压缩包版安装和配置MySQL环境过程》该文章介绍了如何在Windows系统中下载、安装和配置MySQL数据库,包括下载地址、解压文件、创建和配置my.ini文件、设置环境变量... 目录压缩包安装配置下载配置环境变量下载和初始化总结压缩包安装配置下载下载地址:https://d