Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/libraries/network/Server.cc @ 8831

Last change on this file since 8831 was 8706, checked in by dafrick, 13 years ago

Merging presentation branch back into trunk.
There are many new features and also a lot of other changes and bugfixes, if you want to know, digg through the svn log.
Not everything is yet working as it should, but it should be fairly stable. If you habe any bug reports, just send me an email.

  • Property svn:eol-style set to native
File size: 13.3 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++ Implementation: Server
31//
32// Description:
33//
34//
35// Author:  Oliver Scheuss, (C) 2007
36//
37// Copyright: See COPYING file that comes with this distribution
38//
39//
40
41#include "Server.h"
42
[5749]43#define WIN32_LEAN_AND_MEAN
[2773]44#include <enet/enet.h>
[1755]45#include <cassert>
[3214]46#include <string>
[1502]47
[5929]48#include "util/Clock.h"
[3214]49#include "util/Debug.h"
50#include "core/ObjectList.h"
[7284]51#include "core/command/Executor.h"
[1735]52#include "packet/Chat.h"
[3214]53#include "packet/ClassID.h"
54#include "packet/DeleteObjects.h"
55#include "packet/FunctionIDs.h"
56#include "packet/Gamestate.h"
[1735]57#include "packet/Welcome.h"
[2087]58#include "ChatListener.h"
[8327]59// #include "ClientInformation.h"
[3084]60#include "FunctionCallManager.h"
[3214]61#include "GamestateManager.h"
[7801]62#include "WANDiscovery.h"
[1502]63
[2171]64namespace orxonox
[1502]65{
[2087]66  const unsigned int MAX_FAILURES = 20;
[1747]67
[1502]68  /**
69  * Constructor for default values (bindaddress is set to ENET_HOST_ANY
70  *
71  */
[7163]72  Server::Server()
73  {
[3304]74    this->timeSinceLastUpdate_=0;
[1502]75  }
[1747]76
[7163]77  Server::Server(int port)
78  {
[3214]79    this->setPort( port );
[3304]80    this->timeSinceLastUpdate_=0;
[1502]81  }
82
83  /**
84  * Constructor
85  * @param port Port to listen on
86  * @param bindAddress Address to listen on
87  */
[7163]88  Server::Server(int port, const std::string& bindAddress)
89  {
[3214]90    this->setPort( port );
91    this->setBindAddress( bindAddress );
[3304]92    this->timeSinceLastUpdate_=0;
[1502]93  }
94
95  /**
[1907]96  * @brief Destructor
97  */
[7163]98  Server::~Server()
99  {
[1907]100  }
[1502]101
[7801]102
103  /** helper that connects to the master server */
104  void Server::helper_ConnectToMasterserver()
105  {
106//     WANDiscovery::getInstance().msc.sendRequest( MSPROTO_GAME_SERVER " "
107//       MSPROTO_REGISTER_SERVER );
108  }
109
[1502]110  /**
111  * This function opens the server by creating the listener thread
112  */
[7163]113  void Server::open()
114  {
115    Host::setActive(true);
[3214]116    COUT(4) << "opening server" << endl;
117    this->openListener();
[7801]118   
119    /* make discoverable on LAN */
[7163]120    LANDiscoverable::setActivity(true);
[7801]121
122    /* make discoverable on WAN */
123    WANDiscoverable::setActivity(true);
124    /* TODO this needs to be optional, we need a switch from the UI to
125     * enable/disable this
126     */
127//     helper_ConnectToMasterserver();
128
129    /* done */
[1502]130    return;
131  }
132
133  /**
134  * This function closes the server
135  */
[7163]136  void Server::close()
137  {
138    Host::setActive(false);
[3214]139    COUT(4) << "closing server" << endl;
140    this->disconnectClients();
141    this->closeListener();
[7801]142
143    /* tell master server we're closing */
144    COUT(2) << "disconnecting." << endl;
145    WANDiscoverable::setActivity(false);   
146    COUT(2) << "disconnecting done" << endl;
147
[7163]148    LANDiscoverable::setActivity(false);
[1502]149    return;
150  }
151
[7163]152  bool Server::processChat(const std::string& message, unsigned int playerID)
153  {
[8327]154//     ClientInformation *temp = ClientInformation::getBegin();
[1907]155    packet::Chat *chat;
[8327]156//     while(temp){
157    chat = new packet::Chat(message, playerID);
158    chat->setPeerID(NETWORK_PEER_ID_BROADCAST);
159    chat->send( static_cast<Host*>(this) );
160//         COUT(3) << "could not send Chat message to client ID: " << temp->getID() << std::endl;
161//       temp = temp->next();
162//     }
[2087]163//    COUT(1) << "Player " << playerID << ": " << message << std::endl;
[1907]164    return true;
[1502]165  }
166
167
[7801]168  /* handle incoming data */
169  int rephandler( char *addr, ENetEvent *ev )
170  { 
171    /* reply to pings */
172    if( !strncmp( (char *)ev->packet->data, MSPROTO_PING_GAMESERVER, 
173      MSPROTO_PING_GAMESERVER_LEN ) )
174      //this->msc.sendRequest( MSPROTO_ACK );
175      /* NOTE implement this after pollForReply
176       * reimplementation
177       */
178      return 0;
179
180    /* done handling, return all ok code 0 */
181    return 0;
182  }
183
184  void Server::helper_HandleMasterServerRequests()
185  { 
186    /* poll the master server for replies and see whether something
187     * has to be done or changed.
188     */
189    //WANDiscovery::getInstance().msc.pollForReply( rhandler, 10 );
190  }
191
[1502]192  /**
193  * Run this function once every tick
194  * calls processQueue and updateGamestate
195  * @param time time since last tick
196  */
[7163]197  void Server::update(const Clock& time)
198  {
[3304]199    // receive incoming packets
[3214]200    Connection::processQueue();
[7801]201
[7163]202    // receive and process incoming discovery packets
203    LANDiscoverable::update();
[7801]204   
205    // receive and process requests from master server
206    /* todo */
207    //helper_HandleMasterServerRequests();
[6417]208
[8327]209    if ( GamestateManager::hasPeers() )
[3214]210    {
[3304]211      // process incoming gamestates
212      GamestateManager::processGamestates();
[7801]213      FunctionCallManager::processBufferedFunctionCalls();
[6417]214
[3304]215      // send function calls to clients
[7801]216      FunctionCallManager::sendCalls( static_cast<Host*>(this) );
[6417]217
[3304]218      //this steers our network frequency
219      timeSinceLastUpdate_+=time.getDeltaTime();
220      if(timeSinceLastUpdate_>=NETWORK_PERIOD)
221      {
222        timeSinceLastUpdate_ -= static_cast<unsigned int>( timeSinceLastUpdate_ / NETWORK_PERIOD ) * NETWORK_PERIOD;
223        updateGamestate();
224      }
[7801]225//       sendPackets(); // flush the enet queue
[1502]226    }
227  }
[1747]228
[7801]229  void Server::queuePacket(ENetPacket *packet, int clientID, uint8_t channelID)
[7163]230  {
[7801]231    ServerConnection::addPacket(packet, clientID, channelID);
[1735]232  }
[6417]233
[2087]234  /**
[6417]235   * @brief: returns ping time to client in milliseconds
[2087]236   */
[7163]237  unsigned int Server::getRTT(unsigned int clientID)
238  {
[8327]239//     assert(ClientInformation::findClient(clientID));
240//     return ClientInformation::findClient(clientID)->getRTT();
241    // TODO: reimplement
242    return 0;
[2087]243  }
[6417]244
[5961]245  void Server::printRTT()
246  {
[8327]247//     for( ClientInformation* temp=ClientInformation::getBegin(); temp!=0; temp=temp->next() )
248//       COUT(0) << "Round trip time to client with ID: " << temp->getID() << " is " << temp->getRTT() << " ms" << endl;
[5961]249  }
[1502]250
251  /**
[2087]252   * @brief: return packet loss ratio to client (scales from 0 to 1)
253   */
[8351]254  float Server::getPacketLoss(unsigned int clientID)
[7163]255  {
[8327]256//     assert(ClientInformation::findClient(clientID));
257//     return ClientInformation::findClient(clientID)->getPacketLoss();
258    return 0.;
[2087]259  }
[1502]260
261  /**
262  * takes a new snapshot of the gamestate and sends it to the clients
263  */
[7163]264  void Server::updateGamestate()
265  {
[8327]266    if( this->clientIDs_.size()==0 )
[2662]267      //no client connected
[3304]268      return;
269    GamestateManager::update();
[7801]270//     COUT(5) << "Server: one gamestate update complete, goig to sendGameState" << std::endl;
[1502]271    //std::cout << "updated gamestate, sending it" << std::endl;
272    //if(clients->getGamestateID()!=GAMESTATEID_INITIAL)
[7801]273    sendGameStates();
[1907]274    sendObjectDeletes();
[7801]275//     COUT(5) << "Server: one sendGameState turn complete, repeat in next tick" << std::endl;
[1502]276    //std::cout << "sent gamestate" << std::endl;
277  }
278
279  /**
[7801]280  * sends the current gamestate to all peers
[1502]281  */
[7801]282  bool Server::sendGameStates()
[7163]283  {
[7801]284    std::vector<packet::Gamestate*> gamestates = GamestateManager::getGamestates();
285    std::vector<packet::Gamestate*>::iterator it;
286    for( it = gamestates.begin(); it != gamestates.end(); ++it )
287    {
288      (*it)->send(static_cast<Host*>(this));
289    }
[1502]290    return true;
291  }
[1747]292
[7801]293
[7163]294  bool Server::sendObjectDeletes()
295  {
[8327]296//     ClientInformation *temp = ClientInformation::getBegin();
297//     if( temp == NULL )
[2662]298      //no client connected
[8327]299    if( this->clientIDs_.size()==0 )
[2662]300      return true;
[1907]301    packet::DeleteObjects *del = new packet::DeleteObjects();
302    if(!del->fetchIDs())
[5929]303    {
304      delete del;
[1907]305      return true;  //everything ok (no deletes this tick)
[5929]306    }
[1907]307//     COUT(3) << "sending DeleteObjects" << std::endl;
[8327]308//     while(temp != NULL){
309//       if( !(temp->getSynched()) )
310//       {
311//         COUT(5) << "Server: not sending gamestate" << std::endl;
312//         temp=temp->next();
313//         continue;
314//       }
315//       int cid = temp->getID(); //get client id
316//       packet::DeleteObjects *cd = new packet::DeleteObjects(*del);
317//       assert(cd);
318    del->setPeerID(NETWORK_PEER_ID_BROADCAST);
319    if ( !del->send( static_cast<Host*>(this) ) )
320      COUT(3) << "Server: could not broadcast deleteObjects packet" << std::endl;
321//       temp=temp->next();
[1907]322      // gs gets automatically deleted by enet callback
[8327]323//     }
324//     delete del;
[1907]325    return true;
326  }
[1747]327
[1907]328
[8327]329  void Server::addPeer(uint32_t peerID)
[7163]330  {
[8327]331//     static unsigned int newid=1;
332//
333//     COUT(2) << "Server: adding client" << std::endl;
334//     ClientInformation *temp = ClientInformation::insertBack(new ClientInformation);
335//     if(!temp)
336//     {
337//       COUT(2) << "Server: could not add client" << std::endl;
338//     }
339//     temp->setID(newid);
340//     temp->setPeer(event->peer);
[2087]341
342    // inform all the listeners
[8327]343    this->clientIDs_.push_back(peerID);
344    ClientConnectionListener::broadcastClientConnected(peerID);
345    GamestateManager::addPeer(peerID);
[2087]346
[8327]347//     ++newid;
[2087]348
[8327]349    COUT(3) << "Server: added client id: " << peerID << std::endl;
350    createClient(peerID);
[2087]351}
[1747]352
[8327]353  void Server::removePeer(uint32_t peerID)
[5929]354  {
355    COUT(4) << "removing client from list" << std::endl;
[8327]356//     ClientInformation *client = ClientInformation::findClient(&event->peer->address);
357//     if(!client)
358//       return;
359//     else
360//     {
361  std::vector<uint32_t>::iterator it;
362  for( it=this->clientIDs_.begin(); it!=this->clientIDs_.end(); ++it )
363  {
364    if( *it == peerID )
[5929]365    {
[8327]366      this->clientIDs_.erase(it);
367      break;
368    }
369  }
370  ClientConnectionListener::broadcastClientDisconnected(peerID);
371  GamestateManager::removePeer(peerID);
[5929]372      //ServerConnection::disconnectClient( client );
[5961]373      //ClientConnectionListener::broadcastClientDisconnected( client->getID() ); //this is done in ClientInformation now
[8327]374//       delete client;
375//     }
[5929]376  }
[7801]377 
378  void Server::processPacket(packet::Packet* packet)
379  {
380    if( packet->isReliable() )
381    {
[8327]382      if( this->getLastReceivedGamestateID(packet->getPeerID()) >= packet->getRequiredGamestateID() )
[7801]383        packet->process(static_cast<Host*>(this));
384      else
385        this->packetQueue_.push_back(packet);
386    }
387    else
388      packet->process(static_cast<Host*>(this));
389  }
[5929]390
[7801]391
[7163]392  bool Server::createClient(int clientID)
393  {
[8327]394//     ClientInformation *temp = ClientInformation::findClient(clientID);
395//     if(!temp)
396//     {
397//       COUT(2) << "Server. could not create client with id: " << clientID << std::endl;
398//       return false;
399//     }
400//     COUT(4) << "Con.Man: creating client id: " << temp->getID() << std::endl;
[6417]401
[3084]402    // synchronise class ids
[8327]403    syncClassid(clientID);
[6417]404
[3084]405    // now synchronise functionIDs
406    packet::FunctionIDs *fIDs = new packet::FunctionIDs();
[7801]407    fIDs->setPeerID(clientID);
408    bool b = fIDs->send( static_cast<Host*>(this) );
[3084]409    assert(b);
[6417]410
[8327]411//     temp->setSynched(true);
[7801]412    GamestateManager::setSynched(clientID);
413   
[3084]414    COUT(4) << "sending welcome" << std::endl;
[8706]415    packet::Welcome *w = new packet::Welcome(clientID);
[8327]416    w->setPeerID(clientID);
[7801]417    b = w->send( static_cast<Host*>(this) );
[1907]418    assert(b);
[8327]419//     packet::Gamestate *g = new packet::Gamestate();
420//     g->setPeerID(clientID);
421//     b = g->collectData(0,packet::GAMESTATE_MODE_SERVER);
[7801]422//     assert(b);
[8327]423//     if(!b)
424//       return false; //no data for the client
425// //     b = g->compressData();
426// //     assert(b);
427//     b = g->send( static_cast<Host*>(this) );
428//     assert(b);
[1502]429    return true;
430  }
[6417]431
[8327]432  void Server::disconnectClient( uint32_t clientID )
[7163]433  {
[8327]434    ServerConnection::disconnectClient( clientID );
435    GamestateManager::removePeer( clientID );
[5929]436    // inform all the listeners
[5961]437    // ClientConnectionListener::broadcastClientDisconnected(client->getID()); // this is done in ClientInformation now
[1502]438  }
[2087]439
[7163]440  bool Server::chat(const std::string& message)
441  {
[2087]442      return this->sendChat(message, Host::getPlayerID());
443  }
444
[7163]445  bool Server::broadcast(const std::string& message)
446  {
[8327]447      return this->sendChat(message, NETWORK_PEER_ID_BROADCAST);
[2087]448  }
449
[7163]450  bool Server::sendChat(const std::string& message, unsigned int clientID)
451  {
[8327]452//     ClientInformation *temp = ClientInformation::getBegin();
[1907]453    packet::Chat *chat;
[8327]454//     while(temp)
[7163]455    {
[2087]456      chat = new packet::Chat(message, clientID);
[8327]457      chat->setPeerID(NETWORK_PEER_ID_BROADCAST);
458      chat->send( static_cast<Host*>(this) );
459//         COUT(3) << "could not send Chat message to client ID: " << temp->getID() << std::endl;
460//       temp = temp->next();
[1907]461    }
[2087]462//    COUT(1) << "Player " << Host::getPlayerID() << ": " << message << std::endl;
[2171]463    for (ObjectList<ChatListener>::iterator it = ObjectList<ChatListener>::begin(); it != ObjectList<ChatListener>::end(); ++it)
[2087]464      it->incomingChat(message, clientID);
465
[1907]466    return true;
467  }
[1747]468
[7163]469  void Server::syncClassid(unsigned int clientID)
470  {
[3214]471    int failures=0;
472    packet::ClassID *classid = new packet::ClassID();
[7801]473    classid->setPeerID(clientID);
474    while(!classid->send( static_cast<Host*>(this) ) && failures < 10){
[3214]475      failures++;
476    }
477    assert(failures<10);
478    COUT(4) << "syncClassid:\tall synchClassID packets have been sent" << std::endl;
479  }
480
[1502]481}
Note: See TracBrowser for help on using the repository browser.