分析数据在协议栈底层的流程


析数据在协议栈底层的流程:当网卡收到数据后,产生硬件中断,由中断处理程序(一般为网卡驱动程序所注册)从网卡内读取数据,并封装称sk_buff{}结构,
然后把这些数据传递给函数netif_rx()进行进一步的处理。函数netif_rx()根据当前接收队列的拥挤情况,选择丢弃还是接收,如果是接收,则将接收到的sk_buff{}挂到接收队列softnet_data[CPU]->input_pkt_queue上,并调用函数__cpu_raise_softirq()激活软中断NET_RX_SOFTIRQ,相应的处理函数是net_rx_action()。在函数net_rx_action()中根据数据包的协议类型,调用相应的处理函数。对于IP包,处理函数是ip_rcv()。函数ip_rcv()对IP包进行了一系列必要的检查(包括检查校验和),最终调用函数ip_rcv_finish()对数据包进行向上传输。函数ip_rcv_finish()首先调用函数ip_route_input()获取路由,检测该包是发给本机的还是要进行转发的,如果要进行转发,则调用调用函数ip_forward()进行转发,否则调用函数ip_local_deliver()进一步向上传递数据包。函数ip_local_deliver()首先进行了防火墙的过滤工作,最终调用函数ip_local_deliver_finish()向上传递数据。在函数ip_local_deliver_finish()中,会检查是否有匹配协议(如根据IP头判断我们的数据包是TCP包,则要判断是否有接收TCP包的原始套接口。当然,如果有接收所有IP包的原始套接口存在也是可以的)的原始套接口。如果有,则调用函数raw_v4_input()进行处理。在函数raw_v4_input()中,要进一步进行匹配,这次匹配的依据有四个,依次是:协议、源地址、目的地址和接收接口。分别对每一个匹配成功的原始套接口调用函数raw_rcv()传递一个克隆的以sk_buff{}为结构的数据包。接下来的几个函数都很简单,调用顺序依次是raw_rcv()、raw_rcv_skb()和sock_queue_rcv_skb()。这几个函数基本上都是简单的依次调用关系。最后调用函数sock_queue_rcv_skb(),该函数
经过skb_queue_tail()函数将数据包sk_buff{}放入了接收队列sk->receive_queue的末尾。原始套接口的协议栈实现――原始套接口的绑定这里我们简略分析,对原始套接口绑定调用的是函数sk->prot->bind,在原始套接口的创建中我们给出了套接口的sk->prot即structproto结构变量raw_prot,从中可以看出和sk->prot->bind指针实质指向函数raw_bind()。在这个函数中首先判断套接口状态,如果是TCP_CLOSE的话,就退出。然后有对参数进行了一些常规检查。同时,如果发现要绑定的地址是广播或多播的话,也会退出。如果通过了这些检查,就进行一些赋值操作,将用户要绑定的地址赋值到sk->rcv_saddr和sk->saddr中,即:sk->rcv_saddr=sk->saddr=addr->sin_addr.s_addr然后会正常退出。注意,这里没有对端口做任何操作,即使用户指定了要绑定的端口,内核也不予理睬。原始套接口的协议栈实现――原始套接口的连接从原始套接口的创建一节中给出的structproto结构可以看出,原始套接口的连接其实调用的是函数udp_connect(),好兴奋,
终于见到了不
那么"原始"的东西了。在这个函数中,首先对用户的参数进行了一些检查。当然,它也检查了用户指定的网域是否是"AF_INET",如果不是,会返回一个EAFNOSUPPORT错误。然后,该函数调用了函数ip_route_connect()来获取一个到目的地址的路由,如果失败,也会返回错误。接下来的工作看起来就
有点令人难以理解。它检查了套接口是否指定了源地址,如果没有指定,则将寻找到的路由的源地址赋值给这个套接口的源地址,即:if(!sk->saddr)sk->saddr=rt->rt_src;/*Updatesourceaddress*/if(!sk->rcv_saddr)sk->rcv_saddr=rt->rt_src;其中sk代表我们套接口的sock{}结构,rt代表我们找到的路由,是一个structrtable{}结构。最后,就是将目的地址和目的端口赋值到我们的套接口的指定字段中,同时更新套接口状态,即:sk->daddr=rt->rt_dst;sk->dport=usin->sin_port;sk->state=TCP_ESTABLISHED;原始套接口的协议栈实现――原始套接口的关闭根据上面的经验,原始套接口的关闭应该调用函数raw_close(),这个函数只是简单的调用了函数ip_ra_control(),在函数ip_ra_control()中,将该套接口从链表ip_ra_chain中删除,然后释放到该套接口占用的所有空间。原始套接口的应用根据
前面的分析,针对原始套接口的应用,我们可以得出以下结论。绑定的问题可以对原始套接口调用bind函数,但并不常用。该函数仅用来设置本地地址。对
于一个原始套接口而言,端口号是没有意义的。当进行输出的时候,bind设置在原始套接口上所发送的数据报中将要用到的源IP地址(仅当IP_HDRINCL套接口选项未设置时);若不调用bind,则由内核将源IP地址设成外出接口的主IP地址。连接的问题在原始套接口上可调用connect函数,但也不常用。connect函数仅设置目的地址。再重申一遍:端口号对原始套接口而言没有意义。对于输出而言,调用connect之后,由于目的地址已经指定,我们可以调用write或send,而不是sendto了。输出的问题1)普通输出通常通过sendto或sendmsg并指定目的IP地址来完成,如果套接口已经连接,也可以调用write、writev或send。2)如果IP_HDRINCL选项未设置,则内核写的数据起始地址是IP头部之后的第一个字节。因为这种情况下,内核将构造IP头部,并将它安在来自进程数据之前。内核将IPv4头部的协议字段设置成用户在调用socket函数时所给的第三个参数。3)如果IP_HDRINCL选项已设置,则内核写的数据其实地址是IP头部的第一个字节。用户所提供的数据必须包括IP头部。此时进程构造除了以下两项以外的整个IP头部:(1)IPv4标示字段可以设为0,要求内核设置该值。而且仅当该字段为0时,内核才为其设置。(2)IPv4头部校验和由内核来计算和存储。4)如果创建原始套接口时指定了协议类型,即第三个参数protocol,那也并不是说只能发该类型的数据包。如,即使将protocol指定为IPPROTO_TCP,也可以发送用户自己组装的UDP报文,不过此时如果IP_HDRINCL选项未设置,那么内核将会在IP头的协议字段指明后面的报文为TCP报文(不过此时却为UDP报文)。等数据包发送到对方TCP层,一般说来会因为找不到合适的TCP套接口接收该数据包而被丢弃。不过该包可以在目标主机的原始套接口上接收到。5)正如前面所述,任何时候,IP头的校验和都是由内核来设置的。6)内核任何时候那会都不会对IP包以后的字段进行校验和验证。如,即使我们指定第三个参数protocol为IPPROTO_TCP,在数据发送时内核也不会对进行TCP校验和计算和验证。7)如果IP_HDRINCL选项已设置,按照常规,我们应该组建自己的IP头,
但是即使我们没有组建IP头,用sendto或sendmsg并指定目的IP地址来发送数据是照样可以完成的。但是这样的数据包在目标机上用原始套接口是接收不到的,因为在ip_rcv()中要对IP头进行验证,并且要分析校验和,
所以该包会被丢弃,不过在链路层应该能够接收到该数据包。8)如果设置了IP_HDRINCL选项,并且数据包超长,那么数据会被丢弃,并会返回出错码EMSGSIZE。如果未设置IP_HDRINCL选项,并且数据包超长,那么数据包会被分片。输入的问题1)原始套接口可以接收到任何TCP或UDP报文。2)要想接收到原始套接口,首先要接收的数据包必须有一个完整的、正确的IP头,否则不能通过ip_rcv()中的包头检查和检验和验证。3)在原始套接口接收的数据包过程中,内核会对接收的IP包进行校验和验证,但不会对IP包以后的任何字段进行检测和验证。如,我们创建原始套接口时,所指定的protocol参数为IPPROTO_TCP,内核也不会进行TCP校验和验证,而是直接把IP头中协议字段为TCP的所有数据包都复制一份,提交给该原始套接口。4)用原始套接口接收到的TCP包都是进行了IP重组以后,TCP排序以前的报文。5)如果在创建原始套接口时,所指定的protocol参数不为零,(socket的第三个参数),则接收到的数据报的协议字段应该与之匹配。否则该数据报不传递给该套接口。6)如果此原始套接口上绑定了一个本地IP地址,那么接收到的数据报的目的IP地址应该与该绑定的IP地址相匹配,否则该数据包将不传递到该套接口。7)如果此原始套接口通过connect指定了一个对方IP地址,那么接收到的数据包的源IP地址应与该以连接地址相匹配,否则该数据包不传递给该套接口。8)如果一个原始套接口以protocol参数为0的方式创建,并且未调用connect或bind,那么对于内核传递给原始套接口的每一个原始数据报,该套接口都会收到一份拷贝。9)原始套接口接收不到任何的ARP或RARP协议类型的套接口,因为net_rx_action()会把ARP或RARP协议类型的数据包传递给ARP的接收函数类处理,不会传递给IP层的接收函数ip_rcv()。10)原始套接口并不是可以接收到任何的ICMP类型的数据包,因为有些ICMP类型的数据包在传递给原始套接口之前已经被系统所响应,并不再向上层传递。11)如果对方的数据包分片了,由于原始套接口的接收是在IP上层,所以会接收到重组以后的原始IP包。

时间: 2024-08-04 10:22:47

分析数据在协议栈底层的流程的相关文章

Excel表格中使用单变量求解分析数据的方法

  Excel表格中使用单变量求解分析数据的方法           1.打开工作簿,创建工作表,并在工作表中输入数据,同时在B10单元格中输入公式"=SUM(B3:B9)"计算费用总和,如图1所示. 图1 创建工作表 2.在"数据"选项卡的"数据工具"组中单击"模拟分析"按钮,在打开的下拉列表中选择"单变量求解"选项,如图2所示. 图2 选择"单变量求解"选项 3.打开"单变

用SSM框架注册用户并把用户数据存放到数据库的流程

问题描述 用SSM框架注册用户并把用户数据存放到数据库的流程 用SSM框架注册用户并把用户数据存放到数据库的流程 ,不是特别了解层与层之间的关系 , 求大神讲解. 解决方案 http://download.csdn.net/download/u011503113/6256373

《流量的秘密: Google Analytics网站分析与优化技巧(第3版)》一2.3 了解网站分析数据的准确性

2.3 了解网站分析数据的准确性 流量的秘密: Google Analytics网站分析与优化技巧(第3版) 网站分析对于衡量网站的表现相当重要.但是,只有在避免一些常见的数据收集错误(特别是对不同来源的数据进行比较时)的前提下,这些网站分析数据才是准确可靠的. 遗憾的是,相当多的企业只看到了网站分析报告的表面价值.毕竟,这些数据得来不易.但是,现实总是残酷的,网站分析的数据从来都不是百分之百的准确,就算是估计误差范围也很困难. 那么,关键是什么呢? 尽管有各种各样的缺陷,但误差在每周或每个月的

WEB容器调用spark程序分析数据

问题描述 WEB容器调用spark程序分析数据 请教各位大神,如何在web容器中(如tomcat)来调用spark程序啊,网上看到的都是自己把程序生成jar包,然后调用spark-submit来执行!! 解决方案 楼主的问题是怎么解决的,我们现在也要这么做,能不能给点指导,小弟在这先谢谢了 解决方案二: 可以参考 一下 这个 http://www.aboutyun.com/thread-11014-1-1.html

socket传json数据包要怎么个流程

问题描述 socket传json数据包要怎么个流程 希望完成的结果是服务器上的输入数据可以可选择的显示在客户端的相应的显示窗口上.但是现在问题是socket传的是二进制数据,我前边打个包还要转成二进制,要用什么函数.socket的输入格式是怎么样的 解决方案 用的是什么语言啊?java的话直接toString().getBytes()就可以; 解决方案二: 用的什么语言,如果是java的话,用getbytes转换为字节数组 解决方案三: 应该是Java吧,建议用Gson或者Jackson

《流量的秘密 Google Analytics网站分析与商业实战》一第1章 衡量成功的前期准备1.1 网站分析数据的价值

第1章 衡量成功的前期准备 流量的秘密 Google Analytics网站分析与商业实战 你应该知道监测对成功来说是至关重要的,不管是对个人,还是对公司,亦或是对职业发展,都是如此.从很多方面来说,Google Analytics只是一个工具,同用来帮助公司作决策的众多其他数字工具没什么区别. 但是网站分析--即Google Analytics所涉及的领域.技术和行业--有所不同,它的功能和潜力是其他工具都望尘莫及的.为什么?因为Google Analytics不仅可以评估用户参与度.交易量和

需要开发一个可以分析数据生成图表的软件,请问应该用什么平台和语言开发好?

问题描述 需要开发一个可以分析数据生成图表的软件,请问应该用什么平台和语言开发好? 请问应该用什么平台和语言开发好?需要看看哪些东西学习一下.希望大神指点一下,菜鸟一只,想好好学学.就是spss那种的软件,当然比那个要简单很多了. 解决方案 可以用C C++ Java Delphi PowerBuilder VB C# 等语言开发. 解决方案二: 用linq sql分析数据,用mschart devexpress做图,用gridview做表. 解决方案三: C#开发用devexpress控件可以

《Python机器学习实践指南》——1.1 数据科学/机器学习的工作流程

1.1 数据科学/机器学习的工作流程 打造机器学习的应用程序,与标准的工程范例在许多方面都是类似的,不过有一个非常重要的方法有所不同:需要将数据作为原材料来处理.数据项目成功与否,很大程度上依赖于你所获数据的质量,以及它是如何被处理的.由于数据的使用属于数据科学的领域,理解数据科学的工作流程对于我们也有所帮助:整个过程要按照图1-1中的顺序,完成六个步骤:获取,检查和探索,清理和准备,建模,评估和最后的部署. 在这个过程中,还经常需要绕回到之前的步骤,例如检查和准备数据,或者是评估和建模,但图1

分析数据的算法问题-求教一个分析数据的算法

问题描述 求教一个分析数据的算法 销售数据若干行在excel sheet中,我要把相加和为0的分为一组,其余的放在另外一组. 例如,以下sheet中,1,2,8,12应该放进和为0相互消掉的那一组.5和13行和为0,也应该在这一组.余下的都消不掉,放在第二组. 真实的报表数据特多,算法叫我头疼,求教大神给我出出主意吧!什么算法比较好,谢谢!!!跪谢!!! 8000 -2000 1000 2000 -3000 4000 -300 -2000 7000 6000 5000 -4000 3000 14