2011-04-01 03:16:56 +02:00
|
|
|
/* -*- compile-command: "make -j3"; -*- */
|
2005-03-31 04:11:51 +02:00
|
|
|
|
|
|
|
/*
|
2012-01-05 03:14:12 +01:00
|
|
|
* Copyright 2005 - 2012 by Eric House (xwords@eehouse.org). All rights
|
2009-07-30 14:54:17 +02:00
|
|
|
* reserved.
|
2005-03-31 04:11:51 +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) 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
2006-03-21 05:05:33 +01:00
|
|
|
#include <stdio.h>
|
2005-10-23 23:35:19 +02:00
|
|
|
#include <unistd.h>
|
2005-03-31 04:11:51 +02:00
|
|
|
#include <pthread.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <sys/poll.h>
|
|
|
|
#include <errno.h>
|
2008-12-30 06:13:30 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2005-03-31 04:11:51 +02:00
|
|
|
|
|
|
|
#include "tpool.h"
|
|
|
|
#include "xwrelay_priv.h"
|
|
|
|
#include "xwrelay.h"
|
2005-09-02 08:56:34 +02:00
|
|
|
#include "timermgr.h"
|
2005-03-31 04:11:51 +02:00
|
|
|
#include "mlock.h"
|
|
|
|
|
|
|
|
XWThreadPool* XWThreadPool::g_instance = NULL;
|
|
|
|
|
|
|
|
/* static */ XWThreadPool*
|
|
|
|
XWThreadPool::GetTPool()
|
|
|
|
{
|
|
|
|
XWThreadPool* me = g_instance;
|
|
|
|
if ( me == NULL ) {
|
|
|
|
me = new XWThreadPool();
|
|
|
|
g_instance = me;
|
|
|
|
}
|
|
|
|
return me;
|
|
|
|
}
|
|
|
|
|
|
|
|
XWThreadPool::XWThreadPool()
|
2009-07-13 04:58:16 +02:00
|
|
|
: m_timeToDie(false)
|
2005-10-23 17:49:48 +02:00
|
|
|
, m_nThreads(0)
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2005-04-20 14:06:25 +02:00
|
|
|
pthread_rwlock_init( &m_activeSocketsRWLock, NULL );
|
2005-03-31 04:11:51 +02:00
|
|
|
pthread_mutex_init ( &m_queueMutex, NULL );
|
|
|
|
|
|
|
|
pthread_cond_init( &m_queueCondVar, NULL );
|
|
|
|
|
|
|
|
int fd[2];
|
|
|
|
if ( pipe( fd ) ) {
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGERROR, "pipe failed" );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
m_pipeRead = fd[0];
|
|
|
|
m_pipeWrite = fd[1];
|
2009-07-06 03:32:30 +02:00
|
|
|
logf( XW_LOGINFO, "pipes: m_pipeRead: %d; m_pipeWrite: %d",
|
|
|
|
m_pipeRead, m_pipeWrite );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
XWThreadPool::~XWThreadPool()
|
|
|
|
{
|
|
|
|
pthread_cond_destroy( &m_queueCondVar );
|
|
|
|
|
|
|
|
pthread_rwlock_destroy( &m_activeSocketsRWLock );
|
|
|
|
pthread_mutex_destroy ( &m_queueMutex );
|
|
|
|
} /* ~XWThreadPool */
|
|
|
|
|
2005-03-31 04:11:51 +02:00
|
|
|
void
|
2013-01-26 20:54:48 +01:00
|
|
|
XWThreadPool::Setup( int nThreads, kill_func kFunc )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
|
|
|
m_nThreads = nThreads;
|
2013-01-03 06:12:42 +01:00
|
|
|
m_threadInfos = (ThreadInfo*)malloc( nThreads * sizeof(*m_threadInfos) );
|
2010-09-18 17:47:56 +02:00
|
|
|
m_kFunc = kFunc;
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2013-01-12 05:42:04 +01:00
|
|
|
for ( int ii = 0; ii < nThreads; ++ii ) {
|
2013-01-13 19:14:06 +01:00
|
|
|
ThreadInfo* tip = &m_threadInfos[ii];
|
|
|
|
tip->me = this;
|
2013-01-12 05:42:04 +01:00
|
|
|
int result = pthread_create( &tip->thread, NULL, tpool_main, tip );
|
2005-03-31 04:11:51 +02:00
|
|
|
assert( result == 0 );
|
2013-01-12 05:42:04 +01:00
|
|
|
pthread_detach( tip->thread );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
2013-01-12 05:42:04 +01:00
|
|
|
pthread_t thread;
|
2005-03-31 04:11:51 +02:00
|
|
|
int result = pthread_create( &thread, NULL, listener_main, this );
|
|
|
|
assert( result == 0 );
|
2011-06-30 06:37:33 +02:00
|
|
|
result = pthread_detach( thread );
|
|
|
|
assert( result == 0 );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
void
|
|
|
|
XWThreadPool::Stop()
|
|
|
|
{
|
2009-07-13 04:58:16 +02:00
|
|
|
m_timeToDie = true;
|
2005-10-23 17:49:48 +02:00
|
|
|
|
2008-12-30 06:13:30 +01:00
|
|
|
int ii;
|
|
|
|
for ( ii = 0; ii < m_nThreads; ++ii ) {
|
2013-01-13 01:09:24 +01:00
|
|
|
SockInfo si;
|
|
|
|
si.m_type = STYPE_UNKNOWN;
|
|
|
|
enqueue( si );
|
2005-10-23 17:49:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
interrupt_poll();
|
|
|
|
}
|
|
|
|
|
2005-03-31 04:11:51 +02:00
|
|
|
void
|
2013-01-26 20:54:48 +01:00
|
|
|
XWThreadPool::AddSocket( SockType stype, QueueCallback proc, const AddrInfo* from )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
|
|
|
{
|
2005-04-20 14:06:25 +02:00
|
|
|
RWWriteLock ml( &m_activeSocketsRWLock );
|
2012-01-05 03:14:12 +01:00
|
|
|
SockInfo si;
|
|
|
|
si.m_type = stype;
|
2013-01-26 20:54:48 +01:00
|
|
|
si.m_proc = proc;
|
2013-01-13 01:09:24 +01:00
|
|
|
si.m_addr = *from;
|
|
|
|
m_activeSockets.push_back( si );
|
|
|
|
logf( XW_LOGINFO, "%s: %d sockets active", __func__,
|
|
|
|
m_activeSockets.size() );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
interrupt_poll();
|
|
|
|
}
|
|
|
|
|
2008-12-30 06:13:30 +01:00
|
|
|
bool
|
2013-01-13 01:09:24 +01:00
|
|
|
XWThreadPool::RemoveSocket( const AddrInfo* addr )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2013-01-13 01:09:24 +01:00
|
|
|
assert( addr->isTCP() );
|
2008-12-30 06:13:30 +01:00
|
|
|
bool found = false;
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2005-04-20 14:06:25 +02:00
|
|
|
RWWriteLock ml( &m_activeSocketsRWLock );
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2013-01-13 01:09:24 +01:00
|
|
|
logf( XW_LOGINFO, "%s: START: %d sockets active", __func__,
|
|
|
|
m_activeSockets.size() );
|
|
|
|
|
|
|
|
vector<SockInfo>::iterator iter;
|
|
|
|
for ( iter = m_activeSockets.begin();
|
|
|
|
iter != m_activeSockets.end(); ++iter ) {
|
|
|
|
if ( iter->m_addr.equals( *addr ) ) {
|
2005-03-31 04:11:51 +02:00
|
|
|
m_activeSockets.erase( iter );
|
2008-12-30 06:13:30 +01:00
|
|
|
found = true;
|
2005-04-20 14:06:25 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-01-13 01:09:24 +01:00
|
|
|
logf( XW_LOGINFO, "%s: AFTER: %d sockets active", __func__,
|
|
|
|
m_activeSockets.size() );
|
2005-04-20 14:06:25 +02:00
|
|
|
}
|
|
|
|
return found;
|
2005-06-23 06:26:44 +02:00
|
|
|
} /* RemoveSocket */
|
2005-04-20 14:06:25 +02:00
|
|
|
|
|
|
|
void
|
2013-01-13 01:09:24 +01:00
|
|
|
XWThreadPool::CloseSocket( const AddrInfo* addr )
|
2005-04-20 14:06:25 +02:00
|
|
|
{
|
2008-12-30 06:13:30 +01:00
|
|
|
/* bool do_interrupt = false; */
|
2013-01-13 01:09:24 +01:00
|
|
|
assert( addr->isTCP() );
|
|
|
|
if ( !RemoveSocket( addr ) ) {
|
2011-06-30 06:37:33 +02:00
|
|
|
MutexLock ml( &m_queueMutex );
|
2010-09-18 16:44:14 +02:00
|
|
|
deque<QueuePr>::iterator iter = m_queue.begin();
|
2005-04-20 14:06:25 +02:00
|
|
|
while ( iter != m_queue.end() ) {
|
2013-01-13 01:09:24 +01:00
|
|
|
if ( iter->m_info.m_addr.equals( *addr ) ) {
|
2005-04-20 14:06:25 +02:00
|
|
|
m_queue.erase( iter );
|
2008-12-30 06:13:30 +01:00
|
|
|
/* do_interrupt = true; */
|
2005-03-31 04:11:51 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
2009-07-06 03:32:30 +02:00
|
|
|
logf( XW_LOGINFO, "CLOSING socket %d", socket );
|
2013-01-13 19:14:06 +01:00
|
|
|
close( addr->socket() );
|
2005-06-23 06:26:44 +02:00
|
|
|
/* if ( do_interrupt ) { */
|
|
|
|
/* We always need to interrupt the poll because the socket we're closing
|
|
|
|
will be in the list being listened to. That or we need to drop sockets
|
|
|
|
that have been removed on some other thread while the poll call's
|
|
|
|
blocking.*/
|
|
|
|
interrupt_poll();
|
|
|
|
/* } */
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
2010-09-18 17:47:56 +02:00
|
|
|
void
|
2013-01-13 01:09:24 +01:00
|
|
|
XWThreadPool::EnqueueKill( const AddrInfo* addr, const char* const why )
|
2010-09-18 17:47:56 +02:00
|
|
|
{
|
2011-06-21 03:10:42 +02:00
|
|
|
logf( XW_LOGINFO, "%s(%d) reason: %s", __func__, socket, why );
|
2013-01-13 01:09:24 +01:00
|
|
|
if ( addr->isTCP() ) {
|
|
|
|
SockInfo si;
|
|
|
|
si.m_type = STYPE_UNKNOWN;
|
|
|
|
si.m_addr = *addr;
|
|
|
|
enqueue( si, Q_KILL );
|
|
|
|
}
|
2010-09-18 17:47:56 +02:00
|
|
|
}
|
|
|
|
|
2008-12-30 06:13:30 +01:00
|
|
|
bool
|
2013-01-26 20:54:48 +01:00
|
|
|
XWThreadPool::get_process_packet( SockType stype, QueueCallback proc, const AddrInfo* addr )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2009-02-01 17:00:20 +01:00
|
|
|
bool success = false;
|
2005-03-31 04:11:51 +02:00
|
|
|
short packetSize;
|
|
|
|
assert( sizeof(packetSize) == 2 );
|
|
|
|
|
2011-01-21 03:14:56 +01:00
|
|
|
unsigned char buf[MAX_MSG_LEN+1];
|
2013-01-13 19:14:06 +01:00
|
|
|
int nRead = read_packet( addr->socket(), buf, sizeof(buf) );
|
2010-09-21 15:38:43 +02:00
|
|
|
if ( nRead < 0 ) {
|
2013-01-13 01:09:24 +01:00
|
|
|
EnqueueKill( addr, "bad packet" );
|
2013-01-26 20:54:48 +01:00
|
|
|
} else if ( STYPE_PROXY == stype && NULL != proc ) {
|
2011-01-21 03:14:56 +01:00
|
|
|
buf[nRead] = '\0';
|
2013-01-26 20:54:48 +01:00
|
|
|
UdpQueue::get()->handle( addr, buf, nRead+1, proc );
|
|
|
|
} else if ( STYPE_GAME == stype && NULL != proc ) {
|
|
|
|
UdpQueue::get()->handle( addr, buf, nRead, proc );
|
|
|
|
success = true;
|
|
|
|
} else {
|
|
|
|
assert(0);
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
2005-10-14 10:29:58 +02:00
|
|
|
return success;
|
2005-03-31 04:11:51 +02:00
|
|
|
} /* get_process_packet */
|
|
|
|
|
2013-01-03 06:12:42 +01:00
|
|
|
void*
|
2005-03-31 04:11:51 +02:00
|
|
|
XWThreadPool::tpool_main( void* closure )
|
|
|
|
{
|
2010-10-05 05:03:00 +02:00
|
|
|
blockSignals();
|
|
|
|
|
2013-01-03 06:12:42 +01:00
|
|
|
ThreadInfo* tip = (ThreadInfo*)closure;
|
|
|
|
return tip->me->real_tpool_main( tip );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
2005-04-08 16:28:04 +02:00
|
|
|
void*
|
2013-01-03 06:12:42 +01:00
|
|
|
XWThreadPool::real_tpool_main( ThreadInfo* tip )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2005-10-23 17:49:48 +02:00
|
|
|
logf( XW_LOGINFO, "tpool worker thread starting" );
|
2010-09-18 16:44:14 +02:00
|
|
|
int socket = -1;
|
2005-03-31 04:11:51 +02:00
|
|
|
for ( ; ; ) {
|
|
|
|
pthread_mutex_lock( &m_queueMutex );
|
2013-01-13 01:09:24 +01:00
|
|
|
tip->recentTime = 0;
|
2013-01-03 06:12:42 +01:00
|
|
|
|
2013-01-13 01:09:24 +01:00
|
|
|
release_socket_locked( socket );
|
2010-09-18 16:44:14 +02:00
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
while ( !m_timeToDie && m_queue.size() == 0 ) {
|
2005-03-31 04:11:51 +02:00
|
|
|
pthread_cond_wait( &m_queueCondVar, &m_queueMutex );
|
|
|
|
}
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
if ( m_timeToDie ) {
|
2010-09-21 07:11:13 +02:00
|
|
|
logf( XW_LOGINFO, "%s: unlocking b/c m_timeToDie set", __func__ );
|
2010-09-15 06:16:16 +02:00
|
|
|
pthread_mutex_unlock( &m_queueMutex );
|
2005-10-23 17:49:48 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-09-18 16:44:14 +02:00
|
|
|
QueuePr pr;
|
2013-01-13 19:14:06 +01:00
|
|
|
bool gotOne = grab_elem_locked( &pr );
|
2010-09-18 17:47:56 +02:00
|
|
|
|
2013-01-13 01:09:24 +01:00
|
|
|
tip->recentTime = time( NULL );
|
2005-03-31 04:11:51 +02:00
|
|
|
pthread_mutex_unlock( &m_queueMutex );
|
|
|
|
|
2013-01-13 19:14:06 +01:00
|
|
|
if ( gotOne ) {
|
|
|
|
socket = pr.m_info.m_addr.socket();
|
|
|
|
logf( XW_LOGINFO, "worker thread got socket %d from queue", socket );
|
2010-09-18 17:47:56 +02:00
|
|
|
switch ( pr.m_act ) {
|
|
|
|
case Q_READ:
|
2013-01-13 19:14:06 +01:00
|
|
|
assert( socket >= 0 );
|
2013-01-26 20:54:48 +01:00
|
|
|
if ( get_process_packet( pr.m_info.m_type, pr.m_info.m_proc, &pr.m_info.m_addr ) ) {
|
|
|
|
AddSocket( pr.m_info.m_type, pr.m_info.m_proc, &pr.m_info.m_addr );
|
2010-09-18 17:47:56 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case Q_KILL:
|
2013-01-13 01:09:24 +01:00
|
|
|
(*m_kFunc)( &pr.m_info.m_addr );
|
|
|
|
CloseSocket( &pr.m_info.m_addr );
|
2010-09-18 17:47:56 +02:00
|
|
|
break;
|
|
|
|
}
|
2013-01-13 19:37:47 +01:00
|
|
|
} else {
|
|
|
|
socket = -1;
|
2010-09-18 17:47:56 +02:00
|
|
|
}
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
logf( XW_LOGINFO, "tpool worker thread exiting" );
|
2005-04-08 16:28:04 +02:00
|
|
|
return NULL;
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
XWThreadPool::interrupt_poll()
|
|
|
|
{
|
2010-09-09 05:17:32 +02:00
|
|
|
#ifdef LOG_POLL
|
2007-02-06 06:52:22 +01:00
|
|
|
logf( XW_LOGINFO, __func__ );
|
2010-09-09 05:17:32 +02:00
|
|
|
#endif
|
2005-03-31 04:11:51 +02:00
|
|
|
unsigned char byt = 0;
|
|
|
|
int nSent = write( m_pipeWrite, &byt, 1 );
|
|
|
|
if ( nSent != 1 ) {
|
2007-02-03 22:35:07 +01:00
|
|
|
logf( XW_LOGERROR, "errno = %s (%d)", strerror(errno), errno );
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-04-08 16:28:04 +02:00
|
|
|
void*
|
2005-03-31 04:11:51 +02:00
|
|
|
XWThreadPool::real_listener()
|
|
|
|
{
|
2011-12-16 03:41:07 +01:00
|
|
|
int flags = POLLIN | POLLERR | POLLHUP | POLLRDHUP;
|
2005-10-15 17:49:22 +02:00
|
|
|
TimerMgr* tmgr = TimerMgr::GetTimerMgr();
|
2009-07-28 07:08:15 +02:00
|
|
|
int nSocketsAllocd = 1;
|
|
|
|
|
|
|
|
struct pollfd* fds = (pollfd*)calloc( nSocketsAllocd, sizeof(fds[0]) );
|
2012-01-05 03:14:12 +01:00
|
|
|
SockInfo* sinfos = (SockInfo*)calloc( nSocketsAllocd, sizeof(sinfos[0]) );
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2009-07-28 07:08:15 +02:00
|
|
|
char* log = (char*)malloc( 4 * nSocketsAllocd );
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2005-03-31 04:11:51 +02:00
|
|
|
|
|
|
|
for ( ; ; ) {
|
|
|
|
|
2005-04-20 14:06:25 +02:00
|
|
|
pthread_rwlock_rdlock( &m_activeSocketsRWLock );
|
2005-03-31 04:11:51 +02:00
|
|
|
int nSockets = m_activeSockets.size() + 1; /* for pipe */
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2009-07-16 06:43:33 +02:00
|
|
|
int logCapacity = 4 * nSockets;
|
|
|
|
int logLen = 0;
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2009-07-28 07:08:15 +02:00
|
|
|
|
|
|
|
if ( nSockets > nSocketsAllocd ) {
|
|
|
|
fds = (struct pollfd*)realloc( fds, nSockets * sizeof(fds[0]) );
|
2012-01-05 03:14:12 +01:00
|
|
|
sinfos = (SockInfo*)realloc( sinfos, nSockets * sizeof(sinfos[0]) );
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2009-07-28 07:08:15 +02:00
|
|
|
log = (char*)realloc( log, nSockets * 4 );
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2009-07-28 07:08:15 +02:00
|
|
|
nSocketsAllocd = nSockets;
|
|
|
|
}
|
2011-01-20 03:36:24 +01:00
|
|
|
int curfd = 0;
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2011-01-20 03:36:24 +01:00
|
|
|
fds[curfd].fd = m_pipeRead;
|
|
|
|
fds[curfd].events = flags;
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2011-01-20 03:36:24 +01:00
|
|
|
logLen += snprintf( log+logLen, logCapacity - logLen, "%d,",
|
|
|
|
fds[curfd].fd );
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2005-03-31 04:11:51 +02:00
|
|
|
++curfd;
|
|
|
|
|
2013-01-13 01:09:24 +01:00
|
|
|
vector<SockInfo>::iterator iter;
|
2011-01-21 03:14:56 +01:00
|
|
|
for ( iter = m_activeSockets.begin(); iter != m_activeSockets.end();
|
|
|
|
++iter ) {
|
2013-01-13 19:14:06 +01:00
|
|
|
fds[curfd].fd = iter->m_addr.socket();
|
2013-01-13 01:09:24 +01:00
|
|
|
sinfos[curfd] = *iter;
|
2011-01-20 03:36:24 +01:00
|
|
|
fds[curfd].events = flags;
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2009-07-16 06:43:33 +02:00
|
|
|
if ( logCapacity > logLen ) {
|
|
|
|
logLen += snprintf( log+logLen, logCapacity - logLen, "%d,",
|
2011-01-20 03:36:24 +01:00
|
|
|
fds[curfd].fd );
|
2009-07-16 06:43:33 +02:00
|
|
|
}
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2011-01-20 03:36:24 +01:00
|
|
|
assert( curfd < nSockets );
|
2005-03-31 04:11:51 +02:00
|
|
|
++curfd;
|
|
|
|
}
|
2005-04-20 14:06:25 +02:00
|
|
|
pthread_rwlock_unlock( &m_activeSocketsRWLock );
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2005-10-15 17:49:22 +02:00
|
|
|
int nMillis = tmgr->GetPollTimeout();
|
2005-06-23 06:26:44 +02:00
|
|
|
|
2009-08-20 05:50:40 +02:00
|
|
|
#ifdef LOG_POLL
|
2005-10-16 03:19:25 +02:00
|
|
|
logf( XW_LOGINFO, "polling %s nmillis=%d", log, nMillis );
|
2009-08-20 05:50:40 +02:00
|
|
|
#endif
|
2005-10-16 03:19:25 +02:00
|
|
|
int nEvents = poll( fds, nSockets, nMillis );
|
2010-09-09 05:17:32 +02:00
|
|
|
#ifdef LOG_POLL
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGINFO, "back from poll: %d", nEvents );
|
2010-09-09 05:17:32 +02:00
|
|
|
#endif
|
2005-10-23 17:49:48 +02:00
|
|
|
if ( m_timeToDie ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-09-02 08:56:34 +02:00
|
|
|
if ( nEvents == 0 ) {
|
2005-10-15 17:49:22 +02:00
|
|
|
tmgr->FireElapsedTimers();
|
2005-09-02 08:56:34 +02:00
|
|
|
} else if ( nEvents < 0 ) {
|
2009-07-16 06:43:33 +02:00
|
|
|
logf( XW_LOGERROR, "poll failed: errno: %s (%d)",
|
|
|
|
strerror(errno), errno );
|
2005-09-02 08:56:34 +02:00
|
|
|
}
|
2005-03-31 04:11:51 +02:00
|
|
|
|
|
|
|
if ( fds[0].revents != 0 ) {
|
2010-09-09 05:17:32 +02:00
|
|
|
#ifdef LOG_POLL
|
2005-10-02 18:08:42 +02:00
|
|
|
logf( XW_LOGINFO, "poll interrupted" );
|
2010-09-09 05:17:32 +02:00
|
|
|
#endif
|
2005-03-31 04:11:51 +02:00
|
|
|
assert( fds[0].revents == POLLIN );
|
|
|
|
unsigned char byt;
|
|
|
|
read( fds[0].fd, &byt, 1 );
|
|
|
|
--nEvents;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( nEvents > 0 ) {
|
|
|
|
--nSockets;
|
2011-01-20 03:36:24 +01:00
|
|
|
curfd = 1;
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2008-12-30 06:13:30 +01:00
|
|
|
int ii;
|
|
|
|
for ( ii = 0; ii < nSockets && nEvents > 0; ++ii ) {
|
2005-03-31 04:11:51 +02:00
|
|
|
|
2011-01-20 03:36:24 +01:00
|
|
|
if ( fds[curfd].revents != 0 ) {
|
|
|
|
int socket = fds[curfd].fd;
|
2013-01-13 01:09:24 +01:00
|
|
|
const AddrInfo* addr = &sinfos[curfd].m_addr;
|
2013-01-13 19:14:06 +01:00
|
|
|
assert( socket == addr->socket() );
|
2013-01-13 01:09:24 +01:00
|
|
|
if ( !RemoveSocket( addr ) ) {
|
2005-06-23 06:26:44 +02:00
|
|
|
/* no further processing if it's been removed while
|
|
|
|
we've been sleeping in poll */
|
2005-07-05 23:52:24 +02:00
|
|
|
--nEvents;
|
2005-06-23 06:26:44 +02:00
|
|
|
continue;
|
|
|
|
}
|
2005-04-03 19:02:09 +02:00
|
|
|
|
2011-01-20 03:36:24 +01:00
|
|
|
if ( 0 != (fds[curfd].revents & (POLLIN | POLLPRI)) ) {
|
2013-01-13 01:09:24 +01:00
|
|
|
enqueue( sinfos[curfd] );
|
2005-04-03 19:02:09 +02:00
|
|
|
} else {
|
2011-01-20 03:36:24 +01:00
|
|
|
logf( XW_LOGERROR, "odd revents: %x",
|
|
|
|
fds[curfd].revents );
|
2013-01-13 01:09:24 +01:00
|
|
|
EnqueueKill( addr, "error/hup in poll()" );
|
2005-04-03 19:02:09 +02:00
|
|
|
}
|
2005-03-31 04:11:51 +02:00
|
|
|
--nEvents;
|
|
|
|
}
|
|
|
|
++curfd;
|
|
|
|
}
|
|
|
|
assert( nEvents == 0 );
|
|
|
|
}
|
|
|
|
}
|
2005-10-23 17:49:48 +02:00
|
|
|
|
2011-06-30 03:45:02 +02:00
|
|
|
free( fds );
|
2012-01-05 03:14:12 +01:00
|
|
|
free( sinfos );
|
2011-06-30 03:45:02 +02:00
|
|
|
#ifdef LOG_POLL
|
|
|
|
free( log );
|
|
|
|
#endif
|
|
|
|
|
2005-10-23 17:49:48 +02:00
|
|
|
logf( XW_LOGINFO, "real_listener returning" );
|
2005-04-08 16:28:04 +02:00
|
|
|
return NULL;
|
2005-03-31 04:11:51 +02:00
|
|
|
} /* real_listener */
|
|
|
|
|
|
|
|
/* static */ void*
|
|
|
|
XWThreadPool::listener_main( void* closure )
|
|
|
|
{
|
2010-10-05 05:03:00 +02:00
|
|
|
blockSignals();
|
|
|
|
|
2005-03-31 04:11:51 +02:00
|
|
|
XWThreadPool* me = (XWThreadPool*)closure;
|
2005-04-08 16:28:04 +02:00
|
|
|
return me->real_listener();
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-01-13 01:09:24 +01:00
|
|
|
XWThreadPool::enqueue( SockInfo si, QAction act )
|
2005-03-31 04:11:51 +02:00
|
|
|
{
|
2013-01-13 01:09:24 +01:00
|
|
|
QueuePr pr = { act, si };
|
2011-04-01 03:16:56 +02:00
|
|
|
MutexLock ml( &m_queueMutex );
|
2010-09-18 16:44:14 +02:00
|
|
|
m_queue.push_back( pr );
|
2005-03-31 04:11:51 +02:00
|
|
|
|
|
|
|
pthread_cond_signal( &m_queueCondVar );
|
2013-01-03 06:12:42 +01:00
|
|
|
log_hung_threads();
|
2005-03-31 04:11:51 +02:00
|
|
|
}
|
2010-09-18 16:44:14 +02:00
|
|
|
|
2013-01-13 19:14:06 +01:00
|
|
|
bool
|
2010-09-21 07:11:13 +02:00
|
|
|
XWThreadPool::grab_elem_locked( QueuePr* prp )
|
2010-09-18 16:44:14 +02:00
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
deque<QueuePr>::iterator iter;
|
|
|
|
for ( iter = m_queue.begin(); !found && iter != m_queue.end(); ++iter ) {
|
2013-01-13 19:14:06 +01:00
|
|
|
int socket = iter->m_info.m_addr.socket();
|
2010-09-23 15:34:18 +02:00
|
|
|
/* If NOT found */
|
2010-09-21 07:11:13 +02:00
|
|
|
if ( m_sockets_in_use.end() == m_sockets_in_use.find( socket ) ) {
|
2010-09-18 16:44:14 +02:00
|
|
|
*prp = *iter;
|
2010-09-23 15:34:18 +02:00
|
|
|
m_queue.erase( iter ); /* this was a double-free once! */
|
2010-09-18 16:44:14 +02:00
|
|
|
m_sockets_in_use.insert( socket );
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
}
|
2010-09-18 17:47:56 +02:00
|
|
|
|
2010-09-23 15:34:18 +02:00
|
|
|
print_in_use();
|
2013-01-13 19:14:06 +01:00
|
|
|
return found;
|
2010-09-21 07:11:13 +02:00
|
|
|
} /* grab_elem_locked */
|
2010-09-18 16:44:14 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
XWThreadPool::release_socket_locked( int socket )
|
|
|
|
{
|
|
|
|
if ( -1 != socket ) {
|
|
|
|
set<int>::iterator iter = m_sockets_in_use.find( socket );
|
|
|
|
assert( iter != m_sockets_in_use.end() );
|
|
|
|
m_sockets_in_use.erase( iter );
|
|
|
|
}
|
2010-09-23 15:34:18 +02:00
|
|
|
print_in_use();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
XWThreadPool::print_in_use( void )
|
|
|
|
{
|
2013-01-03 06:12:42 +01:00
|
|
|
string str;
|
2010-09-23 15:34:18 +02:00
|
|
|
set<int>::iterator iter;
|
|
|
|
|
|
|
|
for ( iter = m_sockets_in_use.begin();
|
|
|
|
iter != m_sockets_in_use.end(); ++iter ) {
|
2013-01-03 06:12:42 +01:00
|
|
|
string_printf( str, "%d ", *iter );
|
|
|
|
}
|
|
|
|
if ( 0 < str.size() ) {
|
|
|
|
logf( XW_LOGINFO, "Sockets in use: %s", str.c_str() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have the mutex when this is called
|
|
|
|
void
|
|
|
|
XWThreadPool::log_hung_threads( void )
|
|
|
|
{
|
2013-01-12 05:42:04 +01:00
|
|
|
const time_t HUNG_THREASHHOLD = 300; // seconds
|
2013-01-03 06:12:42 +01:00
|
|
|
int ii;
|
|
|
|
time_t now = time( NULL );
|
|
|
|
for ( ii = 0; ii < m_nThreads; ++ii ) {
|
2013-01-13 19:14:06 +01:00
|
|
|
ThreadInfo* tip = &m_threadInfos[ii];
|
|
|
|
time_t recentTime = tip->recentTime;
|
|
|
|
if ( 0 != recentTime ) {
|
|
|
|
time_t howLong = now - recentTime;
|
|
|
|
if ( HUNG_THREASHHOLD < howLong ) {
|
|
|
|
logf( XW_LOGERROR, "thread %d (%p) stopped for %d seconds!",
|
|
|
|
ii, tip->thread, howLong );
|
|
|
|
tip->recentTime = 0; // only log once
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
}
|
2010-09-23 15:34:18 +02:00
|
|
|
}
|
2010-09-18 16:44:14 +02:00
|
|
|
}
|