server.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #include <time.h>
  2. #include <stdio.h>
  3. #include <signal.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <unistd.h>
  7. #include <pthread.h>
  8. #include <arpa/inet.h>
  9. #include <sys/socket.h>
  10. const uint16_t SERVER_PORT = 1092;
  11. const int MAX_CLIENT_NUM = 20;
  12. const int MESSAGE_LEN = 256;
  13. const int HEADER_LEN = 4;
  14. int serv_sock;
  15. int clnt_socks[MAX_CLIENT_NUM];
  16. struct sockaddr_in clnt_addrs[MAX_CLIENT_NUM];
  17. struct message
  18. {
  19. uint8_t opcode, status;
  20. uint16_t length;
  21. char data[MESSAGE_LEN - HEADER_LEN];
  22. };
  23. struct client
  24. {
  25. uint16_t id, port;
  26. uint32_t addr;
  27. };
  28. void send_message(int clnt_sock, uint8_t opcode, uint8_t status, uint16_t length, char data[])
  29. {
  30. struct message msg;
  31. msg.opcode = opcode;
  32. msg.status = status;
  33. msg.length = htons(length);
  34. memcpy(msg.data, data, length);
  35. write(clnt_sock, &msg, MESSAGE_LEN);
  36. }
  37. void *conn(void *id)
  38. {
  39. uint16_t clnt_id = *(uint16_t *)id;
  40. int clnt_sock = clnt_socks[clnt_id];
  41. char data[MESSAGE_LEN];
  42. sprintf(data, "Hello, client %hu.", clnt_id);
  43. send_message(clnt_sock, 0, 1, strlen(data), data);
  44. for (;;)
  45. {
  46. struct message msg;
  47. int expect = MESSAGE_LEN;
  48. for (char *p = (char *)&msg; expect > 0;)
  49. {
  50. int len = read(clnt_sock, p, expect);
  51. if (len <= 0)
  52. {
  53. break;
  54. }
  55. expect -= len;
  56. p += len;
  57. }
  58. if (expect > 0)
  59. {
  60. break;
  61. }
  62. printf("Receive message from client %hu. opcode = %hu.\n", clnt_id, msg.opcode);
  63. switch (msg.opcode)
  64. {
  65. case 1:
  66. {
  67. time_t now;
  68. time(&now);
  69. strcpy(data, ctime(&now));
  70. send_message(clnt_sock, 1, 1, strlen(data) - 1, data);
  71. break;
  72. }
  73. case 2:
  74. {
  75. gethostname(data, MESSAGE_LEN);
  76. send_message(clnt_sock, 2, 1, strlen(data), data);
  77. break;
  78. }
  79. case 3:
  80. {
  81. int cnt = 0;
  82. for (uint16_t i = 0; i < MAX_CLIENT_NUM; i++)
  83. {
  84. if (clnt_socks[i] != -1)
  85. {
  86. ((struct client *)data)[cnt].id = htons(i);
  87. ((struct client *)data)[cnt].addr = clnt_addrs[i].sin_addr.s_addr;
  88. ((struct client *)data)[cnt].port = clnt_addrs[i].sin_port;
  89. cnt++;
  90. }
  91. }
  92. send_message(clnt_sock, 3, 1, sizeof(struct client) * cnt, data);
  93. break;
  94. }
  95. case 4:
  96. {
  97. uint16_t id = ntohs(((uint16_t *)msg.data)[0]);
  98. if (clnt_socks[id] == -1)
  99. {
  100. send_message(clnt_sock, 4, 2, 0, NULL);
  101. }
  102. else
  103. {
  104. ((uint16_t *)msg.data)[0] = htons(clnt_id);
  105. send_message(clnt_sock, 4, 1, 0, NULL);
  106. send_message(clnt_socks[id], 0, 0, strlen(msg.data + 2) + 2, msg.data);
  107. printf(" Forward message to client %hu.\n", id);
  108. }
  109. break;
  110. }
  111. default:
  112. {
  113. sprintf(data, "Invalid operation.");
  114. send_message(clnt_sock, 0, 2, strlen(data), data);
  115. }
  116. }
  117. }
  118. close(clnt_sock);
  119. printf("Client %hu disconnected.\n", clnt_id);
  120. clnt_socks[clnt_id] = -1;
  121. return NULL;
  122. }
  123. void stop_server(int sig)
  124. {
  125. for (uint16_t i = 0; i < MAX_CLIENT_NUM; i++)
  126. {
  127. if (clnt_socks[i] != -1)
  128. {
  129. close(clnt_socks[i]);
  130. printf("Client %hu disconnected.\n", i);
  131. clnt_socks[i] = -1;
  132. }
  133. }
  134. close(serv_sock);
  135. puts("Stop listening.");
  136. exit(0);
  137. }
  138. int main()
  139. {
  140. memset(clnt_socks, -1, sizeof(clnt_socks));
  141. serv_sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  142. if (serv_sock == -1)
  143. {
  144. puts("Create socket failed.");
  145. exit(1);
  146. }
  147. struct sockaddr_in serv_addr;
  148. memset(&serv_addr, 0, sizeof(serv_addr));
  149. serv_addr.sin_family = AF_INET;
  150. serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
  151. serv_addr.sin_port = htons(SERVER_PORT);
  152. if (bind(serv_sock, (const struct sockaddr *)&serv_addr, sizeof(serv_addr)) == -1)
  153. {
  154. puts("Bind socket failed.");
  155. exit(1);
  156. }
  157. if (listen(serv_sock, MAX_CLIENT_NUM) == -1)
  158. {
  159. puts("Listen socket failed.");
  160. exit(1);
  161. }
  162. puts("Start listening on port 1092...");
  163. signal(SIGINT, stop_server);
  164. for (;;)
  165. {
  166. struct sockaddr_in clnt_addr;
  167. socklen_t clnt_addr_size = sizeof(clnt_addr);
  168. int clnt_sock = accept(serv_sock, (struct sockaddr *)&clnt_addr, &clnt_addr_size);
  169. if (clnt_sock == -1)
  170. {
  171. puts("Create socket failed.");
  172. continue;
  173. }
  174. for (uint16_t i = 0; i < MAX_CLIENT_NUM; i++)
  175. {
  176. if (clnt_socks[i] == -1)
  177. {
  178. clnt_socks[i] = clnt_sock;
  179. clnt_addrs[i] = clnt_addr;
  180. pthread_t pid;
  181. if (pthread_create(&pid, NULL, conn, &i) != 0)
  182. {
  183. puts("Create thread failed.");
  184. clnt_socks[i] = -1;
  185. break;
  186. }
  187. printf("Client %hu (%s:%hu) connected. pid = %lu.\n", i, inet_ntoa(clnt_addr.sin_addr), ntohs(clnt_addr.sin_port), pid);
  188. break;
  189. }
  190. }
  191. }
  192. return 0;
  193. }