tcpserver.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. /*
  2. * Copyright (c) 2006-2022, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2022-01-24 ChungHsuan improve code comments
  9. */
  10. #include <rtthread.h>
  11. #include <string.h>
  12. #if !defined(SAL_USING_POSIX)
  13. #error "Please enable SAL_USING_POSIX!"
  14. #else
  15. #include <sys/time.h>
  16. #include <sys/select.h>
  17. #endif
  18. #include <sys/socket.h> /* socket.h header file is needed when using BSD socket */ /* 使用BSD socket,需要包含socket.h头文件 */
  19. #include "netdb.h"
  20. #define DEBUG_TCP_SERVER
  21. #define DBG_TAG "TCP"
  22. #ifdef DEBUG_TCP_SERVER
  23. #define DBG_LVL DBG_LOG
  24. #else
  25. #define DBG_LVL DBG_INFO /* DBG_ERROR */
  26. #endif
  27. #include <rtdbg.h>
  28. #define BUFSZ (1024)
  29. static int started = 0;
  30. static int is_running = 0;
  31. static int port = 5000;
  32. static const char send_data[] = "This is TCP Server from RT-Thread."; /* The message be sent */ /* 发送用到的数据 */
  33. /**
  34. * @brief This function is for creating a tcp server on RT-Thread
  35. */
  36. static void tcpserv(void *arg)
  37. {
  38. int ret;
  39. char *recv_data; /* recv_data is a pointer used to receive data */ /* 用于接收的指针,后面会做一次动态分配以请求可用内存 */
  40. int sock, connected, bytes_received;
  41. struct sockaddr_in server_addr, client_addr;
  42. struct timeval timeout;
  43. fd_set readset, readset_c;
  44. socklen_t sin_size = sizeof(struct sockaddr_in);
  45. recv_data = rt_malloc(BUFSZ + 1);/* Allocate space for recv_data */ /* 分配接收用的数据缓冲 */
  46. if (recv_data == RT_NULL)
  47. {
  48. LOG_E("No memory");
  49. return;
  50. }
  51. /* Before making use of socket, socket should be created first and set the socket created to SOCK_STREAM(TCP) */
  52. /* 一个socket在使用前,需要预先创建出来,指定SOCK_STREAM为TCP的socket */
  53. if ((sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1)
  54. {
  55. LOG_E("Create socket error");
  56. goto __exit;
  57. }
  58. /* Initialize server side address */
  59. /* 初始化服务端地址 */
  60. server_addr.sin_family = AF_INET;
  61. server_addr.sin_port = htons(port); /*Server side port number*//* 服务端工作的端口 */
  62. server_addr.sin_addr.s_addr = INADDR_ANY;
  63. rt_memset(&(server_addr.sin_zero), 0x0, sizeof(server_addr.sin_zero));
  64. /* Bind socket to server side address */
  65. /* 绑定socket到服务端地址 */
  66. if (bind(sock, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
  67. {
  68. LOG_E("Unable to bind");
  69. goto __exit;
  70. }
  71. /* Listen on socket */
  72. /* 在socket上进行监听 */
  73. if (listen(sock, 10) == -1)
  74. {
  75. LOG_E("Listen error");
  76. goto __exit;
  77. }
  78. LOG_I("\nTCPServer Waiting for client on port %d...\n", port);
  79. started = 1;
  80. is_running = 1;
  81. timeout.tv_sec = 3;
  82. timeout.tv_usec = 0;
  83. while (is_running)
  84. {
  85. FD_ZERO(&readset);
  86. FD_SET(sock, &readset);
  87. LOG_I("Waiting for a new connection...");
  88. /* Wait for read or write */
  89. if (select(sock + 1, &readset, RT_NULL, RT_NULL, &timeout) == 0)
  90. continue;
  91. /* Accept a request from client and the function is blocking */
  92. /* 接受一个客户端连接socket的请求,这个函数调用是阻塞式的 */
  93. connected = accept(sock, (struct sockaddr *)&client_addr, &sin_size);
  94. /* Return the socket connected successfully */
  95. /* 返回的是连接成功的socket */
  96. if (connected < 0)
  97. {
  98. LOG_E("accept connection failed! errno = %d", errno);
  99. continue;
  100. }
  101. /* Accept the message which points by client address */
  102. /* 接受返回的client_addr指向了客户端的地址信息 */
  103. LOG_I("I got a connection from (%s , %d)\n",
  104. inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));
  105. /* Handle method of client connection */
  106. /* 客户端连接的处理 */
  107. while (is_running)
  108. {
  109. FD_ZERO(&readset_c);
  110. FD_SET(connected, &readset_c);
  111. /* Wait for read or write */
  112. if (select(connected + 1, &readset_c, RT_NULL, RT_NULL, &timeout) == 0)
  113. continue;
  114. /* Receive message from connected socket. Buffer size is 1024 bytes,but it's not guranteed to receive size exactly 1024 */
  115. /* 从connected socket中接收数据,接收buffer是1024大小,但并不一定能够收到1024大小的数据 */
  116. bytes_received = recv(connected, recv_data, BUFSZ, 0);
  117. if (bytes_received < 0)
  118. {
  119. LOG_E("Received error, close the connect.");
  120. closesocket(connected);
  121. connected = -1;
  122. break;
  123. }
  124. else if (bytes_received == 0)
  125. {
  126. /* Print warning message when recv function returns 0 */
  127. /* 打印recv函数返回值为0的警告信息 */
  128. LOG_W("Received warning, recv function returns 0.");
  129. continue;
  130. }
  131. else
  132. { /* Receive data successfully and append '\0' at the end of message */
  133. /* 有接收到数据,把末端清零 */
  134. recv_data[bytes_received] = '\0';
  135. if (strcmp(recv_data, "q") == 0 || strcmp(recv_data, "Q") == 0)
  136. {
  137. /* If the first letter is 'q' or 'Q', close the connection */
  138. /* 如果是首字母是q或Q,关闭这个连接 */
  139. LOG_I("Got a 'q' or 'Q', close the connect.");
  140. closesocket(connected);
  141. connected = -1;
  142. break;
  143. }
  144. else if (strcmp(recv_data, "exit") == 0)
  145. {
  146. /* If the message received is 'exit', close the whole server side. */
  147. /* 如果接收的是exit,则关闭整个服务端 */
  148. closesocket(connected);
  149. connected = -1;
  150. goto __exit;
  151. }
  152. else
  153. {
  154. /* Show the message in terminal */
  155. /* 在控制终端显示收到的数据 */
  156. LOG_D("Received data = %s", recv_data);
  157. }
  158. }
  159. /* Send message to connected socket */
  160. /* 发送数据到connected socket */
  161. ret = send(connected, send_data, rt_strlen(send_data), 0);
  162. if (ret < 0)
  163. {
  164. LOG_E("send error, close the connect.");
  165. closesocket(connected);
  166. connected = -1;
  167. break;
  168. }
  169. else if (ret == 0)
  170. {
  171. /* Print warning message when send function returns 0 */
  172. /* 打印send函数返回值为0的警告信息 */
  173. LOG_W("Send warning, send function returns 0.");
  174. }
  175. }
  176. }
  177. __exit:
  178. if (recv_data)
  179. {
  180. rt_free(recv_data);
  181. recv_data = RT_NULL;
  182. }
  183. if (connected >= 0)
  184. {
  185. closesocket(connected);
  186. connected = -1;
  187. }
  188. if (sock >= 0)
  189. {
  190. closesocket(sock);
  191. sock = -1;
  192. }
  193. started = 0;
  194. is_running = 0;
  195. return;
  196. }
  197. /**
  198. * @brief The usage description of tcp server on rt-Thread
  199. */
  200. static void usage(void)
  201. {
  202. rt_kprintf("Usage: tcpserver -p <port>\n");
  203. rt_kprintf(" tcpserver --stop\n");
  204. rt_kprintf(" tcpserver --help\n");
  205. rt_kprintf("\n");
  206. rt_kprintf("Miscellaneous:\n");
  207. rt_kprintf(" -p Specify the host port number\n");
  208. rt_kprintf(" --stop Stop tcpserver program\n");
  209. rt_kprintf(" --help Print help information\n");
  210. }
  211. /**
  212. * @brief This function is for testing tcp server on rt-Thread
  213. */
  214. static void tcpserver_test(int argc, char** argv)
  215. {
  216. rt_thread_t tid;
  217. if (argc == 1 || argc > 3)
  218. {
  219. LOG_I("Please check the command you entered!\n");
  220. goto __usage;
  221. }
  222. else
  223. {
  224. if (rt_strcmp(argv[1], "--help") == 0)
  225. {
  226. goto __usage;
  227. }
  228. else if (rt_strcmp(argv[1], "--stop") == 0)
  229. {
  230. is_running = 0;
  231. return;
  232. }
  233. else if (rt_strcmp(argv[1], "-p") == 0)
  234. {
  235. if (started)
  236. {
  237. LOG_I("The tcpserver has started!");
  238. LOG_I("Please stop tcpserver firstly, by: tcpserver --stop");
  239. return;
  240. }
  241. port = atoi(argv[2]);
  242. }
  243. else
  244. {
  245. goto __usage;
  246. }
  247. }
  248. tid = rt_thread_create("tcp_serv",
  249. tcpserv, RT_NULL,
  250. 2048, RT_THREAD_PRIORITY_MAX/3, 20);
  251. if (tid != RT_NULL)
  252. {
  253. rt_thread_startup(tid);
  254. }
  255. return;
  256. __usage:
  257. usage();
  258. }
  259. #ifdef RT_USING_FINSH
  260. MSH_CMD_EXPORT_ALIAS(tcpserver_test, tcpserver,
  261. Start a tcp server. Help: tcpserver --help);
  262. #endif