Chinaunix首页 | 论坛 | 博客
  • 博客访问: 790109
  • 博文数量: 264
  • 博客积分: 592
  • 博客等级: 中士
  • 技术积分: 1574
  • 用 户 组: 普通用户
  • 注册时间: 2011-10-24 22:02
文章分类

全部博文(264)

文章存档

2019年(2)

2018年(1)

2017年(1)

2016年(4)

2015年(14)

2014年(57)

2013年(88)

2012年(97)

分类: LINUX

2014-02-12 15:52:03

转:http://blog.csdn.net/nerdx/article/details/12657661
  1. //ip分片加入到正确的ipq结构  
  2. //调用路径:ip_defrag->ip_frag_queue  
  3.   
  4.   
  5. //  处理过程:  
  6. //      1.正在被释放的ipq,不处理新加入的分片(ipq正在被释放由last_in设置COMPLETE指出)  
  7. //      2.处理分片的合法性  
  8. //          2.1当该封包为最后一个分片时  
  9. //              2.1.1如果之前没有接收到最后一个分片,则该分片在总有效载荷中的结尾位置需要大于等于以推测出的最大长度  
  10. //              2.1.2如果之前已经接收到最后一个分片,则该分片在总有效载荷中的结尾位置需要等于之前接收到的最后一个分片给出的结尾位置  
  11. //          2.2结尾位置对齐到8字节边界,截去多余的字节,希望后续到达的分片补齐  
  12. //      3.更新该分片的skb->data移动到ip有效载荷,skb->tail到8字节  
  13. //      4.处理重叠  
  14. //      5.将分片插入到ipq的分片列表中  
  15. //      6.更新ipq的时间戳,移动ipq到rcu链表尾部  
  16.   
  17. //  重叠的处理:  
  18. //      1.一个分片最多只会与一个前边的分片发生重叠,此时,截去该分片发生重叠的部分  
  19. //      2.一个分片可能会与多个后边的分片发生重叠,此时  
  20. //          2.1 如果后边的一个分片完全被重叠,则释放后边的这个分片  
  21. //          2.2 如果后边的这个分片只有部分被重叠,则从后边的这个分片中截去重叠的部分  
  22. //      3.使被截去缓存区的skb的校验和失效  
  23.   
  24. //  插入ipq->fragments中:  
  25. //      sk_buff通过next域,插入到ipq->fragments中  
  26. 1.1 static void ip_frag_queue(struct ipq *qp, struct sk_buff *skb)  
  27. {  
  28.     struct sk_buff *prev, *next;  
  29.     int flags, offset;  
  30.     int ihl, end;  
  31.     //ipq正在被释放  
  32.     if (qp->last_in & COMPLETE)  
  33.         goto err;  
  34.     //此ip分片(ip有效载荷)在所有有效载荷中的偏移量  
  35.     offset = ntohs(skb->nh.iph->frag_off);  
  36.     flags = offset & ~IP_OFFSET;//DF MF标志  
  37.     offset &= IP_OFFSET;////偏移量为13bit的字段,去掉高3bit的flag  
  38.     offset <<= 3;     //偏移量以8字节为单位  
  39.     ihl = skb->nh.iph->ihl * 4;  
  40.   
  41.     end = offset + skb->len - ihl;//该ip分片的有效载荷相对于总有效载荷的结束位置  
  42.   
  43.     //接收到最后一个分片  
  44.     if ((flags & IP_MF) == 0) {  
  45.         //1.该分片指示的总有效载荷大小不足从已接收到的报文推断出的长度  
  46.         if (end < qp->len ||  
  47.             ((qp->last_in & LAST_IN) && end != qp->len))//2.已经接收到最后一个分片,又接收到一个最后一个分片,但是长度不等  
  48.             goto err;  
  49.         qp->last_in |= LAST_IN;//LAST_IN表示已经接收到最后一个分片  
  50.         qp->len = end;//总有效载荷的长度  
  51.     } else {//非最后一个分片  
  52.         if (end&7) {//结尾位置没有对齐到8字节  
  53.             end &= ~7;//去掉结尾的字节,希望后来的数据补齐  
  54.             if (skb->ip_summed != CHECKSUM_UNNECESSARY)  
  55.                 skb->ip_summed = CHECKSUM_NONE;//由于截去了末尾的字节,因此表示校验和失效  
  56.         }  
  57.         if (end > qp->len) {//非最后一个分片,但是长度超过了总有效载荷长度  
  58.             if (qp->last_in & LAST_IN)  
  59.                 goto err;//有错误  
  60.             qp->len = end;//由于新接收到的分片其结尾字节位置大于最大的结尾位置,更新最大的结尾位置  
  61.         }  
  62.     }  
  63.     if (end == offset)//长度为0分片的  
  64.         goto err;  
  65.   
  66.     //pskb_pull 与 skb_pull的区别:  
  67.     //  1.skb_pull只简单移动 skb->data的指针  
  68.     //  2.pskb_pull考虑当skb->data到skb->tail之间的数据量如果不足够移动时,从frags或者frag_list中向前拷贝  
  69.     if (pskb_pull(skb, ihl) == NULL)//更新skb->data,使其指向ip有效载荷  
  70.         goto err;  
  71.     if (pskb_trim(skb, end-offset))//更新skb->tail指针,使skb->data与skb->tail之间的数据量为end-offset  
  72.         goto err;  
  73.   
  74.     //在ipq->fragments中寻找该分片前边的分片  
  75.     //一个分片前边分片的满足条件:  
  76.     //  1.它的偏移量小于该分片的偏移量  
  77.     //  2.它后边分片的偏移量大于等于该分片的偏移量  
  78.     prev = NULL;  
  79.     for(next = qp->fragments; next != NULL; next = next->next) {  
  80.         if (FRAG_CB(next)->offset >= offset)  
  81.             break;    
  82.         prev = next;  
  83.     }  
  84.   
  85.     //已经接收到该分片前边的分片  
  86.     //该分片最多只会与前边的一个分片重叠  
  87.     if (prev) {  
  88.         int i = (FRAG_CB(prev)->offset + prev->len) - offset;  
  89.         //该分片与前边的分片存在重叠的部分  
  90.         if (i > 0) {  
  91.             offset += i;//更新该分片的offset  
  92.             if (end <= offset)  
  93.                 goto err;  
  94.             if (!pskb_pull(skb, i))//从该分片中删除重叠的部分  
  95.                 goto err;  
  96.             if (skb->ip_summed != CHECKSUM_UNNECESSARY)  
  97.                 skb->ip_summed = CHECKSUM_NONE;//由于该分片长度被截取,因此校验和失效  
  98.         }  
  99.     }  
  100.     //接着之前的搜索,寻找该分片的后边的分片  
  101.     //该分片会与多个后边的分片重叠  
  102.     while (next && FRAG_CB(next)->offset < end) {  
  103.         int i = end - FRAG_CB(next)->offset;//与后边第一个分片重叠的字节数  
  104.   
  105.         //重叠的字节数小于后边分片的长度,说明只与后边一个分片发生了重叠  
  106.         if (i < next->len) {  
  107.             //更新后边分片的data指针,截取重叠的部分  
  108.             if (!pskb_pull(next, i))  
  109.                 goto err;  
  110.             //更新后边分片的偏移量  
  111.             FRAG_CB(next)->offset += i;  
  112.             //由于从后边分片截取了重叠的部分,从已接收到的数据量减去这部分字节  
  113.             qp->meat -= i;  
  114.             if (next->ip_summed != CHECKSUM_UNNECESSARY)  
  115.                 next->ip_summed = CHECKSUM_NONE;//由于截取了后边的这个分片,使其校验和失效  
  116.             break;  
  117.         } else {//否则完全包含了后边的这个分片, 则直接释放掉被重叠的这个分片  
  118.             struct sk_buff *free_it = next;  
  119.   
  120.             next = next->next;  
  121.   
  122.             if (prev)  
  123.                 prev->next = next;  
  124.             else  
  125.                 qp->fragments = next;  
  126.             qp->meat -= free_it->len;  
  127.             frag_kfree_skb(free_it, NULL);  
  128.         }  
  129.     }  
  130.     //强制转换skb->cb为分片控制块  
  131.     //经过与前边,后边的分片比较,最终确定该分片的偏移量  
  132.     //设置该分片的偏移量  
  133.     FRAG_CB(skb)->offset = offset;  
  134.     //将分片添加到ipq的fragments链表中  
  135.     skb->next = next;  
  136.     if (prev)  
  137.         prev->next = skb;  
  138.     else  
  139.         qp->fragments = skb;//此分片为第一个分片  
  140.   
  141.     if (skb->dev)  
  142.         qp->iif = skb->dev->ifindex;  
  143.     skb->dev = NULL;  
  144.     qp->stamp = skb->stamp;//更新ipq的时间戳为最新收到的这个skb的时间戳  
  145.     qp->meat += skb->len;  
  146.     //skb->truesize为skb中所有缓存区占用的总大小  
  147.     atomic_add(skb->truesize, &ip_frag_mem);//增加分片子系统使用的内存量  
  148.     if (offset == 0)  
  149.         qp->last_in |= FIRST_IN;  
  150.   
  151.     write_lock(&ipfrag_lock);  
  152.     list_move_tail(&qp->lru_list, &ipq_lru_list);  
  153.     write_unlock(&ipfrag_lock);  
  154.   
  155.     return;  
  156.   
  157. err:  
  158.     kfree_skb(skb);  
  159. }  
阅读(800) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~