rtlink_port_udp.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2021-7-7 songchao the first version
  9. */
  10. #include <rtthread.h>
  11. #include <rtlink_port.h>
  12. #include <string.h>
  13. #if !defined(SAL_USING_POSIX)
  14. #error "Please enable SAL_USING_POSIX!"
  15. #else
  16. #include <sys/time.h>
  17. #include <sys/select.h>
  18. #endif
  19. #include <sys/socket.h> /* To use the BSD socket, you need to include the socket.h header file */
  20. #include "netdb.h"
  21. #define DBG_TAG "rtlink_port"
  22. #define DBG_LVL DBG_INFO
  23. #include <rtdbg.h>
  24. #define BUFSZ 4096
  25. static struct sockaddr_in local_addr;
  26. static struct sockaddr_in remote_addr;
  27. static uint8_t socket_buffer[BUFSZ];
  28. static int32_t rtlink_socket = -1;
  29. static void rtlink_socket_thread_entry(void *param)
  30. {
  31. fd_set readset;
  32. uint32_t bytes_received;
  33. struct timeval timeout;
  34. socklen_t addr_len;
  35. struct sockaddr_in socket_addr;
  36. addr_len = sizeof(struct sockaddr);
  37. timeout.tv_sec = 0;
  38. timeout.tv_usec = 10000;
  39. while(1)
  40. {
  41. if(rtlink_socket == -1)
  42. {
  43. rt_thread_delay(100);
  44. }
  45. else
  46. {
  47. FD_ZERO(&readset);
  48. FD_SET(rtlink_socket, &readset);
  49. if (select(rtlink_socket + 1, &readset, RT_NULL, RT_NULL, &timeout) == 0)
  50. {
  51. continue;
  52. }
  53. else
  54. {
  55. bytes_received = recvfrom(rtlink_socket, socket_buffer, BUFSZ - 1, 0,
  56. (struct sockaddr *)&socket_addr, &addr_len);
  57. if (bytes_received < 0)
  58. {
  59. LOG_E("Received error, close the socket.");
  60. closesocket(rtlink_socket);
  61. rtlink_socket = -1;
  62. }
  63. else if (bytes_received == 0)
  64. {
  65. LOG_W("Received warning, recv function return 0.");
  66. }
  67. else
  68. {
  69. rt_link_hw_write_cb(&socket_buffer, bytes_received);
  70. }
  71. }
  72. }
  73. }
  74. }
  75. rt_err_t rt_link_port_reconnect(void)
  76. {
  77. return RT_EOK;
  78. }
  79. rt_size_t rt_link_port_send(void *data, rt_size_t length)
  80. {
  81. rt_size_t size = 0;
  82. ssize_t ret;
  83. static rt_size_t udp_send_count = 0;
  84. if(rtlink_socket != -1)
  85. {
  86. ret = sendto(rtlink_socket, data, length, 0,(struct sockaddr *)&remote_addr, sizeof(struct sockaddr));
  87. if (ret < 0)
  88. {
  89. LOG_I("send error, close the socket.");
  90. closesocket(rtlink_socket);
  91. rtlink_socket = -1;
  92. }
  93. else if (ret == 0)
  94. {
  95. LOG_W("Send warning, send function return 0.");
  96. }
  97. else
  98. {
  99. udp_send_count++;
  100. }
  101. rt_thread_delay(1);
  102. }
  103. return size;
  104. }
  105. rt_err_t rt_link_port_init(void)
  106. {
  107. if ((rtlink_socket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) == -1)
  108. {
  109. LOG_E("Create socket error");
  110. return 0;
  111. }
  112. local_addr.sin_family = AF_INET;
  113. local_addr.sin_port = htons(RTLINK_LOCAL_UDP_PORT);
  114. rt_memset(&(local_addr.sin_zero), 0, sizeof(local_addr.sin_zero));
  115. remote_addr.sin_family = AF_INET;
  116. remote_addr.sin_port = htons(RTLINK_REMOTE_UDP_PORT);
  117. remote_addr.sin_addr.s_addr = inet_addr(RTLINK_REMOTE_HOST_IP);
  118. rt_memset(&(remote_addr.sin_zero), 0, sizeof(remote_addr.sin_zero));
  119. if (bind(rtlink_socket, (struct sockaddr *)&local_addr,sizeof(struct sockaddr)) == -1)
  120. {
  121. LOG_E("Unable to bind");
  122. closesocket(rtlink_socket);
  123. rtlink_socket = -1;
  124. return 0;
  125. }
  126. LOG_D("bind to port %d ip_addr %s success\n",RTLINK_LOCAL_UDP_PORT,RTLINK_REMOTE_HOST_IP);
  127. rt_thread_t rtlink_socket_tid;
  128. rtlink_socket_tid = rt_thread_create("rt_link_socket_thread",
  129. rtlink_socket_thread_entry,
  130. RT_NULL,
  131. 4096,
  132. RT_THREAD_PRIORITY_MAX - 2,
  133. 2);
  134. if (rtlink_socket_tid != RT_NULL)
  135. {
  136. rt_thread_startup(rtlink_socket_tid);
  137. }
  138. return RT_EOK;
  139. }
  140. rt_err_t rt_link_port_deinit(void)
  141. {
  142. closesocket(rtlink_socket);
  143. rtlink_socket = -1;
  144. return RT_EOK;
  145. }