-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtcp-socket-server-epoll.cpp
128 lines (118 loc) · 4.06 KB
/
tcp-socket-server-epoll.cpp
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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
#include "sys/types.h"
#include "sys/socket.h"
#include "netdb.h"
#include "string.h"
#include "unistd.h"
#include "fcntl.h"
#include "sys/epoll.h" //linux环境
#include "arpa/inet.h"
#include "errno.h"
#include "stdio.h"
#include "stdlib.h"
#define DEFAULT_PORT 8080
#define MAX_CONN 16
#define MAX_EVENTS 32
#define BUF_SIZE 16
#define MAX_LINE 256
#define AF struct sockaddr
/**
* epoll 可以通过注册 callback 函数的方式,监听某个文件描述符发生变化,就会触发相应的事件
* epoll_create 创建一个 epoll 对象,它的文件描述符对应了内核中的红黑树的结构
* epoll_ctl 添加一个 Socket 时,向红黑树添加一个节点,这个节点存储了关心这个 Socket 变化事件的回调函数列表
* 当 Socket 的事件列表发生变化可以找到挂的 epoll 对象,然后从 epoll 中找到回调并触发它
**/
//实例来源 https://github.com/onestraw/epoll-example/blob/master/epoll.c
static void set_sockaddr(struct sockaddr_in *addr);
static int setnonblocking(int sockfd);
static void epoll_ctl_add(int epfd, int fd, uint32_t events);
int main()
{
int i, n, epfd, nfds;
int listen_sock, conn_sock;
int socklen;
char buf[BUF_SIZE];
struct sockaddr_in srv_addr, cli_addr;
struct epoll_event events[MAX_EVENTS];
listen_sock = socket(AF_INET, SOCK_STREAM, 0);
set_sockaddr(&srv_addr);
bind(listen_sock, (AF *)&srv_addr, sizeof(srv_addr));
setnonblocking(listen_sock);
listen(listen_sock, MAX_CONN);
epfd = epoll_create(1);
epoll_ctl_add(epfd, listen_sock, EPOLLIN | EPOLLOUT | EPOLLET);
socklen = sizeof(cli_addr);
while (1)
{
//变更的事件在events中
nfds = epoll_wait(epfd, events, MAX_EVENTS, -1);
for (int i = 0; i < nfds; i++)
{
if (events[i].data.fd == listen_sock)
{
conn_sock = accept(listen_sock, (AF *)&cli_addr, &socklen);
inet_ntop(AF_INET, (char *)&(cli_addr.sin_addr), buf, sizeof(cli_addr));
printf("[+] connected with %s:%d\n", buf, ntohs(cli_addr.sin_port));
//获取到新的连接加入到epoll中
setnonblocking(conn_sock);
epoll_ctl_add(epfd, conn_sock, EPOLLIN | EPOLLET | EPOLLRDHUP | EPOLLHUP);
}
else if (events[i].events & EPOLLIN)
{
while (1)
{
bzero(buf, sizeof(buf));
n = read(events[i].data.fd, buf, sizeof(buf));
if (n <= 0 || errno == EAGAIN)
break;
else
{
printf("[+] data: %s\n", buf);
//将读取的数据写回到socket中
write(events[i].data.fd, buf, strlen(buf));
}
}
}
else
printf("[+] uexpected\n");
//socket 连接关闭
if (events[i].events & (EPOLLRDHUP | EPOLLHUP))
{
printf("[+] connection closed\n");
//从epoll中删除该socket的监听
epoll_ctl(epfd, EPOLL_CTL_DEL,
events[i].data.fd, NULL);
close(events[i].data.fd);
continue;
}
}
}
}
static void set_sockaddr(struct sockaddr_in *addr)
{
bzero((char *)addr, sizeof(struct sockaddr_in));
addr->sin_family = AF_INET;
addr->sin_addr.s_addr = INADDR_ANY;
addr->sin_port = htons(DEFAULT_PORT);
}
static int setnonblocking(int sockfd)
{
//设置文件描述符的IO操作为非阻塞模式
int ret = fcntl(sockfd, F_SETFD, fcntl(sockfd, F_GETFD, 0) | O_NONBLOCK);
if (ret == -1)
return -1;
return 0;
}
/**
* 将描述符的监听事件添加到epoll上
**/
static void epoll_ctl_add(int epfd, int fd, uint32_t events)
{
struct epoll_event ev;
ev.events = events;
ev.data.fd = fd;
if (epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev) == -1)
{
perror("epoll_ctl()\n");
exit(1);
}
}