/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
- /* Copyright (C) 2003 MySQL AB
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; version 2 of the License.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
- #include <ndb_global.h>
- #include <NdbTCP.h>
- #include <socket_io.h>
- #include <NdbOut.hpp>
- #include <NdbTick.h>
- extern "C"
- int
- read_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
- char * buf, int buflen){
- if(buflen < 1)
- return 0;
-
- fd_set readset;
- FD_ZERO(&readset);
- FD_SET(socket, &readset);
-
- struct timeval timeout;
- timeout.tv_sec = (timeout_millis / 1000);
- timeout.tv_usec = (timeout_millis % 1000) * 1000;
- const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
- if(selectRes == 0)
- return 0;
-
- if(selectRes == -1){
- return -1;
- }
- return recv(socket, &buf[0], buflen, 0);
- }
- extern "C"
- int
- readln_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- char * buf, int buflen, NdbMutex *mutex){
- if(buflen <= 1)
- return 0;
- fd_set readset;
- FD_ZERO(&readset);
- FD_SET(socket, &readset);
- struct timeval timeout;
- timeout.tv_sec = (timeout_millis / 1000);
- timeout.tv_usec = (timeout_millis % 1000) * 1000;
- if(mutex)
- NdbMutex_Unlock(mutex);
- Uint64 tick= NdbTick_CurrentMillisecond();
- const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
- *time= NdbTick_CurrentMillisecond() - tick;
- if(mutex)
- NdbMutex_Lock(mutex);
- if(selectRes == 0){
- return 0;
- }
- if(selectRes == -1){
- return -1;
- }
- char* ptr = buf;
- int len = buflen;
- do
- {
- int t;
- while((t = recv(socket, ptr, len, MSG_PEEK)) == -1 && errno == EINTR);
-
- if(t < 1)
- {
- return -1;
- }
-
- for(int i = 0; i<t; i++)
- {
- if(ptr[i] == '\n')
- {
- /**
- * Now consume
- */
- for (len = 1 + i; len; )
- {
- while ((t = recv(socket, ptr, len, 0)) == -1 && errno == EINTR);
- if (t < 1)
- return -1;
- ptr += t;
- len -= t;
- }
- if (i > 0 && buf[i-1] == '\r')
- {
- buf[i-1] = '\n';
- ptr--;
- }
- ptr[0]= 0;
- return ptr - buf;
- }
- }
-
- for (int tmp = t; tmp; )
- {
- while ((t = recv(socket, ptr, tmp, 0)) == -1 && errno == EINTR);
- if (t < 1)
- {
- return -1;
- }
- ptr += t;
- len -= t;
- tmp -= t;
- }
- FD_ZERO(&readset);
- FD_SET(socket, &readset);
- timeout.tv_sec = ((timeout_millis - *time) / 1000);
- timeout.tv_usec = ((timeout_millis - *time) % 1000) * 1000;
- tick= NdbTick_CurrentMillisecond();
- const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
- *time= NdbTick_CurrentMillisecond() - tick;
- if(selectRes != 1){
- return -1;
- }
- } while (len > 0);
-
- return -1;
- }
- extern "C"
- int
- write_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- const char buf[], int len){
- fd_set writeset;
- FD_ZERO(&writeset);
- FD_SET(socket, &writeset);
- struct timeval timeout;
- timeout.tv_sec = (timeout_millis / 1000);
- timeout.tv_usec = (timeout_millis % 1000) * 1000;
- Uint64 tick= NdbTick_CurrentMillisecond();
- const int selectRes = select(socket + 1, 0, &writeset, 0, &timeout);
- *time= NdbTick_CurrentMillisecond() - tick;
- if(selectRes != 1){
- return -1;
- }
- const char * tmp = &buf[0];
- while(len > 0){
- const int w = send(socket, tmp, len, 0);
- if(w == -1){
- return -1;
- }
- len -= w;
- tmp += w;
-
- if(len == 0)
- break;
-
- FD_ZERO(&writeset);
- FD_SET(socket, &writeset);
- timeout.tv_sec = ((timeout_millis - *time) / 1000);
- timeout.tv_usec = ((timeout_millis - *time) % 1000) * 1000;
- Uint64 tick= NdbTick_CurrentMillisecond();
- const int selectRes2 = select(socket + 1, 0, &writeset, 0, &timeout);
- *time= NdbTick_CurrentMillisecond() - tick;
- if(selectRes2 != 1){
- return -1;
- }
- }
-
- return 0;
- }
- extern "C"
- int
- print_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- const char * fmt, ...){
- va_list ap;
- va_start(ap, fmt);
- int ret = vprint_socket(socket, timeout_millis, time, fmt, ap);
- va_end(ap);
- return ret;
- }
- extern "C"
- int
- println_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- const char * fmt, ...){
- va_list ap;
- va_start(ap, fmt);
- int ret = vprintln_socket(socket, timeout_millis, time, fmt, ap);
- va_end(ap);
- return ret;
- }
- extern "C"
- int
- vprint_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- const char * fmt, va_list ap){
- char buf[1000];
- char *buf2 = buf;
- size_t size;
- if (fmt != 0 && fmt[0] != 0) {
- size = BaseString::vsnprintf(buf, sizeof(buf), fmt, ap);
- /* Check if the output was truncated */
- if(size > sizeof(buf)) {
- buf2 = (char *)malloc(size);
- if(buf2 == NULL)
- return -1;
- BaseString::vsnprintf(buf2, size, fmt, ap);
- }
- } else
- return 0;
- int ret = write_socket(socket, timeout_millis, time, buf2, size);
- if(buf2 != buf)
- free(buf2);
- return ret;
- }
- extern "C"
- int
- vprintln_socket(NDB_SOCKET_TYPE socket, int timeout_millis, int *time,
- const char * fmt, va_list ap){
- char buf[1000];
- char *buf2 = buf;
- size_t size;
- if (fmt != 0 && fmt[0] != 0) {
- size = BaseString::vsnprintf(buf, sizeof(buf), fmt, ap)+1;// extra byte for '/n'
- /* Check if the output was truncated */
- if(size > sizeof(buf)) {
- buf2 = (char *)malloc(size);
- if(buf2 == NULL)
- return -1;
- BaseString::vsnprintf(buf2, size, fmt, ap);
- }
- } else {
- size = 1;
- }
- buf2[size-1]='\n';
- int ret = write_socket(socket, timeout_millis, time, buf2, size);
- if(buf2 != buf)
- free(buf2);
- return ret;
- }
- #ifdef NDB_WIN32
- class INIT_WINSOCK2
- {
- public:
- INIT_WINSOCK2(void);
- ~INIT_WINSOCK2(void);
- private:
- bool m_bAcceptable;
- };
- INIT_WINSOCK2 g_init_winsock2;
- INIT_WINSOCK2::INIT_WINSOCK2(void)
- : m_bAcceptable(false)
- {
- WORD wVersionRequested;
- WSADATA wsaData;
- int err;
-
- wVersionRequested = MAKEWORD( 2, 2 );
-
- err = WSAStartup( wVersionRequested, &wsaData );
- if ( err != 0 ) {
- /* Tell the user that we could not find a usable */
- /* WinSock DLL. */
- m_bAcceptable = false;
- }
-
- /* Confirm that the WinSock DLL supports 2.2.*/
- /* Note that if the DLL supports versions greater */
- /* than 2.2 in addition to 2.2, it will still return */
- /* 2.2 in wVersion since that is the version we */
- /* requested. */
-
- if ( LOBYTE( wsaData.wVersion ) != 2 ||
- HIBYTE( wsaData.wVersion ) != 2 ) {
- /* Tell the user that we could not find a usable */
- /* WinSock DLL. */
- WSACleanup( );
- m_bAcceptable = false;
- }
-
- /* The WinSock DLL is acceptable. Proceed. */
- m_bAcceptable = true;
- }
- INIT_WINSOCK2::~INIT_WINSOCK2(void)
- {
- if(m_bAcceptable)
- {
- m_bAcceptable = false;
- WSACleanup();
- }
- }
- #endif