我在边缘触发模式下使用epoll。为了避免饥饿,代码从一个套接字一次读取MAX_FREAD_LENGTH个字节。之后组装片段直到EOL发生。我注意到当MAX_FREAD_LENGTH很小时epoll卡住了。我认为它适用于任何大小的读取块。它适用于512字节,但有时挂起(意味着没有EPOLLIN事件)。如果我递增MAX_FREAD_LENGTH,它会变得更稳定。我该如何解决这个问题?
非常感谢您考虑我的问题!
EPOLL初始化
int res;
epFd = epoll_create(EPOLL_SIZE);
event.data.fd = serverFd;
event.events = EPOLLIN|EPOLLET;
res=epoll_ctl(epFd, EPOLL_CTL_ADD, serverFd, &event);
if (res == -1){
perror ("epoll_ctl error: ");
return EH_ERROR;
}
events = calloc (MAX_EVENTS, sizeof event);
注册网络活动:
while (TRUE){
int nfds;
do{
nfds = epoll_wait(epFd, events, MAX_EVENTS, -1);
} while (nfds < 0 && errno == EINTR);
int i = 0;
for (;i<nfds;i++){
if ( (events[i].data.fd == serverFd) && (events[i].events & EPOLLIN)){
if ((clientFd = accept(serverFd,(struct sockaddr *) &clientAddr, &clientLen)) < 0){
char log[255];
sprintf(log,"dispatch_net_event: Socket accept failed: %s",strerror(errno));
logger->log(LOGG_ERROR,log);
}
if(newclient(clientFd)!=EH_ERROR){
/* client created */
setnonblocking(fd,NONBLOCKING);
event.data.fd = clientFd;
event.events = EPOLLIN |EPOLLET;
if(epoll_ctl(epFd, EPOLL_CTL_ADD, fd, &event)<0){
fprintf(stderr,"Epoll insertion error (fd=%d): ",clientFd);
return EH_ERROR;
}
continue;
}
else{
logger->log(LOGG_ERROR,"Client creation error");
continue;
}
}
else{
dispatch_event(events[i].data.fd,NET_EVENT);
}
}
}
处理网络事件
#define SMTP_MAX_LINE_LENGTH MAX_FREAD_LENGTH
ssize_t count;
char buf[SMTP_MAX_LINE_LENGTH];
memset(buf,'\0', SMTP_MAX_LINE_LENGTH);
count = read (fd, buf,MAX_FREAD_LENGTH );
if (count==-1){
if (errno == EAGAIN)
return KEEP_IT;
else if (errno == EWOULDBLOCK)
return KEEP_IT;
else{
char log[255];
sprintf(log,"handle_net_event: Read error: %s",strerror(errno));
logger->log(LOGG_ERROR,log);
}
}
else{ /* count > 0 there are data in the buffer */
/* assemble possible partial lines, TRUE if line is whole (end with \r\n)*/
whole=assemble_line(count,client,&buf[0]);
}
/* process the line */
编辑:
我忘了提到,epoll在一个单独的线程中运行,而不是其他部分
答案 0 :(得分:2)
我认为你错误地使用了EPOLLET。尝试没有EPOLLET(即使用水平触发模式),看看是否有效。如果是这样,这意味着你的根本问题是你没有做边缘触发模式要求,即继续读取任何准备好的描述符,直到你获得EAGAIN。