Chinaunix

标题: linux c socket多线程编程 [打印本页]

作者: yel617    时间: 2011-11-28 10:52
标题: linux c socket多线程编程
我现在写一个服务端程序,要求可以链接多个客户端
有报警信息时要能发送报警信息,如果客户端有数据发来时,接收客户端的信息
现在运行起来有问题,当我打开第一个客户端时运行正常,当我断开这个链接,再重新打开一个客户端时,发送时就出现了问题,情况如下:
第一个链接
There is a new socket link coming 4!
alarmtype = 50
data from socket 4, bytes = 1212
data from socket 4, bytes = 1212
Socket link 4 disconnect!
第二个链接
There is a new socket link coming 5!
alarmtype = 50
send error
: Bad file descriptor
data from socket 5, bytes = 1212
data from socket 5, bytes = 1212
代码如下:
void *sockalarm(void *arg)
{
  int sdbytes = 0;
int cnfd = (int)arg;
int connfd;
  rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;
struct sockaddr_in cliaddr;
socklen_t cliaddr_len;
pthread_t rcvthread[pthread_count];
pthread_t sendthread[pthread_count];

  struct sockaddr_in servaddr, cliaddr;
int sockfd;
pthread_t sockthread;

pthread_mutex_init(&mutex, NULL);
pthread_cond_init(&cond, NULL);

sockfd = socket(AF_INET, SOCK_STREAM, 0);
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(SERV_PORT);

bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
listen(sockfd, MAX_NUM);
printf("Accepting connections ...\n");
cliaddr_len = sizeof(cliaddr);
   
while(1)
{
if(pthread_count < MAX_PTHR) {
if( (connfd = accept(cnfd, (struct sockaddr *)&cliaddr, &cliaddr_len)) < 0 ) {
perror("socket accept error\n");
} else {
printf("There is a new socket link coming %d!\n", connfd);
if( pthread_create(&rcvthread[pthread_count], NULL, rcvptz, (void *)connfd) ) {
perror("rcv pthread_create\n");
}
if( pthread_create(&sendthread[pthread_count], NULL, sendalarm, (void *)connfd) ) {
perror("send pthread_create\n");
}
pthread_count += 1;
}
} else {
printf("thread count has big enough!\n");
break;
}          
}
pthread_exit(NULL);
}

void *sendalarm(void *arg)
{
int sdbytes = 0;
int cnfd = (int)arg;
rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;

while (1)
{
pthread_mutex_lock(&mutex);
pthread_cond_wait(&cond, &mutex);
memset(&pkt, 0, sizeof(warn_hdr_t));
pkt.rtcp_head.ver = 2;
pkt.rtcp_head.pt = 204;
  pkt.rtcp_head.len = 1200;
  pkt.rtcp_head.ssrc = 11;
  pkt.rtcp_head.command = 2;
pkt.head = 0xDDCC;
pkt.len = 1188;
printf("alarmtype = %d\n", alarmtype);
pkt.type = alarmtype;
  sdbytes = send(cnfd, (char *)&pkt, RTP_LEN, MSG_DONTWAIT);
if(sdbytes < 0) {
  perror("send error\n");
break;
  }
pthread_mutex_unlock(&mutex);
sleep(1);
}
close(cnfd);
pthread_exit(NULL);
}

void *rcvptz(void *arg)
{
int i;
int rcvbytes = 0;
rtcp_hdr_t rtcp_head;
  warn_hdr_t pkt;
int cnfd = (int)arg;
char sockbuf[RTP_LEN] = {0};
char urtbuff[MAX_DATA] = {0};
  char netbuff[MAX_DATA] = {0};

while (1)
{
rcvbytes = recv(cnfd, sockbuf, RTP_LEN, 0);
if(rcvbytes > 0) {
printf("data from socket %d, bytes = %d\n", cnfd, rcvbytes);
if( rcvbytes > 12 ) {
  rtcp_head = *(rtcp_hdr_t *)sockbuf;
  if((rtcp_head.ver == 2) && (rtcp_head.pt == 204) && (rtcp_head.command == 1)) {
  for(i = 0; i < (rcvbytes - 12); i++) {
  urtbuff[i] = sockbuf[12+i];
// printf("%02x ", urtbuff[i]);
  }
  } else {       
  printf("the header of rtcp pcket is wrong\n");
  }   
  }
} else if(rcvbytes == 0) {
printf("Socket link %d disconnect!\n", cnfd);
break;
} else {
printf("Socket %d \n", cnfd);
perror("rcv error!\n");
break;
}
sleep(1);
}
close(cnfd);
pthread_exit(NULL);       
}
作者: yel617    时间: 2011-11-30 13:47
感觉咋这里提供总没有人给答复
在csdn很快就会有人来,不管答案满不满意,反正会有人答复
让人感觉很好
这个问题已解决
链接中断后,直接退出了,没有解锁
作者: yel617    时间: 2011-11-30 13:48
感觉咋这里提供总没有人给答复
在csdn很快就会有人来,不管答案满不满意,反正会有人答复
让人感觉很好
这个问题已解决
链接中断后,直接退出了,没有解锁
作者: djsxut    时间: 2011-12-18 22:12
代码用[code][/code]括起来,看起来舒服些。可能看看还要整理下代码,时间也不是那么充裕。
作者: djsxut    时间: 2011-12-18 22:16
代码用[code][/code]括起来,看起来舒服些。可能看看还要整理下代码,别人时间也不是那么充裕。
作者: ragolee    时间: 2012-01-05 11:23
int cnfd = (int)arg;
上面的转换不对,应该是
int* cnfd = (int*)arg;
send函数引用cnfd用*cnfd
作者: weifeng270    时间: 2012-05-18 13:09
本帖最后由 weifeng270 于 2012-05-18 13:11 编辑

因为这是垃圾论坛。。

看看我发的贴子。也是自己解决的。

http://bbs.chinaunix.net/thread-3743754-1-1.html




作者: kprc    时间: 2012-05-21 09:46
代码写好点,这个实在没法看。
作者: helpstudy    时间: 2012-06-20 19:02
请问void *rcvptz(void *arg)处理中,为何没有进行加锁处理?线程资源回收也没有处理,对send/recv函数处理,如果数据量过大,该处理方式没有完全处理





欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2