1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
| #include <stdio.h> #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #include <stdlib.h> #include <fcntl.h> #include <sys/epoll.h>
#define MAX_EVENTS 64
int setfd_nonblock(int fd){ int flag = fcntl(fd, F_GETFL, 0); return fcntl(fd, F_SETFL, flag | O_NONBLOCK); }
int main() { int socketfd, clientfd, epollfd; struct sockaddr_in address, client_address; int addrlen, nevent; char buffer[1024] = {0}; struct epoll_event event;
printf("current pid=%d\n", getpid());
epollfd = epoll_create(1);
socketfd = socket(AF_INET,SOCK_STREAM,0); if(socketfd == 0){ perror("socket created failed\n"); exit(-1); }
setfd_nonblock(socketfd);
int optval = 1; if(setsockopt(socketfd, SOL_SOCKET, SO_REUSEPORT, (const void*)&optval, sizeof(optval))){ perror("setsocketopt failed\n"); exit(-1); }
address.sin_family = AF_INET; address.sin_addr.s_addr = INADDR_ANY; address.sin_port = htons(8989); if(bind(socketfd, (const struct sockaddr*)&address,sizeof(address)) < 0){ perror("socket created failed\n"); exit(-1); }
if(listen(socketfd, 511) < 0){ perror("listen error"); exit(-1); } printf("start listening\n");
event.events = EPOLLIN; event.data.fd = socketfd; epoll_ctl(epollfd, EPOLL_CTL_ADD, socketfd, &event); while(1){ struct epoll_event evs[MAX_EVENTS]; nevent = epoll_wait(epollfd, evs, MAX_EVENTS, -1); for(int i = 0 ; i < nevent ; i++){ struct epoll_event active = evs[i]; if(active.data.fd == socketfd){ clientfd = accept(socketfd, (struct sockaddr *)&client_address, (socklen_t *)&addrlen); if(clientfd < 0){ perror("accept error\n"); exit(-1); } printf("accept a client = %d\n", clientfd); event.events = EPOLLIN; event.data.fd = clientfd; epoll_ctl(epollfd, EPOLL_CTL_ADD, clientfd, &event); } if(active.data.fd == clientfd){ switch (active.events) { case EPOLLIN:{ int n = read(clientfd, buffer, 1024); if(n == 0){ printf("client=%d close the connect \n", clientfd); close(clientfd); continue; } printf("read from client: %s\n",buffer); write(clientfd, buffer, n); break; } case EPOLLOUT:{ break; } case EPOLLHUP:{ epoll_ctl(epollfd, EPOLL_CTL_DEL, clientfd, &event); close(clientfd); printf("client %d disconnected", clientfd); } } } } } close(socketfd); close(epollfd); return 0; }
|