/src/Passion/Network/TCPSocket.cpp

https://github.com/necrophcodr/Passion · C++ · 146 lines · 97 code · 25 blank · 24 comment · 9 complexity · 6e36f4c1a19450196e0559c24a73fd45 MD5 · raw file

  1. ////////////////////////////////////////////////////////////
  2. //
  3. // Copyright (C) 2010 Alexander Overvoorde (overv161@gmail.com)
  4. //
  5. // This software is provided 'as-is', without any express or implied
  6. // warranty. In no event will the authors be held liable for any damages
  7. // arising from the use of this software.
  8. //
  9. // Permission is granted to anyone to use this software for any purpose,
  10. // including commercial applications, and to alter it and redistribute it
  11. // freely, subject to the following restrictions:
  12. //
  13. // 1. The origin of this software must not be misrepresented; you must not
  14. // claim that you wrote the original software. If you use this software
  15. // in a product, an acknowledgment in the product documentation would be
  16. // appreciated but is not required.
  17. // 2. Altered source versions must be plainly marked as such, and must not be
  18. // misrepresented as being the original software.
  19. // 3. This notice may not be removed or altered from any source distribution.
  20. //
  21. ////////////////////////////////////////////////////////////
  22. ////////////////////////////////////////////////////////////
  23. // Headers
  24. ////////////////////////////////////////////////////////////
  25. #include <Passion/Network/TCPSocket.hpp>
  26. #ifndef WIN32
  27. #include <sys/types.h>
  28. #include <sys/socket.h>
  29. #include <netinet/in.h>
  30. #include <arpa/inet.h>
  31. #include <netdb.h>
  32. #include <fcntl.h>
  33. #include <unistd.h>
  34. #endif
  35. namespace Passion
  36. {
  37. TCPSocket::TCPSocket()
  38. {
  39. #ifdef WIN32
  40. WSADATA wsaData;
  41. WSAStartup( MAKEWORD( 1, 1 ), &wsaData );
  42. m_socket = socket( AF_INET, SOCK_STREAM, IPPROTO_TCP );
  43. u_long nonblocking = 1;
  44. ioctlsocket( m_socket, FIONBIO, &nonblocking );
  45. #else
  46. m_socket = socket( AF_INET, SOCK_STREAM, 0 );
  47. int flags = fcntl( m_socket, F_GETFL, 0 );
  48. fcntl( m_socket, F_SETFL, flags | O_NONBLOCK );
  49. #endif
  50. }
  51. TCPSocket::~TCPSocket()
  52. {
  53. #ifdef WIN32
  54. WSACleanup();
  55. #endif
  56. }
  57. void TCPSocket::Connect( const char* address, unsigned short port )
  58. {
  59. #ifdef WIN32
  60. LPHOSTENT hostEntry = (LPHOSTENT)gethostbyname( address );
  61. SOCKADDR_IN serverInfo;
  62. serverInfo.sin_family = AF_INET;
  63. serverInfo.sin_addr = *( (LPIN_ADDR)*hostEntry->h_addr_list );
  64. serverInfo.sin_port = htons( port );
  65. connect( m_socket, (LPSOCKADDR)&serverInfo, sizeof( struct sockaddr ) );
  66. #else
  67. struct sockaddr_in server;
  68. struct hostent* hp;
  69. server.sin_family = AF_INET;
  70. hp = gethostbyname( address );
  71. memcpy( (char*)&server.sin_addr, (char*)hp->h_addr, hp->h_length );
  72. server.sin_port = htons( port );
  73. connect( m_socket, (struct sockaddr*)&server, sizeof( server ) );
  74. #endif
  75. m_connected = true;
  76. }
  77. void TCPSocket::Disconnect()
  78. {
  79. #ifdef WIN32
  80. closesocket( m_socket );
  81. #else
  82. close( m_socket );
  83. #endif
  84. m_connected = false;
  85. }
  86. void TCPSocket::Send( const char* buffer, size_t length )
  87. {
  88. if ( length == 0 ) length = strlen( buffer );
  89. send( m_socket, buffer, length, 0 );
  90. }
  91. unsigned int TCPSocket::Receive( char* buffer, size_t length )
  92. {
  93. if ( !Available() )
  94. return 0;
  95. else
  96. return recv( m_socket, buffer, length, 0 );
  97. }
  98. bool TCPSocket::IsConnected()
  99. {
  100. if ( !m_connected ) return false;
  101. fd_set set;
  102. FD_ZERO( &set );
  103. FD_SET( m_socket, &set );
  104. timeval time = { 0, 0 };
  105. return select( m_socket + 1, 0, &set, 0, &time ) != 0;
  106. }
  107. bool TCPSocket::Available()
  108. {
  109. if ( !IsConnected() ) return false;
  110. fd_set set;
  111. FD_ZERO( &set );
  112. FD_SET( m_socket, &set );
  113. timeval time = { 0, 0 };
  114. if ( select( m_socket + 1, &set, 0, 0, &time ) > 0 ) {
  115. char temp;
  116. return recv( m_socket, &temp, 1, MSG_PEEK ) == 1;
  117. } else {
  118. return 0;
  119. }
  120. }
  121. }