当前位置:首页 > 公众号精选 > embed linux share
[导读]科普tcp协议栈对ping命令处理过程

知识回顾

在前面的文章中,我们已经介绍了ip数据包的封装接口,其中主要是以下几个接口:
  • route_init():路由表的初始化
  • ip_rcv():发送以太网帧数据
  • ip_output():发送以太网帧数据 这几个接口是我们封装IP数据接口的基础,最好还是先搞明白原理。

ICMP 协议的格式

无论是在宿舍,还是在办公室,或者运维一个数据中心,我们常常会遇到网络不通的问题。那台机器明明就在那里,你甚至都可以通过机器的终端连上去看。它看着好好的,可是就是连不上去,究竟是哪里出了问题呢?一般情况下,你会想到 ping 一下。
那你知道 ping 是如何工作的吗?
ping 是基于 ICMP 协议工作的。ICMP全称Internet Control Message Protocol,就是互联网控制报文协议。
网络包在异常复杂的网络环境中传输时,常常会遇到各种各样的问题。当遇到问题的时候,总不能“死个不明不白”,要传出消息来,报告情况,这样才可以调整传输策略。这就相当于我们经常看到的电视剧里,古代行军的时候,为将为帅者需要通过侦察兵、哨探或传令兵等人肉的方式来掌握情况,控制整个战局。ICMP 报文是封装在 IP 包里面的。因为传输指令的时候,肯定需要源地址和目标地址。它本身非常简单。因为作为侦查兵,要轻装上阵,不能携带大量的包袱。
从而我们可以确定,ICMP是在ip协议之上的传输层协议,如下图:


ICMP 报文有很多的类型,不同的类型有不同的代码。最常用的类型是主动请求为 8,主动请求的应答为 0。

构造ICMP 报文

接下来,我们使用结构体来定义ICMP 报文,该结构体定义在level-ip的include/icmpv4.h文件中:

查询报文类型

我们经常在电视剧里听到这样的话:主帅说,来人哪!前方战事如何,快去派人打探,一有情况,立即通报!
这种是主帅发起的,主动查看敌情,对应 ICMP 的查询报文类型。例如,常用的ping 就是查询报文,是一种主动请求,并且获得主动应答的 ICMP 协议。所以,ping 发的包也是符合 ICMP 协议格式的,只不过它在后面增加了自己的格式。
对 ping 的主动请求,进行网络抓包,称为ICMP ECHO REQUEST。同理主动请求的回复,称为ICMP ECHO REPLY。比起原生的 ICMP,这里面多了两个字段,一个是标识符。这个很好理解,你派出去两队侦查兵,一队是侦查战况的,一队是去查找水源的,要有个标识才能区分。另一个是序号,你派出去的侦查兵,都要编个号。如果派出去 10 个,回来 10 个,就说明前方战况不错;如果派出去 10 个,回来 2 个,说明情况可能不妙。
在选项数据中,ping 还会存放发送请求的时间值,来计算往返时间,说明路程的长短。
当前,ICMP还有差错报文类型,适合在遇到各种网络故障时使用,我们这里暂不介绍。

ping:查询报文类型的使用

接下来,我们重点来看 ping 的发送和接收过程
  • 主机A执行ping 命令的时候,源主机首先会构建一个 ICMP 请求数据包,ICMP 数据包内包含多个字段。最重要的是两个,第一个是类型字段,对于请求数据包而言该字段为 8;另外一个是顺序号,主要用于区分连续 ping 的时候发出的多个数据包。每发出一个请求数据包,顺序号会自动加 1。为了能够计算往返时间 RTT,它会在报文的数据部分插入发送时间。
  • 主机 B 收到这个数据帧后,先检查它的目的 MAC 地址,并和本机的 MAC 地址对比,如符合,则接收,否则就丢弃。接收后检查该数据帧,将 IP 数据包从帧中提取出来,交给本机的 IP 层。同样,IP 层检查后,将有用的信息提取后交给 ICMP 协议。主机 B 会构建一个 ICMP 应答包,应答数据包的类型字段为 0,顺序号为接收到的请求数据包中的顺序号,然后再发送出去给主机 A。
关于IP层数据收发和MAC层的数据收发已经在前面给大家分析过了,这里不再讲解,我们重点关注IP数据包递交给ICMP层的过程。

ICMP数据接收接口

ICMP数据接收接口为icmpv4_incoming()函数。该函数在以太网数据帧读取接口ip_rcv()函数中调用。我们来了解一下这个函数,如下图:
第4行:从ip数据包中获取icmp报文
第8行:判断icmp的报文类型,如果是查询报文,则调用icmpv4_reply()函数

ICMP数据发送接口

ICMP数据发送接口为icmpv4_reply()函数,当level-ip协议栈接受到icmp查询类型报文后,会调用该接口进行ICMP数据回复。如下图:
第3行:获取ip数据包首部
第8行:用ip数据包的总长度-ip首部长度,得到icmp包总长度
第10行:把sk_buff数据包指针移动到有效数据结束
第11行:把sk_buff数据包指针移动从有效数据的尾巴向前移动icmp包总长度
第13行:获取icmp数据包
第15行:修改icmp协议类型为应答类型
第16行:把类型字段修改为0,表示ping应答
第17行:修改校验数据
第20行:设置目标主机ip
第22行:调用ip数据发送接口,进行icmp数据包发送

总结

通过我们这边文章,我们已经明白了ICMP协议的报文结构,并且对ping命令的处理过程有了一个源码级别的了解。





本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

文件传输协议(File Transfer Protocol,FTP)是用于在网络上进行文件传输的一套标准协议,它工作在 OSI 模型的第七层, TCP 模型的第四层, 即应用层, 使用 TCP 传输而不是 UDP, 客户...

关键字: OSI tcp UDP

对tcp协议栈中ip数据的收发接口进行剖析

关键字: level-ip ip数据包 接口

在嵌入式行业网络编程使用相对较少,主流应用集中在NB-IOT、Lora、Mqtt这一块,原理上一般是通过加入硬件模块或者是使用第三方SDK来实现。

关键字: 网络编程 嵌入式 tcp

讲解一个小巧精致、可以直接在Linux平台下运行的tcp协议栈,带你感受源码剖析tcp的快感!

关键字: level-ip 以太网 Linux

这篇文章主要是讲解level-ip如何实现虚拟网卡接口封装,源码级解析!

关键字: level-ip 虚拟网卡 网卡

生活不止眼前的苟且,还有诗和远方的田野。新的一周又开始了,大白和小林是同事,平时俩人一起喝酒吃肉打游戏居多,当然有时候也讨论下学术和前沿技术。这不,小林听说了个新鲜玩意,然后和大白聊了起来。

关键字: tcp udp 互联网

TCP握手一定是三次?TCP 挥手一定是四次?为什么要有快速重传,超时重传不够用?为什么要有 SACK,为什么要有 D-SACK?Silly Window 又是什么?为什么有滑动窗口流控还需要拥塞控制?快速重传一定要依赖...

关键字: tcp 网络技术

  当网络出现连接故障时,很多网管人员都会首先使用Ping来查找问题的根源。但是很多人只要发现Ping不通,那么就判断是网络出现了故障。这样只通过表面现象进行判断,很可能掩盖真相。那么我们在使用Pin

关键字: ping 网络故障

    本文提出了一种TD-HSUPA系统的TCP优化方法:利用无线网络控制器RNC(Radio Network Controller)解析TCP连接链路上服务器的反馈包信息,

关键字: tcp td-hsupa

  Transmission Control Protocol/Internet Protocol的简写,中译名为传输控制协议/因特网互联协议,又名网络通讯协议,是Internet最基本的协议、

关键字: tcp ip协议
关闭
关闭