/clients.c

https://github.com/zcoder/netio · C · 410 lines · 368 code · 35 blank · 7 comment · 101 complexity · 3bd2df0a9a4c30ce5449dbfd89c8b504 MD5 · raw file

  1. #include <errno.h>
  2. #include <assert.h>
  3. #include <stdlib.h>
  4. #include <unistd.h>
  5. #include <signal.h>
  6. #include <sys/socket.h>
  7. #include <fcntl.h>
  8. #include "clients.h"
  9. #include "servers.h"
  10. #include "params.h"
  11. #include "error.h"
  12. #include "signals.h"
  13. #include "timer.h"
  14. #include "set.h"
  15. #include "run.h"
  16. #include <sys/time.h>
  17. #include <stdio.h>
  18. #define MAX(X,Y) (X>Y)?X:Y
  19. #define MIN(X,Y) (X<Y)?X:Y
  20. struct TClients g_clients = { NULL, 0, 0 };
  21. int set2client(int index)
  22. {
  23. return index - 2 - g_servers.m_count;
  24. }
  25. int client2set(int index)
  26. {
  27. return index + 2 + g_servers.m_count;
  28. }
  29. void client_add(int sock)
  30. {
  31. int events = POLLHUP | POLLERR | POLLNVAL;
  32. struct TClient client;
  33. client.m_sock = sock;
  34. client.m_ipid = -1;
  35. client.m_opid = -1;
  36. client.m_sendbuff = NULL;
  37. client.m_remain = 0;
  38. client.m_timer.tv_sec = 0;
  39. client.m_timer.tv_usec = 0;
  40. switch ( p_iomode )
  41. {
  42. case IOMODE_BLOCK:
  43. shutdown( sock, SHUT_RDWR );
  44. break;
  45. case IOMODE_EVERY:
  46. {
  47. client.m_opid = run( p_iocmd, sock, sock );
  48. client.m_ipid = client.m_opid;
  49. }
  50. break;
  51. case IOMODE_NONE:
  52. {
  53. switch ( p_inmode )
  54. {
  55. case IOMODE_BLOCK:
  56. shutdown( sock, SHUT_WR );
  57. break;
  58. case IOMODE_EVERY:
  59. {
  60. client.m_ipid = run( p_incmd, sock, STDIN_FILENO );
  61. if ( client.m_ipid == -1 )
  62. {
  63. close( sock );
  64. return;
  65. }
  66. }
  67. break;
  68. case IOMODE_ONCE:
  69. case IOMODE_NULL:
  70. case IOMODE_NONE:
  71. break;
  72. }
  73. switch ( p_outmode )
  74. {
  75. case IOMODE_BLOCK:
  76. shutdown( sock, SHUT_RD );
  77. break;
  78. case IOMODE_EVERY:
  79. {
  80. client.m_opid = run( p_outcmd, STDOUT_FILENO, sock );
  81. if ( client.m_opid == -1 )
  82. {
  83. close( sock );
  84. return;
  85. }
  86. }
  87. break;
  88. case IOMODE_ONCE:
  89. case IOMODE_NULL:
  90. case IOMODE_NONE:
  91. break;
  92. }
  93. }
  94. break;
  95. case IOMODE_ONCE:
  96. case IOMODE_NULL:
  97. break;
  98. }
  99. int onone = p_iomode == IOMODE_NONE && p_outmode == IOMODE_NONE;
  100. int oonce = p_iomode == IOMODE_ONCE || p_outmode == IOMODE_ONCE;
  101. int onull = p_iomode == IOMODE_NULL || p_outmode == IOMODE_NULL;
  102. if ( onone || oonce )
  103. {
  104. if ( !(g_set[STDOUT_FILENO].events & POLLOUT) )
  105. {
  106. events |= POLLIN;
  107. }
  108. }
  109. else if ( onull )
  110. {
  111. events |= POLLIN;
  112. }
  113. int newcount = g_clients.m_count + 1;
  114. struct TClient* clients = (struct TClient*)malloc( sizeof(struct TClient) * newcount );
  115. memcpy( clients, g_clients.m_client, sizeof(struct TClient) * g_clients.m_count );
  116. clients[ g_clients.m_count ] = client;
  117. set_add( client2set( 0 ) + g_clients.m_count, sock, events );
  118. free( g_clients.m_client );
  119. g_clients.m_client = clients;
  120. g_clients.m_count = newcount;
  121. int inone = p_iomode == IOMODE_NONE || p_inmode == IOMODE_NONE;
  122. int ionce = p_iomode == IOMODE_ONCE || p_inmode == IOMODE_ONCE;
  123. if ( (g_clients.m_blocked == 0) && (inone || ionce) )
  124. {
  125. g_set[STDIN_FILENO].events |= POLLIN;
  126. signals_cansyncterm();
  127. }
  128. }
  129. void client_del(struct TClient* client)
  130. {
  131. int newcount = g_clients.m_count - 1;
  132. struct TClient* clients = (struct TClient*)malloc( sizeof(struct TClient) * newcount );
  133. int index;
  134. for (index = 0; (g_clients.m_client+index) != client && index < g_clients.m_count; ++index)
  135. {
  136. clients[index] = g_clients.m_client[index];
  137. }
  138. set_del( client2set( 0 ) + index );
  139. for (++index; index < g_clients.m_count; ++index)
  140. {
  141. clients[index-1] = g_clients.m_client[index];
  142. }
  143. free( g_clients.m_client );
  144. g_clients.m_client = clients;
  145. g_clients.m_count = newcount;
  146. }
  147. int clients_loop(char* recvbuff, int nready)
  148. {
  149. int setindex = client2set( 0 );
  150. int clientindex = 0;
  151. while ( nready > 0 && clientindex < g_clients.m_count )
  152. {
  153. struct pollfd* set = g_set + setindex;
  154. struct TClient* client = g_clients.m_client + clientindex;
  155. /* check for events */
  156. if ( !set->revents )
  157. {
  158. ++clientindex;
  159. ++setindex;
  160. continue;
  161. }
  162. /* disconnect client */
  163. if ( set->revents & POLLHUP )
  164. {
  165. --nready;
  166. client_disconnect( client );
  167. continue;
  168. }
  169. /* can send to client */
  170. if ( set->revents & POLLOUT )
  171. {
  172. nready = client_POLLOUT( nready, set, client, recvbuff );
  173. continue;
  174. }
  175. /* read from client */
  176. if ( set->revents & POLLIN )
  177. {
  178. nready = client_POLLIN( nready, set, client, recvbuff );
  179. continue;
  180. }
  181. assert( !set->revents ); //WTF?
  182. }
  183. int inone = p_iomode == IOMODE_NONE || p_inmode == IOMODE_NONE;
  184. int ionce = p_iomode == IOMODE_ONCE || p_inmode == IOMODE_ONCE;
  185. if ( (g_clients.m_blocked == 0) && (inone || ionce) )
  186. {
  187. g_set[STDIN_FILENO].events |= POLLIN;
  188. signals_cansyncterm();
  189. }
  190. return nready;
  191. }
  192. int client_POLLOUT(int nready, struct pollfd* set, struct TClient* client, char* recvbuff)
  193. {
  194. set->revents ^= POLLOUT;
  195. --nready;
  196. const int sock = client->m_sock;
  197. const char* sendbuff = client->m_sendbuff;
  198. const int remain = client->m_remain;
  199. int sendsize = send( sock, sendbuff, remain, MSG_DONTWAIT | MSG_NOSIGNAL );
  200. if ( sendsize == -1 )
  201. {
  202. assert( errno != EWOULDBLOCK ); //WTF?
  203. /* if error */
  204. error_send( errno );
  205. client_disconnect( client );
  206. return nready;
  207. }
  208. if ( sendsize < remain )
  209. {
  210. /* if not all sended */
  211. client->m_sendbuff += sendsize;
  212. client->m_remain -= sendsize;
  213. }
  214. else
  215. {
  216. /* if all sended */
  217. client->m_sendbuff = NULL;
  218. client->m_remain = 0;
  219. set->events &= ~POLLOUT;
  220. --g_clients.m_blocked;
  221. if ( g_set[STDIN_FILENO].fd == -1 ) //stdin closed
  222. {
  223. client_tdisconnect( client );
  224. }
  225. }
  226. return nready;
  227. }
  228. int client_POLLIN(int nready, struct pollfd* set, struct TClient* client, char* recvbuff)
  229. {
  230. set->revents ^= POLLIN;
  231. --nready;
  232. if ( p_iomode == IOMODE_NULL || p_outmode == IOMODE_NULL )
  233. {
  234. ssize_t truncsize = recv( client->m_sock, recvbuff, p_recvbuff, MSG_TRUNC );
  235. if ( truncsize == -1 )
  236. {
  237. error_recv( errno );
  238. client_disconnect( client );
  239. return nready;
  240. }
  241. if ( truncsize == 0 )
  242. {
  243. client_disconnect( client );
  244. return nready;
  245. }
  246. return nready;
  247. }
  248. ssize_t recvsize = recv( set->fd, recvbuff, p_recvbuff, MSG_PEEK );
  249. if ( recvsize == -1 )
  250. {
  251. error_recv( errno );
  252. client_disconnect( client );
  253. return nready;
  254. }
  255. if ( recvsize == 0 )
  256. {
  257. client_disconnect( client );
  258. return nready;
  259. }
  260. ssize_t writesize = write( STDOUT_FILENO, recvbuff, recvsize );
  261. if ( writesize == -1 )
  262. {
  263. error_write( errno );
  264. exit( EXIT_FAILURE); // WTF?
  265. }
  266. ssize_t truncsize = recv( client->m_sock, recvbuff, writesize, MSG_TRUNC );
  267. if ( truncsize == -1 )
  268. {
  269. error_recv( errno );
  270. client_disconnect( client );
  271. return nready;
  272. }
  273. assert( truncsize == writesize ); //WTF?
  274. if ( writesize < recvsize )
  275. {
  276. g_set[STDOUT_FILENO].events |= POLLOUT;
  277. int clientindex = 0;
  278. set = g_set + (2 + g_servers.m_count);
  279. while ( nready > 0 && clientindex < g_clients.m_count )
  280. {
  281. if ( set->revents & POLLIN )
  282. {
  283. --nready;
  284. set->revents ^= POLLIN;
  285. }
  286. set->events &= ~POLLIN;
  287. ++set;
  288. ++clientindex;
  289. }
  290. }
  291. return nready;
  292. }
  293. void client_sendall(char* buff, const size_t buffsize)
  294. {
  295. int clientindex = 0;
  296. int setindex = client2set( 0 );
  297. while (clientindex < g_clients.m_count)
  298. {
  299. struct pollfd* set = g_set + setindex;
  300. struct TClient* client = g_clients.m_client + clientindex;
  301. int sendsize = send( client->m_sock, buff, buffsize, MSG_DONTWAIT );
  302. if ( sendsize == -1 )
  303. {
  304. if ( errno == EWOULDBLOCK )
  305. {
  306. sendsize = 0;
  307. }
  308. else
  309. {
  310. error_send( errno );
  311. client_disconnect( client );
  312. continue;
  313. }
  314. }
  315. if ( sendsize < buffsize )
  316. {
  317. ++g_clients.m_blocked;
  318. client->m_sendbuff = buff + sendsize;
  319. client->m_remain = buffsize - sendsize;
  320. set->events |= POLLOUT;
  321. }
  322. ++clientindex;
  323. ++setindex;
  324. }
  325. }
  326. void client_disconnect(struct TClient* client)
  327. {
  328. close( client->m_sock );
  329. if ( client->m_remain != 0 )
  330. {
  331. --g_clients.m_blocked;
  332. }
  333. if ( client->m_ipid != -1 )
  334. {
  335. kill( client->m_ipid, p_chldterm );
  336. }
  337. if ( client->m_opid != -1 )
  338. {
  339. kill( client->m_opid, p_chldterm );
  340. }
  341. client_del( client );
  342. }
  343. void client_tdisconnect(struct TClient* client)
  344. {
  345. if ( 0 < client->m_remain )
  346. {
  347. return;
  348. }
  349. if ( p_wait.tv_usec == -1 )
  350. {
  351. return;
  352. }
  353. if ( timer_iszero( &p_wait ) )
  354. {
  355. client_disconnect( client );
  356. return;
  357. }
  358. timer_init( client );
  359. }
  360. void client_disconnectall()
  361. {
  362. int index = 0;
  363. while ( index < g_clients.m_count )
  364. {
  365. struct TClient* client = g_clients.m_client + index;
  366. if ( (0 == client->m_timer.tv_sec) && (0 == client->m_timer.tv_usec) && (0 == client->m_remain) )
  367. {
  368. client_tdisconnect( client );
  369. }
  370. else
  371. {
  372. ++index;
  373. }
  374. }
  375. }