Peer.cpp 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. //
  2. // Urho3D Engine
  3. // Copyright (c) 2008-2011 Lasse Öörni
  4. //
  5. // Permission is hereby granted, free of charge, to any person obtaining a copy
  6. // of this software and associated documentation files (the "Software"), to deal
  7. // in the Software without restriction, including without limitation the rights
  8. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. // copies of the Software, and to permit persons to whom the Software is
  10. // furnished to do so, subject to the following conditions:
  11. //
  12. // The above copyright notice and this permission notice shall be included in
  13. // all copies or substantial portions of the Software.
  14. //
  15. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  20. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  21. // THE SOFTWARE.
  22. //
  23. #include "Precompiled.h"
  24. #include "Log.h"
  25. #include "NetworkEvents.h"
  26. #include "Peer.h"
  27. #include "StringUtils.h"
  28. #include <enet/enet.h>
  29. #include "DebugNew.h"
  30. Peer::Peer(ENetPeer* peer, PeerType type, bool sendPacketEvents) :
  31. mPeer(peer),
  32. mType(type),
  33. mConnectionState(CS_CONNECTING),
  34. mPort(0),
  35. mSimulatedPacketLoss(0.0f),
  36. mSimulatedLatency(0),
  37. mHalfSimulatedLatency(0),
  38. mSendPacketEvents(sendPacketEvents),
  39. mUserData(0)
  40. {
  41. }
  42. Peer::~Peer()
  43. {
  44. flushPackets();
  45. onDisconnect();
  46. }
  47. void Peer::setSendPacketEvents(bool enable)
  48. {
  49. mSendPacketEvents = enable;
  50. }
  51. void Peer::send(const VectorBuffer& packet, unsigned char channel, bool reliable, bool inOrder)
  52. {
  53. if ((!mPeer) || (!packet.getSize()))
  54. return;
  55. if ((mSimulatedPacketLoss > 0.0f) && (!reliable))
  56. {
  57. if (random() < mSimulatedPacketLoss)
  58. return;
  59. }
  60. ENetPacket* enetPacket = enet_packet_create(packet.getData(), packet.getSize(), reliable ? ENET_PACKET_FLAG_RELIABLE :
  61. (inOrder ? 0 : ENET_PACKET_FLAG_UNSEQUENCED));
  62. if (!mSimulatedLatency)
  63. enet_peer_send(mPeer, channel, enetPacket);
  64. else
  65. {
  66. QueuedPacket packet;
  67. packet.mPacket = enetPacket;
  68. packet.mChannel = channel;
  69. mSentPackets.push_back(packet);
  70. }
  71. }
  72. void Peer::send(const void* data, unsigned size, unsigned char channel, bool reliable, bool inOrder)
  73. {
  74. if ((!mPeer) || (!data) || (!size))
  75. return;
  76. if ((mSimulatedPacketLoss != 0.0f) && (!reliable))
  77. {
  78. if (random() < mSimulatedPacketLoss)
  79. return;
  80. }
  81. ENetPacket* enetPacket = enet_packet_create(data, size, reliable ? ENET_PACKET_FLAG_RELIABLE : (inOrder ? 0 :
  82. ENET_PACKET_FLAG_UNSEQUENCED));
  83. if (!mSimulatedLatency)
  84. enet_peer_send(mPeer, channel, enetPacket);
  85. else
  86. {
  87. QueuedPacket packet;
  88. packet.mPacket = enetPacket;
  89. packet.mChannel = channel;
  90. mSentPackets.push_back(packet);
  91. }
  92. }
  93. bool Peer::receive(VectorBuffer& packet, unsigned char channel)
  94. {
  95. bool received = false;
  96. bool reliable = false;
  97. if (channel == CHANNEL_ANY)
  98. {
  99. for (std::map<unsigned char, std::list<QueuedPacket> >::iterator i = mPackets.begin(); i != mPackets.end(); ++i)
  100. {
  101. std::list<QueuedPacket>& packetList = i->second;
  102. if ((!packetList.empty()) && (packetList.front().mTimer.getMSec(false) >= mHalfSimulatedLatency))
  103. {
  104. ENetPacket* enetPacket = packetList.front().mPacket;
  105. reliable = (enetPacket->flags & ENET_PACKET_FLAG_RELIABLE) != 0;
  106. packetList.pop_front();
  107. packet.setData(enetPacket->data, enetPacket->dataLength);
  108. enet_packet_destroy(enetPacket);
  109. received = true;
  110. break;
  111. }
  112. }
  113. }
  114. else
  115. {
  116. std::list<QueuedPacket>& packetList = mPackets[channel];
  117. if ((!packetList.empty()) && (packetList.front().mTimer.getMSec(false) >= mHalfSimulatedLatency))
  118. {
  119. ENetPacket* enetPacket = packetList.front().mPacket;
  120. reliable = (enetPacket->flags & ENET_PACKET_FLAG_RELIABLE) != 0;
  121. packetList.pop_front();
  122. packet.setData(enetPacket->data, enetPacket->dataLength);
  123. enet_packet_destroy(enetPacket);
  124. received = true;
  125. }
  126. }
  127. if ((received) && (!reliable) && (mSimulatedPacketLoss > 0.0f))
  128. {
  129. if (random() < mSimulatedPacketLoss)
  130. {
  131. packet.clear();
  132. received = false;
  133. }
  134. }
  135. return received;
  136. }
  137. void Peer::update()
  138. {
  139. // Check send timer of packets with simulated latency and send as necessary
  140. for (std::list<QueuedPacket>::iterator i = mSentPackets.begin(); i != mSentPackets.end();)
  141. {
  142. if (i->mTimer.getMSec(false) >= mHalfSimulatedLatency)
  143. {
  144. enet_peer_send(mPeer, i->mChannel, i->mPacket);
  145. i = mSentPackets.erase(i);
  146. }
  147. else
  148. ++i;
  149. }
  150. }
  151. void Peer::flushPackets()
  152. {
  153. for (std::map<unsigned char, std::list<QueuedPacket> >::iterator i = mPackets.begin(); i != mPackets.end(); ++i)
  154. {
  155. std::list<QueuedPacket>& packetList = i->second;
  156. while (!packetList.empty())
  157. {
  158. ENetPacket* enetPacket = packetList.front().mPacket;
  159. packetList.pop_front();
  160. enet_packet_destroy(enetPacket);
  161. }
  162. }
  163. while (!mSentPackets.empty())
  164. {
  165. ENetPacket* enetPacket = mSentPackets.front().mPacket;
  166. mSentPackets.pop_front();
  167. enet_packet_destroy(enetPacket);
  168. }
  169. }
  170. void Peer::disconnect()
  171. {
  172. if ((mPeer) && (mConnectionState > CS_DISCONNECTING))
  173. {
  174. mConnectionState = CS_DISCONNECTING;
  175. enet_peer_disconnect(mPeer, 0);
  176. }
  177. }
  178. void Peer::forceDisconnect()
  179. {
  180. onDisconnect();
  181. }
  182. void Peer::setSimulatedPacketLoss(float loss)
  183. {
  184. mSimulatedPacketLoss = clamp(loss, 0.0f, 1.0f);
  185. }
  186. void Peer::setSimulatedLatency(unsigned latency)
  187. {
  188. mSimulatedLatency = latency;
  189. mHalfSimulatedLatency = latency >> 1;
  190. }
  191. void Peer::onConnect()
  192. {
  193. if (mPeer)
  194. {
  195. static const int MAX_IPADDRESS = 32;
  196. char ipBuffer[MAX_IPADDRESS];
  197. enet_address_get_host_ip(&mPeer->address, ipBuffer, MAX_IPADDRESS);
  198. mAddress = std::string(ipBuffer);
  199. mPort = mPeer->address.port;
  200. mConnectionState = CS_CONNECTED;
  201. // Send event
  202. using namespace PeerConnected;
  203. VariantMap eventData;
  204. eventData[P_PEER] = (void*)this;
  205. sendEvent(EVENT_PEERCONNECTED, eventData);
  206. LOGINFO(mAddress + ":" + toString(mPort) + " connected");
  207. }
  208. }
  209. void Peer::onDisconnect()
  210. {
  211. if (mPeer)
  212. {
  213. enet_peer_reset(mPeer);
  214. mConnectionState = CS_DISCONNECTED;
  215. mPeer = 0;
  216. // Send event
  217. using namespace PeerDisconnected;
  218. VariantMap eventData;
  219. eventData[P_PEER] = (void*)this;
  220. sendEvent(EVENT_PEERDISCONNECTED, eventData);
  221. if (!mAddress.empty())
  222. LOGINFO(mAddress + ":" + toString(mPort) + " disconnected");
  223. else
  224. LOGINFO("Disconnected");
  225. }
  226. }