Linux SKB结构体中各个长度字段的含义(len, data_len, headlen, pagelen)

结构体sk_buff是Linux内核网络子系统的一个基本结构体,关于它的长度有很多定义和操作,非常容易弄混,这里给出一个简单的说明。

下面这幅图是《深入理解Linux网络技术内幕》里面的图,一个skb用于存储一个报文,如果一个报文特别大的话,线性存储区放不下就需要多个skb来存储,这就是下面frag_list的作用,保存连续的skb,但是如果内核支持分散聚集技术的话,并且报文长度刚好又不大于mtu,就不必重新分配一个skb来存储,可以使用一些内存碎片来存储,就是下面的frags数组表示的内存页面片段。举一个恰当的例子,一帮弟兄去凑份子,凑份子起点比较高,大哥钱不够,这时候如果有小弟(内核支持分散聚集)并且小弟有钱的话就拿上小弟的钱来凑,但是如果加上小弟的钱也不够的话,大哥就只好去找兄弟去了⊙﹏⊙‖∣。份子钱就好比报文大小,大哥不够小弟来凑,没有小弟(不支持分散聚集)或者小弟钱也不够就只好去找兄弟了,这里需要注意的一点是大哥和小弟的总钱数不能大于MTU,如果大于MTU的话,多出的部分就必须重新分配一个skb来存储(总不能一直找小弟吧)。

skb->len      //大哥、小弟和兄弟的总和,即data的总长度,线性和非线性的总和。
skb->data_len //小弟和兄弟的总和,即大哥缺少的份额,非线性数据长度。
skb_headlen() //即大哥长度,线性长度。和skb_headroom()不一样,这个是只头部空间剩余长度。
skb_pagelen() //大哥和小弟的总和,即线性数据长度和页面碎片的长度,不包括分片skb队列长度。

猜你喜欢

转载自blog.csdn.net/fuyuande/article/details/85104180
今日推荐