Chinaunix首页 | 论坛 | 博客
  • 博客访问: 15530508
  • 博文数量: 2005
  • 博客积分: 11986
  • 博客等级: 上将
  • 技术积分: 22535
  • 用 户 组: 普通用户
  • 注册时间: 2007-05-17 13:56
文章分类

全部博文(2005)

文章存档

2014年(2)

2013年(2)

2012年(16)

2011年(66)

2010年(368)

2009年(743)

2008年(491)

2007年(317)

分类: LINUX

2008-06-10 11:30:05

server.c

#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <time.h>
#include <sys/poll.h>
#include <sys/resource.h>

#define ___PORT 1357
#define ___poll_min (10)

static unsigned long ___poll_max = 1024;
static int sockfd_server;
static int max_fd = 0;

static struct pollfd *pfds;
inline void fd_set2(int fd);
inline int fd_get2(int ifd);
inline void fd_clr2(int ifd);
inline void fd_zero2(void);
inline int fd_isset2(int ifd);
inline int fd_isin(int ifd);
void process_data(int sockfd, char *buf);

int main(int argc, char *argv[])
{
    int newsockfd,ifd;
    struct sockaddr_in addr;
    int addr_len;
    time_t local_time;
    int readbytes;
    int ret;
    struct rlimit lim;
    unsigned long user_nums;

    char buffer[256];
    char msg[]="Welcome 2gliethttp server!";

    addr_len = sizeof(struct sockaddr_in);
    if((sockfd_server = socket(PF_INET,SOCK_STREAM,0)) < 0)//申请套接字
    {
        perror("socket");
        return 0;
    }

    memset(&addr, 0, sizeof(addr));
    addr.sin_family =PF_INET;
    addr.sin_port = htons(___PORT);//该套接字对___PORT端口数据进行监听
    addr.sin_addr.s_addr = htonl(INADDR_ANY);//处理来自PC上的任何一块网卡数据

    if(bind(sockfd_server, (struct sockaddr *)&addr, sizeof(addr)) < 0)//将套接字和配置参数绑定在一起
    {
        perror("connect");
        return 0;
    }

    if(listen(sockfd_server, 5) < 0)//同一时刻最多允许5个连接
    {
        perror("listen");
        return 0;
    }

    if(argc > 1) user_nums = strtol(argv[1], 0, 0);
    else user_nums = 0;
    lim.rlim_cur = user_nums + ___poll_min;
    lim.rlim_max = user_nums + ___poll_min;
    if(setrlimit(RLIMIT_NOFILE, &lim))
    {
        perror("setrlimit");
        exit(0);
    }
    ___poll_max = user_nums + ___poll_min - 1;

    printf("poll_max=%d\n",___poll_max);
    fflush(stdout);

    pfds = malloc(sizeof(struct pollfd)*___poll_max);
    fd_zero2();
    fd_set2(sockfd_server);
    
    max_fd = sockfd_server;

    for(;;)
    {
        printf("\npolling...\n");

        ret = poll(pfds, ___poll_max, -1);//如果timeout=0,将一直返回ret=0
        if(ret == 0)
          continue;
        if(ret < 0)
        {
            perror("poll error: ");
            exit(-1);
        }
        
        printf("ret=%d\n",ret);
        fflush(stdout);

        for(ifd = 0;ifd < ___poll_max;ifd++)
        {
            if(fd_isset2(ifd))
            {
                if(sockfd_server == fd_get2(ifd))
                {
                    //监听端口有连接触发请求了,那么处理之
                    if((newsockfd = accept(sockfd_server,(struct sockaddr *)&addr,(socklen_t*)&addr_len)) < 0)
                      perror("accept");
                    else
                    {
                        if(++max_fd >= ___poll_max)
                        {
                            max_fd = ___poll_max;
                            fd_clr2(ifd);
                        }

                        fprintf(stderr,"=====%d=====\n",newsockfd);
                        fd_set2(newsockfd);
                        time(&local_time);
                        sprintf(buffer, "<%d> : %s %s", newsockfd, msg, ctime(&local_time));
                        write(newsockfd, buffer, strlen(buffer));
                        printf("conect: <%d> from %s %s\n", newsockfd, inet_ntoa(addr.sin_addr), ctime(&local_time));
                    }
                }
                else
                {
                    readbytes = read(fd_get2(ifd), buffer, sizeof(buffer));
                    if(readbytes <= 0)
                    {
                        time(&local_time);
                        printf("closed: <%d> %s\n", fd_get2(ifd), ctime(&local_time));
                        close(fd_get2(ifd));
                        fd_clr2(ifd);

                        if(max_fd-- == ___poll_max)
                        {
                            fd_set2(sockfd_server);
                        }
                    }
                    else
                    {
                        buffer[readbytes] = 0;
                        process_data(fd_get2(ifd), buffer);
                    }
                }
            }
        }
    }
}

inline void fd_set2(int fd)
{
    int ifd;
    ifd = fd ;//+ 100;
    pfds[ifd].fd = fd;
    pfds[ifd].events = POLLIN;
}
inline int fd_get2(int ifd)
{
    return pfds[ifd].fd;
}
inline void fd_clr2(int ifd)
{
    pfds[ifd].fd = 0;//必需将pfds[].fd置0,同时poll(,,-1)
    pfds[ifd].events = 0;
}
inline void fd_zero2(void)
{
#if 0
    int i;
    for(i = 0;i < ___poll_max;i++)
    {
        pfds[i].fd = 0;
        pfds[i].events = POLLNVAL;
        pfds[i].revents = 0;
    }
#endif
    memset(pfds, 0, sizeof(pfds));
}
inline int fd_isset2(int ifd)
{
    return pfds[ifd].revents;
}
inline int fd_isin(int ifd)
{
    return pfds[ifd].fd;
}

void process_data(int sockfd, char *buf)
{
    char *pdata;
    int idest_sock,dest_sock;
    int tmp_len;
    char buffer[1024*10];
    int ifd;

    if(strncmp("ls", buf, 2) == 0)
    {
        char *p;
        p = buffer;
        for(ifd = 0;ifd < ___poll_max;ifd++)
        {
            if(fd_isin(ifd))
            {
                if(fd_get2(ifd) == sockfd)p += sprintf(p, "#%d*", ifd);
                p += sprintf(p, "%d\n", fd_get2(ifd));
            }
        }
        write(sockfd, buffer, strlen(buffer));
        return;
    }
    
#if 0
    pdata = strtok(buf, ":");
    if(pdata == NULL)
    {
       
    }
    else
#endif
    {
        pdata = buf;
        dest_sock = strtol(pdata, 0, 0);//atoi(pdata);strtol
        idest_sock = ___poll_max;
        for(ifd = 0;ifd < ___poll_max;ifd++)
        {
            if(fd_get2(ifd) == dest_sock)
            {
                idest_sock = ifd;
                break;
            }
        }

        if(idest_sock < ___poll_max)
        {
                 if(dest_sock < 10)tmp_len = 1;
            else if(dest_sock < 100)tmp_len = 2;
            else if(dest_sock < 1000)tmp_len = 3;
            else tmp_len = 4;
            //pdata = strtok(buf, ":");
            //pdata += strlen(pdata) + 1;
            pdata += tmp_len;

            if(dest_sock == sockfd_server)
            {
                printf("<%d> %s", sockfd, pdata);
                fflush(stdout);
            }
            else
            {
                sprintf(buffer, "<%d> : %s", sockfd, pdata);
                pdata = buffer;
                write(dest_sock, pdata, strlen(pdata));
            }
        }
        else
        {
            sprintf(buffer, "client : %d not exist\n",dest_sock);
            write(sockfd, buffer, strlen(buffer));
        }
    }
}


client.c

#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/poll.h>

#define STDIN 0
#define MAXDATASIZE 10240
#define ___poll_max 2
int main(int argc, char *argv[])
{
    char buf[MAXDATASIZE];
    char send_str[MAXDATASIZE];
    int recvbytes;
    int sockfd,k;
    struct hostent *hp;
    struct sockaddr_in addr;
    char *ip;
    int port;
    int ret;
    struct pollfd pfds[___poll_max];

    if(argc < 3)
    {
        printf("client ip port\n");
        return -1;
    }
    ip = argv[1];
    port = atoi(argv[2]);//strtol(argv[2], 0, 0);
    hp = gethostbyname(ip);
    if (hp == NULL)
    {
        perror("hostent error" );
        return -2;
    }

    if((sockfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror( "socket error" );
        return( -1 );
    }

    memset(&addr,0,sizeof(struct sockaddr_in));
    addr.sin_family=PF_INET;
    addr.sin_port=htons(port);
    //addr.sin_addr.s_addr=inet_addr(ip);
    memcpy(&addr.sin_addr,hp->h_addr_list[0],hp->h_length);
    //addr.sin_addr.s_addr= htonl(INADDR_ANY);
    printf("正在与服务器%s:%d连接....\n",ip,port);
    if(connect(sockfd, (struct sockaddr *)&addr, sizeof(addr)) < 0)
    {
        perror("connect");
        exit(1);
    }
    
    pfds[0].fd = STDIN;
    pfds[1].fd = sockfd;

    pfds[0].events = POLLIN;
    pfds[1].events = POLLIN;


    for(;;)
    {
        ret = poll(pfds, ___poll_max, -1);
        if(ret == 0)
          continue;
        if(ret < 0)
        {
            perror("poll error: ");
            exit(-1);
        }
        
        if(pfds[0].revents)
        {
            fgets(send_str, MAXDATASIZE, stdin);
            send_str[strlen(send_str) - 1] = 0;
            if(strncmp("quit", send_str, 4) == 0)
            {
                close(sockfd);
                exit(0);
            }
            send(sockfd, send_str, strlen(send_str), 0);
        }

        if(pfds[1].revents)
        {
            recvbytes = recv(sockfd, buf, MAXDATASIZE, 0);
            if(recvbytes <= 0)
            {
                close(sockfd);
                exit(0);
            }
            buf[recvbytes] = 0;
            printf("-----------\n%s\n", buf);
            fflush(stdout);
        }
    }
}



文件:poll.tar.gz
大小:2KB
下载:下载

阅读(2170) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~