Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/libraries/network/packet/Packet.cc @ 12050

Last change on this file since 12050 was 12027, checked in by merholzl, 6 years ago

Merged Masterserver, refresh button had to be removed

  • Property svn:eol-style set to native
File size: 7.0 KB
RevLine 
[1711]1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Oliver Scheuss, (C) 2008
24 *   Co-authors:
25 *      ...
26 *
27 */
[1701]28
[1711]29
[1760]30#include "Packet.h"
31
32#include <cassert>
[3214]33#include <cstring>
[5749]34#define WIN32_LEAN_AND_MEAN
[1701]35#include <enet/enet.h>
[7801]36#include <boost/thread/mutex.hpp>
[1701]37
[8858]38#include "util/Output.h"
[1701]39#include "Acknowledgement.h"
40#include "Chat.h"
41#include "ClassID.h"
[3214]42#include "DeleteObjects.h"
[3084]43#include "FunctionCalls.h"
44#include "FunctionIDs.h"
[1709]45#include "Gamestate.h"
[1711]46#include "Welcome.h"
[1701]47#include "network/Host.h"
[8327]48// #include "network/ClientInformation.h"
[1701]49
[2171]50namespace orxonox{
[1701]51
52namespace packet{
[2087]53
[3214]54// Make sure we assume the right values
[11071]55static_assert(static_cast<int>(PacketFlag::Reliable)    == static_cast<int>(ENET_PACKET_FLAG_RELIABLE),    "check enum");
56static_assert(static_cast<int>(PacketFlag::Unsequenced) == static_cast<int>(ENET_PACKET_FLAG_UNSEQUENCED), "check enum");
57static_assert(static_cast<int>(PacketFlag::NoAllocate)  == static_cast<int>(ENET_PACKET_FLAG_NO_ALLOCATE), "check enum");
[3214]58
59#define PACKET_FLAG_DEFAULT PacketFlag::NoAllocate
[1701]60#define _PACKETID           0
[2087]61
[2171]62std::map<size_t, Packet *> Packet::packetMap_;
[7801]63boost::mutex Packet::packetMapMutex_;
[2087]64
[7163]65Packet::Packet()
66{
[1711]67  flags_ = PACKET_FLAG_DEFAULT;
[3280]68  packetDirection_ = Direction::Outgoing;
[7801]69  peerID_=0;
[11071]70  data_=nullptr;
71  enetPacket_=nullptr;
[2087]72  bDataENetAllocated_ = false;
[1701]73}
74
[7801]75Packet::Packet(uint8_t *data, unsigned int peerID)
[7163]76{
[1711]77  flags_ = PACKET_FLAG_DEFAULT;
[3280]78  packetDirection_ = Direction::Incoming;
[7801]79  peerID_=peerID;
[1711]80  data_=data;
[11071]81  enetPacket_=nullptr;
[2087]82  bDataENetAllocated_ = false;
[1701]83}
84
85
[8327]86Packet::Packet(const Packet &p)
87{
[1712]88  enetPacket_=p.enetPacket_;
[1711]89  flags_=p.flags_;
[1730]90  packetDirection_ = p.packetDirection_;
[7801]91  peerID_ = p.peerID_;
[1711]92  if(p.data_){
[1907]93    data_ = new uint8_t[p.getSize()];
[1711]94    memcpy(data_, p.data_, p.getSize());
95  }else
[11071]96    data_=nullptr;
[2087]97  bDataENetAllocated_ = p.bDataENetAllocated_;
[1701]98}
99
[2087]100/**
101@brief
102    Destroys a packet completely.
[6417]103
104    That also means destroying the ENetPacket if one exists. There
[2087]105*/
[8327]106Packet::~Packet()
107{
[2087]108  // Deallocate data_ memory if necessary.
[8327]109  if (this->bDataENetAllocated_)
110  {
[2087]111    // In this case ENet allocated data_ and will destroy it.
112  }
[8327]113  else if (this->data_)
114  {
[3214]115    // This destructor was probably called as a consequence of ENet executing our callback.
[2087]116    // It simply serves us to be able to deallocate the packet content (data_) ourselves since
117    // we have created it in the first place.
118    delete[] this->data_;
119  }
120
121  // Destroy the ENetPacket if necessary.
122  // Note: For the case ENet used the callback to destroy the packet, we have already set
[11071]123  // enetPacket_ to nullptr to avoid destroying it again.
[8327]124  if (this->enetPacket_)
125  {
[2087]126    // enetPacket_->data gets destroyed too by ENet if it was allocated by it.
[1705]127    enet_packet_destroy(enetPacket_);
[1751]128  }
[1705]129}
[1701]130
[12027]131/**
132 * Send the Packet.
133 * @param host The host which sends the packet
134 */
[8327]135bool Packet::send(orxonox::Host* host)
136{
[12027]137  // Deny sending incoming packets
[8327]138  if(packetDirection_ != Direction::Outgoing && packetDirection_ != Direction::Bidirectional )
139  {
[1712]140    assert(0);
[1701]141    return false;
[1730]142  }
[12027]143
[8327]144  if(!enetPacket_)
145  {
[12027]146    // Deny sending empty packets
147    if(!data_) {
[1712]148      assert(0);
[1701]149      return false;
[1712]150    }
[2087]151    // We deliver ENet the data address so that it doesn't memcpy everything again.
152    // --> We have to delete data_ ourselves!
[1711]153    enetPacket_ = enet_packet_create(getData(), getSize(), getFlags());
[1751]154    enetPacket_->freeCallback = &Packet::deletePacket;
[2087]155    // Add the packet to a global list so we can access it again once enet calls our
156    // deletePacket method. We can of course only give a one argument function to the ENet C library.
[2662]157    {
158      // Assures we don't create a packet and destroy it right after in another thread
159      // without having a reference in the packetMap_
[7801]160      Packet::packetMapMutex_.lock();
[12027]161      Packet::packetMap_[reinterpret_cast<size_t>(enetPacket_)] = this;
[7801]162      Packet::packetMapMutex_.unlock();
[2662]163    }
[1701]164  }
[2087]165#ifndef NDEBUG
[11071]166  switch( *(Type *)(data_ + _PACKETID) )
[1715]167  {
[3280]168    case Type::Acknowledgement:
169    case Type::Chat:
170    case Type::ClassID:
171    case Type::Gamestate:
172    case Type::Welcome:
173    case Type::DeleteObjects:
174    case Type::FunctionIDs:
175    case Type::FunctionCalls:
[1751]176      break;
[1715]177    default:
[1751]178      assert(0); //there was some error, if this is the case
[1715]179      break;
[1751]180  }
181#endif
[12027]182
183  // Send via reliable or standard channel respectively
[7801]184  if( this->flags_ & PacketFlag::Reliable )
185    host->addPacket( enetPacket_, peerID_, NETWORK_CHANNEL_DEFAULT);
186  else
187    host->addPacket( enetPacket_, peerID_, NETWORK_CHANNEL_UNRELIABLE);
[12027]188
[1701]189  return true;
190}
191
[12027]192/**
193 * Given an ENetPacket, create an Orxonox packet
194 * @param packet The ENetPacket
195 * @param peerID The sender
196 */
[8327]197Packet *Packet::createPacket(ENetPacket* packet, uint32_t peerID)
198{
[1907]199  uint8_t *data = packet->data;
[11071]200  Packet *p = nullptr;
201  switch( *(Type *)(data + _PACKETID) )
[1701]202  {
[3280]203    case Type::Acknowledgement:
[12027]204      p = new Acknowledgement( data, peerID );
[1701]205      break;
[3280]206    case Type::Chat:
[7801]207      p = new Chat( data, peerID );
[1701]208      break;
[3280]209    case Type::ClassID:
[7801]210      p = new ClassID( data, peerID );
[1701]211      break;
[3280]212    case Type::Gamestate:
[7801]213      p = new Gamestate( data, peerID );
[1701]214      break;
[3280]215    case Type::Welcome:
[7801]216      p = new Welcome( data, peerID );
[1715]217      break;
[3280]218    case Type::DeleteObjects:
[7801]219      p = new DeleteObjects( data, peerID );
[1907]220      break;
[3280]221    case Type::FunctionCalls:
[7801]222      p = new FunctionCalls( data, peerID );
[3084]223      break;
[3280]224    case Type::FunctionIDs:
[7801]225      p = new FunctionIDs( data, peerID );
[3084]226      break;
[1701]227    default:
[7801]228      assert(0);
[1710]229      break;
[1701]230  }
[2087]231
232  // Data was created by ENet
233  p->bDataENetAllocated_ = true;
[3214]234  p->enetPacket_ = packet;
[2087]235
[1711]236  return p;
[1701]237}
238
[2087]239/**
240@brief
241    ENet calls this method whenever it wants to destroy a packet that contains
242    data we allocated ourselves.
243*/
[8327]244void Packet::deletePacket(ENetPacket *enetPacket)
245{
[6105]246  // Get our Packet from a global map with all Packets created in the send() method of Packet.
[7801]247  Packet::packetMapMutex_.lock();
[12027]248
249  std::map<size_t, Packet*>::iterator it = Packet::packetMap_.find(reinterpret_cast<size_t>(enetPacket));
[2087]250  assert(it != packetMap_.end());
[12027]251
[2087]252  // Make sure we don't delete it again in the destructor
[11071]253  it->second->enetPacket_ = nullptr;
[2087]254  delete it->second;
[2171]255  packetMap_.erase(it);
[12027]256
[7801]257  Packet::packetMapMutex_.unlock();
[1711]258}
259
[1701]260} // namespace packet
261
[2171]262} // namespace orxonox
[1701]263
Note: See TracBrowser for help on using the repository browser.