tcp_server.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. /*
  2. * Copyright (C) 2019 Intel Corporation. All rights reserved.
  3. * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. */
  5. #include "socket_utils.h"
  6. #include <arpa/inet.h>
  7. #include <netinet/in.h>
  8. #include <pthread.h>
  9. #include <stdbool.h>
  10. #include <stdio.h>
  11. #include <stdlib.h>
  12. #include <stdint.h>
  13. #include <string.h>
  14. #include <sys/socket.h>
  15. #include <unistd.h>
  16. #ifdef __wasi__
  17. #include <wasi_socket_ext.h>
  18. #endif
  19. #define WORKER_NUM 5
  20. void *
  21. run(void *arg)
  22. {
  23. const char *message = "Say Hi from the Server\n";
  24. int new_socket = *(int *)arg;
  25. int i;
  26. printf("[Server] Communicate with the new connection #%u @ %p ..\n",
  27. new_socket, (void *)(uintptr_t)pthread_self());
  28. for (i = 0; i < 5; i++) {
  29. if (send(new_socket, message, strlen(message), 0) < 0) {
  30. perror("Send failed");
  31. break;
  32. }
  33. }
  34. printf("[Server] Shuting down the new connection #%u ..\n", new_socket);
  35. shutdown(new_socket, SHUT_RDWR);
  36. return NULL;
  37. }
  38. static void
  39. init_sockaddr_inet(struct sockaddr_in *addr)
  40. {
  41. /* 0.0.0.0:1234 */
  42. addr->sin_family = AF_INET;
  43. addr->sin_port = htons(1234);
  44. addr->sin_addr.s_addr = htonl(INADDR_ANY);
  45. }
  46. static void
  47. init_sockaddr_inet6(struct sockaddr_in6 *addr)
  48. {
  49. /* [::]:1234 */
  50. addr->sin6_family = AF_INET6;
  51. addr->sin6_port = htons(1234);
  52. addr->sin6_addr = in6addr_any;
  53. }
  54. int
  55. main(int argc, char *argv[])
  56. {
  57. int socket_fd = -1, addrlen = 0, af;
  58. struct sockaddr_storage addr = { 0 };
  59. unsigned connections = 0;
  60. pthread_t workers[WORKER_NUM] = { 0 };
  61. int client_sock_fds[WORKER_NUM] = { 0 };
  62. char ip_string[64];
  63. if (argc > 1 && strcmp(argv[1], "inet6") == 0) {
  64. af = AF_INET6;
  65. addrlen = sizeof(struct sockaddr_in6);
  66. init_sockaddr_inet6((struct sockaddr_in6 *)&addr);
  67. }
  68. else {
  69. af = AF_INET;
  70. addrlen = sizeof(struct sockaddr_in6);
  71. init_sockaddr_inet((struct sockaddr_in *)&addr);
  72. }
  73. printf("[Server] Create socket\n");
  74. socket_fd = socket(af, SOCK_STREAM, 0);
  75. if (socket_fd < 0) {
  76. perror("Create socket failed");
  77. goto fail;
  78. }
  79. printf("[Server] Bind socket\n");
  80. if (bind(socket_fd, (struct sockaddr *)&addr, addrlen) < 0) {
  81. perror("Bind failed");
  82. goto fail;
  83. }
  84. printf("[Server] Listening on socket\n");
  85. if (listen(socket_fd, 3) < 0) {
  86. perror("Listen failed");
  87. goto fail;
  88. }
  89. printf("[Server] Wait for clients to connect ..\n");
  90. while (connections < WORKER_NUM) {
  91. addrlen = sizeof(struct sockaddr);
  92. client_sock_fds[connections] =
  93. accept(socket_fd, (struct sockaddr *)&addr, (socklen_t *)&addrlen);
  94. if (client_sock_fds[connections] < 0) {
  95. perror("Accept failed");
  96. break;
  97. }
  98. if (sockaddr_to_string((struct sockaddr *)&addr, ip_string,
  99. sizeof(ip_string) / sizeof(ip_string[0]))
  100. != 0) {
  101. printf("[Server] failed to parse client address\n");
  102. goto fail;
  103. }
  104. printf("[Server] Client connected (%s)\n", ip_string);
  105. if (pthread_create(&workers[connections], NULL, run,
  106. &client_sock_fds[connections])) {
  107. perror("Create a worker thread failed");
  108. shutdown(client_sock_fds[connections], SHUT_RDWR);
  109. break;
  110. }
  111. connections++;
  112. }
  113. if (connections == WORKER_NUM) {
  114. printf("[Server] Achieve maximum amount of connections\n");
  115. }
  116. for (int i = 0; i < WORKER_NUM; i++) {
  117. pthread_join(workers[i], NULL);
  118. }
  119. printf("[Server] Shuting down ..\n");
  120. shutdown(socket_fd, SHUT_RDWR);
  121. sleep(3);
  122. printf("[Server] BYE \n");
  123. return EXIT_SUCCESS;
  124. fail:
  125. printf("[Server] Shuting down ..\n");
  126. if (socket_fd >= 0)
  127. close(socket_fd);
  128. sleep(3);
  129. return EXIT_FAILURE;
  130. }