免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 7007 | 回复: 12
打印 上一主题 下一主题

[内核模块] [求助]skb_copy_expand扩展报文头 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-11-29 19:21 |只看该作者 |倒序浏览
本帖最后由 newand 于 2012-11-29 19:34 编辑

现在程序中使用skb_copy_expand步骤是这样的
1. 调用skb_copy_expand多增加一块内容,返回new_skb。
2. 使用skb_put将tail指针下移,增加skb的len。
3. 用memcpy将skb中前40个字节的IP header复制到new_skb中。
4. 用memcpy将skb中40字节往后的内容复制到new_skb的第40+extend header地址处。
5. 设置extend header的值。

但是运行的结果是,每次抓包,报文的长度不会改变,只是因为第5步改变的内容导致报文出错。
  1. /*
  2. * this programe is working as plugin of netfilter which will pin a timestamp to
  3. * the destination header of ipv6 packet every x seconds when the packet is
  4. * the object multicast packet.
  5. */


  6. #include <linux/netfilter.h>
  7. #include <linux/netfilter_ipv6.h>
  8. #include <linux/netfilter_ipv6/ip6_tables.h>
  9. #include <linux/module.h>
  10. #include <linux/moduleparam.h>
  11. #include <linux/kernel.h>
  12. #include <linux/inet.h>
  13. #include <linux/ip.h>
  14. #include <linux/udp.h>
  15. #include <net/checksum.h>
  16. #include <net/udp.h>
  17. #include <net/ipv6.h>
  18. #include <linux/time.h>

  19. #include <asm/byteorder.h>

  20. MODULE_LICENSE("GPL");
  21. MODULE_AUTHOR("qj");
  22. MODULE_DESCRIPTION("Change the header of ipv6 packet");

  23. #define PRINT(fmt,args...) printk("INFO: " fmt, ##args)


  24. /* IP6 Hooks */
  25. /* After promisc drops#include <asm/byteorder.h>
  26. , checksum checks. */
  27. #define NF_IP6_PRE_ROUTING  0
  28. /* If the packet is destined for this box. */
  29. #define NF_IP6_LOCAL_IN     1
  30. /* If the packet is destined for another interface. */
  31. #define NF_IP6_FORWARD      2
  32. /* Packets coming from a local process. */
  33. #define NF_IP6_LOCAL_OUT        3
  34. /* Packets about to hit the wire. */
  35. #define NF_IP6_POST_ROUTING 4


  36. /*sequence of measeure#include <asm/byteorder.h>
  37. sample packet*/
  38. static int sample_seq = 0;

  39. /*Next header destination header*/
  40. struct ip6_dst_hdr
  41. {
  42.                 uint8_t ip6d_nxt;
  43.                 uint8_t ip6d_len;
  44.                 uint8_t ip6d_opt_type;
  45.                 uint8_t ip6d_opt_len;
  46.                 uint32_t ip6d_ssn;
  47.                 uint32_t ip6d_sec;
  48.                 uint32_t ip6d_usec;
  49. };


  50. inline
  51. void print_6addr(const struct in6_addr *addr)
  52. {
  53.     PRINT("%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n",
  54.                  (int)addr->s6_addr[0], (int)addr->s6_addr[1],
  55.                  (int)addr->s6_addr[2], (int)addr->s6_addr[3],
  56.                  (int)addr->s6_addr[4], (int)addr->s6_addr[5],
  57.                  (int)addr->s6_addr[6], (int)addr->s6_addr[7],
  58.                  (int)addr->s6_addr[8], (int)addr->s6_addr[9],
  59.                  (int)addr->s6_addr[10], (int)addr->s6_addr[11],
  60.                  (int)addr->s6_addr[12], (int)addr->s6_addr[13],
  61.                  (int)addr->s6_addr[14], (int)addr->s6_addr[15]);
  62. }
  63. char *in_ntoa(__u32 in)
  64.   {
  65.           static char buff[18];
  66.           char *p;
  67.   
  68.           p = (char *) &in;
  69.           sprintf(buff, "%d.%d.%d.%d",
  70.                   (p[0] & 255), (p[1] & 255), (p[2] & 255), (p[3] & 255));
  71.           return(buff);
  72.   }

  73. struct sk_buff *
  74. ip6_reconstruct_pkt(struct sk_buff *skb)
  75. {
  76.         struct sk_buff *new_skb, *temp_skb;
  77.         struct ipv6hdr * ip6_hdr = ipv6_hdr(skb);
  78.         struct ip6_dst_hdr *ip6_dst;
  79.         struct timeval tv;//get time

  80.         ip6_hdr->nexthdr = 0x60;

  81.         //copy from the *skb to new_skb
  82.         new_skb = skb_copy_expand(skb, skb_headroom(skb),
  83.                                 skb_tailroom(skb) + sizeof(struct ip6_dst_hdr),
  84.                                 GFP_ATOMIC);

  85.         if(new_skb == NULL)
  86.         {
  87.                 PRINT("Allocate new sk_buffer error!\n");
  88.                 return NULL;
  89.         }

  90.         //set the new_skb
  91.         if(skb->sk != NULL)
  92.                 skb_set_owner_w(new_skb, skb->sk);

  93.         //move tail to tail + sizeof(ip6_dst_hdr)
  94.         skb_put(new_skb,sizeof(struct ip6_dst_hdr));
  95.   
  96.         //insert a ip6_dst_hdr between the memory
  97.         memcpy (new_skb->data, skb->data, 40);//the ip header has 40 bytes
  98.              memcpy (new_skb->data + 40 + sizeof (struct ip6_dst_hdr),
  99.                                   skb->data + 40, skb->len - 40);

  100.         skb = new_skb;//skb pointer to new_skb

  101.             //turn the space to ip6_dst struct
  102.         ip6_dst = (struct ip6_dst_hdr *)(new_skb->data + 40);
  103.         memset(ip6_dst,0,sizeof(struct ip6_dst_hdr));
  104.         //add ipv6 destination header
  105.         ip6_dst->ip6d_nxt = htonl(0x17);
  106.         ip6_dst->ip6d_len = htonl(0x15);
  107.         ip6_dst->ip6d_opt_type = htonl(0x00);// type of option
  108.         ip6_dst->ip6d_opt_len = htonl(0x0C);
  109.        
  110.         do_gettimeofday(&tv);
  111.              ip6_dst->ip6d_sec = htonl(tv.tv_sec);
  112.         ip6_dst->ip6d_usec = htonl(tv.tv_usec);
  113.         sample_seq++;
  114.         if(sample_seq==0xffffffff)//if overflow ,reset
  115.         {
  116.            sample_seq=0;
  117.         }
  118.         ip6_dst->ip6d_ssn = htonl(sample_seq);

  119.         return skb;
  120. }

  121. /*Hook which will call the encapsulate func when condition satisfied
  122. *condition1: IPv6 Multicast packets
  123. *condition2: every 20ms
  124. */
  125. static unsigned int
  126. ip6_multi_modify(unsigned int hooknum,
  127.                                 struct sk_buff *skb,
  128.                                 const struct net_device *in,
  129.                                 const struct net_device *out,
  130.                                 int (*okfn)(struct sk_buff*))
  131. {
  132.     struct sk_buff *sk = skb;
  133.         struct ipv6hdr *ip6_hdr = ipv6_hdr(skb);//header

  134.         //because nh only supportted in kernels below 2.6
  135.         //after 2.6, it often use network_header to express nh
  136.         //struct ipv6hdr *ip6_hdr = (struct ipv6hdr*)skb->nh.ipv6h;
  137.         if(ip6_hdr->version == 6)
  138.         {
  139.                 struct in6_addr destip = ip6_hdr->daddr;//destination ip
  140.                 //TODO:use module_para or /proc to replace here
  141.                 if(destip.s6_addr[0] == 0xff && destip.s6_addr[1] == 0x15)

  142.                 {
  143.                         if(skb_tailroom(sk) >= 40)
  144.                         {
  145.                                 PRINT("tailroom is enough\n");
  146.                         }
  147.                         else
  148.                         {
  149.                                 PRINT("not enough\n");
  150.                         }
  151.                    skb = ip6_reconstruct_pkt(skb);
  152.                         if(skb == NULL)
  153.                         {
  154.                                 return NF_STOLEN;
  155.                         }
  156.                         ip6_hdr= ipv6_hdr(skb);

  157.                         if(!skb)
  158.                                 return NF_STOLEN;
  159.                 }
  160.         }
  161.         return NF_ACCEPT;
  162. }


  163. /*Initialize the hook*/
  164. static struct nf_hook_ops nf_out_modify =
  165. {
  166.         .hook = ip6_multi_modify,
  167.         .hooknum = NF_IP6_PRE_ROUTING,//Check all the forwarded packets
  168.         .pf = PF_INET6,
  169.         .priority = NF_IP6_PRI_FIRST,
  170. };

  171. /*Initialize the module*/
  172. static int __init ip6_multi_init(void)
  173. {
  174.         int ret;
  175.         ret = nf_register_hook(&nf_out_modify);
  176.         PRINT("IPV6 multicast packet modify module init.\n");
  177.         return 0; //success
  178. }

  179. /*Clear the module*/
  180. static void __exit ip6_multi_exit(void)
  181. {
  182.         nf_unregister_hook(&nf_out_modify);
  183.         PRINT("IPV6 multicast packet modify module exit.\n");
  184. }

  185. module_init(ip6_multi_init);
  186. module_exit(ip6_multi_exit);
复制代码
实在没找出问题在哪,请各位大牛帮忙看看。

论坛徽章:
0
2 [报告]
发表于 2012-11-30 14:07 |只看该作者
没有拿代码做实验,至少有一个潜在的问题。
skb_linear.

论坛徽章:
0
3 [报告]
发表于 2012-11-30 14:11 |只看该作者
回复 1# newand


    还有个原有的skb没有释放的问题。

论坛徽章:
0
4 [报告]
发表于 2012-11-30 16:11 |只看该作者
本帖最后由 newand 于 2012-11-30 16:30 编辑

回复 2# oscarvei


    你好,刚接触内核网络编程,大部分都是看网上的示例或者去github上找代码。比较少用skb_linear,查了skb_linearize的作用是- convert paged skb to linear one。(我不是特别明白,请问有没有什么书或者资料推荐吗?)

另外关于释放skb的问题
我之前用kfree来在 skb = new_skb;之前释放skb,不过会导致死机的问题。
是不是还需要其他操作?

还有一个问题就是,抓包时发现通过skb_copy_expand新产生报文是没有的,还需要调用什么函数,将它返回的skb加入到队列吗?

论坛徽章:
0
5 [报告]
发表于 2012-12-01 10:38 |只看该作者
kfree不能释放skb    skb释放用__kfree_skb

论坛徽章:
0
6 [报告]
发表于 2012-12-01 13:47 |只看该作者
一些操作的函数 可以参考内核代码

论坛徽章:
0
7 [报告]
发表于 2012-12-01 13:53 |只看该作者
回复 6# SCDXMOE


    请问,我扩展了报文报头的长度,用skb_put将tail指针下移,然后用memcpy将数据拷贝。除了这些操作还应该有其他能够设置报文长度的操作吗?
因为我咋client抓到的报文,长度还是没有更改之前的长度。

论坛徽章:
0
8 [报告]
发表于 2012-12-01 14:37 |只看该作者
你要确定 你用了skb_put之后 要相应的更改skb->len

论坛徽章:
0
9 [报告]
发表于 2012-12-01 15:03 |只看该作者
本帖最后由 newand 于 2012-12-01 15:10 编辑

回复 8# SCDXMOE


    skb_put的函数中会对skb->len做增加操作。
  1. unsigned char *skb_put(struct sk_buff *skb, unsigned int len)
  2. {
  3.     unsigned char *tmp = skb_tail_pointer(skb);
  4.     SKB_LINEAR_ASSERT(skb);
  5.     skb->tail += len;
  6.     skb->len  += len;
  7.     if (unlikely(skb->tail > skb->end))
  8.         skb_over_panic(skb, len, __builtin_return_address(0));
  9.     return tmp;
  10. }
复制代码
skb->len的大小是随着skb->put的调用而变化的,
不过skb->truesize没变,但是我手动设置了skb->truesize之后,在client抓到的报文,长度不因为增加了ip首部而变化。

论坛徽章:
0
10 [报告]
发表于 2012-12-01 16:17 |只看该作者
skb->truesize =len+sizeof(skbuff)是随着len的长度变化er变化的 怎么可能是没有变化呢
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP