免费注册 查看新帖 |

Chinaunix

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

关于条件锁和互斥锁的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-09-09 13:29 |只看该作者 |倒序浏览
本帖最后由 syoubin_sai 于 2011-09-09 14:35 编辑

用多线程实现了一个ping程序。但发送线程与接收线程根本启动不了,怎么回事?
条件锁和互斥锁用的不对?请高手指点。

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <strings.h>
  5. #include <errno.h>
  6. #include <sys/socket.h>
  7. #include <sys/time.h>
  8. #include <signal.h>
  9. #include <netinet/in.h>
  10. #include <netinet/ip.h>
  11. #include <netinet/ip_icmp.h>
  12. #include <netdb.h>
  13. #include <pthread.h>
  14. #include <sys/types.h>

  15. #define PACKET_SIZE 4096
  16. #define RECV_TIME_OUT 10

  17. char send_pkt[PACKET_SIZE];
  18. char recv_pkt[PACKET_SIZE];

  19. int n_send =0, n_recv = 0;
  20. int sock_fd;
  21. pid_t pid;
  22. struct sockaddr_in dest_addr, recv_addr;
  23. struct timeval tv_recv;
  24. int timer = 0;
  25. int send_tag = 0;
  26. int recv_tag = 1;
  27. pthread_mutex_t ping_lock = PTHREAD_MUTEX_INITIALIZER;
  28. pthread_cond_t send_cond = PTHREAD_COND_INITIALIZER;
  29. pthread_cond_t recv_cond = PTHREAD_COND_INITIALIZER;

  30. void statistics(int sig_no);
  31. unsigned short cal_cksum(unsigned short *addr, int len);
  32. int pkt(int pkt_no);
  33. void *recv_pkt_th(void *arg);
  34. void *send_pkt_th(void *arg);
  35. int unpkt(char *buf, int len);
  36. void tv_sub(struct timeval *out, struct timeval *in);

  37. void statistics(int signo)
  38. {
  39.         printf("\n------------PING statistics-----------\n");
  40.         printf("%d packets transmitted, %d received, %%%d lost\n",
  41.                  n_send, n_recv, (n_send - n_recv) / n_send * 100 );
  42.         pthread_mutex_unlock(&ping_lock);
  43.         pthread_cond_destroy(&send_cond);
  44.         pthread_cond_destroy(&recv_cond);
  45.         close(sock_fd);
  46.         exit(1);
  47. }

  48. unsigned short cal_cksum(unsigned short *addr, int len)
  49. {
  50.         int nleft = len;
  51.         int sum = 0;
  52.         unsigned short *w = addr;
  53.         unsigned short cksum = 0;
  54.        
  55.         while(nleft > 1)
  56.         {
  57.                 sum += *w++;
  58.                 nleft -= 2;
  59.         }
  60.         if(nleft == 1)
  61.         {
  62.                 *(unsigned char *)(&cksum) = *(unsigned char *)w;
  63.                 sum += cksum;
  64.         }
  65.         sum = (sum >> 16) + (sum & 0xffff);
  66.         sum += (sum >> 16);
  67.         cksum = ~sum;

  68.         return cksum;
  69. }

  70. int pkt(int pkt_no)
  71. {
  72.         int i, pkt_sz;
  73.         struct icmp *icmp;
  74.         struct timeval *tval;
  75.        
  76.         icmp = (struct icmp *)send_pkt;
  77.         icmp->icmp_type = ICMP_ECHO;
  78.         icmp->icmp_code = 0;
  79.         icmp->icmp_cksum = 0;
  80.         icmp->icmp_seq = pkt_no;
  81.         icmp->icmp_id = pid;
  82.         pkt_sz = 64;
  83.         tval = (struct timeval *)icmp->icmp_data;
  84.         gettimeofday(tval, NULL);
  85.         icmp->icmp_cksum = cal_cksum((unsigned short *)icmp, pkt_sz);

  86.         return pkt_sz;
  87. }

  88. void *send_pkt_th(void *arg)
  89. {
  90.         int pkt_sz;
  91. printf("D");
  92.         while(1){
  93.                 pthread_mutex_lock(&ping_lock);
  94.                 n_send ++;
  95.                 pkt_sz = pkt(n_send);
  96.                 if(sendto(sock_fd, send_pkt, pkt_sz, 0, (struct sockaddr *)&dest_addr,
  97.                      sizeof(dest_addr)) == -1){
  98.                         perror("sendto error");
  99.                         goto L_END;
  100.                 }
  101. //                sleep(1);
  102.                 recv_tag = 0;
  103.                 pthread_cond_signal(&recv_cond);
  104. printf("C");
  105.                 while(send_tag == 0){
  106.                         pthread_cond_wait(&send_cond, &ping_lock);
  107.                 }
  108.                 send_tag = 0;
  109.                 L_END:
  110.                         pthread_mutex_unlock(&ping_lock);
  111.         }

  112.         return ((void *)0);
  113. }

  114. void *recv_pkt_th(void *arg)
  115. {
  116.         int n, fromlen;
  117.         fromlen = sizeof(recv_addr);
  118. printf("A");
  119.         while(1)
  120.         {
  121.                 pthread_mutex_lock(&ping_lock);
  122. printf("B");
  123.                 while(recv_tag == 1){
  124.                         pthread_cond_wait(&recv_cond, &ping_lock);
  125.                 }
  126.                 recv_tag = 1;
  127.                 if(n = recvfrom(sock_fd, recv_pkt, sizeof(recv_pkt), 0,
  128.                      (struct sockaddr *)&recv_addr, &fromlen) == -1){
  129.                         if(errno == EINTR){
  130.                                 goto L_END;
  131.                         }
  132.                         perror("recvfrom error");
  133.                         goto L_END;
  134.                 }
  135.                 gettimeofday(&tv_recv, NULL);
  136.                 if(unpkt(recv_pkt, n) == -1){
  137.                         goto L_END;
  138.                 }
  139.                 n_recv ++;
  140.                 if(timer == RECV_TIME_OUT){
  141.                         send_tag = 1;
  142.                         timer = 0;
  143.                         printf("From %s icmp_seq=%d Destination Host Unreachable",
  144.                                 inet_ntoa(dest_addr.sin_addr), n_send);
  145.                 }
  146.                 if(send_tag == 1){
  147.                         pthread_cond_signal(&send_cond);
  148.                 }
  149.                 L_END:
  150.                         timer ++;
  151.                         pthread_mutex_unlock(&ping_lock);
  152.         }

  153.         return ((void *)0);
  154. }

  155. int unpkt(char *buf, int len)
  156. {
  157.         int i, iphdrlen;
  158.         struct ip *ip;
  159.         struct icmp *icmp;
  160.         struct timeval *tv_send;
  161.         double rtt;

  162.         ip = (struct ip *)buf;
  163.         iphdrlen = ip->ip_hl << 2;
  164.         icmp = (struct icmp *)(buf + iphdrlen);
  165.         len -= iphdrlen;

  166.         if(len < 8){
  167.                 return -1;
  168.         }
  169.         if((icmp->icmp_type == ICMP_ECHOREPLY)
  170.            && (icmp->icmp_id == pid)) {
  171.                 tv_send = (struct timeval *)icmp->icmp_data;
  172.                 tv_sub(&tv_recv, tv_send);
  173.                 rtt = tv_recv.tv_sec * 1000 + tv_recv.tv_usec / 1000;
  174.                 printf("%d byte from %s:icmp_seq = %u ttl = %d rtt = %.3f ms\n",
  175.                 len, inet_ntoa(dest_addr.sin_addr), icmp->icmp_seq, ip->ip_ttl, rtt);
  176.                 send_tag = 1;
  177.         } else {
  178.                 return -1;
  179.         }
  180. }

  181. void tv_sub(struct timeval *out, struct timeval *in)
  182. {
  183.         if(out->tv_usec -= in->tv_usec){
  184.                 --out->tv_sec;
  185.                 out->tv_usec += 1000000;
  186.         }
  187.         out->tv_sec -= in->tv_sec;
  188. }

  189. int main(int argc, char *argv[ ])
  190. {
  191.         struct hostent *host;
  192.         struct protoent *protocol;
  193.         unsigned int inaddr = 0l;
  194.         int waittime = 5;
  195.         int size = 50 * 1024;
  196.         pthread_t thread_id;
  197.         pthread_attr_t attr;
  198.         int retval;

  199.         if(argc < 2) {
  200.                 printf("usage:%s hostname/IP address\n", argv[0]);
  201.                 exit(1);
  202.         }
  203.         if((protocol = getprotobyname("icmp")) == NULL ){
  204.                 perror("getprotobyname");
  205.                 exit(1);
  206.         }
  207.         if((sock_fd = socket(AF_INET, SOCK_RAW, protocol->p_proto)) == -1){
  208.                 perror("socket error");
  209.                 exit(1);
  210.         }
  211.         setuid(getuid());
  212.         setsockopt(sock_fd, SOL_SOCKET, SO_RCVBUF, &size, sizeof(size));
  213.         bzero(&dest_addr, sizeof(dest_addr));
  214.         dest_addr.sin_family = AF_INET;
  215.         if((inaddr = inet_addr(argv[1])) == INADDR_NONE){
  216.                 if((host = gethostbyname(argv[1])) == NULL){
  217.                         perror("gethostbyname error");
  218.                         exit(1);
  219.                 }
  220.                 memcpy((char *)&dest_addr.sin_addr, host->h_addr, host->h_length);
  221.         } else{
  222.                 memcpy((char *)&dest_addr, (char *)&inaddr, host->h_length);
  223.         }

  224.         pid = getpid();
  225. printf("x");
  226.         printf("PING %s(%s): %d bytes data in ICMP packets.\n", argv[1],
  227.                 inet_ntoa(dest_addr.sin_addr), 56);
  228. printf("0");
  229.         (void)pthread_attr_init(&attr);
  230.         (void)pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
  231. printf("1");
  232.         if((retval = pthread_create(&thread_id, &attr, send_pkt_th, NULL)) != 0){
  233.                 perror("pthread_create error");
  234.                 exit(1);
  235.         }

  236. printf("2");
  237.         if((retval = pthread_create(&thread_id, &attr, recv_pkt_th, NULL)) != 0){
  238.                 perror("pthread_create error");
  239.                 exit(1);
  240.         }
  241.           signal(SIGTERM, statistics);
  242.         signal(SIGINT, statistics);
  243.         sleep(10);

  244. printf("3");
  245.         return 0;
  246. }
复制代码

论坛徽章:
0
2 [报告]
发表于 2011-09-09 16:50 |只看该作者
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP