USB linux NCM usbnet驱动详解

2024-05-27 11:58
文章标签 linux 详解 驱动 usb ncm usbnet

本文主要是介绍USB linux NCM usbnet驱动详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

cdc_ncm -> usbnet device -> net_device

enum流程,cdc_ncm的probe中直接调用usbnet_probe,usbnet_probe中会分配network_device和usbnet,并调用cdc_ncm的bind回调进行cdc_ncm的附加descriptor的解析,之后调用init_status对NCM的
第一个interface的interrupt ep进行urb的分配和注册。这个urb的complete的callback会调用cdc_ncm的status回调进行相应处理,主要是两种event, USB_CDC_NOTIFY_NETWORK_CONNECTION和USB_CDC_NOTIFY_SPEED_CHANGE.
USB_CDC_NOTIFY_NETWORK_CONNECTION:根据传来的wValue判断出是disconnect和connect,然后调用netif_carrier_on/netif_carrier_off来通知上层。
USB_CDC_NOTIFY_SPEED_CHANGE:简单的打印一句log而已。
最后注册network_device(register_netdev和netif_device_attach)。
cdc_ncm_bind主要就是根据NCM的descriptor来初始化一个cdc_ncm_ctx对象。

cdc_ncm_check_connect是在ifconfig时调用到了network_device的open函数时候被调用,主要来判断ctx是否为空,因为disconnect的时候,它就会被设置为NULL。

cdc_ncm_manage_power的实现主要就是用来开启/关闭remote wakeup功能的。


接受流程:
rx_submit, 会alloc_skb, 然后usb_fill_bulk_urb,最后usb_submit_urb。
usb_submit_urb如果submit成功,则将skb提交到rxq上。
urb的rx_complete里首先会设置state为rx_done,然后会判断urb->status, 如果是0说明接受成功,state = rx_cleanup。然后调用defer_bh。defer_bh会把skb从rxq上移除,然后将其
加入到done的list中。如果done中的qlen等于1, 则tasklet_schedule(&dev->bh),即usbnet_bh被调用。
usbnet_bh会首先从done中dequeue一个skb,然后判断如果state状态是rx_done,则设置为rx_cleanup,然后调用rx_process去处理。如果是rx_cleanup则free_urb,free_skb。
rx_process会调用driver_info->rx_fixup去让cdc_ncm驱动进行处理,处理完成后,因为cdc_ncm设置了FLAG_MULTI_PACKET,所以直接调用dev_kfree_skb_any去释放掉了skb。如果没有设置,则调用usbnet_skb_return。

cdc_ncm_rx_fixup处理:主要就是将skb中的data,按照NTB格式解析:
1,在NTH中取得NTB length,如果大于cdc_ncm驱动支持的最大size,则返回错误。
2,解析NTH中的wNdpIndex取得第一个NDP的索引。
3,memcpy,将第一个ndp拷贝到rx_ncm.ndp16中
4,接下来解析ndp,首先ndp会包含一个或多个dpe,每个dpe包含一个frame的网络数据。
1019     nframes = ((le16_to_cpu(ctx->rx_ncm.ndp16.wLength) -
1020                     sizeof(struct usb_cdc_ncm_ndp16)) /
1021                     sizeof(struct usb_cdc_ncm_dpe16));
计算出有多少个frames
5,判断nframes是否大于CDC_NCM_DPT_DATAGRAMS_MAX,如果大于则 nframes = CDC_NCM_DPT_DATAGRAMS_MAX;
6,将所有的npe都拷贝到rx_ncm.dpe16中
7,一个for循环,开始解析每一个dpe,对每一个dpe都调用skb_clone创建一个skb备份,然后调用usbnet_skb_return,通过netif_rx将dataparam中的数据,提交给network subsystem。

发送流程:
network_device的ndo_start_xmit:
1,首先调用cdc_ncm的tx_fixup函数。 cdc_ncm_tx_fixup会调用cdc_ncm_fill_tx_frame函数将上层传下来的skb转换为NTB,linux的cdc_ncm的NTB格式为: NTH + NDP + datapayload。其中NDP会包含n个datagram(index+len)。
2,创建urb,并在skb->cb创建skb_data。并处理发送ZLP,就是有些device要求发送一个zero packet来作为结束请求。
3,提交urb,成功返回后,将对应的skb加到txq中,如果txq中的skb大于maxSegmentSize,那么调用netif_stop_queue来告诉上层不要再发数据了,快爆了。
在urb的tx_complete中,最后会调用usbnet_bh,usbnet_bh中最后判断出txq中的skb小于maxSegmentSize后,会调用netif_wake_queue来恢复上层的数据传送。

这篇关于USB linux NCM usbnet驱动详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

javaSE类和对象进阶用法举例详解

《javaSE类和对象进阶用法举例详解》JavaSE的面向对象编程是软件开发中的基石,它通过类和对象的概念,实现了代码的模块化、可复用性和灵活性,:本文主要介绍javaSE类和对象进阶用法的相关资... 目录前言一、封装1.访问限定符2.包2.1包的概念2.2导入包2.3自定义包2.4常见的包二、stati

Linux之UDP和TCP报头管理方式

《Linux之UDP和TCP报头管理方式》文章系统讲解了传输层协议UDP与TCP的核心区别:UDP无连接、不可靠,适合实时传输(如视频),通过端口号标识应用;TCP有连接、可靠,通过确认应答、序号、窗... 目录一、关于端口号1.1 端口号的理解1.2 端口号范围的划分1.3 认识知名端口号1.4 一个进程

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析

Linux权限管理与ACL访问控制详解

《Linux权限管理与ACL访问控制详解》Linux权限管理涵盖基本rwx权限(通过chmod设置)、特殊权限(SUID/SGID/StickyBit)及ACL精细授权,由umask决定默认权限,需合... 目录一、基本权限概述1. 基本权限与数字对应关系二、权限管理命令(chmod)1. 字符模式语法2.

在Linux系统上连接GitHub的方法步骤(适用2025年)

《在Linux系统上连接GitHub的方法步骤(适用2025年)》在2025年,使用Linux系统连接GitHub的推荐方式是通过SSH(SecureShell)协议进行身份验证,这种方式不仅安全,还... 目录步骤一:检查并安装 Git步骤二:生成 SSH 密钥步骤三:将 SSH 公钥添加到 github

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我