UDPSocket.cpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. #include <iostream>
  2. #include <cassert>
  3. #include "NetCrown.h"
  4. #include "IPv4Address.h"
  5. #include "UDPSocket.h"
  6. namespace crown
  7. {
  8. namespace network
  9. {
  10. UDPSocket::UDPSocket()
  11. {
  12. m_socket = 0;
  13. }
  14. UDPSocket::~UDPSocket()
  15. {
  16. close();
  17. }
  18. bool UDPSocket::open(unsigned short port)
  19. {
  20. assert(!is_open());
  21. m_socket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
  22. if (m_socket <= 0)
  23. {
  24. cout << "Failed to create socket." << endl;
  25. m_socket = 0;
  26. return false;
  27. }
  28. // bind to port
  29. sockaddr_in address;
  30. address.sin_family = AF_INET;
  31. address.sin_addr.s_addr = INADDR_ANY;
  32. address.sin_port = htons(port);
  33. if ( bind( m_socket, (const sockaddr*)&address, sizeof(sockaddr_in)) < 0)
  34. {
  35. cout << "Failed to bind socket" << endl;
  36. close();
  37. return false;
  38. }
  39. // set non-blocking io
  40. #if PLATFORM == PLATFORM_MAC || PLATFORM == PLATFORM_UNIX
  41. int non_blocking = 1;
  42. if (fcntl( m_socket, F_SETFL, O_NONBLOCK, non_blocking ) == -1)
  43. {
  44. cout << "Failed to set non-blocking socket" << endl;
  45. close();
  46. return false;
  47. }
  48. #elif PLATFORM == PLATFORM_WINDOWS
  49. DWORD non_blocking = 1;
  50. if (ioctlsocket( socket, FIONBIO, &non_blocking ) != 0)
  51. {
  52. cout << "Failed to set non-blocking socket" << endl;
  53. close();
  54. return false;
  55. }
  56. #endif
  57. return true;
  58. }
  59. bool UDPSocket::send(IPv4Address &receiver, const void* data, int size)
  60. {
  61. assert(data);
  62. assert(size > 0);
  63. if (m_socket == 0)
  64. {
  65. return false;
  66. }
  67. assert(receiver.get_address() != 0);
  68. assert(receiver.get_port() != 0);
  69. sockaddr_in address;
  70. address.sin_family = AF_INET;
  71. address.sin_addr.s_addr = htonl(receiver.get_address());
  72. address.sin_port = htons(receiver.get_port());
  73. int sent_bytes = sendto(m_socket, (const char*)data, size, 0, (sockaddr*)&address, sizeof(sockaddr_in));
  74. return sent_bytes == size;
  75. }
  76. int UDPSocket::receive(IPv4Address &sender, void* data, int size)
  77. {
  78. assert(data);
  79. assert(size > 0);
  80. if (m_socket == 0)
  81. {
  82. return false;
  83. }
  84. #if PLATFORM == PLATFORM_WINDOWS
  85. typedef int socklen_t;
  86. #endif
  87. sockaddr_in from;
  88. socklen_t from_length = sizeof(from);
  89. int received_bytes = recvfrom(m_socket, (char*)data, size, 0, (sockaddr*)&from, &from_length);
  90. if (received_bytes <= 0)
  91. {
  92. return 0;
  93. }
  94. unsigned int address = ntohl(from.sin_addr.s_addr);
  95. unsigned short port = ntohs(from.sin_port);
  96. sender = IPv4Address(address, port);
  97. return received_bytes;
  98. }
  99. void UDPSocket::close()
  100. {
  101. if ( m_socket != 0 )
  102. {
  103. #if PLATFORM == PLATFORM_MAC || PLATFORM == PLATFORM_UNIX
  104. ::close(m_socket);
  105. #elif PLATFORM == PLATFORM_WINDOWS
  106. closesocket(m_socket);
  107. #endif
  108. m_socket = 0;
  109. }
  110. }
  111. bool UDPSocket::is_open()
  112. {
  113. return m_socket != 0;
  114. }
  115. }
  116. }