wasgxu 发表于 2015-05-08 18:30

epoll_data里的ptr问题.

最近刚开始学linux编程,今天用epoll做一个简单的TCP服务器的时候,出现了问题,自己摸索了好久,一直没有解决,希望有哪位大侠帮帮我,,问题是这样的.

按照网上的资料,这是网上的一段代码..
typedef struct _socket_data{
        int fd;
        int stype;
}socket_data,*psocket_data;


int main(int argc, char **argv)
{
        intservPort = 6888;
        int listenq = 1024;

        int listenfd, connfd, kdpfd, nfds, n, nread, curfds,acceptCount = 0;
        struct sockaddr_in servaddr, cliaddr;
        socklen_t socklen = sizeof(struct sockaddr_in);
        struct epoll_event ev;
        struct epoll_event events;
        struct rlimit rt;
        char buf;

        /* 设置每个进程允许打开的最大文件数 */
        rt.rlim_max = rt.rlim_cur = MAXEPOLLSIZE;
        if (setrlimit(RLIMIT_NOFILE, &rt) == -1)
        {
                perror("setrlimit error");
                return -1;
        }


        bzero(&servaddr, sizeof(servaddr));
        servaddr.sin_family = AF_INET;
        servaddr.sin_addr.s_addr = htonl (INADDR_ANY);
        servaddr.sin_port = htons (servPort);

        listenfd = socket(AF_INET, SOCK_STREAM, 0);
        if (listenfd == -1) {
                perror("can't create socket file");
                return -1;
        }

        int opt = 1;
        setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));

        if (setnonblocking(listenfd) < 0) {
                perror("setnonblock error");
        }

        if (bind(listenfd, (struct sockaddr *) &servaddr, sizeof(struct sockaddr)) == -1)
        {
                perror("bind error");
                return -1;
        }
        if (listen(listenfd, listenq) == -1)
        {
                perror("listen error");
                return -1;
        }
        /* 创建 epoll 句柄,把监听 socket 加入到 epoll 集合里 */
        psocket_data sdata=(psocket_data)malloc(sizeof(socket_data));
        sdata->stype=1;
        sdata->fd = listenfd;
        kdpfd = epoll_create(MAXEPOLLSIZE);
        //ev.data.fd = listenfd;
        ev.data.ptr = ()sdata;
        ev.events = EPOLLIN | EPOLLET;
        if (epoll_ctl(kdpfd, EPOLL_CTL_ADD, listenfd, &ev) < 0)
        {
                fprintf(stderr, "epoll set insertion error: fd=%d\n", listenfd);
                return -1;
        }
        curfds = 1;

        printf("epollserver startup,port %d, max connection is %d, backlog is %d\n", servPort, MAXEPOLLSIZE, listenq);

        for (;;) {
                /* 等待有事件发生 */
                nfds = epoll_wait(kdpfd, events, curfds, -1);
                if (nfds == -1)
                {
                        perror("epoll_wait");
                        continue;
                }
                /* 处理所有事件 */
                for (n = 0; n < nfds; ++n)
                {
                        if (events.data.fd == listenfd)
                        {
                                connfd = accept(listenfd, (struct sockaddr *)&cliaddr,&socklen);
                                if (connfd < 0)
                                {
                                        perror("accept error");
                                        continue;
                                }

                                sprintf(buf, "accept form %s:%d\n", inet_ntoa(cliaddr.sin_addr), cliaddr.sin_port);
                                printf("%d:%s", ++acceptCount, buf);

                                if (curfds >= MAXEPOLLSIZE) {
                                        fprintf(stderr, "too many connection, more than %d\n", MAXEPOLLSIZE);
                                        shutdown(connfd, SHUT_RDWR);
                                        close(connfd);
                                        continue;
                                }
                                if (setnonblocking(connfd) < 0) {
                                        perror("setnonblocking error");
                                }
                                ev.events = EPOLLIN | EPOLLET;
                                ev.data.fd = connfd;
                                if (epoll_ctl(kdpfd, EPOLL_CTL_ADD, connfd, &ev) < 0)
                                {
                                        fprintf(stderr, "add socket '%d' to epoll failed: %s\n", connfd, strerror(errno));
                                        return -1;
                                }
                                curfds++;
                                continue;
                        }
                        // 处理客户端请求
                        if (handle(events.data.fd) < 0) {
                                epoll_ctl(kdpfd, EPOLL_CTL_DEL, events.data.fd,&ev);
                                curfds--;


                        }
                }
        }
        close(listenfd);
        return 0;
}源代码是直接把句柄赋给了ev.data.fd
//ev.data.fd = listenfd;

然后编译运行是正常的...但是我需要修改下代码,把ptr指向我定义的一个结构,然后,
客户端连接的时候,就会一直
epoll_wait: Invalid argument
如果取消对data.ptr的赋值就正常,也查找了很多网上的资料,他们也是那样赋值的,但却没看到谁说这样有问题.所以不知道我哪里做错了,希望大家指点一下,谢谢.

下面循环部分的events.data.fd == listenfd判断肯定是错误的,但是我感觉我的代码终止在了perror("epoll_wait");这里,所以我就没修改,
是哪里有问题呢.

hellioncu 发表于 2015-05-11 16:04

data是一个union,你用了fd,就不能再用ptr了

wasgxu 发表于 2015-05-14 18:01

我知道了,因为perror持续打印,把第一条出错语句盖掉了,所以我一直没看到第一句出错,,现在问题找到了,是因为我一直觉得他执行不到events.data.fd == listenfd这一句,但其实就是这句出错,因为data.fd不能用了.解决了,谢谢.
页: [1]
查看完整版本: epoll_data里的ptr问题.