- 论坛徽章:
- 0
|
本帖最后由 kangle000 于 2010-07-19 06:59 编辑
以下代码是在2.6.31内核中执行的,根据send_rst改写的。但是代码存在以下几个问题:
1、捕包发现ip头的version字段为0,offset的值不为0,校验出错。
2、udp头的内容好像并没有更改。
3、us指针指向的内容并没有改变
4、经常死机
请大家帮我看看是哪里的问题- #ifndef __KERNEL__
- #define __KERNEL__
- #endif
- #ifndef MODULE
- #define MODULE
- #endif
- #include <linux/module.h>
- #include <linux/kernel.h>
- #include <linux/skbuff.h>
- #include <linux/netfilter.h>
- #include <linux/netfilter_ipv4.h>
- #include <linux/ip.h>
- #include <linux/if_ether.h>
- #include <linux/netdevice.h>
- #include <linux/if_packet.h>
- #include <net/tcp.h>
- #include <net/udp.h>
- #include <asm/string.h>
- #define ETH "eth0"
- #define ETH_HLEN 14
- #define UDP_HLEN 8
- #define loc_swap(a,b) {(a)=(a)^(b);(b)=(a)^(b);(a)=(a)^(b);}
- #define DNSPORT 53
- #define MAC_FMT "%02x:%02x:%02x:%02x:%02x:%02x"
- #define MAC_ARG(x) ((u8*)(x))[0],((u8*)(x))[1],((u8*)(x))[2],((u8*)(x))[3],((u8*)(x))[4],((u8*)(x))[5]
- #define FLAGS 0x8003
- typedef struct _rfc1035_header rfc1035_header;
- struct _rfc1035_header {
- unsigned short id;
- unsigned int qr:1;
- unsigned int opcode:4;
- unsigned int aa:1;
- unsigned int tc:1;
- unsigned int rd:1;
- unsigned int ra:1;
- unsigned int rcode:4;
- unsigned short qdcount;
- unsigned short ancount;
- unsigned short nscount;
- unsigned short arcount;
- };
- unsigned int modify_dns(unsigned int hooknum,
- struct sk_buff *skb,
- const struct net_device *in,
- const struct net_device *out,
- int (*okfn)(struct sk_buff *))
- {
- struct net_device *dev;
- struct iphdr *iph = NULL;
- struct ethhdr *eth = NULL;
- struct udphdr *udph = NULL;
- unsigned char mac_temp[ETH_ALEN] = {0};
- unsigned int iph_len;
- char *dns_data;
- int data_len;
- rfc1035_header *qh;
- unsigned short *us;
- printk("before dev = in\n");
- dev = dev_get_by_name(&init_net, "eth0");
- if(NULL == dev)
- goto out;
- if(NULL == skb)
- return NF_ACCEPT;
- printk("before decide packet type\n");
- if((IPPROTO_UDP != (ip_hdr(skb)->protocol))
- && (DNSPORT != udp_hdr(skb)->dest))
- return NF_ACCEPT;
-
- iph = (struct iphdr *)ip_hdr(skb);
- iph_len = iph->ihl << 2;
- printk("iphdr_len=%d data_len=%d sip=%04x dip=%04x\n", iph_len, ntohs(iph->tot_len), iph->saddr, iph->daddr);
- udph = udp_hdr(skb);
- printk("source=%hu dest=%hu len=%d\n", ntohs(udph->source), ntohs(udph->dest), ntohs(udph->len));
- data_len = ntohs(udph->len) - UDP_HLEN;
- dns_data = (char *)udph + UDP_HLEN;
- printk("before modify dns data\n");
- qh = (rfc1035_header *) dns_data;
- us = (unsigned short *)(dns_data + 2);
- *us = *us | FLAGS;
- printk("flags=%hu\n", *us);
-
- loc_swap(iph->saddr, iph->daddr);
- loc_swap(udph->source, udph->dest);
- iph->ihl = sizeof(struct iphdr)/4;
- iph->tot_len = htons(skb->len);
-
- printk("source=%uh dest=%uh len=%d\n", ntohs(udph->source), ntohs(udph->dest), ntohs(udph->len));
- printk("before udp checksum\n");
- udph->check = 0;
- udph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, ntohs(udph->len), IPPROTO_UDP, 0);
- iph->frag_off = htons("IP_DF");
- iph->id = 0;
- printk("before ip checksum\n");
- iph->check = 0;
- iph->check = ip_fast_csum((unsigned char *)iph, iph_len);
-
- printk("before skb_push\n");
- eth = (struct ethhdr *)skb_push(skb, ETH_HLEN);
- if(NULL == eth)
- goto out;
- if(NULL != eth->h_source)
- printk("SOURCE:" MAC_FMT "\n", MAC_ARG(eth->h_source));
- if(NULL != eth->h_dest)
- printk("DEST:" MAC_FMT "\n", MAC_ARG(eth->h_dest));
- printk("before swap mac adress\n");
- memcpy(mac_temp, (unsigned char *)eth->h_dest, ETH_ALEN);
- memcpy(eth->h_dest, (unsigned char *)eth->h_source, ETH_ALEN);
- memcpy(eth->h_source, mac_temp, ETH_ALEN);
- printk("before send packet\n");
- if (0 > dev_queue_xmit(skb))
- goto out;
- return NF_ACCEPT;
- out:
- printk("in out: ");
- if (NULL != skb)
- {
- printk("skb not null\n");
- dev_put(dev);
- kfree_skb(skb);
- }
- return NF_ACCEPT;
- }
- static struct nf_hook_ops dns_ops[] = {
- {
- .hook = modify_dns,
- .owner = THIS_MODULE,
- .pf = PF_INET,
- .hooknum = NF_IP_PRE_ROUTING,
- .priority = NF_IP_PRE_ROUTING,
- }
- };
- static int __init init(void)
- {
- int ret;
- printk("before nf_register_hooks\n");
- ret = nf_register_hooks(dns_ops, ARRAY_SIZE(dns_ops));
- if(ret < 0)
- {
- printk("can't register dns_ops!\n");
- return ret;
- }
- printk("insmod dns_ops ok!\n");
- return 0;
- }
- static void __exit fini(void)
- {
- nf_unregister_hooks(dns_ops, ARRAY_SIZE(dns_ops));
- printk("rmmod dns_ops ok!\n");
- }
- module_init(init);
- module_exit(fini);
- MODULE_LICENSE("GPL");
- MODULE_AUTHOR("KAL");
- MODULE_DESCRIPTION("MODIFY DNS PACKETS");
复制代码 |
|