2005-03-19 23:14:27 +01:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
|
2005-03-25 03:59:44 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2005 by Eric House (fixin@peak.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) any later version.
|
|
|
|
*
|
|
|
|
* 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-03-19 23:14:27 +01:00
|
|
|
#include <string>
|
|
|
|
#include <map>
|
|
|
|
#include <assert.h>
|
2005-03-25 03:59:44 +01:00
|
|
|
#include <pthread.h>
|
2005-07-06 00:05:37 +02:00
|
|
|
#include <netinet/in.h>
|
2005-03-19 23:14:27 +01:00
|
|
|
|
|
|
|
#include "cref.h"
|
|
|
|
#include "xwrelay.h"
|
2005-03-25 03:59:44 +01:00
|
|
|
#include "mlock.h"
|
2005-04-20 14:13:20 +02:00
|
|
|
#include "tpool.h"
|
2005-07-06 00:05:37 +02:00
|
|
|
#include "states.h"
|
2005-09-02 08:56:34 +02:00
|
|
|
#include "timermgr.h"
|
|
|
|
#include "configs.h"
|
|
|
|
#include "crefmgr.h"
|
2005-03-19 23:14:27 +01:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2005-07-06 02:54:38 +02:00
|
|
|
pthread_mutex_t g_IdsMutex = PTHREAD_MUTEX_INITIALIZER;
|
2005-03-25 03:59:44 +01:00
|
|
|
CookieID CookieRef::ms_nextConnectionID = 1000;
|
2005-03-19 23:14:27 +01:00
|
|
|
|
2005-04-20 14:13:20 +02:00
|
|
|
/*****************************************************************************
|
|
|
|
* SocketsIterator class
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
SocketsIterator::SocketsIterator( SocketMap::iterator iter,
|
|
|
|
SocketMap::iterator end,
|
|
|
|
pthread_mutex_t* rwlock )
|
2005-04-20 14:13:20 +02:00
|
|
|
: m_iter( iter )
|
2005-09-02 08:56:34 +02:00
|
|
|
, m_end( end )
|
|
|
|
, m_mutex( rwlock )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SocketsIterator::~SocketsIterator()
|
2005-04-20 14:13:20 +02:00
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
pthread_mutex_unlock( m_mutex );
|
2005-04-20 14:13:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
SocketsIterator::Next()
|
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
int socket = 0;
|
|
|
|
if ( m_iter != m_end ) {
|
|
|
|
socket = m_iter->first;
|
|
|
|
++m_iter;
|
|
|
|
}
|
2005-04-20 14:13:20 +02:00
|
|
|
return socket;
|
2005-03-19 23:14:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
* CookieRef class
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-07-06 02:54:38 +02:00
|
|
|
CookieRef::CookieRef( string s, CookieID id )
|
2005-09-02 08:56:34 +02:00
|
|
|
: m_heatbeat(RelayConfigs::GetConfigs()->GetHeartbeatInterval())
|
|
|
|
, m_name(s)
|
2005-07-06 00:05:37 +02:00
|
|
|
, m_totalSent(0)
|
|
|
|
, m_curState(XW_ST_INITED)
|
2005-03-19 23:14:27 +01:00
|
|
|
{
|
2005-04-20 14:13:20 +02:00
|
|
|
pthread_rwlock_init( &m_sockets_rwlock, NULL );
|
2005-07-06 00:05:37 +02:00
|
|
|
pthread_mutex_init( &m_EventsMutex, NULL );
|
2005-07-06 02:54:38 +02:00
|
|
|
|
|
|
|
if ( id == 0 ) {
|
|
|
|
MutexLock ml( &g_IdsMutex );
|
|
|
|
m_connectionID = ms_nextConnectionID++; /* needs a mutex!!! */
|
|
|
|
} else {
|
|
|
|
m_connectionID = id;
|
|
|
|
}
|
2005-03-19 23:14:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CookieRef::~CookieRef()
|
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
cancelAllConnectedTimer();
|
|
|
|
|
2005-04-20 14:13:20 +02:00
|
|
|
/* get rid of any sockets still contained */
|
|
|
|
XWThreadPool* tPool = XWThreadPool::GetTPool();
|
|
|
|
|
|
|
|
for ( ; ; ) {
|
|
|
|
RWWriteLock rwl( &m_sockets_rwlock );
|
2005-06-23 06:26:44 +02:00
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.begin();
|
2005-04-20 14:13:20 +02:00
|
|
|
|
|
|
|
if ( iter == m_hostSockets.end() ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-06-23 06:26:44 +02:00
|
|
|
int socket = iter->second.m_socket;
|
2005-04-20 14:13:20 +02:00
|
|
|
tPool->CloseSocket( socket );
|
|
|
|
m_hostSockets.erase( iter );
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_rwlock_destroy( &m_sockets_rwlock );
|
2005-03-19 23:14:27 +01:00
|
|
|
logf( "CookieRef for %d being deleted", m_connectionID );
|
2005-09-02 08:56:34 +02:00
|
|
|
|
|
|
|
pthread_mutex_destroy( &m_EventsMutex );
|
|
|
|
pthread_rwlock_destroy( &m_sockets_rwlock );
|
|
|
|
} /* ~CookieRef */
|
2005-03-19 23:14:27 +01:00
|
|
|
|
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::_Connect( int socket, HostID srcID )
|
2005-03-19 23:14:27 +01:00
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
CRefMgr::Get()->Associate( socket, this );
|
2005-07-06 00:05:37 +02:00
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushConnectEvent( socket, srcID );
|
|
|
|
handleEvents();
|
2005-03-19 23:14:27 +01:00
|
|
|
}
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::_Reconnect( int socket, HostID srcID )
|
|
|
|
{
|
|
|
|
CRefMgr::Get()->Associate( socket, this );
|
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushReconnectEvent( socket, srcID );
|
|
|
|
handleEvents();
|
|
|
|
}
|
|
|
|
|
2005-03-19 23:14:27 +01:00
|
|
|
int
|
|
|
|
CookieRef::SocketForHost( HostID dest )
|
|
|
|
{
|
|
|
|
int socket;
|
2005-06-23 06:26:44 +02:00
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.find( dest );
|
2005-03-19 23:14:27 +01:00
|
|
|
if ( iter == m_hostSockets.end() ) {
|
|
|
|
socket = -1;
|
|
|
|
} else {
|
2005-06-23 06:26:44 +02:00
|
|
|
socket = iter->second.m_socket;
|
2005-03-19 23:14:27 +01:00
|
|
|
logf( "socketForHost(%x) => %d", dest, socket );
|
|
|
|
}
|
|
|
|
logf( "returning socket=%d for hostid=%x", socket, dest );
|
|
|
|
return socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::removeSocket( const CRefEvent* evt )
|
2005-03-19 23:14:27 +01:00
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
int socket = evt->u.rmsock.socket;
|
2005-04-20 14:13:20 +02:00
|
|
|
int count;
|
|
|
|
{
|
|
|
|
RWWriteLock rwl( &m_sockets_rwlock );
|
|
|
|
|
|
|
|
count = CountSockets();
|
|
|
|
assert( count > 0 );
|
2005-06-23 06:26:44 +02:00
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.begin();
|
2005-04-20 14:13:20 +02:00
|
|
|
while ( iter != m_hostSockets.end() ) {
|
2005-06-23 06:26:44 +02:00
|
|
|
if ( iter->second.m_socket == socket ) {
|
2005-04-20 14:13:20 +02:00
|
|
|
m_hostSockets.erase(iter);
|
|
|
|
--count;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++iter;
|
2005-03-19 23:14:27 +01:00
|
|
|
}
|
|
|
|
}
|
2005-03-25 03:59:44 +01:00
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
/* Does this belong here or at a higher level? */
|
|
|
|
XWThreadPool::GetTPool()->CloseSocket( socket );
|
|
|
|
|
2005-03-25 03:59:44 +01:00
|
|
|
if ( count == 0 ) {
|
2005-09-02 08:56:34 +02:00
|
|
|
pushLastSocketGoneEvent();
|
2005-03-25 03:59:44 +01:00
|
|
|
}
|
2005-09-02 08:56:34 +02:00
|
|
|
} /* Remove */
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
CookieRef::HasSocket( int socket )
|
|
|
|
{
|
|
|
|
int found = 0;
|
|
|
|
logf( "CookieRef::HasSocket" );
|
|
|
|
RWReadLock rwl( &m_sockets_rwlock );
|
|
|
|
|
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.begin();
|
|
|
|
while ( iter != m_hostSockets.end() ) {
|
|
|
|
if ( iter->second.m_socket == socket ) {
|
|
|
|
found = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
} /* HasSocket */
|
2005-03-19 23:14:27 +01:00
|
|
|
|
2005-06-23 06:26:44 +02:00
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::_HandleHeartbeat( HostID id, int socket )
|
2005-06-23 06:26:44 +02:00
|
|
|
{
|
2005-07-06 00:05:37 +02:00
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushHeartbeatEvent( id, socket );
|
|
|
|
handleEvents();
|
|
|
|
} /* HandleHeartbeat */
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::CheckHeartbeats( time_t now, vector<int>* victims )
|
|
|
|
{
|
|
|
|
logf( "CookieRef::CheckHeartbeats" );
|
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushHeartTimerEvent( now, victims );
|
|
|
|
handleEvents();
|
|
|
|
} /* CheckHeartbeats */
|
|
|
|
|
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::_Forward( HostID src, HostID dest, unsigned char* buf, int buflen )
|
2005-07-06 00:05:37 +02:00
|
|
|
{
|
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushForwardEvent( src, dest, buf, buflen );
|
|
|
|
handleEvents();
|
|
|
|
} /* Forward */
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::_Remove( int socket )
|
|
|
|
{
|
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
pushRemoveSocketEvent( socket );
|
|
|
|
handleEvents();
|
|
|
|
} /* Forward */
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushConnectEvent( int socket, HostID srcID )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_CONNECTMSG;
|
|
|
|
evt.u.con.socket = socket;
|
|
|
|
evt.u.con.srcID = srcID;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
} /* pushConnectEvent */
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushReconnectEvent( int socket, HostID srcID )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_RECONNECTMSG;
|
|
|
|
evt.u.recon.socket = socket;
|
|
|
|
evt.u.recon.srcID = srcID;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
} /* pushConnectEvent */
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushHeartbeatEvent( HostID id, int socket )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_HEARTMSG;
|
|
|
|
evt.u.heart.id = id;
|
|
|
|
evt.u.heart.socket = socket;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::pushHeartTimerEvent( time_t now, vector<int>* victims )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_HEARTTIMER;
|
|
|
|
evt.u.htime.now = now;
|
|
|
|
evt.u.htime.victims = victims;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::pushForwardEvent( HostID src, HostID dest,
|
|
|
|
unsigned char* buf, int buflen )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_FORWARDMSG;
|
|
|
|
evt.u.fwd.src = src;
|
|
|
|
evt.u.fwd.dest = dest;
|
|
|
|
evt.u.fwd.buf = buf;
|
|
|
|
evt.u.fwd.buflen = buflen;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushRemoveSocketEvent( int socket )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_REMOVESOCKET;
|
|
|
|
evt.u.rmsock.socket = socket;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
2005-08-02 06:57:13 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushDestBadEvent()
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_DESTBAD;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::pushDestOkEvent( const CRefEvent* oldEvt )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
memcpy( &evt, oldEvt, sizeof(evt) );
|
|
|
|
evt.type = XW_EVENT_DESTOK;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::pushCanLockEvent( const CRefEvent* oldEvt )
|
2005-07-06 00:05:37 +02:00
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
CRefEvent evt;
|
|
|
|
memcpy( &evt, oldEvt, sizeof(evt) );
|
|
|
|
evt.type = XW_EVENT_CAN_LOCK;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
2005-07-06 00:05:37 +02:00
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::pushCantLockEvent( const CRefEvent* oldEvt )
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
memcpy( &evt, oldEvt, sizeof(evt) );
|
|
|
|
evt.type = XW_EVENT_CANT_LOCK;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::pushLastSocketGoneEvent()
|
|
|
|
{
|
|
|
|
CRefEvent evt;
|
|
|
|
evt.type = XW_EVENT_NOMORESOCKETS;
|
|
|
|
m_eventQueue.push_front( evt );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::handleEvents()
|
|
|
|
{
|
|
|
|
/* Assumption: has mutex!!!! */
|
|
|
|
while ( m_eventQueue.size () > 0 ) {
|
2005-07-06 00:05:37 +02:00
|
|
|
CRefEvent evt = m_eventQueue.front();
|
|
|
|
m_eventQueue.pop_front();
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
XW_RELAY_ACTION takeAction;
|
2005-07-06 00:05:37 +02:00
|
|
|
if ( getFromTable( m_curState, evt.type, &takeAction, &m_nextState ) ) {
|
|
|
|
|
|
|
|
logf( "moving from state %s to state %s for event %s",
|
|
|
|
stateString(m_curState), stateString(m_nextState),
|
|
|
|
eventString(evt.type) );
|
|
|
|
|
|
|
|
switch( takeAction ) {
|
2005-09-02 08:56:34 +02:00
|
|
|
case XW_ACTION_SEND_1ST_RSP:
|
|
|
|
setAllConnectedTimer();
|
|
|
|
/* fallthru */
|
2005-07-06 00:05:37 +02:00
|
|
|
case XW_ACTION_SENDRSP:
|
|
|
|
sendResponse( &evt );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case XW_ACTION_FWD:
|
|
|
|
forward( &evt );
|
|
|
|
break;
|
|
|
|
|
2005-08-02 06:57:13 +02:00
|
|
|
case XW_ACTION_CHECKDEST:
|
|
|
|
checkDest( &evt );
|
|
|
|
break;
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
case XW_ACTION_CHECK_CAN_LOCK:
|
|
|
|
checkFromServer( &evt );
|
|
|
|
break;
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
case XW_ACTION_DISCONNECTALL:
|
|
|
|
disconnectAll( &evt );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case XW_ACTION_NOTEHEART:
|
|
|
|
noteHeartbeat( &evt );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case XW_ACTION_CHECKHEART:
|
|
|
|
checkHeartbeats( &evt );
|
|
|
|
break;
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
case XW_ACTION_REMOVESOCKET:
|
|
|
|
removeSocket( &evt );
|
|
|
|
break;
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
case XW_ACTION_HEARTOK:
|
2005-08-02 06:57:13 +02:00
|
|
|
case XW_ACTION_NONE:
|
|
|
|
/* nothing to do for these */
|
2005-07-06 00:05:37 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_curState = m_nextState;
|
2005-09-02 08:56:34 +02:00
|
|
|
} else {
|
|
|
|
assert(0);
|
2005-07-06 00:05:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} /* handleEvents */
|
|
|
|
|
|
|
|
static void
|
|
|
|
send_with_length( int socket, unsigned char* buf, int bufLen )
|
|
|
|
{
|
|
|
|
SocketWriteLock slock( socket );
|
2005-07-06 03:36:52 +02:00
|
|
|
|
|
|
|
if ( !send_with_length_unsafe( socket, buf, bufLen ) ) {
|
|
|
|
/* ok that the slock above is still in scope */
|
2005-07-06 00:05:37 +02:00
|
|
|
killSocket( socket, "couldn't send" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
putNetShort( unsigned char** bufpp, unsigned short s )
|
|
|
|
{
|
|
|
|
s = htons( s );
|
|
|
|
memcpy( *bufpp, &s, sizeof(s) );
|
|
|
|
*bufpp += sizeof(s);
|
|
|
|
}
|
|
|
|
|
2005-07-06 01:02:15 +02:00
|
|
|
static void
|
|
|
|
putNetLong( unsigned char** bufpp, unsigned long s )
|
|
|
|
{
|
|
|
|
s = htonl( s );
|
|
|
|
memcpy( *bufpp, &s, sizeof(s) );
|
|
|
|
*bufpp += sizeof(s);
|
|
|
|
assert( sizeof(s) == 4 ); /* otherwise need to hardcode */
|
|
|
|
}
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::setAllConnectedTimer()
|
|
|
|
{
|
|
|
|
time_t inHowLong;
|
|
|
|
inHowLong = RelayConfigs::GetConfigs()->GetAllConnectedInterval();
|
|
|
|
TimerMgr::getTimerMgr()->setTimer( inHowLong,
|
|
|
|
s_checkAllConnected, this, 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::cancelAllConnectedTimer()
|
|
|
|
{
|
|
|
|
TimerMgr::getTimerMgr()->clearTimer( s_checkAllConnected, this );
|
|
|
|
}
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
void
|
|
|
|
CookieRef::sendResponse( const CRefEvent* evt )
|
|
|
|
{
|
|
|
|
int socket = evt->u.con.socket;
|
|
|
|
HostID id = evt->u.con.srcID;
|
|
|
|
|
|
|
|
assert( id != HOST_ID_NONE );
|
|
|
|
logf( "remembering pair: hostid=%x, socket=%d", id, socket );
|
|
|
|
RWWriteLock ml( &m_sockets_rwlock );
|
|
|
|
HostRec hr(socket);
|
|
|
|
m_hostSockets.insert( pair<HostID,HostRec>(id,hr) );
|
|
|
|
|
|
|
|
/* Now send the response */
|
2005-07-06 01:02:15 +02:00
|
|
|
unsigned char buf[7];
|
2005-07-06 00:05:37 +02:00
|
|
|
unsigned char* bufp = buf;
|
|
|
|
|
|
|
|
*bufp++ = XWRELAY_CONNECTRESP;
|
|
|
|
putNetShort( &bufp, GetHeartbeat() );
|
2005-07-06 01:02:15 +02:00
|
|
|
putNetLong( &bufp, GetCookieID() );
|
2005-07-06 00:05:37 +02:00
|
|
|
|
|
|
|
send_with_length( socket, buf, sizeof(buf) );
|
|
|
|
RecordSent( sizeof(buf), socket );
|
2005-08-09 01:33:51 +02:00
|
|
|
logf( "sent XWRELAY_CONNECTRESP" );
|
2005-07-06 00:05:37 +02:00
|
|
|
} /* sendResponse */
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::forward( const CRefEvent* evt )
|
|
|
|
{
|
|
|
|
unsigned char* buf = evt->u.fwd.buf;
|
|
|
|
int buflen = evt->u.fwd.buflen;
|
|
|
|
HostID src = evt->u.fwd.src;
|
|
|
|
HostID dest = evt->u.fwd.dest;
|
|
|
|
|
|
|
|
int destSocket = SocketForHost( dest );
|
|
|
|
|
2005-08-02 06:57:13 +02:00
|
|
|
if ( destSocket != -1 ) {
|
|
|
|
/* This is an ugly hack!!!! */
|
|
|
|
*buf = XWRELAY_MSG_FROMRELAY;
|
|
|
|
send_with_length( destSocket, buf, buflen );
|
2005-07-06 00:05:37 +02:00
|
|
|
|
2005-08-02 06:57:13 +02:00
|
|
|
/* also note that we've heard from src recently */
|
|
|
|
pushHeartbeatEvent( src, SocketForHost(src) );
|
|
|
|
} else {
|
|
|
|
/* We're not really connected yet! */
|
|
|
|
}
|
2005-07-06 00:05:37 +02:00
|
|
|
} /* forward */
|
|
|
|
|
2005-08-02 06:57:13 +02:00
|
|
|
void
|
|
|
|
CookieRef::checkDest( const CRefEvent* evt )
|
|
|
|
{
|
|
|
|
HostID dest = evt->u.fwd.dest;
|
|
|
|
int destSocket = SocketForHost( dest );
|
|
|
|
if ( destSocket == -1 ) {
|
|
|
|
pushDestBadEvent();
|
|
|
|
} else {
|
|
|
|
pushDestOkEvent( evt );
|
|
|
|
}
|
|
|
|
} /* checkDest */
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
void
|
|
|
|
CookieRef::checkFromServer( const CRefEvent* evt )
|
|
|
|
{
|
|
|
|
HostID src = evt->u.fwd.src;
|
|
|
|
if ( src == HOST_ID_SERVER ) {
|
|
|
|
pushCanLockEvent( evt );
|
|
|
|
} else {
|
|
|
|
pushCantLockEvent( evt );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-06 00:05:37 +02:00
|
|
|
void
|
|
|
|
CookieRef::disconnectAll( const CRefEvent* evt )
|
|
|
|
{
|
2005-09-02 08:56:34 +02:00
|
|
|
logf( "disconnectAll" );
|
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.begin();
|
|
|
|
while ( iter != m_hostSockets.end() ) {
|
|
|
|
pushRemoveSocketEvent( iter->second.m_socket );
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
logf( "disconnectAll done" );
|
2005-07-06 00:05:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::noteHeartbeat( const CRefEvent* evt )
|
|
|
|
{
|
|
|
|
int socket = evt->u.heart.socket;
|
|
|
|
HostID id = evt->u.heart.id;
|
|
|
|
|
2005-06-23 06:26:44 +02:00
|
|
|
RWWriteLock rwl( &m_sockets_rwlock );
|
|
|
|
|
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.find(id);
|
2005-09-02 08:56:34 +02:00
|
|
|
if ( iter == m_hostSockets.end() ) {
|
|
|
|
logf( "no socket for HostID %d", id );
|
|
|
|
} else {
|
2005-06-23 06:26:44 +02:00
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
/* PENDING If the message came on an unexpected socket, kill the
|
|
|
|
connection. An attack is the most likely explanation. */
|
|
|
|
assert( iter->second.m_socket == socket );
|
2005-06-23 06:26:44 +02:00
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
logf( "upping m_lastHeartbeat from %d to %d",
|
|
|
|
iter->second.m_lastHeartbeat, now() );
|
|
|
|
iter->second.m_lastHeartbeat = now();
|
|
|
|
}
|
2005-07-06 00:05:37 +02:00
|
|
|
} /* noteHeartbeat */
|
2005-06-23 06:26:44 +02:00
|
|
|
|
|
|
|
void
|
2005-07-06 00:05:37 +02:00
|
|
|
CookieRef::checkHeartbeats( const CRefEvent* evt )
|
2005-06-23 06:26:44 +02:00
|
|
|
{
|
2005-07-06 00:05:37 +02:00
|
|
|
int vcount = 0;
|
|
|
|
vector<int>* victims = evt->u.htime.victims;
|
|
|
|
time_t now = evt->u.htime.now;
|
2005-06-23 06:26:44 +02:00
|
|
|
|
|
|
|
RWWriteLock rwl( &m_sockets_rwlock );
|
|
|
|
|
|
|
|
map<HostID,HostRec>::iterator iter = m_hostSockets.begin();
|
|
|
|
while ( iter != m_hostSockets.end() ) {
|
|
|
|
time_t last = iter->second.m_lastHeartbeat;
|
2005-09-02 08:56:34 +02:00
|
|
|
if ( (now - last) > GetHeartbeat() * 2 ) {
|
2005-06-23 06:26:44 +02:00
|
|
|
victims->push_back( iter->second.m_socket );
|
2005-07-06 00:05:37 +02:00
|
|
|
++vcount;
|
2005-06-23 06:26:44 +02:00
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
logf( "CookieRef::CheckHeartbeats done" );
|
2005-07-06 00:05:37 +02:00
|
|
|
|
|
|
|
/* Post an event */
|
|
|
|
CRefEvent newEvt;
|
|
|
|
newEvt.type = vcount > 0 ? XW_EVENT_HEARTFAILED : XW_EVENT_HEARTOK;
|
|
|
|
m_eventQueue.push_front( newEvt );
|
|
|
|
} /* checkHeartbeats */
|
2005-06-23 06:26:44 +02:00
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
/* timer callback */
|
|
|
|
/* static */ void
|
|
|
|
CookieRef::s_checkAllConnected( void* closure )
|
|
|
|
{
|
|
|
|
/* Need to ensure */
|
|
|
|
CookieRef* self = (CookieRef*)closure;
|
|
|
|
SafeCref scr(self);
|
|
|
|
if ( scr.IsValid() ) {
|
|
|
|
scr.CheckAllConnected();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-04-20 14:13:20 +02:00
|
|
|
void
|
2005-09-02 08:56:34 +02:00
|
|
|
CookieRef::_CheckAllConnected()
|
|
|
|
{
|
|
|
|
logf( "checkAllConnected" );
|
|
|
|
MutexLock ml( &m_EventsMutex );
|
|
|
|
CRefEvent newEvt;
|
|
|
|
newEvt.type = XW_EVENT_CONNTIMER;
|
|
|
|
m_eventQueue.push_front( newEvt );
|
|
|
|
handleEvents();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CookieRef::_PrintCookieInfo( string& out )
|
2005-04-20 14:13:20 +02:00
|
|
|
{
|
|
|
|
out += "Name: ";
|
|
|
|
out += Name();
|
|
|
|
out += "\n";
|
|
|
|
out += "ID: ";
|
|
|
|
char buf[64];
|
2005-07-06 01:02:15 +02:00
|
|
|
snprintf( buf, sizeof(buf), "%ld\n", GetCookieID() );
|
2005-04-20 14:13:20 +02:00
|
|
|
out += buf;
|
|
|
|
|
|
|
|
snprintf( buf, sizeof(buf), "Bytes sent: %d\n", m_totalSent );
|
|
|
|
out += buf;
|
|
|
|
|
|
|
|
/* n messages */
|
|
|
|
/* n bytes */
|
|
|
|
/* open since when */
|
|
|
|
/* sockets */
|
|
|
|
|
|
|
|
} /* PrintCookieInfo */
|