Chinaunix首页 | 论坛 | 博客
  • 博客访问: 778835
  • 博文数量: 215
  • 博客积分: 291
  • 博客等级: 二等列兵
  • 技术积分: 1031
  • 用 户 组: 普通用户
  • 注册时间: 2010-05-12 18:17
文章分类

全部博文(215)

文章存档

2016年(16)

2015年(16)

2014年(123)

2013年(60)

分类: 网络与安全

2014-01-13 14:17:11

原文地址:IPv4的头校验算法 作者:forgaoqiang

IPv4版本的头部信息比较复杂,含有校验等信息,但是也导致路由器需要进行计算,增加了延时,IPv6中取消了对应的校验和。



这个图很好的显示了Ipv4的IP包内的组成,可以看出头部总长度是20字节。

然后我们看下Ipv4和Ipv6版本的IP头部的区别:


直接说下重点吧,就是关于IP头部的计算问题:

这是我随手抓的一个包,就以它为例进行计算:

IP的所有字段:
4500 0042 3038 0000 4011 6257 c0a8 0afa af90 [6ce9]  

全部相加: 29314 =>  然后把溢出的第一位移动到最后一位 得到 9316
得到二进制后人后取反:
1001 0011 0001 0110
0110 1100 1110 1001 => 6CE9(正解)

二进制反码求和,即为先进行二进制求和,然后对和取反。


下面是最常用的C语言的校验算法,虽然Linux内核使用更高效的汇编写成的,但是这个C语言版本的在其他位置更常用。


  1. /*求校验和函数*/
  2. USHORT CheckSum(USHORT *buffer, int size)
  3. {
  4.     unsigned long cksum=0;
  5.     while (size > 1)
  6.     {
  7.         cksum += *buffer++;
  8.         size -= sizeof(USHORT);
  9.     }
  10.     if (size)
  11.     {
  12.         cksum += *(UCHAR*)buffer;
  13.     }
  14.     /*对每个16bit进行二进制反码求和*/
  15.     cksum = (cksum >> 16) + (cksum & 0xffff);
  16.     cksum += (cksum >>16);
  17.     return (USHORT)(~cksum);
  18. }


阅读(2109) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~