skb_copy_bits

int skb_copy_bits(const struct sk_buff *skb, int offset, void *to, int len)
{
 int i, copy;
 int start = skb_headlen(skb); //主数据包的基本数据块长度
                               //skb->len : 全部数据块的总长度

 if (offset > (int)skb->len - len) //offset + len > skb->len
  goto fault;

 /* Copy header. */ 
 if ((copy = start - offset) > 0) { //基本数据块长度 - 偏移处: 判断是否要拷贝头部
  if (copy > len) //需拷贝的数据长度完全在基本数据块内, 所以完全是线性拷贝
   copy = len;
  memcpy(to, skb->data + offset, copy); 
  if ((len -= copy) == 0)
   return 0;
  offset += copy; //如果拷贝的数据存在于散列在内存页面的分散数据块或者分段数据块中
  to     += copy;
 }
    //到这里说明还没有全部复制完成,需要复制分散数据块
 for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) {
  int end;

  BUG_TRAP(start <= offset + len); // start - offset <= len 说明分散数据块内有东西 

  end = start + skb_shinfo(skb)->frags[i].size; //基本数据块+当前分散数据块
  if ((copy = end - offset) > 0) {
   u8 *vaddr;

   if (copy > len)
    copy = len;

   vaddr = kmap_skb_frag(&skb_shinfo(skb)->frags[i]);//取得分散数据块所在的页面
   memcpy(to,
          vaddr + skb_shinfo(skb)->frags[i].page_offset+
          offset - start, copy); //复制分散数据块中的内容
   kunmap_skb_frag(vaddr); //冲刷分散数据块所在的页面

   if ((len -= copy) == 0)
    return 0;
   offset += copy;
   to     += copy;
  }
  start = end;
 }

  //到这里,说明还未复制完成,需要复制原有分段数据包的数据块

 /* 如果还没有 copy 完足够长度的数据,还要遍历 frag_list 继续  copy 数据,而且 copy 数据的方式是递归调用该函数 */

 if (skb_shinfo(skb)->frag_list) {
  struct sk_buff *list = skb_shinfo(skb)->frag_list;

  for (; list; list = list->next) {
   int end;

   BUG_TRAP(start <= offset + len);

   end = start + list->len;
   if ((copy = end - offset) > 0) {
    if (copy > len)
     copy = len;
    //由于frag_list 的成员仍然是 sk_buff结构体,因此这部分数据的
    //copy方式同本函数
    if (skb_copy_bits(list, offset - start, 
        to, copy))
     goto fault;
    if ((len -= copy) == 0)
     return 0;
    offset += copy;
    to     += copy;
   }
   start = end;
  }
 }
 if (!len)
  return 0;

fault:
 return -EFAULT;
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值