当前位置:  技术问答>linux和unix

请问netfilter中tcp报文的信息什么时候更新?

    来源: 互联网  发布时间:2016-07-06

    本文导语:  本帖最后由 wryse 于 2009-09-30 17:19:46 编辑 我在NF_INET_PRE_ROUTING这个点挂载的钩子函数,现在能正常获取到数据包,检查内容时已经能判断出当前包是一个tcp报文了,但发现sk_buff结构中的transport_header的值和network_header的...

本帖最后由 wryse 于 2009-09-30 17:19:46 编辑
我在NF_INET_PRE_ROUTING这个点挂载的钩子函数,现在能正常获取到数据包,检查内容时已经能判断出当前包是一个tcp报文了,但发现sk_buff结构中的transport_header的值和network_header的值完全相等,导致我无法取到tcp报文的报头。请问一下这是因为系统还没有将tcp报文的信息更新到sk_buff结构里去的原因么?如果是,那么怎样做才能正确取到报头,直接按ip包头算偏移量读取数据么?有没有更好的做法?如果不是,那么会是什么原因?
以下是部分关键代码,盼高手指点:

struct sk_buff *msg_buf = skb;
struct iphdr *msg_ip_hdr = NULL;
struct tcphdr *msg_tcp_hdr = NULL;

msg_ip_hdr = ip_hdr(msg_buf); //ip包包头,这个应该不会有问题,毕竟源目的地址以及长度都正确取到了

......  //一些取ip,长度的计算,结果正确,过程中没有改变过msg_buf和msg_ip_hdr的值

msg_tcp_hdr = tcp_hdr(msg_buf); //这个值取出来和前一段的值完全一样
//msg_tcp_hdr = (struct tcphdr*)(msg_buf->network_header); //直接取,虽然知道没有区别但还是试了一下,同样的结果
msg_tcp_hdr_len = tcp_hdrlen(msg_buf); //前面两个值一样,导致这行悲剧了
//msg_tcp_hdr_len = msg_tcp_hdr->doff * 4;
msg_tcp_payload_len = msg_ip_tot_len - msg_ip_hdr_len; //这里通过ip包的内容算出来的是正确的长度

//输出,第一行的输出里四个值全都相等,囧rz
printk("SKB transport header: %x, network header: %x, IP header pointer: %x, TCP header pointer: %xn",
msg_buf->transport_header, msg_buf->network_header,
msg_ip_hdr, msg_tcp_hdr);
printk("TCP total payload len: %u, TCP header len: %u. TCP res1: %un",
msg_tcp_payload_len, msg_tcp_hdr_len, msg_tcp_hdr->res1);

|
看了下kernel的code,当网卡收到数据包后交由softirq处理,在netif_receive_skb函数中同时调用了skb_reset_network_header和skb_reset_transport_header,这时network_header就和transport_header相等。

在netfilter的NF_INET_LOCAL_IN hook点之后会调用ip_local_deliver_finish, 在ip_local_deliver_finish中会重新计算transport_header。也就是在数据包从第三层(IP)进入第四层(TCP)之前设置transport_header。


static int ip_local_deliver_finish(struct sk_buff* skb)
{
    struct net *net = dev_net(skb->dev);
    __skb_pull(skb, ip_hdrlen(skb));

    /* Point into the IP datagram, just pass the header. */
    skb_reset_transport_header(skb);
    ....
    ....
}

    
 
 
 
本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • 请问什么时候发出SIGKILL信号、什么时候发出SIGTERM信号?
  • 我在安装 redhat7.2 的时候选择了 GNOME 界面,启动的时候选择了图形界面启动,请问如何更改?
  • 请问一下,安装FreeBSD的时候,我得键盘和鼠标都是USB口的,安装的时候找不到鼠标和键盘怎么办?急!
  • 为什么我在安装linux9的时候,机器检测硬件的时候,出现“超频保护中”的提示,请问大家有没有碰到此类的问题
  • 请问vsftp那里下载东西的时候提示输入密码怎样去掉呢?
  • 请问大家在写jsp代码的时候都用什么编辑器?
  • 用Redhat7.3下的Emacs,请问调试编译程序的时候,有无象msdn那样的东西
  • 请问redhat9.0下怎么在启动的时候直接进入命令行方式而不是xwindows方式??谢谢
  • 请问red hat 多少版本以后安装的时候可以装在硬盘任何柱面以后的分区。
  • 请问linux下半部处理的时候是否要关闭中断?
  • 各位,请问一下,我在第一运行JBuilder的时候,它向我要Login name和password应该怎么输入呀?
  • 在solaris8上编译完成,链接的时候出现Segmentation Fault - core dumped这种错误,请问是什么原因??
  • 请问怎么可以让程序出segment fault 的时候自动生成core 文件? 以前习惯调试core ,现在怎么突然没有了?
  • 请问读/proc/stat的时候是否需要加锁,会不会读到错误数据?
  • 我装LINUX时候选择开机字符界面,而且进入KDE,请问怎么进入GNOME呀
  • 请问:我要链接哪些oci库才能成功编译.我在编译的时候老是报OCI函数找不到.
  • 请问在linux中,怎样在进入目录的时候自动设定一些环境变量?
  • 请问linux里面的六个虚拟终端是不是在安装的时候就默认有的
  • 请问怎么样才能在两个组串比较的时候对大小写不敏感呢?
  • 请问:使用RMI的时候如何知道对方的IP地址?(zhoudan)
  • 请问:我知道路由器的telnet密码,但忘记了enable 密码,请问如何是好?
  • 请问那里有SYBASE的jbdb 2.0下载;jspsmartupload可以直接将文件上传到数据库,请问如何使用
  • 请问最新的reahat9.0是基于什么核心的?2.4?2.6?请问那里能下载?
  • 请问:请问哪里有关于linux基本操作命令讲解的资料下载,最好是幻灯片格式的.
  • 请问,我试图用#admintool&图形工具命令来安装sun workshop5.0,为什么进入的却是用户管理界面?请问具体该如何在solaris下安装应用软件
  • 请问在Redhat 9里,我从登录就是图形介面,请问如何在图形介面内进入命令行方式呢,谢谢
  • 请问玩过SOLARIS的高手门,在不正常关机后,就不能启动到windows公用桌面了,只能在命令提示模式下了,请问怎么解决这个问题啊?急~!~!
  • 请问:我在redhat下装了bochs-2.2.1-1.rpm,.装了后,想设置一下,但找不到bochsrc.fda.bxrc,请问这个文件在哪个曰录下啊。
  • 请问:在配置Qt时,很多文档都说在.profile,.login里加东西,但是我好像没有发现有这两个文件上,请问这些文件在哪个目录下啊
  • 请问:在GCC里的C程序里的变量的声明是不是只能在前面,而且相同类型的变量的声明只能放在一起?如果不是,请问怎么样可以解决这个问题.
  • 请问各位大虾,小弟今天开始学jsp了,这学期我们有java课,所以已经下载了jdk(好象是1.2),请问我的98环境怎么配置jsp环境呀?我的jdk可以运行.java程序,别的我就不知道了....谢谢!


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3