- 论坛徽章:
- 0
|
本帖最后由 syoubin_sai 于 2011-09-09 14:35 编辑
用多线程实现了一个ping程序。但发送线程与接收线程根本启动不了,怎么回事?
条件锁和互斥锁用的不对?请高手指点。
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <strings.h>
- #include <errno.h>
- #include <sys/socket.h>
- #include <sys/time.h>
- #include <signal.h>
- #include <netinet/in.h>
- #include <netinet/ip.h>
- #include <netinet/ip_icmp.h>
- #include <netdb.h>
- #include <pthread.h>
- #include <sys/types.h>
- #define PACKET_SIZE 4096
- #define RECV_TIME_OUT 10
- char send_pkt[PACKET_SIZE];
- char recv_pkt[PACKET_SIZE];
- int n_send =0, n_recv = 0;
- int sock_fd;
- pid_t pid;
- struct sockaddr_in dest_addr, recv_addr;
- struct timeval tv_recv;
- int timer = 0;
- int send_tag = 0;
- int recv_tag = 1;
- pthread_mutex_t ping_lock = PTHREAD_MUTEX_INITIALIZER;
- pthread_cond_t send_cond = PTHREAD_COND_INITIALIZER;
- pthread_cond_t recv_cond = PTHREAD_COND_INITIALIZER;
- void statistics(int sig_no);
- unsigned short cal_cksum(unsigned short *addr, int len);
- int pkt(int pkt_no);
- void *recv_pkt_th(void *arg);
- void *send_pkt_th(void *arg);
- int unpkt(char *buf, int len);
- void tv_sub(struct timeval *out, struct timeval *in);
- void statistics(int signo)
- {
- printf("\n------------PING statistics-----------\n");
- printf("%d packets transmitted, %d received, %%%d lost\n",
- n_send, n_recv, (n_send - n_recv) / n_send * 100 );
- pthread_mutex_unlock(&ping_lock);
- pthread_cond_destroy(&send_cond);
- pthread_cond_destroy(&recv_cond);
- close(sock_fd);
- exit(1);
- }
- unsigned short cal_cksum(unsigned short *addr, int len)
- {
- int nleft = len;
- int sum = 0;
- unsigned short *w = addr;
- unsigned short cksum = 0;
-
- while(nleft > 1)
- {
- sum += *w++;
- nleft -= 2;
- }
- if(nleft == 1)
- {
- *(unsigned char *)(&cksum) = *(unsigned char *)w;
- sum += cksum;
- }
- sum = (sum >> 16) + (sum & 0xffff);
- sum += (sum >> 16);
- cksum = ~sum;
- return cksum;
- }
- int pkt(int pkt_no)
- {
- int i, pkt_sz;
- struct icmp *icmp;
- struct timeval *tval;
-
- icmp = (struct icmp *)send_pkt;
- icmp->icmp_type = ICMP_ECHO;
- icmp->icmp_code = 0;
- icmp->icmp_cksum = 0;
- icmp->icmp_seq = pkt_no;
- icmp->icmp_id = pid;
- pkt_sz = 64;
- tval = (struct timeval *)icmp->icmp_data;
- gettimeofday(tval, NULL);
- icmp->icmp_cksum = cal_cksum((unsigned short *)icmp, pkt_sz);
- return pkt_sz;
- }
- void *send_pkt_th(void *arg)
- {
- int pkt_sz;
- printf("D");
- while(1){
- pthread_mutex_lock(&ping_lock);
- n_send ++;
- pkt_sz = pkt(n_send);
- if(sendto(sock_fd, send_pkt, pkt_sz, 0, (struct sockaddr *)&dest_addr,
- sizeof(dest_addr)) == -1){
- perror("sendto error");
- goto L_END;
- }
- // sleep(1);
- recv_tag = 0;
- pthread_cond_signal(&recv_cond);
- printf("C");
- while(send_tag == 0){
- pthread_cond_wait(&send_cond, &ping_lock);
- }
- send_tag = 0;
- L_END:
- pthread_mutex_unlock(&ping_lock);
- }
- return ((void *)0);
- }
- void *recv_pkt_th(void *arg)
- {
- int n, fromlen;
- fromlen = sizeof(recv_addr);
- printf("A");
- while(1)
- {
- pthread_mutex_lock(&ping_lock);
- printf("B");
- while(recv_tag == 1){
- pthread_cond_wait(&recv_cond, &ping_lock);
- }
- recv_tag = 1;
- if(n = recvfrom(sock_fd, recv_pkt, sizeof(recv_pkt), 0,
- (struct sockaddr *)&recv_addr, &fromlen) == -1){
- if(errno == EINTR){
- goto L_END;
- }
- perror("recvfrom error");
- goto L_END;
- }
- gettimeofday(&tv_recv, NULL);
- if(unpkt(recv_pkt, n) == -1){
- goto L_END;
- }
- n_recv ++;
- if(timer == RECV_TIME_OUT){
- send_tag = 1;
- timer = 0;
- printf("From %s icmp_seq=%d Destination Host Unreachable",
- inet_ntoa(dest_addr.sin_addr), n_send);
- }
- if(send_tag == 1){
- pthread_cond_signal(&send_cond);
- }
- L_END:
- timer ++;
- pthread_mutex_unlock(&ping_lock);
- }
- return ((void *)0);
- }
- int unpkt(char *buf, int len)
- {
- int i, iphdrlen;
- struct ip *ip;
- struct icmp *icmp;
- struct timeval *tv_send;
- double rtt;
- ip = (struct ip *)buf;
- iphdrlen = ip->ip_hl << 2;
- icmp = (struct icmp *)(buf + iphdrlen);
- len -= iphdrlen;
- if(len < 8){
- return -1;
- }
- if((icmp->icmp_type == ICMP_ECHOREPLY)
- && (icmp->icmp_id == pid)) {
- tv_send = (struct timeval *)icmp->icmp_data;
- tv_sub(&tv_recv, tv_send);
- rtt = tv_recv.tv_sec * 1000 + tv_recv.tv_usec / 1000;
- printf("%d byte from %s:icmp_seq = %u ttl = %d rtt = %.3f ms\n",
- len, inet_ntoa(dest_addr.sin_addr), icmp->icmp_seq, ip->ip_ttl, rtt);
- send_tag = 1;
- } else {
- return -1;
- }
- }
- void tv_sub(struct timeval *out, struct timeval *in)
- {
- if(out->tv_usec -= in->tv_usec){
- --out->tv_sec;
- out->tv_usec += 1000000;
- }
- out->tv_sec -= in->tv_sec;
- }
- int main(int argc, char *argv[ ])
- {
- struct hostent *host;
- struct protoent *protocol;
- unsigned int inaddr = 0l;
- int waittime = 5;
- int size = 50 * 1024;
- pthread_t thread_id;
- pthread_attr_t attr;
- int retval;
- if(argc < 2) {
- printf("usage:%s hostname/IP address\n", argv[0]);
- exit(1);
- }
- if((protocol = getprotobyname("icmp")) == NULL ){
- perror("getprotobyname");
- exit(1);
- }
- if((sock_fd = socket(AF_INET, SOCK_RAW, protocol->p_proto)) == -1){
- perror("socket error");
- exit(1);
- }
- setuid(getuid());
- setsockopt(sock_fd, SOL_SOCKET, SO_RCVBUF, &size, sizeof(size));
- bzero(&dest_addr, sizeof(dest_addr));
- dest_addr.sin_family = AF_INET;
- if((inaddr = inet_addr(argv[1])) == INADDR_NONE){
- if((host = gethostbyname(argv[1])) == NULL){
- perror("gethostbyname error");
- exit(1);
- }
- memcpy((char *)&dest_addr.sin_addr, host->h_addr, host->h_length);
- } else{
- memcpy((char *)&dest_addr, (char *)&inaddr, host->h_length);
- }
- pid = getpid();
- printf("x");
- printf("PING %s(%s): %d bytes data in ICMP packets.\n", argv[1],
- inet_ntoa(dest_addr.sin_addr), 56);
- printf("0");
- (void)pthread_attr_init(&attr);
- (void)pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
- printf("1");
- if((retval = pthread_create(&thread_id, &attr, send_pkt_th, NULL)) != 0){
- perror("pthread_create error");
- exit(1);
- }
- printf("2");
- if((retval = pthread_create(&thread_id, &attr, recv_pkt_th, NULL)) != 0){
- perror("pthread_create error");
- exit(1);
- }
- signal(SIGTERM, statistics);
- signal(SIGINT, statistics);
- sleep(10);
- printf("3");
- return 0;
- }
复制代码 |
|