2005-09-02 08:35:25 +02:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
|
|
|
|
/*
|
2009-07-30 14:39:45 +02:00
|
|
|
* Copyright 2005-2009 by Eric House (xwords@eehouse.org). All rights
|
|
|
|
* reserved.
|
2005-09-02 08:35:25 +02:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
2005-09-02 09:18:39 +02:00
|
|
|
#include <assert.h>
|
2006-03-21 05:05:33 +01:00
|
|
|
#include <stdio.h>
|
2008-12-30 06:13:30 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2005-09-02 09:18:39 +02:00
|
|
|
#include <pthread.h>
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
#include "crefmgr.h"
|
|
|
|
#include "cref.h"
|
|
|
|
#include "mlock.h"
|
2005-10-15 18:30:10 +02:00
|
|
|
#include "configs.h"
|
|
|
|
#include "timermgr.h"
|
2005-09-02 08:35:25 +02:00
|
|
|
|
|
|
|
class SocketStuff {
|
|
|
|
public:
|
|
|
|
SocketStuff( CookieRef* cref )
|
|
|
|
: m_cref(cref)
|
|
|
|
{
|
|
|
|
pthread_mutex_init( &m_writeMutex, NULL );
|
|
|
|
}
|
|
|
|
~SocketStuff() { pthread_mutex_destroy( &m_writeMutex ); }
|
|
|
|
CookieRef* m_cref;
|
|
|
|
pthread_mutex_t m_writeMutex; /* so only one thread writes at a time */
|
|
|
|
};
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
static CRefMgr* s_instance = NULL;
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
/* static */ CRefMgr*
|
|
|
|
CRefMgr::Get()
|
|
|
|
{
|
2005-10-23 17:49:48 +02:00
|
|
|
if ( s_instance == NULL ) {
|
|
|
|
s_instance = new CRefMgr();
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
return s_instance;
|
2005-09-02 08:35:25 +02:00
|
|
|
} /* Get */
|
|
|
|
|
|
|
|
CRefMgr::CRefMgr()
|
2005-10-01 18:33:45 +02:00
|
|
|
: m_nextCID(0)
|
2009-07-31 14:56:04 +02:00
|
|
|
, m_startTime(time(NULL))
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
|
|
|
/* should be using pthread_once() here */
|
2007-02-06 06:52:22 +01:00
|
|
|
pthread_mutex_init( &m_SocketStuffMutex, NULL );
|
2009-08-21 14:00:09 +02:00
|
|
|
pthread_mutex_init( &m_nextCIDMutex, NULL );
|
2010-05-28 04:31:27 +02:00
|
|
|
pthread_mutex_init( &m_roomsFilledMutex, NULL );
|
2009-07-28 07:15:26 +02:00
|
|
|
pthread_mutex_init( &m_freeList_mutex, NULL );
|
2005-09-02 08:35:25 +02:00
|
|
|
pthread_rwlock_init( &m_cookieMapRWLock, NULL );
|
|
|
|
}
|
|
|
|
|
|
|
|
CRefMgr::~CRefMgr()
|
|
|
|
{
|
2005-10-23 17:49:48 +02:00
|
|
|
assert( this == s_instance );
|
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
pthread_mutex_destroy( &m_freeList_mutex );
|
2005-10-23 17:49:48 +02:00
|
|
|
pthread_rwlock_destroy( &m_cookieMapRWLock );
|
|
|
|
|
2009-06-29 14:58:30 +02:00
|
|
|
SocketMap::iterator iter;
|
|
|
|
for ( iter = m_SocketStuff.begin(); iter != m_SocketStuff.end(); ++iter ) {
|
|
|
|
SocketStuff* stuff = iter->second;
|
|
|
|
delete stuff;
|
|
|
|
}
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
s_instance = NULL;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
void
|
|
|
|
CRefMgr::CloseAll()
|
|
|
|
{
|
|
|
|
/* Get every cref instance, shut it down */
|
2005-10-30 06:16:35 +01:00
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
for ( ; ; ) {
|
|
|
|
CookieRef* cref = NULL;
|
2005-10-30 06:16:35 +01:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
RWWriteLock rwl( &m_cookieMapRWLock );
|
|
|
|
CookieMap::iterator iter = m_cookieMap.begin();
|
|
|
|
if ( iter == m_cookieMap.end() ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cref = iter->second;
|
|
|
|
{
|
|
|
|
SafeCref scr( cref ); /* cref */
|
|
|
|
scr.Shutdown();
|
|
|
|
}
|
2005-10-30 06:16:35 +01:00
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
}
|
|
|
|
} /* CloseAll */
|
|
|
|
|
2009-09-26 16:37:49 +02:00
|
|
|
/* Find a game to which this guy belongs. Return NULL if none's found and
|
|
|
|
* presumably a new one will be created.
|
|
|
|
*
|
|
|
|
* Match by connName if provided. If I match, great. But what if there isn't
|
|
|
|
* room, which would normally mean I'm already there and this is a duplicate
|
|
|
|
* packet. Should a dup be simply dropped, or should I reply? If the dup
|
|
|
|
* happened because our earlier reply was dropped then we should in fact
|
|
|
|
* reply.
|
|
|
|
*
|
|
|
|
* If match is by cookie (called Room in the UI) we need to be careful. If
|
|
|
|
* this is an established game, meaning that at some point all devices were
|
|
|
|
* there and a connName was established, then check if the incoming seed is in
|
|
|
|
* that connName, otherwise reject it (to form its own game.) But otherwise,
|
|
|
|
* if either we can't match seeds yet or this guy's does match, let him in.
|
|
|
|
* Again there's the duplicate packet issue if the game is "full" already.
|
|
|
|
*
|
|
|
|
* Match can also occur on cookie only -- device has no connName perhaps
|
|
|
|
* because the relay crashed before sending it -- where the device belongs in
|
|
|
|
* an existing game. That we detect by checking if the new arrival's seed is
|
|
|
|
* a component of the candidate's connName.
|
2009-08-21 14:00:09 +02:00
|
|
|
*/
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
CookieRef*
|
2009-08-21 14:00:09 +02:00
|
|
|
CRefMgr::FindOpenGameFor( const char* cookie, const char* connName,
|
2009-09-26 16:37:49 +02:00
|
|
|
HostID hid, int socket, int nPlayersH, int nPlayersT,
|
2010-09-10 10:59:37 +02:00
|
|
|
int gameSeed, int langCode, bool* alreadyHere )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-11-02 02:01:47 +01:00
|
|
|
logf( XW_LOGINFO, "%s(cookie=%s,connName=%s,hid=%d,seed=%x,socket=%d,"
|
|
|
|
"here=%d,total=%d)", __func__, cookie, connName, hid, gameSeed,
|
|
|
|
socket, nPlayersH, nPlayersT );
|
2009-08-21 14:00:09 +02:00
|
|
|
CookieRef* found = NULL;
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
assert( !!cookie || !!connName );
|
|
|
|
RWReadLock rwl( &m_cookieMapRWLock );
|
2009-08-21 14:00:09 +02:00
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
CookieMap::iterator iter;
|
|
|
|
for ( iter = m_cookieMap.begin();
|
|
|
|
NULL == found && iter != m_cookieMap.end();
|
|
|
|
++iter ) {
|
|
|
|
CookieRef* cref = iter->second;
|
2009-08-21 14:00:09 +02:00
|
|
|
|
2010-09-10 22:45:40 +02:00
|
|
|
/* Reject immediately if language code or proposed game size don't
|
|
|
|
match. */
|
|
|
|
if ( (cref->GetLangCode() != langCode)
|
|
|
|
|| (cref->GetPlayersSought() != nPlayersT) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
if ( !!connName && 0 == strcmp( cref->ConnName(), connName ) ) {
|
|
|
|
found = cref;
|
|
|
|
/* if ( cref->Lock() ) { */
|
|
|
|
|
|
|
|
/* assert( !cookie || */
|
|
|
|
/* 0 == strcasecmp( cookie, cref->Cookie() ) ); */
|
|
|
|
/* if ( cref->SeedBelongs( gameSeed ) ) { */
|
|
|
|
/* logf( XW_LOGINFO, "%s: SeedBelongs: dup packet?", */
|
|
|
|
/* __func__ ); */
|
|
|
|
/* *alreadyHere = true; */
|
|
|
|
/* found = cref; */
|
|
|
|
/* } else if ( cref->GameOpen( cookie, false, */
|
|
|
|
/* alreadyHere ) ) { */
|
|
|
|
/* found = cref; */
|
|
|
|
/* } else { */
|
|
|
|
/* /\* drop if we match on connName and it's not */
|
|
|
|
/* wanted; must be dup. *\/ */
|
|
|
|
/* *alreadyHere = true; */
|
|
|
|
/* } */
|
|
|
|
/* cref->Unlock(); */
|
|
|
|
/* } */
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !found && !!cookie ) {
|
|
|
|
if ( 0 == strcasecmp( cref->Cookie(), cookie ) ) {
|
|
|
|
if ( cref->Lock() ) {
|
|
|
|
assert( cref->ConnName()[0] );
|
|
|
|
if ( cref->AlreadyHere( gameSeed, -1 ) ) {
|
|
|
|
found = cref;
|
|
|
|
*alreadyHere = true;
|
|
|
|
} else if ( cref->GameOpen( cookie ) ) {
|
|
|
|
found = cref;
|
|
|
|
} else if ( cref->HasSocket_locked(socket) ) {
|
|
|
|
assert( 0 ); /* should have dumped the socket */
|
|
|
|
logf( XW_LOGINFO, "%s: HasSocket case", __func__);
|
|
|
|
found = cref;
|
2009-08-21 14:00:09 +02:00
|
|
|
}
|
2010-09-10 10:30:40 +02:00
|
|
|
cref->Unlock();
|
2005-10-01 18:33:45 +02:00
|
|
|
}
|
|
|
|
}
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
2010-09-10 10:30:40 +02:00
|
|
|
} /* for */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2009-08-21 14:00:09 +02:00
|
|
|
logf( XW_LOGINFO, "%s=>%p", __func__, found );
|
|
|
|
return found;
|
2005-10-01 18:33:45 +02:00
|
|
|
} /* FindOpenGameFor */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieID
|
|
|
|
CRefMgr::nextCID( const char* connName )
|
|
|
|
{
|
|
|
|
/* Later may want to guarantee that wrap-around doesn't cause an overlap.
|
|
|
|
But that's really only a theoretical possibility. */
|
2009-08-21 14:00:09 +02:00
|
|
|
MutexLock ml(&m_nextCIDMutex);
|
2005-10-01 18:33:45 +02:00
|
|
|
return ++m_nextCID;
|
|
|
|
} /* nextCID */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2010-05-28 04:31:27 +02:00
|
|
|
void
|
|
|
|
CRefMgr::IncrementFullCount( void )
|
2009-07-06 03:50:51 +02:00
|
|
|
{
|
2010-05-28 04:31:27 +02:00
|
|
|
MutexLock ml( &m_roomsFilledMutex );
|
|
|
|
++m_nRoomsFilled;
|
2009-07-06 03:50:51 +02:00
|
|
|
}
|
|
|
|
|
2010-05-28 04:31:27 +02:00
|
|
|
int
|
|
|
|
CRefMgr::GetNumRoomsFilled( void )
|
|
|
|
{
|
|
|
|
MutexLock ml(&m_roomsFilledMutex);
|
|
|
|
return m_nRoomsFilled;
|
|
|
|
}
|
2009-07-13 05:01:22 +02:00
|
|
|
|
|
|
|
int
|
|
|
|
CRefMgr::GetSize( void )
|
|
|
|
{
|
|
|
|
return m_cookieMap.size();
|
|
|
|
}
|
|
|
|
|
2009-08-21 14:32:57 +02:00
|
|
|
void
|
|
|
|
CRefMgr::GetStats( CrefMgrInfo& mgrInfo )
|
|
|
|
{
|
2010-05-28 04:31:27 +02:00
|
|
|
mgrInfo.m_nRoomsFilled = GetNumRoomsFilled();
|
2009-08-21 14:32:57 +02:00
|
|
|
mgrInfo.m_startTimeSpawn = m_startTime;
|
|
|
|
|
2009-12-16 03:43:52 +01:00
|
|
|
if ( 0 == m_ports.length() ) {
|
|
|
|
RelayConfigs* cfg = RelayConfigs::GetConfigs();
|
|
|
|
vector<int> ints;
|
2010-08-12 15:42:33 +02:00
|
|
|
if ( cfg->GetValueFor( "GAME_PORTS", ints ) ) {
|
2009-12-16 03:43:52 +01:00
|
|
|
vector<int>::const_iterator iter;
|
|
|
|
for ( iter = ints.begin(); ; ) {
|
|
|
|
char buf[8];
|
|
|
|
snprintf( buf, sizeof(buf), "%d", *iter );
|
|
|
|
m_ports += buf;
|
|
|
|
++iter;
|
|
|
|
if ( iter == ints.end() ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
m_ports += ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mgrInfo.m_ports = m_ports.c_str();
|
|
|
|
|
2009-08-21 14:32:57 +02:00
|
|
|
RWReadLock rwl( &m_cookieMapRWLock );
|
|
|
|
mgrInfo.m_nCrefsCurrent = m_cookieMap.size();
|
|
|
|
|
|
|
|
CookieMap::iterator iter;
|
|
|
|
for ( iter = m_cookieMap.begin(); iter != m_cookieMap.end(); ++iter ) {
|
|
|
|
CookieRef* cref = iter->second;
|
|
|
|
|
|
|
|
CrefInfo info;
|
|
|
|
info.m_cookie = cref->Cookie();
|
|
|
|
info.m_connName = cref->ConnName();
|
|
|
|
info.m_cookieID = cref->GetCookieID();
|
|
|
|
info.m_curState = cref->CurState();
|
|
|
|
info.m_totalSent = cref->GetTotalSent();
|
|
|
|
info.m_nPlayersSought = cref->GetPlayersSought();
|
|
|
|
info.m_nPlayersHere = cref->GetPlayersHere();
|
|
|
|
info.m_startTime = cref->GetStarttime();
|
2010-09-10 10:59:37 +02:00
|
|
|
info.m_langCode = cref->GetLangCode();
|
2009-08-21 14:32:57 +02:00
|
|
|
|
|
|
|
SafeCref sc(cref);
|
2009-09-26 16:37:49 +02:00
|
|
|
sc.GetHostsConnected( &info.m_hostsIds, &info.m_hostSeeds, &info.m_hostIps );
|
2009-08-21 14:32:57 +02:00
|
|
|
|
|
|
|
mgrInfo.m_crefInfo.push_back( info );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
CookieID
|
2005-10-01 18:33:45 +02:00
|
|
|
CRefMgr::cookieIDForConnName( const char* connName )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieID cid = 0;
|
|
|
|
/* for now, just walk the existing data structure and see if the thing's
|
|
|
|
in use. If it isn't, return a new id. */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
|
|
|
RWReadLock rwl( &m_cookieMapRWLock );
|
|
|
|
|
|
|
|
CookieMap::iterator iter = m_cookieMap.begin();
|
|
|
|
while ( iter != m_cookieMap.end() ) {
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieRef* cref = iter->second;
|
|
|
|
if ( 0 == strcmp( cref->ConnName(), connName ) ) {
|
|
|
|
cid = iter->first;
|
|
|
|
break;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
2005-10-01 18:33:45 +02:00
|
|
|
|
|
|
|
return cid;
|
|
|
|
} /* cookieIDForConnName */
|
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
void
|
|
|
|
CRefMgr::addToFreeList( CookieRef* cref )
|
|
|
|
{
|
|
|
|
MutexLock ml( &m_freeList_mutex );
|
|
|
|
m_freeList.push_back( cref );
|
|
|
|
}
|
|
|
|
|
|
|
|
CookieRef*
|
|
|
|
CRefMgr::getFromFreeList( void )
|
|
|
|
{
|
|
|
|
CookieRef* cref = NULL;
|
|
|
|
MutexLock ml( &m_freeList_mutex );
|
|
|
|
if ( m_freeList.size() > 0 ) {
|
|
|
|
cref = m_freeList.front();
|
|
|
|
m_freeList.pop_front();
|
|
|
|
}
|
|
|
|
return cref;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieRef*
|
2009-08-21 14:00:09 +02:00
|
|
|
CRefMgr::getMakeCookieRef_locked( const char* cookie, const char* connName,
|
|
|
|
HostID hid, int socket, int nPlayersH,
|
2010-09-10 10:59:37 +02:00
|
|
|
int nPlayersT, int langCode, int gameSeed )
|
2005-10-01 18:33:45 +02:00
|
|
|
{
|
|
|
|
CookieRef* cref;
|
|
|
|
|
2009-08-21 14:00:09 +02:00
|
|
|
/* We have a cookie from a new connection or from a reconnect. This may
|
|
|
|
be the first time it's been seen, or there may be a game currently in
|
|
|
|
the XW_ST_CONNECTING state, or it may be a dupe of a connect packet.
|
|
|
|
If there's a game, cool. Otherwise add a new one. Pass the connName
|
|
|
|
which will be used if set, but if not set we'll be generating another
|
|
|
|
later when the game is complete.
|
|
|
|
*/
|
2005-10-01 18:33:45 +02:00
|
|
|
|
2009-09-26 16:37:49 +02:00
|
|
|
bool alreadyHere;
|
|
|
|
cref = FindOpenGameFor( cookie, connName, hid, socket, nPlayersH, nPlayersT,
|
2010-09-10 10:59:37 +02:00
|
|
|
gameSeed, langCode, &alreadyHere );
|
2009-09-26 16:37:49 +02:00
|
|
|
if ( cref == NULL && !alreadyHere ) {
|
2010-09-10 10:59:37 +02:00
|
|
|
cref = AddNew( cookie, connName, nextCID( NULL ), langCode );
|
2005-10-01 18:33:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return cref;
|
|
|
|
} /* getMakeCookieRef_locked */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2009-02-01 17:00:20 +01:00
|
|
|
bool
|
2005-09-02 08:35:25 +02:00
|
|
|
CRefMgr::Associate( int socket, CookieRef* cref )
|
|
|
|
{
|
|
|
|
MutexLock ml( &m_SocketStuffMutex );
|
2009-11-02 02:01:47 +01:00
|
|
|
return Associate_locked( socket, cref );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
CRefMgr::Associate_locked( int socket, CookieRef* cref )
|
|
|
|
{
|
|
|
|
bool isNew = false;
|
2005-09-02 08:35:25 +02:00
|
|
|
SocketMap::iterator iter = m_SocketStuff.find( socket );
|
2009-02-01 17:00:20 +01:00
|
|
|
/* This isn't enough. Must provide a way to reuse sockets should a
|
|
|
|
genuinely different connection appear. Now maybe we already remove
|
|
|
|
this reference when a socket is closed. Test this! Or assert
|
|
|
|
something here. Bottom line: need to swallow repeated/duplicate
|
|
|
|
connect messages from same host. */
|
|
|
|
if ( iter == m_SocketStuff.end() ) {
|
|
|
|
SocketStuff* stuff = new SocketStuff( cref );
|
|
|
|
m_SocketStuff.insert( pair< int, SocketStuff* >( socket, stuff ) );
|
|
|
|
isNew = true;
|
|
|
|
} else {
|
|
|
|
logf( XW_LOGERROR, "Already have cref/threadID pair for socket %d; "
|
|
|
|
"error???", socket );
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
2009-02-01 17:00:20 +01:00
|
|
|
return isNew;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
|
2005-09-05 17:50:49 +02:00
|
|
|
void
|
2009-11-02 02:01:47 +01:00
|
|
|
CRefMgr::Disassociate_locked( int socket, CookieRef* cref )
|
2005-09-05 17:50:49 +02:00
|
|
|
{
|
|
|
|
SocketMap::iterator iter = m_SocketStuff.find( socket );
|
|
|
|
if ( iter == m_SocketStuff.end() ) {
|
2009-07-28 07:15:26 +02:00
|
|
|
logf( XW_LOGERROR, "can't find SocketStuff for socket %d", socket );
|
2005-09-05 17:50:49 +02:00
|
|
|
} else {
|
|
|
|
SocketStuff* stuff = iter->second;
|
2009-07-06 03:50:51 +02:00
|
|
|
assert( cref == NULL || stuff->m_cref == cref );
|
2005-09-05 17:50:49 +02:00
|
|
|
delete stuff;
|
|
|
|
m_SocketStuff.erase( iter );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-02 02:01:47 +01:00
|
|
|
void
|
|
|
|
CRefMgr::Disassociate( int socket, CookieRef* cref )
|
|
|
|
{
|
|
|
|
MutexLock ml( &m_SocketStuffMutex );
|
|
|
|
Disassociate_locked( socket, cref );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CRefMgr::MoveSockets( vector<int> sockets, CookieRef* cref )
|
|
|
|
{
|
|
|
|
MutexLock ml( &m_SocketStuffMutex );
|
|
|
|
vector<int>::iterator iter;
|
|
|
|
for ( iter = sockets.begin(); iter != sockets.end(); ++iter ) {
|
|
|
|
Disassociate_locked( *iter, NULL );
|
|
|
|
Associate_locked( *iter, cref );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
#if 0
|
2005-09-02 08:35:25 +02:00
|
|
|
pthread_mutex_t*
|
|
|
|
CRefMgr::GetWriteMutexForSocket( int socket )
|
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
pthread_mutex_t* mutex = NULL;
|
2005-09-02 08:35:25 +02:00
|
|
|
MutexLock ml( &m_SocketStuffMutex );
|
|
|
|
SocketMap::iterator iter = m_SocketStuff.find( socket );
|
|
|
|
if ( iter != m_SocketStuff.end() ) {
|
|
|
|
SocketStuff* stuff = iter->second;
|
2009-07-28 07:15:26 +02:00
|
|
|
/* this is dangerous! What if we want to nuke SocketStuff while this
|
|
|
|
is locked? And shouldn't it be the case that only one thread at a
|
|
|
|
time can be trying to write to one of a given cref's sockets since
|
|
|
|
only one thread at a time is handling a cref? */
|
|
|
|
mutex = &stuff->m_writeMutex;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGERROR, "GetWriteMutexForSocket: not found" );
|
2009-07-28 07:15:26 +02:00
|
|
|
return mutex;
|
2005-09-02 08:35:25 +02:00
|
|
|
} /* GetWriteMutexForSocket */
|
2009-07-28 07:15:26 +02:00
|
|
|
#endif
|
2005-09-02 08:35:25 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
CRefMgr::RemoveSocketRefs( int socket )
|
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
{
|
|
|
|
SafeCref scr( socket );
|
|
|
|
scr.Remove( socket );
|
|
|
|
}
|
2009-07-06 03:50:51 +02:00
|
|
|
|
|
|
|
Disassociate( socket, NULL );
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CRefMgr::PrintSocketInfo( int socket, string& out )
|
|
|
|
{
|
|
|
|
SafeCref scr( socket );
|
2005-10-16 03:19:25 +02:00
|
|
|
const char* name = scr.Cookie();
|
2005-10-01 18:33:45 +02:00
|
|
|
if ( name != NULL && name[0] != '\0' ) {
|
2005-09-02 08:35:25 +02:00
|
|
|
char buf[64];
|
|
|
|
|
|
|
|
snprintf( buf, sizeof(buf), "* socket: %d\n", socket );
|
|
|
|
out += buf;
|
|
|
|
|
2005-10-01 18:33:45 +02:00
|
|
|
snprintf( buf, sizeof(buf), " in cookie: %s\n", name );
|
2005-09-02 08:35:25 +02:00
|
|
|
out += buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ SocketsIterator
|
|
|
|
CRefMgr::MakeSocketsIterator()
|
|
|
|
{
|
|
|
|
pthread_mutex_lock( &m_SocketStuffMutex );
|
|
|
|
SocketsIterator iter( m_SocketStuff.begin(), m_SocketStuff.end(),
|
|
|
|
&m_SocketStuffMutex );
|
|
|
|
return iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
CookieRef*
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::getCookieRef( CookieID cookieID )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
return getCookieRef_impl( cookieID );
|
|
|
|
} /* getCookieRef */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
|
|
|
CookieRef*
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::getCookieRef( int socket )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
|
|
|
CookieRef* cref = NULL;
|
2009-07-28 07:15:26 +02:00
|
|
|
MutexLock ml( &m_SocketStuffMutex );
|
|
|
|
SocketMap::iterator iter = m_SocketStuff.find( socket );
|
|
|
|
if ( iter != m_SocketStuff.end() ) {
|
|
|
|
SocketStuff* stuff = iter->second;
|
|
|
|
cref = stuff->m_cref;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return cref;
|
2009-07-28 07:15:26 +02:00
|
|
|
} /* getCookieRef */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2007-11-19 00:38:56 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2005-10-15 18:30:10 +02:00
|
|
|
/* static */ void
|
|
|
|
CRefMgr::heartbeatProc( void* closure )
|
|
|
|
{
|
|
|
|
CRefMgr* self = (CRefMgr*)closure;
|
2009-07-31 14:56:04 +02:00
|
|
|
self->checkHeartbeats( ::uptime() );
|
2005-10-15 18:30:10 +02:00
|
|
|
} /* heartbeatProc */
|
2007-11-19 00:38:56 +01:00
|
|
|
#endif
|
2005-10-15 18:30:10 +02:00
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
CookieRef*
|
2010-09-10 10:59:37 +02:00
|
|
|
CRefMgr::AddNew( const char* cookie, const char* connName, CookieID id,
|
|
|
|
int langCode )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-11-02 02:01:47 +01:00
|
|
|
/* PENDING: should this return a locked cref? */
|
|
|
|
logf( XW_LOGINFO, "%s( cookie=%s, connName=%s, cid=%d)", __func__,
|
|
|
|
cookie, connName, id );
|
2009-07-28 07:15:26 +02:00
|
|
|
|
|
|
|
CookieRef* ref = getFromFreeList();
|
2005-10-02 17:39:38 +02:00
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
RWWriteLock rwl( &m_cookieMapRWLock );
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGINFO, "making new cref: %d", id );
|
2009-07-28 07:15:26 +02:00
|
|
|
|
|
|
|
if ( !!ref ) {
|
2010-09-09 05:17:32 +02:00
|
|
|
logf( XW_LOGVERBOSE1, "using from free list" );
|
2010-09-10 10:59:37 +02:00
|
|
|
ref->ReInit( cookie, connName, id, langCode );
|
2009-07-28 07:15:26 +02:00
|
|
|
} else {
|
2010-09-09 05:17:32 +02:00
|
|
|
logf( XW_LOGVERBOSE1, "calling constructor" );
|
2010-09-10 10:59:37 +02:00
|
|
|
ref = new CookieRef( cookie, connName, id, langCode );
|
2009-07-28 07:15:26 +02:00
|
|
|
}
|
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
ref->assignConnName();
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
m_cookieMap.insert( pair<CookieID, CookieRef*>(ref->GetCookieID(), ref ) );
|
2009-09-17 05:42:12 +02:00
|
|
|
logf( XW_LOGINFO, "%s: paired cookie %s/connName %s with cid %d", __func__,
|
2005-10-01 18:33:45 +02:00
|
|
|
(cookie?cookie:"NULL"), connName, ref->GetCookieID() );
|
2005-10-15 18:30:10 +02:00
|
|
|
|
2007-11-19 00:38:56 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2005-10-15 18:30:10 +02:00
|
|
|
if ( m_cookieMap.size() == 1 ) {
|
|
|
|
RelayConfigs* cfg = RelayConfigs::GetConfigs();
|
2009-03-02 02:50:14 +01:00
|
|
|
int heartbeat;
|
|
|
|
cfg->GetValueFor( "HEARTBEAT", &heartbeat );
|
2005-10-15 18:30:10 +02:00
|
|
|
TimerMgr::GetTimerMgr()->SetTimer( heartbeat, heartbeatProc, this,
|
|
|
|
heartbeat );
|
|
|
|
}
|
2007-11-19 00:38:56 +01:00
|
|
|
#endif
|
2005-10-15 18:30:10 +02:00
|
|
|
|
2009-09-14 03:55:03 +02:00
|
|
|
logf( XW_LOGINFO, "%s=>%p", __func__, ref );
|
2005-09-02 08:35:25 +02:00
|
|
|
return ref;
|
|
|
|
} /* AddNew */
|
|
|
|
|
|
|
|
void
|
2009-07-31 14:56:04 +02:00
|
|
|
CRefMgr::Recycle_locked( CookieRef* cref )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
logf( XW_LOGINFO, "%s(cref=%p,cookie=%s)", __func__, cref, cref->Cookie() );
|
|
|
|
CookieID id = cref->GetCookieID();
|
|
|
|
cref->Clear();
|
|
|
|
addToFreeList( cref );
|
|
|
|
|
|
|
|
cref->Unlock();
|
|
|
|
|
|
|
|
/* don't grab this lock until after releasing cref's lock; otherwise
|
|
|
|
deadlock happens. */
|
2005-09-02 08:35:25 +02:00
|
|
|
RWWriteLock rwl( &m_cookieMapRWLock );
|
|
|
|
|
|
|
|
CookieMap::iterator iter = m_cookieMap.begin();
|
|
|
|
while ( iter != m_cookieMap.end() ) {
|
|
|
|
CookieRef* ref = iter->second;
|
|
|
|
if ( ref == cref ) {
|
2009-07-28 07:15:26 +02:00
|
|
|
logf( XW_LOGINFO, "%s: erasing cref cid %d", __func__, id );
|
2005-09-02 08:35:25 +02:00
|
|
|
m_cookieMap.erase( iter );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
|
2005-10-15 18:30:10 +02:00
|
|
|
|
2007-11-19 00:38:56 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2005-10-15 18:30:10 +02:00
|
|
|
if ( m_cookieMap.size() == 0 ) {
|
|
|
|
TimerMgr::GetTimerMgr()->ClearTimer( heartbeatProc, this );
|
|
|
|
}
|
2007-11-19 00:38:56 +01:00
|
|
|
#endif
|
2009-07-28 07:15:26 +02:00
|
|
|
} /* CRefMgr::Recycle */
|
2005-09-02 08:35:25 +02:00
|
|
|
|
|
|
|
void
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::Recycle( CookieID id )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
CookieRef* cref = getCookieRef( id );
|
2005-09-02 08:35:25 +02:00
|
|
|
if ( cref != NULL ) {
|
2009-07-31 14:56:04 +02:00
|
|
|
cref->Lock();
|
|
|
|
Recycle_locked( cref );
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
} /* Delete */
|
|
|
|
|
|
|
|
void
|
2009-07-28 07:15:26 +02:00
|
|
|
CRefMgr::Recycle( const char* connName )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieID id = cookieIDForConnName( connName );
|
2009-07-28 07:15:26 +02:00
|
|
|
Recycle( id );
|
2005-09-02 08:35:25 +02:00
|
|
|
} /* Delete */
|
|
|
|
|
|
|
|
CookieRef*
|
|
|
|
CRefMgr::getCookieRef_impl( CookieID cookieID )
|
|
|
|
{
|
|
|
|
CookieRef* ref = NULL;
|
|
|
|
RWReadLock rwl( &m_cookieMapRWLock );
|
|
|
|
|
|
|
|
CookieMap::iterator iter = m_cookieMap.find( cookieID );
|
|
|
|
while ( iter != m_cookieMap.end() ) {
|
2009-02-28 17:15:59 +01:00
|
|
|
CookieRef* second = iter->second;
|
|
|
|
if ( second->GetCookieID() == cookieID ) {
|
|
|
|
ref = second;
|
2005-09-02 08:35:25 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
return ref;
|
|
|
|
}
|
|
|
|
|
2007-11-19 00:38:56 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2005-09-02 08:35:25 +02:00
|
|
|
void
|
2005-10-15 18:30:10 +02:00
|
|
|
CRefMgr::checkHeartbeats( time_t now )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2005-09-04 22:36:53 +02:00
|
|
|
vector<CookieRef*> crefs;
|
|
|
|
|
|
|
|
{
|
|
|
|
RWReadLock rwl( &m_cookieMapRWLock );
|
|
|
|
CookieMap::iterator iter = m_cookieMap.begin();
|
|
|
|
while ( iter != m_cookieMap.end() ) {
|
|
|
|
crefs.push_back(iter->second);
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-12-30 06:13:30 +01:00
|
|
|
unsigned int ii;
|
|
|
|
for ( ii = 0; ii < crefs.size(); ++ii ) {
|
|
|
|
SafeCref scr( crefs[ii] );
|
2005-10-01 18:33:45 +02:00
|
|
|
scr.CheckHeartbeats( now );
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
2005-10-15 18:30:10 +02:00
|
|
|
} /* checkHeartbeats */
|
2007-11-19 00:38:56 +01:00
|
|
|
#endif
|
2005-09-02 08:35:25 +02:00
|
|
|
|
2009-07-31 14:56:04 +02:00
|
|
|
time_t
|
|
|
|
CRefMgr::uptime( void )
|
|
|
|
{
|
|
|
|
return time(NULL) - m_startTime;
|
|
|
|
}
|
|
|
|
|
2005-09-02 08:35:25 +02:00
|
|
|
/* static */ CookieMapIterator
|
|
|
|
CRefMgr::GetCookieIterator()
|
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
CookieMapIterator iter(&m_cookieMapRWLock);
|
2005-09-02 08:35:25 +02:00
|
|
|
return iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
CookieMapIterator::CookieMapIterator(pthread_rwlock_t* rwlock)
|
|
|
|
: m_rwl( rwlock )
|
|
|
|
,_iter( CRefMgr::Get()->m_cookieMap.begin() )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
CookieID
|
|
|
|
CookieMapIterator::Next()
|
|
|
|
{
|
|
|
|
CookieID id = 0;
|
|
|
|
if ( _iter != CRefMgr::Get()->m_cookieMap.end() ) {
|
|
|
|
CookieRef* cref = _iter->second;
|
|
|
|
id = cref->GetCookieID();
|
|
|
|
++_iter;
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// SafeCref
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
/* connect case */
|
|
|
|
SafeCref::SafeCref( const char* cookie, int socket, int nPlayersH, int nPlayersS,
|
|
|
|
unsigned short gameSeed, int langCode, bool wantsPublic,
|
|
|
|
bool makePublic )
|
|
|
|
: m_cref( NULL )
|
|
|
|
, m_mgr( CRefMgr::Get() )
|
|
|
|
, m_isValid( false )
|
|
|
|
{
|
|
|
|
CookieRef* cref;
|
|
|
|
|
|
|
|
cref = m_mgr->getMakeCookieRef_locked( cookie, NULL, 0, socket,
|
2010-09-10 10:59:37 +02:00
|
|
|
nPlayersH, nPlayersS, langCode,
|
|
|
|
gameSeed );
|
2010-09-10 10:30:40 +02:00
|
|
|
if ( cref != NULL ) {
|
|
|
|
m_locked = cref->Lock();
|
|
|
|
m_cref = cref;
|
|
|
|
m_isValid = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* REconnect case */
|
|
|
|
SafeCref::SafeCref( const char* connName, HostID hid,
|
|
|
|
int socket, int nPlayersH, int nPlayersS,
|
2010-09-11 10:23:39 +02:00
|
|
|
unsigned short gameSeed, int langCode )
|
2009-07-28 07:15:26 +02:00
|
|
|
: m_cref( NULL )
|
|
|
|
, m_mgr( CRefMgr::Get() )
|
|
|
|
, m_isValid( false )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2005-10-01 18:33:45 +02:00
|
|
|
CookieRef* cref;
|
|
|
|
|
2010-09-10 10:30:40 +02:00
|
|
|
cref = m_mgr->getMakeCookieRef_locked( NULL, connName, hid, socket, nPlayersH,
|
2010-09-11 10:23:39 +02:00
|
|
|
nPlayersS, langCode, gameSeed );
|
2005-09-02 08:35:25 +02:00
|
|
|
if ( cref != NULL ) {
|
2009-07-30 14:39:45 +02:00
|
|
|
m_locked = cref->Lock();
|
2009-07-28 07:15:26 +02:00
|
|
|
m_cref = cref;
|
|
|
|
m_isValid = true;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-28 07:15:26 +02:00
|
|
|
SafeCref::SafeCref( CookieID connID, bool failOk )
|
|
|
|
: m_cref( NULL )
|
|
|
|
, m_mgr( CRefMgr::Get() )
|
|
|
|
, m_isValid( false )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
CookieRef* cref = m_mgr->getCookieRef( connID );
|
|
|
|
if ( cref != NULL ) { /* known cookie? */
|
|
|
|
m_locked = cref->Lock();
|
|
|
|
m_isValid = m_locked && connID == cref->GetCookieID();
|
|
|
|
m_cref = cref;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SafeCref::SafeCref( int socket )
|
2009-07-28 07:15:26 +02:00
|
|
|
: m_cref( NULL )
|
|
|
|
, m_mgr( CRefMgr::Get() )
|
|
|
|
, m_isValid( false )
|
|
|
|
{
|
|
|
|
CookieRef* cref = m_mgr->getCookieRef( socket );
|
|
|
|
if ( cref != NULL ) { /* known socket? */
|
|
|
|
m_locked = cref->Lock();
|
|
|
|
m_isValid = m_locked && cref->HasSocket_locked( socket );
|
|
|
|
m_cref = cref;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SafeCref::SafeCref( CookieRef* cref )
|
2009-07-28 07:15:26 +02:00
|
|
|
: m_cref( NULL )
|
|
|
|
, m_mgr( CRefMgr::Get() )
|
|
|
|
, m_isValid( false )
|
2005-09-02 08:35:25 +02:00
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
m_locked = cref->Lock();
|
|
|
|
m_isValid = m_locked;
|
|
|
|
m_cref = cref;
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
SafeCref::~SafeCref()
|
|
|
|
{
|
2009-07-28 07:15:26 +02:00
|
|
|
if ( m_cref != NULL && m_locked ) {
|
2005-09-02 08:35:25 +02:00
|
|
|
if ( m_cref->ShouldDie() ) {
|
2009-07-31 14:56:04 +02:00
|
|
|
m_mgr->Recycle_locked( m_cref );
|
2005-09-02 08:35:25 +02:00
|
|
|
} else {
|
2009-07-28 07:15:26 +02:00
|
|
|
m_cref->Unlock();
|
2005-09-02 08:35:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|