mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-15 15:41:24 +01:00
deal with case where guest begins new game but connects to host that's
running the old game: the host may reply, but guest must not take that as evidence of a connection having been established: don't remove sent messages (this isn't an ACK), and don't assert later on.
This commit is contained in:
parent
e6315e118f
commit
a762f27846
1 changed files with 53 additions and 50 deletions
103
common/comms.c
103
common/comms.c
|
@ -205,6 +205,7 @@ comms_reset( CommsCtxt* comms, XP_Bool isServer,
|
||||||
XP_U16 XP_UNUSED_RELAY(nPlayersHere),
|
XP_U16 XP_UNUSED_RELAY(nPlayersHere),
|
||||||
XP_U16 XP_UNUSED_RELAY(nPlayersTotal) )
|
XP_U16 XP_UNUSED_RELAY(nPlayersTotal) )
|
||||||
{
|
{
|
||||||
|
LOG_FUNC();
|
||||||
#ifdef XWFEATURE_RELAY
|
#ifdef XWFEATURE_RELAY
|
||||||
relayDisconnect( comms );
|
relayDisconnect( comms );
|
||||||
#endif
|
#endif
|
||||||
|
@ -223,6 +224,7 @@ comms_reset( CommsCtxt* comms, XP_Bool isServer,
|
||||||
comms->r.nPlayersTotal = nPlayersTotal;
|
comms->r.nPlayersTotal = nPlayersTotal;
|
||||||
relayConnect( comms );
|
relayConnect( comms );
|
||||||
#endif
|
#endif
|
||||||
|
LOG_RETURN_VOID();
|
||||||
} /* comms_reset */
|
} /* comms_reset */
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -643,55 +645,59 @@ printQueue( CommsCtxt* comms )
|
||||||
|
|
||||||
/* We've received on some channel a message with a certain ID. This means
|
/* We've received on some channel a message with a certain ID. This means
|
||||||
* that all messages sent on that channel with lower IDs have been received
|
* that all messages sent on that channel with lower IDs have been received
|
||||||
* and can be removed from our queue.
|
* and can be removed from our queue. BUT: if this ID is higher than any
|
||||||
|
* we've sent, don't remove. We may be starting a new game but have a server
|
||||||
|
* that's still on the old one.
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
removeFromQueue( CommsCtxt* comms, XP_PlayerAddr channelNo, MsgID msgID )
|
removeFromQueue( CommsCtxt* comms, XP_PlayerAddr channelNo, MsgID msgID )
|
||||||
{
|
{
|
||||||
MsgQueueElem* elem;
|
XP_STATUSF( "%s: remove msgs <= " XP_LD " for channel %d (queueLen: %d)",
|
||||||
MsgQueueElem* keep = NULL;
|
__func__, msgID, channelNo, comms->queueLen );
|
||||||
MsgQueueElem* tail = NULL;
|
|
||||||
MsgQueueElem* keepHead = NULL;
|
|
||||||
MsgQueueElem* next;
|
|
||||||
|
|
||||||
XP_STATUSF( "looking to remove msgs prior or equal to " XP_LD
|
if ( (channelNo == 0) || !!getRecordFor(comms, channelNo) ) {
|
||||||
" for channel %d (queue len now %d)",
|
MsgQueueElem* elem;
|
||||||
msgID, channelNo, comms->queueLen );
|
MsgQueueElem* next;
|
||||||
|
MsgQueueElem* keep = NULL;
|
||||||
|
MsgQueueElem* tail = NULL;
|
||||||
|
MsgQueueElem* keepHead = NULL;
|
||||||
|
|
||||||
for ( elem = comms->msgQueueHead; !!elem; elem = next ) {
|
for ( elem = comms->msgQueueHead; !!elem; elem = next ) {
|
||||||
next = elem->next;
|
next = elem->next;
|
||||||
|
|
||||||
/* remove the 0-channel message if we've established a channel number.
|
/* remove the 0-channel message if we've established a channel
|
||||||
Only clients should have any 0-channel messages in the queue, and
|
number. Only clients should have any 0-channel messages in the
|
||||||
receiving something from the server is an implicit ACK */
|
queue, and receiving something from the server is an implicit
|
||||||
|
ACK -- IFF it isn't left over from the last game. */
|
||||||
|
|
||||||
if ( elem->channelNo == 0 && channelNo != 0 ) {
|
if ( (elem->channelNo == 0) && (channelNo != 0) ) {
|
||||||
XP_ASSERT( !comms->isServer ); /* I've seen this fail once */
|
XP_ASSERT( !comms->isServer ); /* I've seen this fail once */
|
||||||
XP_ASSERT( elem->msgID == 0 ); /* will the check below pass? */
|
XP_ASSERT( elem->msgID == 0 ); /* will the check below pass? */
|
||||||
} else if ( elem->channelNo != channelNo ) {
|
} else if ( elem->channelNo != channelNo ) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( elem->msgID <= msgID ) {
|
/* here */
|
||||||
XP_FREE( comms->mpool, elem->msg );
|
if ( elem->msgID <= msgID ) {
|
||||||
XP_FREE( comms->mpool, elem );
|
XP_FREE( comms->mpool, elem->msg );
|
||||||
--comms->queueLen;
|
XP_FREE( comms->mpool, elem );
|
||||||
} else {
|
--comms->queueLen;
|
||||||
if ( !!keepHead ) {
|
} else {
|
||||||
XP_ASSERT( !!keep );
|
if ( !!keepHead ) {
|
||||||
keep->next = elem;
|
XP_ASSERT( !!keep );
|
||||||
} else {
|
keep->next = elem;
|
||||||
keepHead = elem;
|
} else {
|
||||||
|
keepHead = elem;
|
||||||
|
}
|
||||||
|
keep = elem;
|
||||||
|
tail = elem;
|
||||||
}
|
}
|
||||||
keep = elem;
|
|
||||||
tail = elem;
|
|
||||||
}
|
}
|
||||||
|
comms->msgQueueHead = keepHead;
|
||||||
|
comms->msgQueueTail = tail;
|
||||||
}
|
}
|
||||||
|
|
||||||
comms->msgQueueHead = keepHead;
|
XP_STATUSF( "%s: queueLen now %d", __func__, comms->queueLen );
|
||||||
comms->msgQueueTail = tail;
|
|
||||||
|
|
||||||
XP_STATUSF( "%s: queueLen now %d", __FUNCTION__, comms->queueLen );
|
|
||||||
|
|
||||||
XP_ASSERT( comms->queueLen > 0 || comms->msgQueueHead == NULL );
|
XP_ASSERT( comms->queueLen > 0 || comms->msgQueueHead == NULL );
|
||||||
|
|
||||||
|
@ -717,7 +723,7 @@ sendMsg( CommsCtxt* comms, MsgQueueElem* elem )
|
||||||
result = send_via_relay( comms, XWRELAY_MSG_TORELAY, destID,
|
result = send_via_relay( comms, XWRELAY_MSG_TORELAY, destID,
|
||||||
elem->msg, elem->len );
|
elem->msg, elem->len );
|
||||||
} else {
|
} else {
|
||||||
XP_LOGF( "skipping message: not connected" );
|
XP_LOGF( "%s: skipping message: not connected", __func__ );
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
} else {
|
} else {
|
||||||
|
@ -915,7 +921,7 @@ comms_checkIncomingStream( CommsCtxt* comms, XWStreamCtxt* stream,
|
||||||
if ( !done ) {
|
if ( !done ) {
|
||||||
if ( stream_getSize( stream ) >= sizeof(connID) ) {
|
if ( stream_getSize( stream ) >= sizeof(connID) ) {
|
||||||
connID = stream_getU32( stream );
|
connID = stream_getU32( stream );
|
||||||
XP_STATUSF( "read connID of %lx", connID );
|
XP_STATUSF( "%s: read connID of %lx", __func__, connID );
|
||||||
|
|
||||||
if ( comms->connID == connID || comms->connID == CONN_ID_NONE ) {
|
if ( comms->connID == connID || comms->connID == CONN_ID_NONE ) {
|
||||||
if ( stream_getSize( stream ) >= sizeof(channelNo)
|
if ( stream_getSize( stream ) >= sizeof(channelNo)
|
||||||
|
@ -980,7 +986,6 @@ comms_checkIncomingStream( CommsCtxt* comms, XWStreamCtxt* stream,
|
||||||
validMessage = XP_FALSE;
|
validMessage = XP_FALSE;
|
||||||
}
|
}
|
||||||
} else if ( msgID > 1 ) {
|
} else if ( msgID > 1 ) {
|
||||||
XP_ASSERT( 0 );
|
|
||||||
validMessage = XP_FALSE;
|
validMessage = XP_FALSE;
|
||||||
}
|
}
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
|
@ -1141,18 +1146,16 @@ channelToAddress( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
||||||
static AddressRecord*
|
static AddressRecord*
|
||||||
getRecordFor( CommsCtxt* comms, XP_PlayerAddr channelNo )
|
getRecordFor( CommsCtxt* comms, XP_PlayerAddr channelNo )
|
||||||
{
|
{
|
||||||
AddressRecord* recs;
|
AddressRecord* recs = NULL;
|
||||||
|
|
||||||
if ( channelNo == CHANNEL_NONE ) {
|
if ( channelNo != CHANNEL_NONE ) {
|
||||||
return (AddressRecord*)NULL;
|
for ( recs = comms->recs; !!recs; recs = recs->next ) {
|
||||||
}
|
if ( recs->channelNo == channelNo ) {
|
||||||
|
break;
|
||||||
for ( recs = comms->recs; !!recs; recs = recs->next ) {
|
}
|
||||||
if ( recs->channelNo == channelNo ) {
|
|
||||||
return recs;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (AddressRecord*)NULL;
|
return recs;
|
||||||
} /* getRecordFor */
|
} /* getRecordFor */
|
||||||
|
|
||||||
static XP_U16
|
static XP_U16
|
||||||
|
@ -1276,7 +1279,7 @@ send_via_relay( CommsCtxt* comms, XWRELAY_Cmd cmd, XWHostID destID,
|
||||||
static void
|
static void
|
||||||
relayConnect( CommsCtxt* comms )
|
relayConnect( CommsCtxt* comms )
|
||||||
{
|
{
|
||||||
XP_LOGF( "relayConnect called" );
|
LOG_FUNC();
|
||||||
if ( comms->addr.conType == COMMS_CONN_RELAY && !comms->r.connecting ) {
|
if ( comms->addr.conType == COMMS_CONN_RELAY && !comms->r.connecting ) {
|
||||||
comms->r.connecting = XP_TRUE;
|
comms->r.connecting = XP_TRUE;
|
||||||
send_via_relay( comms,
|
send_via_relay( comms,
|
||||||
|
|
Loading…
Reference in a new issue