本文主要是介绍skb操作之网卡收包,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
好久没有写博客了,今天来聊一聊收包过程中的skb是如何操作的,内容比较简单
2023.3.6也是一个小有特别的日子
收包过程中的skb变化
驱动申请skb
当nic有数据包到来时,驱动会调用netdev_alloc_skb
函数去申请一个skb,该skb就用于存放网卡后续拷贝过来的数据,接下来的流程图中skb的长度都假设未进行字节对齐,驱动调用完该接口后,skb如下图所示:
网卡DMA拷贝
当驱动申请完skb后,接下来便是驱动将skb的data指针告诉网卡,网卡便会往skb的data指针处进行数据的DMA拷贝,拷贝完毕后,网卡会告知cpu一共DMA拷贝了多少字节数,以便cpu进行后续操作,如下图所示,网卡拷贝了real_len的数据到了skb的线性缓冲区
驱动操作skb
驱动拿到已经拷贝完毕的skb后,便开始了对skb进行一系列的操作,最终将其送往网络子系统进行处理,主要为以下几个步骤:
skb_reserve(skb, 2)
//可有可无,驱动为了性能,可能会继续reserve两个字节skb_put(skb, real_len)
//根据网卡拷贝的字节数,对skb指针进行操作eth_type_trans(skb,dev)
//确定skb的协议类似,该函数也同步确定了skb的mac头的偏移
skb_put(skb, real_len)
eth_type_trans(skb,dev)
调用该函数后主要做了三件事:
- 是通过一系列判断,决定该包数据的协议(802.3、802.2等)
- 通过一系列判断,决定该包数据的类型,广播等
- 先重置skb的以太网头(记录以太网头的偏移),然后将data指针指向ip头(
skb_pull_inline
)
__netif_receive_skb_core
最后,调用该函数将skb送完协议栈就结束啦,后面ip头、tcp头的处理交给内核吧
ip层拿到skb后,首先要调用以下两个函数:
skb_reset_network_header(skb)
//记录ip头的偏移skb_reset_mac_len(skb)
//记录mac头的长度
ip_rcv
这里ip_rcv其实就是注册到pt_prev->func(skb, skb->dev, pt_prev, orig_dev)
的钩子函数
在该函数里设置了传输层的头指针
skb->transport_header = skb->network_header + iph->ihl*4
tcp_v4_rcv
最终将skb递交给tcp层处理啦!
这篇关于skb操作之网卡收包的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!