/storage/ndb/src/common/util/socket_io.cpp

https://github.com/suhailsherif/MySQL-5.1- · C++ · 332 lines · 247 code · 56 blank · 29 comment · 72 complexity · 7a25aef31c23381ce8573ff65521f0df MD5 · raw file

  1. /* Copyright (C) 2003 MySQL AB
  2. This program is free software; you can redistribute it and/or modify
  3. it under the terms of the GNU General Public License as published by
  4. the Free Software Foundation; version 2 of the License.
  5. This program is distributed in the hope that it will be useful,
  6. but WITHOUT ANY WARRANTY; without even the implied warranty of
  7. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  8. GNU General Public License for more details.
  9. You should have received a copy of the GNU General Public License
  10. along with this program; if not, write to the Free Software
  11. Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
  12. #include <ndb_global.h>
  13. #include <NdbTCP.h>
  14. #include <socket_io.h>
  15. #include <NdbOut.hpp>
  16. #include <NdbTick.h>
  17. extern "C"
  18. int
  19. read_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
  20. char * buf, int buflen){
  21. if(buflen < 1)
  22. return 0;
  23. fd_set readset;
  24. FD_ZERO(&readset);
  25. FD_SET(socket, &readset);
  26. struct timeval timeout;
  27. timeout.tv_sec = (timeout_millis / 1000);
  28. timeout.tv_usec = (timeout_millis % 1000) * 1000;
  29. const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
  30. if(selectRes == 0)
  31. return 0;
  32. if(selectRes == -1){
  33. return -1;
  34. }
  35. return recv(socket, &buf[0], buflen, 0);
  36. }
  37. extern "C"
  38. int
  39. readln_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  40. char * buf, int buflen, NdbMutex *mutex){
  41. if(buflen <= 1)
  42. return 0;
  43. fd_set readset;
  44. FD_ZERO(&readset);
  45. FD_SET(socket, &readset);
  46. struct timeval timeout;
  47. timeout.tv_sec = (timeout_millis / 1000);
  48. timeout.tv_usec = (timeout_millis % 1000) * 1000;
  49. if(mutex)
  50. NdbMutex_Unlock(mutex);
  51. Uint64 tick= NdbTick_CurrentMillisecond();
  52. const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
  53. *time= NdbTick_CurrentMillisecond() - tick;
  54. if(mutex)
  55. NdbMutex_Lock(mutex);
  56. if(selectRes == 0){
  57. return 0;
  58. }
  59. if(selectRes == -1){
  60. return -1;
  61. }
  62. char* ptr = buf;
  63. int len = buflen;
  64. do
  65. {
  66. int t;
  67. while((t = recv(socket, ptr, len, MSG_PEEK)) == -1 && errno == EINTR);
  68. if(t < 1)
  69. {
  70. return -1;
  71. }
  72. for(int i = 0; i<t; i++)
  73. {
  74. if(ptr[i] == '\n')
  75. {
  76. /**
  77. * Now consume
  78. */
  79. for (len = 1 + i; len; )
  80. {
  81. while ((t = recv(socket, ptr, len, 0)) == -1 && errno == EINTR);
  82. if (t < 1)
  83. return -1;
  84. ptr += t;
  85. len -= t;
  86. }
  87. if (i > 0 && buf[i-1] == '\r')
  88. {
  89. buf[i-1] = '\n';
  90. ptr--;
  91. }
  92. ptr[0]= 0;
  93. return ptr - buf;
  94. }
  95. }
  96. for (int tmp = t; tmp; )
  97. {
  98. while ((t = recv(socket, ptr, tmp, 0)) == -1 && errno == EINTR);
  99. if (t < 1)
  100. {
  101. return -1;
  102. }
  103. ptr += t;
  104. len -= t;
  105. tmp -= t;
  106. }
  107. FD_ZERO(&readset);
  108. FD_SET(socket, &readset);
  109. timeout.tv_sec = ((timeout_millis - *time) / 1000);
  110. timeout.tv_usec = ((timeout_millis - *time) % 1000) * 1000;
  111. tick= NdbTick_CurrentMillisecond();
  112. const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
  113. *time= NdbTick_CurrentMillisecond() - tick;
  114. if(selectRes != 1){
  115. return -1;
  116. }
  117. } while (len > 0);
  118. return -1;
  119. }
  120. extern "C"
  121. int
  122. write_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  123. const char buf[], int len){
  124. fd_set writeset;
  125. FD_ZERO(&writeset);
  126. FD_SET(socket, &writeset);
  127. struct timeval timeout;
  128. timeout.tv_sec = (timeout_millis / 1000);
  129. timeout.tv_usec = (timeout_millis % 1000) * 1000;
  130. Uint64 tick= NdbTick_CurrentMillisecond();
  131. const int selectRes = select(socket + 1, 0, &writeset, 0, &timeout);
  132. *time= NdbTick_CurrentMillisecond() - tick;
  133. if(selectRes != 1){
  134. return -1;
  135. }
  136. const char * tmp = &buf[0];
  137. while(len > 0){
  138. const int w = send(socket, tmp, len, 0);
  139. if(w == -1){
  140. return -1;
  141. }
  142. len -= w;
  143. tmp += w;
  144. if(len == 0)
  145. break;
  146. FD_ZERO(&writeset);
  147. FD_SET(socket, &writeset);
  148. timeout.tv_sec = ((timeout_millis - *time) / 1000);
  149. timeout.tv_usec = ((timeout_millis - *time) % 1000) * 1000;
  150. Uint64 tick= NdbTick_CurrentMillisecond();
  151. const int selectRes2 = select(socket + 1, 0, &writeset, 0, &timeout);
  152. *time= NdbTick_CurrentMillisecond() - tick;
  153. if(selectRes2 != 1){
  154. return -1;
  155. }
  156. }
  157. return 0;
  158. }
  159. extern "C"
  160. int
  161. print_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  162. const char * fmt, ...){
  163. va_list ap;
  164. va_start(ap, fmt);
  165. int ret = vprint_socket(socket, timeout_millis, time, fmt, ap);
  166. va_end(ap);
  167. return ret;
  168. }
  169. extern "C"
  170. int
  171. println_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  172. const char * fmt, ...){
  173. va_list ap;
  174. va_start(ap, fmt);
  175. int ret = vprintln_socket(socket, timeout_millis, time, fmt, ap);
  176. va_end(ap);
  177. return ret;
  178. }
  179. extern "C"
  180. int
  181. vprint_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  182. const char * fmt, va_list ap){
  183. char buf[1000];
  184. char *buf2 = buf;
  185. size_t size;
  186. if (fmt != 0 && fmt[0] != 0) {
  187. size = BaseString::vsnprintf(buf, sizeof(buf), fmt, ap);
  188. /* Check if the output was truncated */
  189. if(size > sizeof(buf)) {
  190. buf2 = (char *)malloc(size);
  191. if(buf2 == NULL)
  192. return -1;
  193. BaseString::vsnprintf(buf2, size, fmt, ap);
  194. }
  195. } else
  196. return 0;
  197. int ret = write_socket(socket, timeout_millis, time, buf2, size);
  198. if(buf2 != buf)
  199. free(buf2);
  200. return ret;
  201. }
  202. extern "C"
  203. int
  204. vprintln_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
  205. const char * fmt, va_list ap){
  206. char buf[1000];
  207. char *buf2 = buf;
  208. size_t size;
  209. if (fmt != 0 && fmt[0] != 0) {
  210. size = BaseString::vsnprintf(buf, sizeof(buf), fmt, ap)+1;// extra byte for '/n'
  211. /* Check if the output was truncated */
  212. if(size > sizeof(buf)) {
  213. buf2 = (char *)malloc(size);
  214. if(buf2 == NULL)
  215. return -1;
  216. BaseString::vsnprintf(buf2, size, fmt, ap);
  217. }
  218. } else {
  219. size = 1;
  220. }
  221. buf2[size-1]='\n';
  222. int ret = write_socket(socket, timeout_millis, time, buf2, size);
  223. if(buf2 != buf)
  224. free(buf2);
  225. return ret;
  226. }
  227. #ifdef NDB_WIN32
  228. class INIT_WINSOCK2
  229. {
  230. public:
  231. INIT_WINSOCK2(void);
  232. ~INIT_WINSOCK2(void);
  233. private:
  234. bool m_bAcceptable;
  235. };
  236. INIT_WINSOCK2 g_init_winsock2;
  237. INIT_WINSOCK2::INIT_WINSOCK2(void)
  238. : m_bAcceptable(false)
  239. {
  240. WORD wVersionRequested;
  241. WSADATA wsaData;
  242. int err;
  243. wVersionRequested = MAKEWORD( 2, 2 );
  244. err = WSAStartup( wVersionRequested, &wsaData );
  245. if ( err != 0 ) {
  246. /* Tell the user that we could not find a usable */
  247. /* WinSock DLL. */
  248. m_bAcceptable = false;
  249. }
  250. /* Confirm that the WinSock DLL supports 2.2.*/
  251. /* Note that if the DLL supports versions greater */
  252. /* than 2.2 in addition to 2.2, it will still return */
  253. /* 2.2 in wVersion since that is the version we */
  254. /* requested. */
  255. if ( LOBYTE( wsaData.wVersion ) != 2 ||
  256. HIBYTE( wsaData.wVersion ) != 2 ) {
  257. /* Tell the user that we could not find a usable */
  258. /* WinSock DLL. */
  259. WSACleanup( );
  260. m_bAcceptable = false;
  261. }
  262. /* The WinSock DLL is acceptable. Proceed. */
  263. m_bAcceptable = true;
  264. }
  265. INIT_WINSOCK2::~INIT_WINSOCK2(void)
  266. {
  267. if(m_bAcceptable)
  268. {
  269. m_bAcceptable = false;
  270. WSACleanup();
  271. }
  272. }
  273. #endif