Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/network/ConnectionManager.cc @ 1749

Last change on this file since 1749 was 1747, checked in by landauf, 17 years ago

merged core3 back to trunk

  • Property svn:eol-style set to native
File size: 11.7 KB
RevLine 
[1282]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) 2007
24 *   Co-authors:
25 *      ...
26 *
27 */
28
29//
30// C++ Interface: ConnectionManager
31//
32// Description: The Class ConnectionManager manages the servers conenctions to the clients.
33// each connection is provided by a new process. communication between master process and
34// connection processes is provided by ...
35//
36//
37// Author:  Oliver Scheuss
38//
39
40#include <iostream>
[1735]41#include <assert.h>
[1282]42// boost.thread library for multithreading support
43#include <boost/bind.hpp>
44
45#include "core/CoreIncludes.h"
46#include "core/BaseObject.h"
[1747]47#include "core/Iterator.h"
[1502]48#include "objects/SpaceShip.h"
[1282]49#include "util/Math.h"
[1502]50#include "util/Sleep.h"
[1282]51#include "ClientInformation.h"
52#include "ConnectionManager.h"
53#include "Synchronisable.h"
[1735]54#include "packet/ClassID.h"
[1282]55
56namespace std
57{
58  bool operator< (ENetAddress a, ENetAddress b) {
59    if(a.host <= b.host)
60      return true;
61    else
62      return false;
63  }
64}
65
66namespace network
67{
68  //boost::thread_group network_threads;
[1747]69
[1735]70  ConnectionManager *ConnectionManager::instance_=0;
[1747]71
[1735]72  ConnectionManager::ConnectionManager():receiverThread_(0){
73    assert(instance_==0);
74    instance_=this;
[1282]75    quit=false;
76    bindAddress.host = ENET_HOST_ANY;
77    bindAddress.port = NETWORK_PORT;
78  }
[1735]79  boost::recursive_mutex ConnectionManager::enet_mutex;
[1747]80
[1735]81//   ConnectionManager::ConnectionManager(ClientInformation *head) : receiverThread_(0) {
82//     assert(instance_==0);
83//     instance_=this;
84//     quit=false;
85//     bindAddress.host = ENET_HOST_ANY;
86//     bindAddress.port = NETWORK_PORT;
87//   }
[1747]88
[1735]89  ConnectionManager::ConnectionManager(int port){
90    assert(instance_==0);
91    instance_=this;
[1502]92    quit=false;
93    bindAddress.host = ENET_HOST_ANY;
94    bindAddress.port = port;
95  }
[1282]96
[1735]97  ConnectionManager::ConnectionManager(int port, std::string address) :receiverThread_(0) {
98    assert(instance_==0);
99    instance_=this;
[1282]100    quit=false;
101    enet_address_set_host (& bindAddress, address.c_str());
102    bindAddress.port = NETWORK_PORT;
103  }
104
[1735]105  ConnectionManager::ConnectionManager(int port, const char *address) : receiverThread_(0) {
106    assert(instance_==0);
107    instance_=this;
[1282]108    quit=false;
109    enet_address_set_host (& bindAddress, address);
110    bindAddress.port = NETWORK_PORT;
111  }
[1747]112
[1735]113  ConnectionManager::~ConnectionManager(){
114    instance_=0;
115    if(!quit)
116      quitListener();
117  }
[1282]118
[1502]119  /*ENetPacket *ConnectionManager::getPacket(ENetAddress &address) {
[1282]120    if(!buffer.isEmpty())
121      return buffer.pop(address);
122    else
123      return NULL;
[1502]124  }*/
[1282]125/**
126This function only pops the first element in PacketBuffer (first in first out)
127used by processQueue in Server.cc
128*/
[1502]129  /*ENetPacket *ConnectionManager::getPacket(int &clientID) {
[1282]130    ENetAddress address;
131    ENetPacket *packet=getPacket(address);
132    ClientInformation *temp =head_->findClient(&address);
[1360]133    if(!temp)
134      return NULL;
[1282]135    clientID=temp->getID();
136    return packet;
[1502]137  }*/
[1747]138
[1502]139  ENetEvent *ConnectionManager::getEvent(){
140    if(!buffer.isEmpty())
141      return buffer.pop();
142    else
143      return NULL;
[1282]144  }
145
146  bool ConnectionManager::queueEmpty() {
147    return buffer.isEmpty();
148  }
149
150  void ConnectionManager::createListener() {
151    receiverThread_ = new boost::thread(boost::bind(&ConnectionManager::receiverThread, this));
152    //network_threads.create_thread(boost::bind(boost::mem_fn(&ConnectionManager::receiverThread), this));
153         //boost::thread thr(boost::bind(boost::mem_fn(&ConnectionManager::receiverThread), this));
154    return;
155  }
156
157  bool ConnectionManager::quitListener() {
158    quit=true;
159    //network_threads.join_all();
160    receiverThread_->join();
161    return true;
162  }
[1747]163
[1735]164//   bool ConnectionManager::addPacket(Packet::Packet *packet){
165//     ClientInformation *temp = instance_->head_->findClient(packet->getClientID());
166//     if(!temp){
167//       COUT(3) << "C.Man: addPacket findClient failed" << std::endl;
168//       return false;
169//     }
170//     ENetPacket *packet = new ENetPacket;
171//     //  TODO: finish implementation
172//   }
[1747]173//
174
[1282]175  bool ConnectionManager::addPacket(ENetPacket *packet, ENetPeer *peer) {
[1735]176    boost::recursive_mutex::scoped_lock lock(instance_->enet_mutex);
177    if(enet_peer_send(peer, NETWORK_DEFAULT_CHANNEL, packet)!=0)
[1282]178      return false;
179    return true;
180  }
181
182  bool ConnectionManager::addPacket(ENetPacket *packet, int clientID) {
[1735]183    ClientInformation *temp = ClientInformation::findClient(clientID);
[1502]184    if(!temp){
185      COUT(3) << "C.Man: addPacket findClient failed" << std::endl;
[1282]186      return false;
[1502]187    }
[1735]188    return addPacket(packet, temp->getPeer());
[1282]189  }
190
191  bool ConnectionManager::addPacketAll(ENetPacket *packet) {
[1735]192    if(!instance_)
193      return false;
194    boost::recursive_mutex::scoped_lock lock(instance_->enet_mutex);
195    for(ClientInformation *i=ClientInformation::getBegin()->next(); i!=0; i=i->next()){
[1534]196      COUT(3) << "adding broadcast packet for client: " << i->getID() << std::endl;
197      if(enet_peer_send(i->getPeer(), 0, packet)!=0)
[1282]198        return false;
199    }
200    return true;
201  }
202
[1502]203  // we actually dont need that function, because host_service does that for us
[1282]204  bool ConnectionManager::sendPackets() {
[1735]205    if(server==NULL || !instance_)
[1282]206      return false;
[1735]207    boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]208    enet_host_flush(server);
209    lock.unlock();
210    return true;
[1282]211  }
212
213  void ConnectionManager::receiverThread() {
214    // what about some error-handling here ?
[1502]215    ENetEvent *event;
[1282]216    atexit(enet_deinitialize);
[1502]217    { //scope of the mutex
[1735]218      boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]219      enet_initialize();
220      server = enet_host_create(&bindAddress, NETWORK_MAX_CONNECTIONS, 0, 0);
221      lock.unlock();
222    }
[1282]223    if(server==NULL){
224      // add some error handling here ==========================
225      quit=true;
226      return;
227    }
228
[1502]229    event = new ENetEvent;
[1282]230    while(!quit){
[1502]231      { //mutex scope
[1735]232        boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]233        if(enet_host_service(server, event, NETWORK_WAIT_TIMEOUT)<0){
234          // we should never reach this point
235          quit=true;
236          continue;
237          // add some error handling here ========================
238        }
239        lock.unlock();
[1282]240      }
241      switch(event->type){
242        // log handling ================
243        case ENET_EVENT_TYPE_CONNECT:
[1502]244          COUT(3) << "adding event_type_connect to queue" << std::endl;
245        case ENET_EVENT_TYPE_DISCONNECT:
246          //addClient(event);
[1282]247          //this is a workaround to ensure thread safety
[1502]248          //COUT(5) << "Con.Man: connection event has occured" << std::endl;
249          //break;
[1282]250        case ENET_EVENT_TYPE_RECEIVE:
251          //std::cout << "received data" << std::endl;
252          COUT(5) << "Con.Man: receive event has occured" << std::endl;
253          // only add, if client has connected yet and not been disconnected
[1502]254          //if(head_->findClient(&event->peer->address))
[1282]255            processData(event);
[1502]256            event = new ENetEvent;
257//           else
258//             COUT(3) << "received a packet from a client we don't know" << std::endl;
[1282]259          break;
[1502]260        //case ENET_EVENT_TYPE_DISCONNECT:
261          //clientDisconnect(event->peer);
262          //break;
[1282]263        case ENET_EVENT_TYPE_NONE:
[1502]264          //receiverThread_->yield();
265          usleep(1000);
[1282]266          break;
267      }
268//       usleep(100);
[1502]269      //receiverThread_->yield(); //TODO: find apropriate
[1282]270    }
271    disconnectClients();
272    // if we're finishied, destroy server
[1502]273    {
[1735]274      boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]275      enet_host_destroy(server);
276      lock.unlock();
277    }
[1282]278  }
[1747]279
[1282]280  //### added some bugfixes here, but we cannot test them because
281  //### the server crashes everytime because of some gamestates
282  //### (trying to resolve that now)
283  void ConnectionManager::disconnectClients() {
284    ENetEvent event;
[1735]285    ClientInformation *temp = ClientInformation::getBegin()->next();
[1282]286    while(temp!=0){
[1502]287      {
[1735]288        boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]289        enet_peer_disconnect(temp->getPeer(), 0);
290        lock.unlock();
291      }
[1282]292      temp = temp->next();
293    }
294    //bugfix: might be the reason why server crashes when clients disconnects
[1735]295    temp = ClientInformation::getBegin()->next();
296    boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]297    while( temp!=0 && enet_host_service(server, &event, NETWORK_WAIT_TIMEOUT) >= 0){
[1282]298      switch (event.type)
299      {
300      case ENET_EVENT_TYPE_NONE: break;
301      case ENET_EVENT_TYPE_CONNECT: break;
302      case ENET_EVENT_TYPE_RECEIVE:
303        enet_packet_destroy(event.packet);
304        break;
305      case ENET_EVENT_TYPE_DISCONNECT:
306        COUT(4) << "disconnecting all clients" << std::endl;
[1735]307        if(ClientInformation::findClient(&(event.peer->address)))
308          delete ClientInformation::findClient(&(event.peer->address));
[1282]309        //maybe needs bugfix: might also be a reason for the server to crash
310        temp = temp->next();
311        break;
312      }
313    }
314    return;
315  }
316
317  bool ConnectionManager::processData(ENetEvent *event) {
318    // just add packet to the buffer
319    // this can be extended with some preprocessing
320    return buffer.push(event);
321  }
322
323
[1502]324
[1282]325  int ConnectionManager::getClientID(ENetPeer peer) {
326    return getClientID(peer.address);
327  }
328
329  int ConnectionManager::getClientID(ENetAddress address) {
[1735]330    return ClientInformation::findClient(&address)->getID();
[1282]331  }
332
333  ENetPeer *ConnectionManager::getClientPeer(int clientID) {
[1735]334    return ClientInformation::findClient(clientID)->getPeer();
[1282]335  }
336
[1735]337  /**
[1747]338   *
339   * @param clientID
[1735]340   */
341  void ConnectionManager::syncClassid(unsigned int clientID) {
[1360]342    unsigned int network_id=0, failures=0;
[1282]343    std::string classname;
344    orxonox::Identifier *id;
345    std::map<std::string, orxonox::Identifier*>::const_iterator it = orxonox::Factory::getFactoryBegin();
346    while(it != orxonox::Factory::getFactoryEnd()){
347      id = (*it).second;
348      if(id == NULL)
349        continue;
350      classname = id->getName();
351      network_id = id->getNetworkID();
[1360]352      if(network_id==0)
353        COUT(3) << "we got a null class id: " << id->getName() << std::endl;
[1282]354      COUT(4) << "Con.Man:syncClassid:\tnetwork_id: " << network_id << ", classname: " << classname << std::endl;
355
[1735]356      packet::ClassID *classid = new packet::ClassID( network_id, classname );
357      classid->setClientID(clientID);
358      while(!classid->send() && failures < 10){
[1360]359        failures++;
360      }
[1282]361      ++it;
362    }
[1502]363    //sendPackets();
[1282]364    COUT(4) << "syncClassid:\tall synchClassID packets have been sent" << std::endl;
365  }
366
[1747]367
368
[1282]369  bool ConnectionManager::removeShip(ClientInformation *client){
[1735]370    unsigned int id=client->getShipID();
[1747]371    orxonox::ObjectList<orxonox::SpaceShip>::iterator it;
372    for(it = orxonox::ObjectList<orxonox::SpaceShip>::begin(); it; ++it){
[1282]373      if(it->objectID!=id)
374        continue;
375      delete *it;
376    }
377    return true;
378  }
[1747]379
380
[1282]381  void ConnectionManager::disconnectClient(ClientInformation *client){
[1502]382    {
[1735]383      boost::recursive_mutex::scoped_lock lock(enet_mutex);
[1502]384      enet_peer_disconnect(client->getPeer(), 0);
385      lock.unlock();
386    }
[1282]387    removeShip(client);
388  }
389
[1747]390
[1282]391}
Note: See TracBrowser for help on using the repository browser.