Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/network2/src/libraries/network/packet/Packet.cc @ 6698

Last change on this file since 6698 was 6450, checked in by scheusso, 15 years ago

further traffic reduction:

  • synchronisableheaders are now smaller( by 2 bytes per header )
  • variableID of SynchronisableVariables are now uint8_t → max 256 synchronised variables per Synchronisable
  • Property svn:eol-style set to native
File size: 7.3 KB
Line 
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 */
28
29
30#include "Packet.h"
31
32#include <cassert>
33#include <cstring>
34#define WIN32_LEAN_AND_MEAN
35#include <enet/enet.h>
36#include <boost/static_assert.hpp>
37
38#include "util/Debug.h"
39#include "Acknowledgement.h"
40#include "Chat.h"
41#include "ClassID.h"
42#include "DeleteObjects.h"
43#include "FunctionCalls.h"
44#include "FunctionIDs.h"
45#include "Gamestate.h"
46#include "Welcome.h"
47#include "network/Host.h"
48#include "network/ClientInformation.h"
49
50namespace orxonox{
51
52namespace packet{
53
54// Make sure we assume the right values
55BOOST_STATIC_ASSERT(static_cast<int>(PacketFlag::Reliable)   == static_cast<int>(ENET_PACKET_FLAG_RELIABLE));
56BOOST_STATIC_ASSERT(static_cast<int>(PacketFlag::Unsequence) == static_cast<int>(ENET_PACKET_FLAG_UNSEQUENCED));
57BOOST_STATIC_ASSERT(static_cast<int>(PacketFlag::NoAllocate) == static_cast<int>(ENET_PACKET_FLAG_NO_ALLOCATE));
58
59#define PACKET_FLAG_DEFAULT PacketFlag::NoAllocate
60#define _PACKETID           0
61
62std::map<size_t, Packet *> Packet::packetMap_;
63
64Packet::Packet(){
65  flags_ = PACKET_FLAG_DEFAULT;
66  packetDirection_ = Direction::Outgoing;
67  clientID_=0;
68  data_=0;
69  enetPacket_=0;
70  bDataENetAllocated_ = false;
71}
72
73Packet::Packet(uint8_t *data, unsigned int clientID){
74  flags_ = PACKET_FLAG_DEFAULT;
75  packetDirection_ = Direction::Incoming;
76  clientID_=clientID;
77  data_=data;
78  enetPacket_=0;
79  bDataENetAllocated_ = false;
80}
81
82
83Packet::Packet(const Packet &p){
84  enetPacket_=p.enetPacket_;
85  flags_=p.flags_;
86  packetDirection_ = p.packetDirection_;
87  clientID_ = p.clientID_;
88  if(p.data_){
89    data_ = new uint8_t[p.getSize()];
90    memcpy(data_, p.data_, p.getSize());
91  }else
92    data_=0;
93  bDataENetAllocated_ = p.bDataENetAllocated_;
94}
95
96/**
97@brief
98    Destroys a packet completely.
99
100    That also means destroying the ENetPacket if one exists. There
101*/
102Packet::~Packet(){
103  // Deallocate data_ memory if necessary.
104  if (this->bDataENetAllocated_){
105    // In this case ENet allocated data_ and will destroy it.
106  }
107  else if (this->data_) {
108    // This destructor was probably called as a consequence of ENet executing our callback.
109    // It simply serves us to be able to deallocate the packet content (data_) ourselves since
110    // we have created it in the first place.
111    delete[] this->data_;
112  }
113
114  // Destroy the ENetPacket if necessary.
115  // Note: For the case ENet used the callback to destroy the packet, we have already set
116  // enetPacket_ to NULL to avoid destroying it again.
117  if (this->enetPacket_){
118    // enetPacket_->data gets destroyed too by ENet if it was allocated by it.
119    enet_packet_destroy(enetPacket_);
120  }
121}
122
123bool Packet::send(){
124  if(packetDirection_ != Direction::Outgoing && packetDirection_ != Direction::Bidirectional ){
125    assert(0);
126    return false;
127  }
128  if(!enetPacket_){
129    if(!data_){
130      assert(0);
131      return false;
132    }
133    // We deliver ENet the data address so that it doesn't memcpy everything again.
134    // --> We have to delete data_ ourselves!
135    enetPacket_ = enet_packet_create(getData(), getSize(), getFlags());
136    enetPacket_->freeCallback = &Packet::deletePacket;
137    // Add the packet to a global list so we can access it again once enet calls our
138    // deletePacket method. We can of course only give a one argument function to the ENet C library.
139    {
140      // Assures we don't create a packet and destroy it right after in another thread
141      // without having a reference in the packetMap_
142      packetMap_[reinterpret_cast<size_t>(enetPacket_)] = this;
143    }
144  }
145#ifndef NDEBUG
146  switch( *(Type::Value *)(data_ + _PACKETID) )
147  {
148    case Type::Acknowledgement:
149    case Type::Chat:
150    case Type::ClassID:
151    case Type::Gamestate:
152    case Type::Welcome:
153    case Type::DeleteObjects:
154    case Type::FunctionIDs:
155    case Type::FunctionCalls:
156      break;
157    default:
158      assert(0); //there was some error, if this is the case
159      break;
160  }
161#endif
162//  ENetPacket *temp = enetPacket_;
163//  enetPacket_ = 0; // otherwise we have a double free because enet already handles the deallocation of the packet
164  if(!Host::addPacket( enetPacket_, clientID_))
165    enet_packet_destroy(this->enetPacket_); // if we could not add the packet to the enet queue delete it manually
166  return true;
167}
168
169Packet *Packet::createPacket(ENetPacket *packet, ENetPeer *peer){
170  uint8_t *data = packet->data;
171  assert(ClientInformation::findClient(&peer->address)->getID() != static_cast<unsigned int>(-2) || !Host::isServer());
172  unsigned int clientID = ClientInformation::findClient(&peer->address)->getID();
173  Packet *p = 0;
174  COUT(6) << "packet type: " << *(Type::Value *)&data[_PACKETID] << std::endl;
175  switch( *(Type::Value *)(data + _PACKETID) )
176  {
177    case Type::Acknowledgement:
178      COUT(5) << "ack" << std::endl;
179      p = new Acknowledgement( data, clientID );
180      break;
181    case Type::Chat:
182      COUT(5) << "chat" << std::endl;
183      p = new Chat( data, clientID );
184      break;
185    case Type::ClassID:
186      COUT(5) << "classid" << std::endl;
187      p = new ClassID( data, clientID );
188      break;
189    case Type::Gamestate:
190      COUT(5) << "gamestate" << std::endl;
191      // TODO: remove brackets
192      p = new Gamestate( data, clientID );
193      break;
194    case Type::Welcome:
195      COUT(5) << "welcome" << std::endl;
196      p = new Welcome( data, clientID );
197      break;
198    case Type::DeleteObjects:
199      COUT(5) << "deleteobjects" << std::endl;
200      p = new DeleteObjects( data, clientID );
201      break;
202    case Type::FunctionCalls:
203      COUT(5) << "functionCalls" << std::endl;
204      p = new FunctionCalls( data, clientID );
205      break;
206    case Type::FunctionIDs:
207      COUT(5) << "functionIDs" << std::endl;
208      p = new FunctionIDs( data, clientID );
209      break;
210    default:
211      assert(0); //TODO: repair this
212      break;
213  }
214
215  // Data was created by ENet
216  p->bDataENetAllocated_ = true;
217  p->enetPacket_ = packet;
218
219  return p;
220}
221
222/**
223@brief
224    ENet calls this method whenever it wants to destroy a packet that contains
225    data we allocated ourselves.
226*/
227void Packet::deletePacket(ENetPacket *enetPacket){
228  // Get our Packet from a global map with all Packets created in the send() method of Packet.
229  std::map<size_t, Packet*>::iterator it = packetMap_.find(reinterpret_cast<size_t>(enetPacket));
230  assert(it != packetMap_.end());
231  // Make sure we don't delete it again in the destructor
232  it->second->enetPacket_ = 0;
233  delete it->second;
234  packetMap_.erase(it);
235  COUT(6) << "PacketMap size: " << packetMap_.size() << std::endl;
236}
237
238} // namespace packet
239
240} // namespace orxonox
241
Note: See TracBrowser for help on using the repository browser.