-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathselectServer.c
137 lines (117 loc) · 3.03 KB
/
selectServer.c
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
129
130
131
132
133
134
135
136
137
/*************************************************************************
> File Name: selectServer.c
> Author: lidongmeng
> Mail:lidongmeng0213@163.com
> Created Time: Fri 11 Sep 2015 04:37:43 AM PDT
************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <sys/select.h>
#include <errno.h>
#include <string.h>
#define MAXLINE 1024
void handle(int * clientSockFd, int maxfd, fd_set *rset, fd_set * allset);
int main(int argc, char ** argv) {
int serverPort = 6888;
int listenBackLog = 1024;
int listenfd, connfd;
struct sockaddr_in serverAddr, clientAddr;
socklen_t clientLen = sizeof(struct sockaddr_in);
int nready, nread;
char buf[MAXLINE];
int clientSockFd[FD_SETSIZE];
fd_set allset, rset;
int maxfd;
listenfd = socket(AF_INET, SOCK_STREAM, 0);
if (listenfd < 0) {
perror("listen error");
return -1;
}
int opt = 1;
if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) < 0) {
perror("set sockopt error");
}
bzero(&serverAddr, sizeof(serverAddr));
serverAddr.sin_family = AF_INET;
serverAddr.sin_addr.s_addr = htonl(INADDR_ANY);
serverAddr.sin_port = htons(serverPort);
if (bind(listenfd, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) == -1) {
perror("bind error");
return -1;
}
if (listen(listenfd, listenBackLog) < 0) {
perror("listen error");
return -1;
}
int i = 0;
for (i = 0; i < FD_SETSIZE; ++i) {
clientSockFd[i] = -1;
}
FD_ZERO(&allset);
FD_SET(listenfd, &allset);
maxfd = listenfd;
while (1) {
rset = allset;
nready = select(maxfd+1, &rset, NULL, NULL, NULL);
if (nready < 0) {
perror("select error");
continue;
}
if (FD_ISSET(listenfd, &rset)) {
connfd = accept(listenfd, (struct sockaddr*)&clientAddr, &clientLen);
if (connfd < 0) {
perror("accept error");
continue;
}
sprintf(buf, "accept form %s:%d\n", inet_ntoa(clientAddr.sin_addr), clientAddr.sin_port);
printf(buf, "");
for (i = 0; i < FD_SETSIZE; ++i) {
if (clientSockFd[i] == -1) {
clientSockFd[i] = connfd;
break;
}
}
if (i == FD_SETSIZE) {
printf("Too many connection\n");
close(connfd);
continue;
}
if (connfd > maxfd) {
maxfd = connfd;
}
FD_SET(connfd, &allset);
if (--nready <= 0)
continue;
}
handle(clientSockFd, maxfd, &rset, &allset);
}
}
void handle(int * clientSockFds, int maxfd, fd_set * rset, fd_set * allset) {
int nread;
int i;
char buf[MAXLINE];
for (i = 0; i < maxfd; ++i) {
if (FD_ISSET(clientSockFds[i], rset)) {
nread = read(clientSockFds[i], buf, MAXLINE);
if (nread < 0) {
perror("read error");
close(clientSockFds[i]);
FD_CLR(clientSockFds[i], allset);
clientSockFds[i] = -1;
continue;
}
if (nread == 0) {
printf("client closet the connection\n");
close(clientSockFds[i]);
FD_CLR(clientSockFds[i], allset);
clientSockFds[i] = -1;
continue;
}
write(clientSockFds[i], buf, nread);
}
}
}