当前位置:   article > 正文

【完整代码】详解ICMP协议校验和的计算方式,清晰易懂_icmp校验和计算

icmp校验和计算

摸鱼啦,说的就是你,学习编程从入门到放弃。掌握编程思维方式,远胜于死记硬背编程语法,不能再迷路了。


ICMP协议报文结构(标准)

ICMP协议常用的回送与回送响应结构(ping)

图一 ICMP协议报文结构


ICMP协议校验和的计算方式

对于ICMP协议校验和的计算方式,参考RFC官方文档给出的说明:

  1. The checksum is the 16-bit ones's complement of the one's
  2. complement sum of the ICMP message starting with the ICMP Type.
  3. For computing the checksum , the checksum field should be zero.
  4. If the total length is odd, the received data is padded with one
  5. octet of zeros for computing the checksum.  This checksum may be
  6. replaced in the future.

总体的思路和步骤如下:

一、获取ICMP报文(首部+数据部分)

二、将ICMP报文中的校验和字段置为0。

三、将ICMP协议报文中的每两个字节(16位,需要注意大小端问题)两两相加,得到一个累加和。若报文长度为奇数,则最后一个字节(8-bit)作为高8位,再用0填充一个字节(低8-bit)扩展到16-bit,之后再和前面的累加和继续相加得到一个新的累加和。

四、(若有溢出)将累加和的高16位和低16位相加,直到最后只剩下16位。

五、将最后得到的16位结果取反(按位取反)作为校验和的值。


ICMP协议校验和计算示例

以ICMP协议的ping回送响应报文(大端)为例:

0x08 0x00 0xeb 0xc9 0x00 0x0e 0x00  0x1f 0x00 0x01 0x02 0x03 0x04 0x05 0x06 

1. 先将ICMP协议的校验和字段置为0

0x08 0x00 0x00 0x00 0x00 0x0e 0x00 0x1f 0x00 0x01 0x02 0x03 0x04 0x05 0x06 

2. 将ICMP协议报文中的每两个字节(16位)(需要注意大小端问题,网络字节序是大端模式,如0xebc9, 对于小端主机来说,组成的是0xc9eb,应转化为主机字节序:0xebc9)两两相加,得到一个累加和。若报文长度为奇数,则最后一个字节(8-bit)作为高8位,再用0填充一个字节(低8-bit)扩展到16-bit,之后再和前面的累加和继续相加得到一个新的累加和。

0x0800 + 0x0000 + 0x000e + 0x001f + 0x0001+ 0x0203 + 0x0405 + 0x0600 = 0x1436

3. 将累加和的高16位和低16位相加,直到最后只剩下16位

0x0000 + 0x1436 = 0x1436

4. 将最后得到的16位结果取反(按位取反)作为校验和字段的值

0001 0100 0011 0110 (0x1436 取反)
1110 1011 1100 1001 (0xebc9)


[C/C++] ICMP协议校验和计算代码示例

  1. /**                                                            
  2.  * icmp_calc_checksum:
  3.  * @size: the icmp data packet length
  4.  * @icmp_data: icmp protocol packet both header and data
  5.  *
  6.  * Calc icmp's checksum:
  7.  * if we divide the ICMP data packet is 16 bit words and sum each of them up
  8.  * then hihg 16bit add low 16bit to sum get a value,  
  9.  * If the total length is odd, 
  10.  * the last byte is padded with one octet of zeros for computing the checksum.
  11.  * Then hihg 16bit add low 16bit to sum get a value,
  12.  * finally do a one's complementing 
  13.  * then the value generated out of this operation would be the checksum.
  14.  * 
  15.  * Return: unsigned short checksum
  16.  */
  17. unsigned short icmp_calc_checksum(char * icmp_packet, int size)
  18. {
  19.  unsigned short * sum = (unsigned short *)icmp_packet;
  20.  unsigned int checksum = 0;
  21.  while (size > 1) {
  22.   checksum += ntohs(*sum++);
  23.   size -= sizeof(unsigned short);
  24.  }
  25.  if (size) {
  26.   *sum = *((unsigned char*)sum);
  27.   checksum += ((*sum << 8) & 0xFF00);
  28.  }
  29.  checksum = (checksum >> 16) + (checksum & 0xffff);
  30.  checksum += checksum >> 16;
  31.  
  32.  return (unsigned short)(~checksum);
  33. }

总结

所有文章中给出的代码和示例,都是通过测试验证后贡献给大家的,文章中有些地方可能会出现笔误,欢迎童鞋们及时指出并加以更正。

关注小编,每篇文章都有不一样的体验和感悟,可以留言或者私信小编沟通交流,一起学习呦。


创作不易,动动发财的小手点个免费的关注再走呗!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/386037
推荐阅读
相关标签
  

闽ICP备14008679号