操作tailroom中用户数据块区域:skb_put用于修改指向数据区末尾的指针tail:
void *skb_put(struct sk_buff *skb, unsigned int len)
{
void *tmp = skb_tail_pointer(skb);
SKB_LINEAR_ASSERT(skb);
skb- >tail += len;
skb- >len += len;
if (unlikely(skb- >tail > skb- >end))
skb_over_panic(skb, len, __builtin_return_address(0));
return tmp;
}
可以看到 tail指针的移动是扩大数据区域 ,即数据区向下扩大len字节,并更新数据区长度len。
增加headroom区域的协议头: skb_push函数用于移动data指针,增加头部协议, 与skb_reserve()类似,也并没有真正向数据缓存区中添加数据,而只是移动数据缓存区的头指针data。数据由其他函数复制到数据缓存区中。 函数如下:
void *skb_push(struct sk_buff *skb, unsigned int len)
{
skb- >data -= len;
skb- >len += len;
if (unlikely(skb- >data< skb- >head))
skb_under_panic(skb, len, __builtin_return_address(0));
return skb- >data;
}
如下两张图分别是由传输层、网络层,数据包向下传递时data指针移动,进行头部协议的封装。
可以看到在数据包封装的过程中,每一层移动data指针进行数据报头的封装。
数据报文解封装,解除协议头: skb_pull通过将data指针向下移动,进行数据报文的解封装,函数如下所示:
static inline void *__skb_pull(struct sk_buff *skb, unsigned int len)
{
skb- >len -= len;
BUG_ON(skb- >len < skb- >data_len);
return skb- >data += len;
}
如下图所示,在收包流程上,向上层协议,如下网络层向传输层传送的时候,调用skb_pull进行数据包的解封装。
以上就是struct sk_buff的四大指针的相关操作,通过分析可得:
全部0条评论
快来发表一下你的评论吧 !