/release/src/router/mysql/extra/yassl/src/socket_wrapper.cpp

https://gitlab.com/envieidoc/advancedtomato2 · C++ · 208 lines · 134 code · 46 blank · 28 comment · 21 complexity · a77431f7d004fcb8a7a7597a6e913cc8 MD5 · raw file

  1. /*
  2. Copyright (c) 2005, 2012, Oracle and/or its affiliates. All rights reserved.
  3. This program is free software; you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation; version 2 of the License.
  6. This program is distributed in the hope that it will be useful,
  7. but WITHOUT ANY WARRANTY; without even the implied warranty of
  8. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  9. GNU General Public License for more details.
  10. You should have received a copy of the GNU General Public License
  11. along with this program; see the file COPYING. If not, write to the
  12. Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston,
  13. MA 02110-1301 USA.
  14. */
  15. /* The socket wrapper source implements a Socket class that hides the
  16. * differences between Berkely style sockets and Windows sockets, allowing
  17. * transparent TCP access.
  18. */
  19. #include "runtime.hpp"
  20. #include "socket_wrapper.hpp"
  21. #ifndef _WIN32
  22. #include <errno.h>
  23. #include <netdb.h>
  24. #include <unistd.h>
  25. #include <arpa/inet.h>
  26. #include <netinet/in.h>
  27. #include <sys/ioctl.h>
  28. #include <string.h>
  29. #include <fcntl.h>
  30. #endif // _WIN32
  31. #if defined(__sun) || defined(__SCO_VERSION__)
  32. #include <sys/filio.h>
  33. #endif
  34. #ifdef _WIN32
  35. const int SOCKET_EINVAL = WSAEINVAL;
  36. const int SOCKET_EWOULDBLOCK = WSAEWOULDBLOCK;
  37. const int SOCKET_EAGAIN = WSAEWOULDBLOCK;
  38. #else
  39. const int SOCKET_EINVAL = EINVAL;
  40. const int SOCKET_EWOULDBLOCK = EWOULDBLOCK;
  41. const int SOCKET_EAGAIN = EAGAIN;
  42. #endif // _WIN32
  43. namespace yaSSL {
  44. Socket::Socket(socket_t s)
  45. : socket_(s), wouldBlock_(false), nonBlocking_(false)
  46. {}
  47. void Socket::set_fd(socket_t s)
  48. {
  49. socket_ = s;
  50. }
  51. socket_t Socket::get_fd() const
  52. {
  53. return socket_;
  54. }
  55. Socket::~Socket()
  56. {
  57. // don't close automatically now
  58. }
  59. void Socket::closeSocket()
  60. {
  61. if (socket_ != INVALID_SOCKET) {
  62. #ifdef _WIN32
  63. closesocket(socket_);
  64. #else
  65. close(socket_);
  66. #endif
  67. socket_ = INVALID_SOCKET;
  68. }
  69. }
  70. uint Socket::get_ready() const
  71. {
  72. #ifdef _WIN32
  73. unsigned long ready = 0;
  74. ioctlsocket(socket_, FIONREAD, &ready);
  75. #else
  76. /*
  77. 64-bit Solaris requires the variable passed to
  78. FIONREAD be a 32-bit value.
  79. */
  80. unsigned int ready = 0;
  81. ioctl(socket_, FIONREAD, &ready);
  82. #endif
  83. return ready;
  84. }
  85. uint Socket::send(const byte* buf, unsigned int sz, unsigned int& written,
  86. int flags)
  87. {
  88. const byte* pos = buf;
  89. const byte* end = pos + sz;
  90. wouldBlock_ = false;
  91. while (pos != end) {
  92. int sent = ::send(socket_, reinterpret_cast<const char *>(pos),
  93. static_cast<int>(end - pos), flags);
  94. if (sent == -1) {
  95. if (get_lastError() == SOCKET_EWOULDBLOCK ||
  96. get_lastError() == SOCKET_EAGAIN) {
  97. wouldBlock_ = true; // would have blocked this time only
  98. nonBlocking_ = true; // nonblocking, win32 only way to tell
  99. return 0;
  100. }
  101. return static_cast<uint>(-1);
  102. }
  103. pos += sent;
  104. written += sent;
  105. }
  106. return sz;
  107. }
  108. uint Socket::receive(byte* buf, unsigned int sz, int flags)
  109. {
  110. wouldBlock_ = false;
  111. int recvd = ::recv(socket_, reinterpret_cast<char *>(buf), sz, flags);
  112. // idea to seperate error from would block by arnetheduck@gmail.com
  113. if (recvd == -1) {
  114. if (get_lastError() == SOCKET_EWOULDBLOCK ||
  115. get_lastError() == SOCKET_EAGAIN) {
  116. wouldBlock_ = true; // would have blocked this time only
  117. nonBlocking_ = true; // socket nonblocking, win32 only way to tell
  118. return 0;
  119. }
  120. }
  121. else if (recvd == 0)
  122. return static_cast<uint>(-1);
  123. return recvd;
  124. }
  125. // wait if blocking for input, return false for error
  126. bool Socket::wait()
  127. {
  128. byte b;
  129. return receive(&b, 1, MSG_PEEK) != static_cast<uint>(-1);
  130. }
  131. void Socket::shutDown(int how)
  132. {
  133. shutdown(socket_, how);
  134. }
  135. int Socket::get_lastError()
  136. {
  137. #ifdef _WIN32
  138. return WSAGetLastError();
  139. #else
  140. return errno;
  141. #endif
  142. }
  143. bool Socket::WouldBlock() const
  144. {
  145. return wouldBlock_;
  146. }
  147. bool Socket::IsNonBlocking() const
  148. {
  149. return nonBlocking_;
  150. }
  151. void Socket::set_lastError(int errorCode)
  152. {
  153. #ifdef _WIN32
  154. WSASetLastError(errorCode);
  155. #else
  156. errno = errorCode;
  157. #endif
  158. }
  159. } // namespace