【C语言】linux内核napi_gro_receive和netif_napi_add

news/2024/7/27 8:14:35/文章来源:https://blog.csdn.net/eidolon_foot/article/details/136298896

napi_gro_receive

一、注释

// napi_gro_receive是网络设备接口的一个函数,它被NAPI(New API)网络轮询机制使用,用于接收和处理接收到的数据包。
// 这个函数通过通用接收分组(GRO,Generic Receive Offload)技术来合并多个接收到的数据包,以减少CPU的使用率并提高吞吐量。
gro_result_t napi_gro_receive(struct napi_struct *napi, struct sk_buff *skb)
{// 将skb(socket buffer,数据包缓冲区)关联到当前的napi结构skb_mark_napi_id(skb, napi);// 记录跟踪点,通知开始接收GRO数据包的操作(若系统编译时启用了跟踪点的话)trace_napi_gro_receive_entry(skb);// 重置数据包的GRO偏移量,准备对其进行GRO处理skb_gro_reset_offset(skb);// 调用dev_gro_receive函数来处理具体的GRO逻辑// 并使用napi_skb_finish来处理dev_gro_receive返回的结果,完成GRO处理return napi_skb_finish(dev_gro_receive(napi, skb), skb);
}
// 导出napi_gro_receive符号,使得它可以被模块化的内核代码使用
EXPORT_SYMBOL(napi_gro_receive);

这个函数的作用是将一个接收到的`skb`(数据包缓冲区)与一个`napi`结构(代表网络设备的轮询机制)相关联,并对数据包进行处理以适配GRO。处理后,数据包可能会被合并(分组到一起)以提高网络的处理效率。其中涉及到的跟踪点(如`trace_napi_gro_receive_entry`)用于网络调试和性能分析,如果内核配置支持ftrace或其他调试工具时会生成相应的跟踪信息。最后,`EXPORT_SYMBOL`宏确保了该函数可以被其他内核模块调用。

二、讲解

这段代码是Linux内核网络栈中处理接收包的函数,其主要作用是进行通用接收分段(Generic Receive Offload,简称GRO)的处理。以下是对该函数的中文讲解:

// 定义函数napi_gro_receive,gro_result_t是GRO处理结果的类型,该函数用于处理接收到的网络包
gro_result_t napi_gro_receive(struct napi_struct *napi, struct sk_buff *skb)
{// 将此skb(socket缓冲区)与当前的NAPI(New API,一种提高网络接收性能的接口)上下文关联起来skb_mark_napi_id(skb, napi);// 跟踪函数napi_gro_receive起始的相关信息,用于事件调试trace_napi_gro_receive_entry(skb);// 重置skb中用于GRO处理的偏移指示,准备进行聚合处理skb_gro_reset_offset(skb);// 将skb传递给dev_gro_receive函数进行具体的GRO处理,并通过napi_skb_finish函数处理dev_gro_receive的结果// dev_gro_receive函数负责将多个分段的数据包聚合成一个大的数据包,以提高处理效率return napi_skb_finish(dev_gro_receive(napi, skb), skb);
}
// 通过EXPORT_SYMBOL宏导出napi_gro_receive符号,允许其他模块调用这个函数
EXPORT_SYMBOL(napi_gro_receive);

大致流程为:
1. 当网络包通过napi结构接收时,首先标记该网络包与napi的关联,这一步有助于后续处理该包时保持上下文信息。
2. 如果内核配置了事件追踪,记录接收事件的起始状态,以便于开发者调试。
3. 重置网络包的GRO偏移量,准备进行后续的聚合处理。
4. 然后将网络包传递给`dev_gro_receive`实现GRO处理,这个函数的工作是尝试把多个类似的数据包合并为一个大的数据包,从而减少每个数据包的处理开销,提高整体性能。
5. 最后处理聚合结果,并使用`napi_skb_finish`返回处理后的网络包。
此函数通常被网络设备驱动在其NAPI轮询函数中调用,用来处理数据包的接收和聚合工作。通过这种方式,可以显著提高数据包接收的效率,特别是在高速网络接口上。

netif_napi_add

一、中文注释

/*** netif_napi_add - 将NAPI结构添加到网络设备中* @dev: 指向与NAPI关联的网络设备结构体的指针* @napi: 指向要添加的NAPI结构体的指针* @poll: 指向在轮询模式下处理网络数据包的函数的指针* @weight: 定义在单个poll调用中NAPI结构体可以处理的网络数据包的最大数量** 此函数初始化NAPI结构体并将其添加到网络设备的napi_list中。* 在网络设备注册期间,通常需要调用此函数以为设备设置轮询模式处理。*/
void netif_napi_add(struct net_device *dev, struct napi_struct *napi,int (*poll)(struct napi_struct *, int), int weight)
{// 初始化NAPI结构体的poll_listINIT_LIST_HEAD(&napi->poll_list);// 初始化高精度定时器hrtimer_init(&napi->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL_PINNED);// 设置定时器触发的函数为napi_watchdognapi->timer.function = napi_watchdog;// 初始化NAPI的GRO(Generic Receive Offload)散列init_gro_hash(napi);// 将NAPI结构体的skb成员初始化为NULLnapi->skb = NULL;// 设置NAPI结构体的轮询函数napi->poll = poll;// 检查传入的weight值是否超过了NAPI_POLL_WEIGHT的最大值if (weight > NAPI_POLL_WEIGHT)pr_err_once("netif_napi_add() called with weight %d on device %s\n",weight, dev->name);// 设置NAPI结构体的权重napi->weight = weight;// 将NAPI结构体添加到网络设备的napi_list中list_add(&napi->dev_list, &dev->napi_list);// 将NAPI结构体与网络设备关联napi->dev = dev;#ifdef CONFIG_NETPOLL// 初始化poll_owner字段,用于网络轮询控制napi->poll_owner = -1;
#endif// 设置NAPI结构体的状态为SCHED(表示已经安排了轮询)set_bit(NAPI_STATE_SCHED, &napi->state);// 将NAPI结构体添加到全局的hash表中napi_hash_add(napi);
}

二、讲解

这个`netif_napi_add`函数是Linux网络驱动程序中使用的,用来初始化和添加一个NAPI结构到指定的网络设备中。NAPI是“New API”的简称,是用来改善网络驱动处理高速数据包的一种机制。下面是该函数的讲解:

void netif_napi_add(struct net_device *dev, struct napi_struct *napi,int (*poll)(struct napi_struct *, int), int weight)
{// 初始化napi结构中的poll_list链表头INIT_LIST_HEAD(&napi->poll_list);// 初始化napi结构中的高分辨率定时器hrtimer_init(&napi->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL_PINNED);// 设置高分辨率定时器的回调函数为napi_watchdognapi->timer.function = napi_watchdog;// 初始化接收分组的散列表(用于GRO, 即通用接收分组)init_gro_hash(napi);// 把napi结构中的skb指针置为空napi->skb = NULL;// 设置NAPI结构的poll回调函数napi->poll = poll;// 如果权重大于NAPI_POLL_WEIGHT(定义的最大权重),打印错误信息。权重用来限制poll函数一次处理的最大数据包数量if (weight > NAPI_POLL_WEIGHT)pr_err_once("netif_napi_add() called with weight %d on device %s\n",weight, dev->name);// 设置NAPI结构的权重napi->weight = weight;// 将这个NAPI结构添加到网络设备napi_list的末尾list_add(&napi->dev_list, &dev->napi_list);// 将网络设备指针保存到NAPI结构中napi->dev = dev;// 如果有配置NETPOLL(一种网络轮询模式),则初始化poll_owner为-1
#ifdef CONFIG_NETPOLLnapi->poll_owner = -1;
#endif// 设置NAPI状态为"已调度"(表示这个NAPI结构已经被加入到系统中并且可以被调度执行)set_bit(NAPI_STATE_SCHED, &napi->state);// 将NAPI结构添加到全局的哈希表中,这样它就可以被网络子系统正确地管理和调度了napi_hash_add(napi);
}

总的来说,此函数用于设置网络设备的NAPI结构,配置必要的参数和函数指针,并将其注册到系统中,使得在接收网络数据包时,可以使用这个NAPI结构来调用相关函数处理数据包。这可以降低中断频率,提升网络吞吐性能。

相关链接:

【C语言】linux内核ipoib模块 - ipoib_napi_add-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_997193.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024.3.7 FreeRTOS 作业

思维导图 练习题 1.使用ADC采样光敏电阻数值,如何根据这个数值调节LED灯亮度。 //打开定时器3的通道3,并且设置为PWM功能HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_3);/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1…

MES数据采集设备

在智能制造日益盛行的今天,MES(制造执行系统)作为连接计划与生产现场的关键环节,其重要性不言而喻。而MES数据采集设备则是MES系统的核心组件,负责实时、准确地获取生产现场的各种数据,为企业的生产决策提供…

three.js如何实现简易3D机房?(二)模型加载的过渡动画

接上一篇: three.js如何实现简易3D机房?(一)基础准备-下:http://t.csdnimg.cn/TTI1P 目录 六、自定义过渡动画 1.过渡动画组件 2.模型加载时使用 根据模型大小,可以自定义模型加载的过渡动画效果&am…

Django Web架构:全面掌握Django模型字段(下)

Django Web架构 全面掌握Django模型字段(下) - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite:http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article…

item_fee-获得淘宝商品快递费用 API调用说明获取测试key

item_fee-获得淘宝商品快递费用 .通过传入商品id、区域id,来获取该商品的快递费用。 公共参数 点此获取API请求地址 名称类型必须描述keyString是调用key(必须以GET方式拼接在URL中)secretString是调用密钥api_nameString是API接口名称&a…

JavaScript基础4之原型的原型继承、原型链和理解对象的数据属性、访问器属性

JavaScript基础 原型原型继承问题解决 原型链isPrototypeOf()Object.getPrototypeOf() 理解对象数据属性访问器属性 原型 原型继承 继承是面向对象编程的另一个特征,通过继承进一步提升代码封装的程度,JavaScript中大多是借助原型对象实现继承的特性。…

按图搜索淘宝商品(拍立淘)

使用淘宝拍立淘接口的步骤通常包括: 注册成为淘宝开放平台的开发者:在淘宝开放平台网站上注册账号并完成认证。 创建应用以获取API密钥:在您的开发者控制台中创建一个应用,并获取用于API调用的密钥,如Client ID和Clie…

OpenHarmony教程指南—Ability的启动模式

介绍 本示例展示了在一个Stage模型中,实现standard、singleton、specified多种模式场景。 本实例参考开发指南 。 本实例需要使用aa工具 查看应用Ability 模式信息。 效果预览 使用说明 1、standard模式: 1)进入首页,点击番茄…

FreeRTOS day2

使用ADC采样光敏电阻数值,如何根据这个数值调节LED灯亮度 总结DMA空闲中断接收数据的使用方法 首先要要选择串口然后配置串口的参数,配置MDA通道选择接受数据,配置空闲中断,定义一个数据接收的容器,启动MDA传输当串口…

学习大语言模型(LLM),从这里开始

在见识了ChatGPT的各种强大能力后,不少 NLP一线从业人员很自然地想到,以后开发者只要借助 ChatGPT,就可以做到现在大部分NLP工程师在做的事,比如文本分类、实体抽取、文本推理等。甚至随着大语言模型(largelanguagemod…

力扣刷题Days11第二题--141. 环形链表(js)

目录 1,题目 2,代码 2.1快慢指针 2.2,哈希表 3,学习与总结 3.1自己尝试写快慢指针 反思 1,题目 给你一个链表的头节点 head ,判断链表中是否有环。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达&…

时光机关:探秘Java中的Timer和TimerTask

欢迎来到我的博客,代码的世界里,每一行都是一个故事 时光机关:探秘Java中的Timer和TimerTask 前言Timer和TimerTask的基本概念Timer:TimerTask:为何它们是 Java 中任务调度的得力工具: Timer的使用方法创建…

python中的文件操作2

文件遍历 在Python中,遍历文件通常指的是逐行读取文件中的内容。这种方式对于处理大型文件特别有用,因为它不需要一次性将整个文件加载到内存中。下面是几种常见的遍历文件内容的方法: 1. 使用with语句和for循环 这是最推荐的方式&#xf…

appium解锁android真机系统的屏幕

在使用appium进行app自动化操作的过程中,经常遇到的第一个难题就是如何解锁系统屏幕,也就是亮屏解锁。 实际上解决办法如下:在desired_capabilities中增加两个参数unlockType和unlockKey,类似的示例代码如下: desire…

外包干了2年,技术退步明显

先说一下自己的情况,研究生,19年进入广州某软件公司,干了接近4年的功能测试,今年年初,感觉自己不能够在这样下去了,长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试&#xf…

Stable Diffusion 模型分享:Realistic Stock Photo(真实的库存照片)

本文收录于《AI绘画从入门到精通》专栏,专栏总目录:点这里。 文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八下载地址模型介绍

利用Python爬取高德地图全国地铁站点信息

利用Python中的requests库进行地铁站点信息的获取,同时将数据保存在本机excel中 # 首先引入所需要的包 import requests from bs4 import BeautifulSoup import pandas as pd import json# 发送 GET 请求获取网页内容 url http://map.amap.com/subway/index.html response r…

vue3页面内容切换(类似登录、注册内容切换)

一、内容描述 页面有俩块内容,分别是验证码登录页面内容,账号密码登录页面内容。有俩种处理方式,一个是写俩个页面跳转使用,还有一种是一个页面俩个内容,切换的只是不同的内容,相同的内容保留。一般都是选择…

比肩Gen-2,全新开源文生视频模型

著名开源平台Stability.ai在官网宣布,推出全新文生视频的扩散模型Stable Video Diffusion,已开源了该项目并公布了论文。 据悉,用户通过文本或图像就能生成高精准,14帧和25帧的短视频。目前,Stable Video Diffusion处…

【常见索引使用】⭐️Mysql中索引的类型以及使用方式和失效场景

目录 一、前言 二、数据准备 三、索引的分类 四、索引示例 示例1、主键索引(Primary Key Index)与 唯一索引(Unique Index) 示例2、前缀索引(Prefix Index) 示例3、联合索引(复合索引&am…