记录一次epoll事件未触发的经历,由于是tcp通信,使用了io多路复用epoll,客户端只发送数据,不处理服务端响应,服务端处理对应的事件,并且会响应消息给客户端,遇到了EPOLLOUT事件未触发的情况,进行排查。
经过查阅相关资料得知,socket通信时是全双工的,都有send、recv缓冲区,当recv缓冲区满了,就会响应ACK给对方,将发送窗口置为0,这个时候EPOLLOUT事件就无法触发,EPOLLOUT触发的前提条件是发送缓冲区可写,即对应的内核缓冲区为可写状态,这个时候不满足条件,EPOLLOUT事件就触发不了,响应消息就发送不过去。
附相关代码
客户端代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<string.h>
#include<errno.h>
#define MAXLINE 1024
void cli_send_msg(FILE *fp, int sockfd)
{
char sendline[MAXLINE], recvline[MAXLINE];
//while (fgets(sendline, MAXLINE, fp) != NULL) {
while(1) {
memset(sendline, 97, MAXLINE);
int n = write(sockfd, sendline, strlen(sendline));
if (n < 0) {
fprintf(stderr, "write err errno(%d) %s\n", errno, strerror(errno));
}
memset(sendline, 0, sizeof(sendline));
usleep(10);
//if (read(sockfd, recvline, MAXLINE) == -1) {
// fprintf(stderr, "read err errno(%d) %s\n", errno, strerror(errno));
// close(sockfd);
//}
//memset(recvline, 0, MAXLINE);
//Fputs(recvline, stdout);
}
}
int main(int argc, const char **argv)
{
int sockfd;
struct socka