fix race: if we get a cid from the DB while another thread has claimed

that cid, then by the time we can claim it there's no longer space for
us.  Call new method after the claim succeeds to make sure there's
still room, and if not relinquish and loop back to the DB for another
try.
This commit is contained in:
Andy2 2011-06-23 19:00:32 -07:00
parent c51e5e41f4
commit 9ff149fe2d

View file

@ -272,6 +272,9 @@ CRefMgr::getMakeCookieRef( const char* cookie, HostID hid, int socket,
if ( NULL == cinfo->GetRef() ) { if ( NULL == cinfo->GetRef() ) {
m_cidlock->Relinquish( cinfo, true ); m_cidlock->Relinquish( cinfo, true );
continue; continue;
} else if ( !cinfo->GetRef()->HaveRoom( nPlayersH ) ) {
m_cidlock->Relinquish( cinfo, false );
continue;
} }
} else { } else {
cinfo = m_cidlock->Claim(); cinfo = m_cidlock->Claim();
@ -595,8 +598,6 @@ CRefMgr::Recycle_locked( CookieRef* cref )
cref->Unlock(); cref->Unlock();
sleep(2);
/* don't grab this lock until after releasing cref's lock; otherwise /* don't grab this lock until after releasing cref's lock; otherwise
deadlock happens. */ deadlock happens. */
RWWriteLock rwl( &m_cookieMapRWLock ); RWWriteLock rwl( &m_cookieMapRWLock );
@ -757,6 +758,7 @@ SafeCref::SafeCref( const char* connName, const char* cookie, HostID hid,
nPlayersS, gameSeed, langCode, nPlayersS, gameSeed, langCode,
wantsPublic || makePublic, &isDead ); wantsPublic || makePublic, &isDead );
if ( cinfo != NULL ) { if ( cinfo != NULL ) {
assert( cinfo->GetCid() == cinfo->GetRef()->GetCookieID() );
m_locked = cinfo->GetRef()->Lock(); m_locked = cinfo->GetRef()->Lock();
m_cinfo = cinfo; m_cinfo = cinfo;
m_isValid = true; m_isValid = true;
@ -773,6 +775,7 @@ SafeCref::SafeCref( const char* const connName )
bool isDead = false; bool isDead = false;
CidInfo* cinfo = m_mgr->getMakeCookieRef( connName, &isDead ); CidInfo* cinfo = m_mgr->getMakeCookieRef( connName, &isDead );
if ( cinfo != NULL ) { if ( cinfo != NULL ) {
assert( cinfo->GetCid() == cinfo->GetRef()->GetCookieID() );
m_locked = cinfo->GetRef()->Lock(); m_locked = cinfo->GetRef()->Lock();
m_cinfo = cinfo; m_cinfo = cinfo;
m_isValid = true; m_isValid = true;
@ -788,6 +791,7 @@ SafeCref::SafeCref( CookieID connID, bool failOk )
CidInfo* cinfo = m_mgr->getCookieRef( connID ); CidInfo* cinfo = m_mgr->getCookieRef( connID );
if ( cinfo != NULL ) { /* known cookie? */ if ( cinfo != NULL ) { /* known cookie? */
CookieRef* cref = cinfo->GetRef(); CookieRef* cref = cinfo->GetRef();
assert( cinfo->GetCid() == cref->GetCookieID() );
m_locked = cref->Lock(); m_locked = cref->Lock();
m_isValid = m_locked && connID == cref->GetCookieID(); m_isValid = m_locked && connID == cref->GetCookieID();
m_cinfo = cinfo; m_cinfo = cinfo;
@ -802,6 +806,7 @@ SafeCref::SafeCref( int socket )
CidInfo* cinfo = m_mgr->getCookieRef( socket ); CidInfo* cinfo = m_mgr->getCookieRef( socket );
if ( cinfo != NULL ) { /* known socket? */ if ( cinfo != NULL ) { /* known socket? */
CookieRef* cref = cinfo->GetRef(); CookieRef* cref = cinfo->GetRef();
assert( cinfo->GetCid() == cref->GetCookieID() );
m_locked = cref->Lock(); m_locked = cref->Lock();
m_isValid = m_locked && cref->HasSocket_locked( socket ); m_isValid = m_locked && cref->HasSocket_locked( socket );
m_cinfo = cinfo; m_cinfo = cinfo;
@ -825,6 +830,7 @@ SafeCref::~SafeCref()
bool recycle = false; bool recycle = false;
if ( m_locked ) { if ( m_locked ) {
CookieRef* cref = m_cinfo->GetRef(); CookieRef* cref = m_cinfo->GetRef();
assert( m_cinfo->GetCid() == cref->GetCookieID() );
recycle = cref->ShouldDie(); recycle = cref->ShouldDie();
if ( recycle ) { if ( recycle ) {
m_mgr->Recycle_locked( cref ); m_mgr->Recycle_locked( cref );