server.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  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 uint16_t MAX_CLIENT_NUM = 20;
  12. const uint16_t BUF_SIZE = 256;
  13. int serv_sock; // server socket
  14. int clnt_socks[MAX_CLIENT_NUM]; // client sockets
  15. struct sockaddr_in clnt_addrs[MAX_CLIENT_NUM]; // client addresses
  16. struct buffer
  17. {
  18. uint16_t ptr, len;
  19. char buf[BUF_SIZE];
  20. } clnt_buf[MAX_CLIENT_NUM]; // client buffers
  21. // get char from client socket
  22. int sgetc(uint16_t clnt_id)
  23. {
  24. struct buffer *b = &clnt_buf[clnt_id];
  25. if (b->ptr == b->len)
  26. {
  27. if (b->len == BUF_SIZE)
  28. {
  29. b->ptr = b->len = 0;
  30. }
  31. ssize_t n = read(clnt_socks[clnt_id], b->buf, BUF_SIZE - b->len);
  32. if (n == -1)
  33. {
  34. return -1;
  35. }
  36. if (n == 0)
  37. {
  38. return -2;
  39. }
  40. }
  41. return b->buf[b->ptr++];
  42. }
  43. void *conn(void *id)
  44. {
  45. uint16_t clnt_id = *(uint16_t *)id;
  46. int clnt_sock = clnt_socks[clnt_id];
  47. for (;;)
  48. {
  49. }
  50. close(clnt_sock);
  51. printf("Client %hu disconnected.\n", clnt_id);
  52. clnt_socks[clnt_id] = -1;
  53. return NULL;
  54. }
  55. void stop_server(int sig)
  56. {
  57. for (uint16_t i = 0; i < MAX_CLIENT_NUM; i++)
  58. {
  59. if (clnt_socks[i] != -1)
  60. {
  61. close(clnt_socks[i]);
  62. printf("Client %hu disconnected.\n", i);
  63. clnt_socks[i] = -1;
  64. }
  65. }
  66. close(serv_sock);
  67. puts("Stop listening.");
  68. exit(0);
  69. }
  70. int main()
  71. {
  72. memset(clnt_socks, -1, sizeof(clnt_socks));
  73. serv_sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
  74. if (serv_sock == -1)
  75. {
  76. puts("Create socket failed.");
  77. exit(1);
  78. }
  79. struct sockaddr_in serv_addr;
  80. memset(&serv_addr, 0, sizeof(serv_addr));
  81. serv_addr.sin_family = AF_INET;
  82. serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
  83. serv_addr.sin_port = htons(SERVER_PORT);
  84. if (bind(serv_sock, (const struct sockaddr *)&serv_addr, sizeof(serv_addr)) == -1)
  85. {
  86. puts("Bind socket failed.");
  87. exit(1);
  88. }
  89. if (listen(serv_sock, MAX_CLIENT_NUM) == -1)
  90. {
  91. puts("Listen socket failed.");
  92. exit(1);
  93. }
  94. puts("Start listening on port 1092...");
  95. signal(SIGINT, stop_server);
  96. for (;;)
  97. {
  98. struct sockaddr_in clnt_addr;
  99. socklen_t clnt_addr_size = sizeof(clnt_addr);
  100. int clnt_sock = accept(serv_sock, (struct sockaddr *)&clnt_addr, &clnt_addr_size);
  101. if (clnt_sock == -1)
  102. {
  103. puts("Create socket failed.");
  104. continue;
  105. }
  106. for (uint16_t i = 0; i < MAX_CLIENT_NUM; i++)
  107. {
  108. if (clnt_socks[i] == -1)
  109. {
  110. clnt_socks[i] = clnt_sock;
  111. clnt_addrs[i] = clnt_addr;
  112. pthread_t pid;
  113. if (pthread_create(&pid, NULL, conn, &i) != 0)
  114. {
  115. puts("Create thread failed.");
  116. clnt_socks[i] = -1;
  117. break;
  118. }
  119. printf("Client %hu (%s:%hu) connected. pid = %lu.\n", i, inet_ntoa(clnt_addr.sin_addr), ntohs(clnt_addr.sin_port), pid);
  120. break;
  121. }
  122. }
  123. }
  124. return 0;
  125. }