/ARDroneLib/VP_SDK/VP_Com/vp_com_socket_utils.c

https://github.com/zhengwy888/ARDrone-SDK_MF · C · 223 lines · 146 code · 49 blank · 28 comment · 31 complexity · e1ab9a4dcb69863e23cc7ca87fb320b1 MD5 · raw file

  1. #include <VP_Com/vp_com_socket.h>
  2. #include <VP_Com/vp_com_error.h>
  3. #include <VP_Os/vp_os_malloc.h>
  4. #include <VP_Os/vp_os_print.h>
  5. #include <VP_Os/vp_os_signal.h>
  6. #include <fcntl.h>
  7. #include <errno.h>
  8. #ifdef __linux__
  9. #include <sys/socket.h>
  10. #include <sys/ioctl.h>
  11. #include <netinet/in.h>
  12. #include <netinet/tcp.h>
  13. #include <unistd.h>
  14. #define CYGPKG_NET 1
  15. #endif
  16. int32_t vp_com_fill_read_fs(vp_com_socket_t* sockets, int32_t num_sockets, int32_t max, fd_set* read_fs )
  17. {
  18. while( num_sockets > 0 )
  19. {
  20. if( !sockets->is_disable )
  21. {
  22. int32_t s = (int32_t) sockets->priv;
  23. FD_SET( s, read_fs); // add the socket
  24. if( s > max )
  25. max = s;
  26. }
  27. sockets ++;
  28. num_sockets--;
  29. }
  30. return max;
  31. }
  32. void vp_com_close_client_sockets(vp_com_socket_t* client_sockets, int32_t num_client_sockets)
  33. {
  34. int32_t s;
  35. // Select timed out - We close all sockets because it should mean we lost connection with client
  36. while( num_client_sockets > 0 )
  37. {
  38. if( !client_sockets->is_disable )
  39. {
  40. s = (int32_t) client_sockets->priv;
  41. DEBUG_PRINT_SDK("[VP_COM_SERVER] Closing socket %d\n", (int)s);
  42. client_sockets->select( client_sockets->server,
  43. client_sockets,
  44. VP_COM_SOCKET_SELECT_DISABLE,
  45. (Write) vp_com_write_socket );
  46. if( client_sockets->protocol == VP_COM_TCP )
  47. {
  48. close( s );
  49. }
  50. vp_os_memset( client_sockets, 0, sizeof(vp_com_socket_t) );
  51. client_sockets->is_disable = TRUE;
  52. }
  53. client_sockets++;
  54. num_client_sockets--;
  55. }
  56. }
  57. C_RESULT vp_com_client_open_socket(vp_com_socket_t* server_socket, vp_com_socket_t* client_socket)
  58. {
  59. C_RESULT res;
  60. struct sockaddr_in raddr = { 0 }; // remote address
  61. socklen_t l = sizeof(raddr);
  62. int32_t s = (int32_t) server_socket->priv;
  63. Write write = (Write) (server_socket->protocol == VP_COM_TCP ? vp_com_write_socket : vp_com_write_udp_socket);
  64. vp_os_memcpy( client_socket, server_socket, sizeof(vp_com_socket_t) );
  65. res = server_socket->select( server_socket, client_socket, VP_COM_SOCKET_SELECT_ENABLE, write );
  66. if( SUCCEED(res) )
  67. {
  68. if( server_socket->protocol == VP_COM_TCP )
  69. {
  70. client_socket->priv = (void*)accept( s, (struct sockaddr*)&raddr, &l );
  71. }
  72. DEBUG_PRINT_SDK("[VP_COM_SERVER] Opening socket for server %d\n", (int)s);
  73. client_socket->server = server_socket;
  74. }
  75. else
  76. {
  77. DEBUG_PRINT_SDK("[VP_COM_SERVER] Failed to open socket for server %d\n", (int)s);
  78. vp_os_memset( client_socket, 0, sizeof(vp_com_socket_t) );
  79. }
  80. return res;
  81. }
  82. void vp_com_client_receive( vp_com_socket_t *client_socket )
  83. {
  84. static int8_t *local_buffer=NULL ;//[VP_COM_THREAD_LOCAL_BUFFER_MAX_SIZE];
  85. static int8_t *new_buffer=NULL;
  86. static int32_t local_buffer_length=0;
  87. struct sockaddr_in from;
  88. socklen_t fromlen;
  89. int32_t s, received,available;
  90. s = (int32_t) client_socket->priv;
  91. fromlen = sizeof(from);
  92. switch(client_socket->protocol)
  93. {
  94. case VP_COM_TCP:
  95. /* TCP : read as much information as we can */
  96. /* Resize the buffer */
  97. if (local_buffer==NULL || local_buffer_length<VP_COM_THREAD_LOCAL_BUFFER_SIZE) {
  98. new_buffer=(int8_t *)vp_os_realloc(local_buffer,VP_COM_THREAD_LOCAL_BUFFER_SIZE);
  99. if (new_buffer==NULL) { PRINT("Not enough memory to store TCP data.\n"); return; }
  100. local_buffer = new_buffer;
  101. local_buffer_length = VP_COM_THREAD_LOCAL_BUFFER_SIZE;
  102. }
  103. /* Fetch data */
  104. received = recvfrom(s, (char*)local_buffer,local_buffer_length, 0, (struct sockaddr*)&from, &fromlen);
  105. /* We close the socket if an error occurred */
  106. if( received <= 0)
  107. {
  108. PRINT("\nClosing socket on port %i.\n",client_socket->port);
  109. client_socket->select( client_socket->server, client_socket, VP_COM_SOCKET_SELECT_DISABLE, (Write) vp_com_write_socket );
  110. close( s );
  111. vp_os_memset( client_socket, 0, sizeof(vp_com_socket_t) );
  112. client_socket->is_disable = TRUE;
  113. }
  114. /* If some data were received, we pass them to the application. */
  115. else if( received>=0 && client_socket->read != NULL )
  116. {
  117. client_socket->remotePort = ntohs(from.sin_port);
  118. client_socket->read( (void*) client_socket, local_buffer, &received, (&from)->sin_addr.s_addr );
  119. }
  120. break; // End of TCP processing
  121. case VP_COM_UDP:
  122. /* UDP : read one packet */
  123. /* Query packet size */
  124. available = recvfrom(s, (char*)local_buffer,0, MSG_PEEK|MSG_TRUNC, (struct sockaddr*)&from, &fromlen);
  125. /* Resize the buffer */
  126. if(available>0){
  127. if (available>VP_COM_THREAD_LOCAL_BUFFER_MAX_SIZE) {
  128. PRINT("UDP packet is bigger than maximum authorized size. Dropping packet.\n"); return;
  129. }
  130. if (local_buffer==NULL || local_buffer_length<available) {
  131. new_buffer=(int8_t *)vp_os_realloc(local_buffer,available);
  132. if (new_buffer==NULL) {
  133. PRINT("UDP packet is bigger than available memory. Dropping packet.\n"); return;
  134. }
  135. local_buffer = new_buffer; local_buffer_length = available;
  136. }}
  137. else { return ; }
  138. /* Fetch data */
  139. received = recvfrom(s, (char*)local_buffer,local_buffer_length, 0, (struct sockaddr*)&from, &fromlen);
  140. /* Closes the socket if an error occurred on an UDP socket */
  141. if( received < 0 )
  142. {
  143. PRINT("\nClosing socket on port %i.\n",client_socket->port);
  144. client_socket->select( client_socket->server, client_socket, VP_COM_SOCKET_SELECT_DISABLE, (Write) vp_com_write_socket );
  145. close( s );
  146. vp_os_memset( client_socket, 0, sizeof(vp_com_socket_t) );
  147. client_socket->is_disable = TRUE;
  148. }
  149. /* If some data were received, we pass them to the application. */
  150. else if( received>=0 && client_socket->read != NULL )
  151. {
  152. client_socket->remotePort = ntohs(from.sin_port);
  153. client_socket->read( (void*) client_socket, local_buffer, &received, (&from)->sin_addr.s_addr );
  154. }
  155. break; // End of UDP processing
  156. default: break;
  157. }
  158. //DEBUG_STEPH - debugging the fact that an empty UDP packet crashes the Mykonos program
  159. /* PRINT("Received (%i bytes) from (ip %i.%i.%i.%i) on (port %i) - ",
  160. received,
  161. ((((struct sockaddr_in*)&from)->sin_addr.s_addr))&0x0FF,
  162. ((((struct sockaddr_in*)&from)->sin_addr.s_addr)>>8)&0x0FF,
  163. ((((struct sockaddr_in*)&from)->sin_addr.s_addr)>>16)&0x0FF,
  164. ((((struct sockaddr_in*)&from)->sin_addr.s_addr)>>24)&0x0FF,
  165. ntohs(((struct sockaddr_in*)&from)->sin_port)
  166. );
  167. switch(client_socket->protocol){
  168. case VP_COM_UDP: printf(" UDP"); break;
  169. case VP_COM_TCP: printf(" TCP"); break;
  170. default : printf("unknown\n"); break;
  171. }
  172. */
  173. }