-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathpoll_chatserver.c
More file actions
121 lines (99 loc) · 2.71 KB
/
poll_chatserver.c
File metadata and controls
121 lines (99 loc) · 2.71 KB
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
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <poll.h>
#define PORT "9034"
/*
* Конвертация сокета в строку IP-адреса:
* addr: struct sockaddr_in или struct sockaddr_in6
*/
const char* inet_ntop2(void* addr, char* buf, size_t size)
{
struct sockaddr_storage* sas = addr;
struct sockaddr_in* sa4;
struct sockaddr_in6* sa6;
void* src;
switch(sas->ss_family) {
case AF_INET:
sa4 = addr;
src = &(sa4->sin_addr);
break;
case AF_INET6:
sa6 = addr;
src = &(sa6->sin6_addr);
break;
default:
return NULL;
}
return inet_ntop(sas->ss_family, src, buf, size);
}
int get_listener_socket(void)
{
struct addrinfo hints;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
struct addrinfo* ai;
int gai_res = getaddrinfo(NULL, PORT, &hints, &ai);
if(gai_res != 0) {
fprintf(stderr, "pollserver: %s\n", gai_strerror(gai_res));
exit(1);
}
int listener_sock;
struct addrinfo* p;
for(p = ai; p != NULL; p = p->ai_next) {
listener_sock = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if(listener_sock < 0) {
continue;
}
int yes = 1;
setsockopt(listener_sock, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int));
int bind_res = bind(listener_sock, p->ai_addr, p->ai_addrlen);
if(bind_res < 0) {
close(listener_sock);
continue;
}
break;
}
if(p == NULL) {
return -1;
}
freeaddrinfo(ai);
int listen_res = listen(listener_sock, 10);
if(listen_res == -1) {
return -1;
}
return listener_sock;
}
/* функция добавления нового дескриптора во множество struct pollfd */
void add_to_pfds(struct pollfd** pfds, int newfd, int* fd_count, int* fd_size )
{
/* если не хватает размера (нет комнаты), увеличиваем: */
if(*fd_count == *fd_size) {
*fd_size *= 2;
*pfds = realloc(*pfds, sizeof(**pfds) * (*fd_size));
}
(*pfds)[*fd_count].fd = newfd;
(*pfds)[*fd_count].events = POLLIN;
(*pfds)[*fd_count].revents = 0;
(*fd_count)++;
}
/* удаление дескриптора по индексу во множестве */
void del_from_pfds(struct pollfd pfds[], int i, int* fd_count )
{
/* хак, удаление, когда порядок не важен: */
/* перемещаем на удаленный индекс хвост массива */
pfds[i] = pfds[*fd_count-1];
(*fd_count)--;
}
/*
* Обработка входящих подключений
*/
void handle_new_connection(int listener, int* fd_count, )