/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */ /* * Copyright 2005-2009 by Eric House (xwords@eehouse.org). All rights * reserved. * * 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. */ #include #include #include #include #include #include "crefmgr.h" #include "cref.h" #include "mlock.h" #include "configs.h" #include "timermgr.h" 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 */ }; static CRefMgr* s_instance = NULL; /* static */ CRefMgr* CRefMgr::Get() { if ( s_instance == NULL ) { s_instance = new CRefMgr(); } return s_instance; } /* Get */ CRefMgr::CRefMgr() : m_nextCID(0) , m_startTime(time(NULL)) { /* should be using pthread_once() here */ pthread_mutex_init( &m_SocketStuffMutex, NULL ); pthread_mutex_init( &m_nextCIDMutex, NULL ); pthread_mutex_init( &m_roomsFilledMutex, NULL ); pthread_mutex_init( &m_freeList_mutex, NULL ); pthread_rwlock_init( &m_cookieMapRWLock, NULL ); m_db = DBMgr::Get(); } CRefMgr::~CRefMgr() { assert( this == s_instance ); delete m_db; pthread_mutex_destroy( &m_freeList_mutex ); pthread_rwlock_destroy( &m_cookieMapRWLock ); SocketMap::iterator iter; for ( iter = m_SocketStuff.begin(); iter != m_SocketStuff.end(); ++iter ) { SocketStuff* stuff = iter->second; delete stuff; } s_instance = NULL; } void CRefMgr::CloseAll() { /* Get every cref instance, shut it down */ for ( ; ; ) { CookieRef* cref = NULL; { 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(); } } } } /* CloseAll */ /* 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. */ CookieRef* CRefMgr::FindOpenGameFor( const char* cookie, const char* connName, HostID hid, int socket, int nPlayersH, int nPlayersT, int gameSeed, int langCode, bool* alreadyHere ) { #if 1 CookieRef* found = NULL; CookieID cid = m_db->FindOpen( cookie, langCode, nPlayersT, nPlayersH ); if ( cid > 0 ) { found = getCookieRef_impl( cid ); } return found; #else 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 ); CookieRef* found = NULL; assert( !!cookie || !!connName ); RWReadLock rwl( &m_cookieMapRWLock ); CookieMap::iterator iter; for ( iter = m_cookieMap.begin(); NULL == found && iter != m_cookieMap.end(); ++iter ) { CookieRef* cref = iter->second; /* Reject immediately if language code or proposed game size don't match. */ if ( (cref->GetLangCode() != langCode) || (cref->GetPlayersSought() != nPlayersT) ) { continue; } 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; } cref->Unlock(); } } } } /* for */ logf( XW_LOGINFO, "%s=>%p", __func__, found ); return found; #endif } /* FindOpenGameFor */ 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. */ MutexLock ml(&m_nextCIDMutex); return ++m_nextCID; } /* nextCID */ void CRefMgr::IncrementFullCount( void ) { MutexLock ml( &m_roomsFilledMutex ); ++m_nRoomsFilled; } int CRefMgr::GetNumRoomsFilled( void ) { MutexLock ml(&m_roomsFilledMutex); return m_nRoomsFilled; } int CRefMgr::GetSize( void ) { return m_cookieMap.size(); } void CRefMgr::GetStats( CrefMgrInfo& mgrInfo ) { mgrInfo.m_nRoomsFilled = GetNumRoomsFilled(); mgrInfo.m_startTimeSpawn = m_startTime; if ( 0 == m_ports.length() ) { RelayConfigs* cfg = RelayConfigs::GetConfigs(); vector ints; if ( cfg->GetValueFor( "GAME_PORTS", ints ) ) { vector::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(); 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(); info.m_langCode = cref->GetLangCode(); SafeCref sc(cref); sc.GetHostsConnected( &info.m_hostsIds, &info.m_hostSeeds, &info.m_hostIps ); mgrInfo.m_crefInfo.push_back( info ); } } CookieID CRefMgr::cookieIDForConnName( const char* connName ) { 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. */ RWReadLock rwl( &m_cookieMapRWLock ); CookieMap::iterator iter = m_cookieMap.begin(); while ( iter != m_cookieMap.end() ) { CookieRef* cref = iter->second; if ( 0 == strcmp( cref->ConnName(), connName ) ) { cid = iter->first; break; } ++iter; } return cid; } /* cookieIDForConnName */ 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; } CookieRef* CRefMgr::getMakeCookieRef_locked( const char* cookie, const char* connName, HostID hid, int socket, int nPlayersH, int nPlayersT, int langCode, int gameSeed ) { CookieRef* cref; /* 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. */ bool alreadyHere; cref = FindOpenGameFor( cookie, connName, hid, socket, nPlayersH, nPlayersT, gameSeed, langCode, &alreadyHere ); if ( cref == NULL && !alreadyHere ) { CookieID cid = nextCID( NULL ); cref = AddNew( cookie, connName, cid, langCode, nPlayersT ); m_db->AddNew( cookie, cref->ConnName(), cid, langCode, nPlayersT ); } return cref; } /* getMakeCookieRef_locked */ bool CRefMgr::Associate( int socket, CookieRef* cref ) { MutexLock ml( &m_SocketStuffMutex ); return Associate_locked( socket, cref ); } bool CRefMgr::Associate_locked( int socket, CookieRef* cref ) { bool isNew = false; SocketMap::iterator iter = m_SocketStuff.find( socket ); /* 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 ); } return isNew; } void CRefMgr::Disassociate_locked( int socket, CookieRef* cref ) { SocketMap::iterator iter = m_SocketStuff.find( socket ); if ( iter == m_SocketStuff.end() ) { logf( XW_LOGERROR, "can't find SocketStuff for socket %d", socket ); } else { SocketStuff* stuff = iter->second; assert( cref == NULL || stuff->m_cref == cref ); delete stuff; m_SocketStuff.erase( iter ); } } void CRefMgr::Disassociate( int socket, CookieRef* cref ) { MutexLock ml( &m_SocketStuffMutex ); Disassociate_locked( socket, cref ); } void CRefMgr::MoveSockets( vector sockets, CookieRef* cref ) { MutexLock ml( &m_SocketStuffMutex ); vector::iterator iter; for ( iter = sockets.begin(); iter != sockets.end(); ++iter ) { Disassociate_locked( *iter, NULL ); Associate_locked( *iter, cref ); } } #if 0 pthread_mutex_t* CRefMgr::GetWriteMutexForSocket( int socket ) { pthread_mutex_t* mutex = NULL; MutexLock ml( &m_SocketStuffMutex ); SocketMap::iterator iter = m_SocketStuff.find( socket ); if ( iter != m_SocketStuff.end() ) { SocketStuff* stuff = iter->second; /* 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; } logf( XW_LOGERROR, "GetWriteMutexForSocket: not found" ); return mutex; } /* GetWriteMutexForSocket */ #endif void CRefMgr::RemoveSocketRefs( int socket ) { { SafeCref scr( socket ); scr.Remove( socket ); } Disassociate( socket, NULL ); } void CRefMgr::PrintSocketInfo( int socket, string& out ) { SafeCref scr( socket ); const char* name = scr.Cookie(); if ( name != NULL && name[0] != '\0' ) { char buf[64]; snprintf( buf, sizeof(buf), "* socket: %d\n", socket ); out += buf; snprintf( buf, sizeof(buf), " in cookie: %s\n", name ); out += buf; } } /* static */ SocketsIterator CRefMgr::MakeSocketsIterator() { pthread_mutex_lock( &m_SocketStuffMutex ); SocketsIterator iter( m_SocketStuff.begin(), m_SocketStuff.end(), &m_SocketStuffMutex ); return iter; } CookieRef* CRefMgr::getCookieRef( CookieID cookieID ) { return getCookieRef_impl( cookieID ); } /* getCookieRef */ CookieRef* CRefMgr::getCookieRef( int socket ) { CookieRef* cref = NULL; MutexLock ml( &m_SocketStuffMutex ); SocketMap::iterator iter = m_SocketStuff.find( socket ); if ( iter != m_SocketStuff.end() ) { SocketStuff* stuff = iter->second; cref = stuff->m_cref; } return cref; } /* getCookieRef */ #ifdef RELAY_HEARTBEAT /* static */ void CRefMgr::heartbeatProc( void* closure ) { CRefMgr* self = (CRefMgr*)closure; self->checkHeartbeats( ::uptime() ); } /* heartbeatProc */ #endif CookieRef* CRefMgr::AddNew( const char* cookie, const char* connName, CookieID id, int langCode, int nPlayers ) { /* PENDING: should this return a locked cref? */ logf( XW_LOGINFO, "%s( cookie=%s, connName=%s, cid=%d)", __func__, cookie, connName, id ); CookieRef* ref = getFromFreeList(); RWWriteLock rwl( &m_cookieMapRWLock ); logf( XW_LOGINFO, "making new cref: %d", id ); if ( !!ref ) { logf( XW_LOGVERBOSE1, "using from free list" ); ref->ReInit( cookie, connName, id, langCode, nPlayers ); } else { logf( XW_LOGVERBOSE1, "calling constructor" ); ref = new CookieRef( cookie, connName, id, langCode, nPlayers ); } ref->assignConnName(); m_cookieMap.insert( pair(ref->GetCookieID(), ref ) ); logf( XW_LOGINFO, "%s: paired cookie %s/connName %s with cid %d", __func__, (cookie?cookie:"NULL"), connName, ref->GetCookieID() ); #ifdef RELAY_HEARTBEAT if ( m_cookieMap.size() == 1 ) { RelayConfigs* cfg = RelayConfigs::GetConfigs(); int heartbeat; cfg->GetValueFor( "HEARTBEAT", &heartbeat ); TimerMgr::GetTimerMgr()->SetTimer( heartbeat, heartbeatProc, this, heartbeat ); } #endif logf( XW_LOGINFO, "%s=>%p", __func__, ref ); return ref; } /* AddNew */ void CRefMgr::Recycle_locked( CookieRef* cref ) { 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. */ RWWriteLock rwl( &m_cookieMapRWLock ); CookieMap::iterator iter = m_cookieMap.begin(); while ( iter != m_cookieMap.end() ) { CookieRef* ref = iter->second; if ( ref == cref ) { logf( XW_LOGINFO, "%s: erasing cref cid %d", __func__, id ); m_cookieMap.erase( iter ); break; } ++iter; } #ifdef RELAY_HEARTBEAT if ( m_cookieMap.size() == 0 ) { TimerMgr::GetTimerMgr()->ClearTimer( heartbeatProc, this ); } #endif } /* CRefMgr::Recycle */ void CRefMgr::Recycle( CookieID id ) { CookieRef* cref = getCookieRef( id ); if ( cref != NULL ) { cref->Lock(); Recycle_locked( cref ); } } /* Delete */ void CRefMgr::Recycle( const char* connName ) { CookieID id = cookieIDForConnName( connName ); Recycle( id ); } /* 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() ) { CookieRef* second = iter->second; if ( second->GetCookieID() == cookieID ) { ref = second; break; } ++iter; } return ref; } #ifdef RELAY_HEARTBEAT void CRefMgr::checkHeartbeats( time_t now ) { vector crefs; { RWReadLock rwl( &m_cookieMapRWLock ); CookieMap::iterator iter = m_cookieMap.begin(); while ( iter != m_cookieMap.end() ) { crefs.push_back(iter->second); ++iter; } } unsigned int ii; for ( ii = 0; ii < crefs.size(); ++ii ) { SafeCref scr( crefs[ii] ); scr.CheckHeartbeats( now ); } } /* checkHeartbeats */ #endif time_t CRefMgr::uptime( void ) { return time(NULL) - m_startTime; } /* static */ CookieMapIterator CRefMgr::GetCookieIterator() { CookieMapIterator iter(&m_cookieMapRWLock); return iter; } CookieMapIterator::CookieMapIterator(pthread_rwlock_t* rwlock) : m_rwl( rwlock ) ,_iter( CRefMgr::Get()->m_cookieMap.begin() ) { } CookieID CookieMapIterator::Next() { CookieID id = 0; if ( _iter != CRefMgr::Get()->m_cookieMap.end() ) { CookieRef* cref = _iter->second; id = cref->GetCookieID(); ++_iter; } return id; } ////////////////////////////////////////////////////////////////////////////// // SafeCref ////////////////////////////////////////////////////////////////////////////// /* 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, nPlayersH, nPlayersS, langCode, gameSeed ); 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, unsigned short gameSeed, int langCode ) : m_cref( NULL ) , m_mgr( CRefMgr::Get() ) , m_isValid( false ) { CookieRef* cref; cref = m_mgr->getMakeCookieRef_locked( NULL, connName, hid, socket, nPlayersH, nPlayersS, langCode, gameSeed ); if ( cref != NULL ) { m_locked = cref->Lock(); m_cref = cref; m_isValid = true; } } SafeCref::SafeCref( CookieID connID, bool failOk ) : m_cref( NULL ) , m_mgr( CRefMgr::Get() ) , m_isValid( false ) { 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; } } SafeCref::SafeCref( int socket ) : 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; } } SafeCref::SafeCref( CookieRef* cref ) : m_cref( NULL ) , m_mgr( CRefMgr::Get() ) , m_isValid( false ) { m_locked = cref->Lock(); m_isValid = m_locked; m_cref = cref; } SafeCref::~SafeCref() { if ( m_cref != NULL && m_locked ) { if ( m_cref->ShouldDie() ) { m_mgr->Recycle_locked( m_cref ); } else { m_cref->Unlock(); } } }