Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/network/ClientConnection.cc @ 3198

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

merged netp4 back to trunk

  • Property svn:eol-style set to native
File size: 7.8 KB
RevLine 
[1502]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:
[3084]23 *      Oliver Scheuss
[1502]24 *   Co-authors:
25 *      ...
26 *
27 */
28
29//
30// C++ Interface: ClientConnection
31//
32// Description: The Class ClientConnection 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 "ClientConnection.h"
41
[2773]42#include <enet/enet.h>
[1502]43#include <iostream>
[3084]44#include <cassert>
[1502]45// boost.thread library for multithreading support
[1755]46#include <boost/thread/thread.hpp>
[1502]47#include <boost/bind.hpp>
[2773]48#include <boost/thread/recursive_mutex.hpp>
[1502]49
50#include "util/Sleep.h"
[1747]51#include "util/Debug.h"
[1502]52
[2171]53namespace orxonox
[1502]54{
55  //static boost::thread_group network_threads;
56
[2773]57  static boost::recursive_mutex enet_mutex_g;
[1502]58
[2087]59  ClientConnection::ClientConnection(int port, const std::string& address) {
[3084]60    quit_=false;
[1502]61    server=NULL;
[2773]62    serverAddress = new ENetAddress();
63    enet_address_set_host(serverAddress, address.c_str());
64    serverAddress->port = port;
[1502]65    established=false;
66  }
67
68  ClientConnection::ClientConnection(int port, const char *address) {
[3084]69    quit_=false;
[1502]70    server=NULL;
[2773]71    serverAddress = new ENetAddress();
72    enet_address_set_host(serverAddress, address);
73    serverAddress->port = port;
[1502]74    established=false;
75  }
76
77  bool ClientConnection::waitEstablished(int milisec) {
78    for(int i=0; i<=milisec && !established; i++)
[2087]79      msleep(1);
[1502]80
81    return established;
82  }
83
[1907]84  ClientConnection::~ClientConnection(){
85    if(established)
86      closeConnection();
[2773]87    delete serverAddress; // surely was created
[1502]88  }
89
90  ENetEvent *ClientConnection::getEvent(){
91    if(!buffer.isEmpty())
92      return buffer.pop();
93    else
94      return NULL;
95  }
96
97  bool ClientConnection::queueEmpty() {
98    return buffer.isEmpty();
99  }
100
101  bool ClientConnection::createConnection() {
102    receiverThread_ = new boost::thread(boost::bind(&ClientConnection::receiverThread, this));
103    //network_threads.create_thread(boost::bind(boost::mem_fn(&ClientConnection::receiverThread), this));
104    // wait 10 seconds for the connection to be established
[1534]105    return waitEstablished(NETWORK_CLIENT_CONNECT_TIMEOUT);
[1502]106  }
107
108  bool ClientConnection::closeConnection() {
[3084]109    quit_=true;
[1502]110    //network_threads.join_all();
111    receiverThread_->join();
112    established=false;
113    return true;
114  }
115
116
117  bool ClientConnection::addPacket(ENetPacket *packet) {
118    if(server==NULL)
119      return false;
120    if(packet==NULL){
121      COUT(3) << "Cl.con: addpacket: invalid packet" << std::endl;
122      return false;
123    }
[2773]124    boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1502]125    if(enet_peer_send(server, 0, packet)<0)
126      return false;
127    else
128      return true;
129  }
130
131  bool ClientConnection::sendPackets() {
132    if(server==NULL)
133      return false;
[2773]134    boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1502]135    enet_host_flush(client);
136    lock.unlock();
137    return true;
138  }
139
140  void ClientConnection::receiverThread() {
141    // what about some error-handling here ?
142    atexit(enet_deinitialize);
143    ENetEvent *event;
144    {
[2773]145      boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1502]146      enet_initialize();
147      client = enet_host_create(NULL, NETWORK_CLIENT_MAX_CONNECTIONS, 0, 0);
148      lock.unlock();
149    }
150    if(client==NULL) {
151      COUT(2) << "ClientConnection: could not create client host" << std::endl;
152      // add some error handling here ==========================
[3084]153      quit_=true;
[1502]154    }
155    //connect to the server
156    if(!establishConnection()){
157      COUT(2) << "clientConn: receiver thread: could not establishConnection" << std::endl;
[3084]158      quit_=true;
[1502]159      return;
160    }
161    event = new ENetEvent;
162    //main loop
[3084]163    while(!quit_){
[1502]164      //std::cout << "connection loop" << std::endl;
165      {
[2773]166        boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1534]167        if(enet_host_service(client, event, NETWORK_CLIENT_WAIT_TIME)<0){
[1502]168          // we should never reach this point
[3084]169//              assert(0);
170          printf("ClientConnection: ENet returned with an error!\n");
[3198]171//           quit_=true;
172          continue;
[1502]173          // add some error handling here ========================
174        }
175        lock.unlock();
176      }
177      switch(event->type){
178        // log handling ================
179      case ENET_EVENT_TYPE_CONNECT:
180        break;
181      case ENET_EVENT_TYPE_RECEIVE:
[2662]182        //COUT(5) << "Cl.Con: receiver-Thread while loop: got new packet" << std::endl;
[1502]183        if ( !processData(event) ) COUT(2) << "Current packet was not pushed to packetBuffer -> ev ongoing SegFault" << std::endl;
[2662]184        //COUT(5) << "Cl.Con: processed Data in receiver-thread while loop" << std::endl;
[1502]185        event = new ENetEvent;
186        break;
187      case ENET_EVENT_TYPE_DISCONNECT:
[3084]188        quit_=true;
189        printf("Received disconnect Packet from Server!\n");
[1502]190        // server closed the connection
191        return;
192        break;
193      case ENET_EVENT_TYPE_NONE:
194        //receiverThread_->yield();
[2087]195        msleep(1);
[1502]196        break;
197      }
198    }
[3198]199    delete event;
[1502]200    // now disconnect
201
202    if(!disconnectConnection())
[3198]203    {
204      printf("could not disconnect properly\n");
[1502]205      // if disconnecting failed destroy conn.
[2773]206      boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1502]207      enet_peer_reset(server);
[3198]208    }
209    else
210      printf("properly disconnected\n");
[1502]211    return;
212  }
213
214  bool ClientConnection::disconnectConnection() {
215    ENetEvent event;
[3084]216    if(this->quit_)
217      return true;
[2773]218    boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
[1502]219    enet_peer_disconnect(server, 0);
[3084]220    while(enet_host_service(client, &event, NETWORK_CLIENT_WAIT_TIME) >= 0){
[1502]221      switch (event.type)
222      {
223      case ENET_EVENT_TYPE_NONE:
224      case ENET_EVENT_TYPE_CONNECT:
225      case ENET_EVENT_TYPE_RECEIVE:
226        enet_packet_destroy(event.packet);
227        break;
228      case ENET_EVENT_TYPE_DISCONNECT:
[3198]229        printf("received disconnect confirmation from server");
[1502]230        return true;
231      }
232    }
233    enet_peer_reset(server);
234    return false;
235  }
236
237  bool ClientConnection::establishConnection() {
238    ENetEvent event;
239    // connect to peer (server is type ENetPeer*)
[2773]240    boost::recursive_mutex::scoped_lock lock(enet_mutex_g);
241    server = enet_host_connect(client, serverAddress, NETWORK_CLIENT_CHANNELS);
[1502]242    if(server==NULL) {
243      COUT(2) << "ClientConnection: server == NULL" << std::endl;
244      // error handling
245      return false;
246    }
247    // handshake
[3084]248    while(enet_host_service(client, &event, NETWORK_CLIENT_WAIT_TIME)>=0 && !quit_){
[1534]249      if( event.type == ENET_EVENT_TYPE_CONNECT ){
250        established=true;
251        return true;
252      }
[1502]253    }
[1534]254    COUT(2) << "ClientConnection: enet_host_service < 0 or event.type != ENET_EVENT_TYPE_CONNECT # EVENT:" << event.type << std::endl;
255    return false;
[1502]256  }
257
258  bool ClientConnection::processData(ENetEvent *event) {
259    COUT(5) << "Cl.Con: got packet, pushing to queue" << std::endl;
260    // just add packet to the buffer
261    // this can be extended with some preprocessing
262    return buffer.push(event);
263  }
264
265}
Note: See TracBrowser for help on using the repository browser.