packet_peer_udp_winsock.cpp 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. #include "packet_peer_udp_winsock.h"
  2. #include <winsock2.h>
  3. int PacketPeerUDPWinsock::get_available_packet_count() const {
  4. Error err = const_cast<PacketPeerUDPWinsock*>(this)->_poll(false);
  5. if (err!=OK)
  6. return 0;
  7. return queue_count;
  8. }
  9. Error PacketPeerUDPWinsock::get_packet(const uint8_t **r_buffer,int &r_buffer_size) const{
  10. Error err = const_cast<PacketPeerUDPWinsock*>(this)->_poll(false);
  11. if (err!=OK)
  12. return err;
  13. if (queue_count==0)
  14. return ERR_UNAVAILABLE;
  15. uint32_t size;
  16. rb.read((uint8_t*)&packet_ip.host,4,true);
  17. rb.read((uint8_t*)&packet_port,4,true);
  18. rb.read((uint8_t*)&size,4,true);
  19. rb.read(packet_buffer,size,true);
  20. --queue_count;
  21. *r_buffer=packet_buffer;
  22. r_buffer_size=size;
  23. return OK;
  24. }
  25. Error PacketPeerUDPWinsock::put_packet(const uint8_t *p_buffer,int p_buffer_size){
  26. int sock = _get_socket();
  27. ERR_FAIL_COND_V( sock == -1, FAILED );
  28. struct sockaddr_in addr;
  29. addr.sin_family = AF_INET;
  30. addr.sin_port = htons(peer_port);
  31. addr.sin_addr = *((struct in_addr*)&peer_addr.host);
  32. _set_blocking(true);
  33. errno = 0;
  34. int err;
  35. while ( (err = sendto(sock, (const char*)p_buffer, p_buffer_size, 0, (struct sockaddr*)&addr, sizeof(addr))) != p_buffer_size) {
  36. if (WSAGetLastError() != WSAEWOULDBLOCK) {
  37. return FAILED;
  38. };
  39. }
  40. return OK;
  41. }
  42. int PacketPeerUDPWinsock::get_max_packet_size() const{
  43. return 512; // uhm maybe not
  44. }
  45. void PacketPeerUDPWinsock::_set_blocking(bool p_blocking) {
  46. //am no windows expert
  47. //hope this is the right thing
  48. if (blocking==p_blocking)
  49. return;
  50. blocking=p_blocking;
  51. unsigned long par = blocking?0:1;
  52. if (ioctlsocket(sockfd, FIONBIO, &par)) {
  53. perror("setting non-block mode");
  54. //close();
  55. //return -1;
  56. };
  57. }
  58. Error PacketPeerUDPWinsock::listen(int p_port, int p_recv_buffer_size){
  59. close();
  60. int sock = _get_socket();
  61. if (sock == -1 )
  62. return ERR_CANT_CREATE;
  63. sockaddr_in addr = {0};
  64. addr.sin_family = AF_INET;
  65. addr.sin_port = htons(p_port);
  66. addr.sin_addr.s_addr = INADDR_ANY;
  67. if (bind(sock, (struct sockaddr*)&addr, sizeof(sockaddr_in)) == -1 ) {
  68. close();
  69. return ERR_UNAVAILABLE;
  70. }
  71. blocking=true;
  72. printf("UDP Connection listening on port %i\n", p_port);
  73. rb.resize(nearest_shift(p_recv_buffer_size));
  74. return OK;
  75. }
  76. void PacketPeerUDPWinsock::close(){
  77. if (sockfd != -1)
  78. ::closesocket(sockfd);
  79. sockfd=-1;
  80. rb.resize(8);
  81. queue_count=0;
  82. }
  83. Error PacketPeerUDPWinsock::wait() {
  84. return _poll(true);
  85. }
  86. Error PacketPeerUDPWinsock::_poll(bool p_wait) {
  87. _set_blocking(p_wait);
  88. struct sockaddr_in from = {0};
  89. int len = sizeof(struct sockaddr_in);
  90. int ret;
  91. while ( (ret = recvfrom(sockfd, (char*)recv_buffer, MIN(sizeof(recv_buffer),rb.data_left()-12), 0, (struct sockaddr*)&from, &len)) > 0) {
  92. rb.write((uint8_t*)&from.sin_addr, 4);
  93. uint32_t port = ntohs(from.sin_port);
  94. rb.write((uint8_t*)&port, 4);
  95. rb.write((uint8_t*)&ret, 4);
  96. rb.write(recv_buffer, ret);
  97. len = sizeof(struct sockaddr_in);
  98. ++queue_count;
  99. };
  100. if (ret == 0 || (ret == SOCKET_ERROR && WSAGetLastError() != WSAEWOULDBLOCK) ) {
  101. close();
  102. return FAILED;
  103. };
  104. return OK;
  105. }
  106. bool PacketPeerUDPWinsock::is_listening() const{
  107. return sockfd!=-1;
  108. }
  109. IP_Address PacketPeerUDPWinsock::get_packet_address() const {
  110. return packet_ip;
  111. }
  112. int PacketPeerUDPWinsock::get_packet_port() const{
  113. return packet_port;
  114. }
  115. int PacketPeerUDPWinsock::_get_socket() {
  116. if (sockfd != -1)
  117. return sockfd;
  118. sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
  119. ERR_FAIL_COND_V( sockfd == -1, -1 );
  120. //fcntl(sockfd, F_SETFL, O_NONBLOCK);
  121. return sockfd;
  122. }
  123. void PacketPeerUDPWinsock::set_send_address(const IP_Address& p_address,int p_port) {
  124. peer_addr=p_address;
  125. peer_port=p_port;
  126. }
  127. void PacketPeerUDPWinsock::make_default() {
  128. PacketPeerUDP::_create = PacketPeerUDPWinsock::_create;
  129. };
  130. PacketPeerUDP* PacketPeerUDPWinsock::_create() {
  131. return memnew(PacketPeerUDPWinsock);
  132. };
  133. PacketPeerUDPWinsock::PacketPeerUDPWinsock() {
  134. sockfd=-1;
  135. packet_port=0;
  136. queue_count=0;
  137. peer_port=0;
  138. }
  139. PacketPeerUDPWinsock::~PacketPeerUDPWinsock() {
  140. close();
  141. }