/* -*- compile-command: "make -j3"; -*- */ /* * Copyright 2005 - 2012 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) 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> #include <stdio.h> #include <unistd.h> #include <pthread.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <sys/poll.h> #include <errno.h> #include <string.h> #include <stdlib.h> #include "tpool.h" #include "xwrelay_priv.h" #include "xwrelay.h" #include "timermgr.h" #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() : m_timeToDie(false) , m_nThreads(0) { pthread_rwlock_init( &m_activeSocketsRWLock, NULL ); pthread_mutex_init ( &m_queueMutex, NULL ); pthread_cond_init( &m_queueCondVar, NULL ); int fd[2]; if ( pipe( fd ) ) { logf( XW_LOGERROR, "pipe failed" ); } m_pipeRead = fd[0]; m_pipeWrite = fd[1]; logf( XW_LOGINFO, "pipes: m_pipeRead: %d; m_pipeWrite: %d", m_pipeRead, m_pipeWrite ); } XWThreadPool::~XWThreadPool() { pthread_cond_destroy( &m_queueCondVar ); pthread_rwlock_destroy( &m_activeSocketsRWLock ); pthread_mutex_destroy ( &m_queueMutex ); } /* ~XWThreadPool */ void XWThreadPool::Setup( int nThreads, packet_func pFunc, kill_func kFunc ) { m_nThreads = nThreads; m_pFunc = pFunc; m_kFunc = kFunc; pthread_t thread; int ii; for ( ii = 0; ii < nThreads; ++ii ) { int result = pthread_create( &thread, NULL, tpool_main, this ); assert( result == 0 ); pthread_detach( thread ); } int result = pthread_create( &thread, NULL, listener_main, this ); assert( result == 0 ); result = pthread_detach( thread ); assert( result == 0 ); } void XWThreadPool::Stop() { m_timeToDie = true; int ii; for ( ii = 0; ii < m_nThreads; ++ii ) { SockInfo si = { STYPE_UNKNOWN, {0} }; enqueue( 0, si ); } interrupt_poll(); } void XWThreadPool::AddSocket( int socket, SockType stype, in_addr& from ) { { RWWriteLock ml( &m_activeSocketsRWLock ); SockInfo si; si.m_type = stype; si.m_addr = from; m_activeSockets.push_back( pair<int,SockInfo>(socket, si) ); } interrupt_poll(); } bool XWThreadPool::RemoveSocket( int socket ) { bool found = false; { RWWriteLock ml( &m_activeSocketsRWLock ); vector< pair<int,SockInfo> >::iterator iter = m_activeSockets.begin(); while ( iter != m_activeSockets.end() ) { if ( iter->first == socket ) { m_activeSockets.erase( iter ); found = true; break; } ++iter; } } return found; } /* RemoveSocket */ void XWThreadPool::CloseSocket( int socket ) { /* bool do_interrupt = false; */ if ( !RemoveSocket( socket ) ) { MutexLock ml( &m_queueMutex ); deque<QueuePr>::iterator iter = m_queue.begin(); while ( iter != m_queue.end() ) { if ( iter->m_socket == socket ) { m_queue.erase( iter ); /* do_interrupt = true; */ break; } ++iter; } } logf( XW_LOGINFO, "CLOSING socket %d", socket ); close( socket ); /* 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(); /* } */ } void XWThreadPool::EnqueueKill( int socket, const char* const why ) { logf( XW_LOGINFO, "%s(%d) reason: %s", __func__, socket, why ); SockInfo si = { STYPE_UNKNOWN, {0} }; enqueue( socket, si, Q_KILL ); } bool XWThreadPool::get_process_packet( int socket, SockType stype, in_addr& addr ) { bool success = false; short packetSize; assert( sizeof(packetSize) == 2 ); unsigned char buf[MAX_MSG_LEN+1]; int nRead = read_packet( socket, buf, sizeof(buf) ); if ( nRead < 0 ) { EnqueueKill( socket, "bad packet" ); } else if ( STYPE_GAME == stype ) { logf( XW_LOGINFO, "calling m_pFunc" ); success = (*m_pFunc)( buf, nRead, socket, addr ); } else { buf[nRead] = '\0'; handle_proxy_packet( buf, nRead, socket, addr ); CloseSocket( socket ); } return success; } /* get_process_packet */ /* static */ void* XWThreadPool::tpool_main( void* closure ) { blockSignals(); XWThreadPool* me = (XWThreadPool*)closure; return me->real_tpool_main(); } void* XWThreadPool::real_tpool_main() { logf( XW_LOGINFO, "tpool worker thread starting" ); int socket = -1; for ( ; ; ) { pthread_mutex_lock( &m_queueMutex ); release_socket_locked( socket ); while ( !m_timeToDie && m_queue.size() == 0 ) { pthread_cond_wait( &m_queueCondVar, &m_queueMutex ); } if ( m_timeToDie ) { logf( XW_LOGINFO, "%s: unlocking b/c m_timeToDie set", __func__ ); pthread_mutex_unlock( &m_queueMutex ); break; } QueuePr pr; grab_elem_locked( &pr ); pthread_mutex_unlock( &m_queueMutex ); if ( pr.m_socket >= 0 ) { logf( XW_LOGINFO, "worker thread got socket %d from queue", pr.m_socket ); switch ( pr.m_act ) { case Q_READ: if ( get_process_packet( pr.m_socket, pr.m_info.m_type, pr.m_info.m_addr ) ) { AddSocket( pr.m_socket, pr.m_info.m_type, pr.m_info.m_addr ); } break; case Q_KILL: (*m_kFunc)( pr.m_socket ); CloseSocket( pr.m_socket ); break; } } socket = pr.m_socket; } logf( XW_LOGINFO, "tpool worker thread exiting" ); return NULL; } void XWThreadPool::interrupt_poll() { #ifdef LOG_POLL logf( XW_LOGINFO, __func__ ); #endif unsigned char byt = 0; int nSent = write( m_pipeWrite, &byt, 1 ); if ( nSent != 1 ) { logf( XW_LOGERROR, "errno = %s (%d)", strerror(errno), errno ); } } void* XWThreadPool::real_listener() { int flags = POLLIN | POLLERR | POLLHUP | POLLRDHUP; TimerMgr* tmgr = TimerMgr::GetTimerMgr(); int nSocketsAllocd = 1; struct pollfd* fds = (pollfd*)calloc( nSocketsAllocd, sizeof(fds[0]) ); SockInfo* sinfos = (SockInfo*)calloc( nSocketsAllocd, sizeof(sinfos[0]) ); #ifdef LOG_POLL char* log = (char*)malloc( 4 * nSocketsAllocd ); #endif for ( ; ; ) { pthread_rwlock_rdlock( &m_activeSocketsRWLock ); int nSockets = m_activeSockets.size() + 1; /* for pipe */ #ifdef LOG_POLL int logCapacity = 4 * nSockets; int logLen = 0; #endif if ( nSockets > nSocketsAllocd ) { fds = (struct pollfd*)realloc( fds, nSockets * sizeof(fds[0]) ); sinfos = (SockInfo*)realloc( sinfos, nSockets * sizeof(sinfos[0]) ); #ifdef LOG_POLL log = (char*)realloc( log, nSockets * 4 ); #endif nSocketsAllocd = nSockets; } int curfd = 0; fds[curfd].fd = m_pipeRead; fds[curfd].events = flags; #ifdef LOG_POLL logLen += snprintf( log+logLen, logCapacity - logLen, "%d,", fds[curfd].fd ); #endif ++curfd; vector< pair<int,SockInfo> >::iterator iter; for ( iter = m_activeSockets.begin(); iter != m_activeSockets.end(); ++iter ) { fds[curfd].fd = iter->first; sinfos[curfd] = iter->second; fds[curfd].events = flags; #ifdef LOG_POLL if ( logCapacity > logLen ) { logLen += snprintf( log+logLen, logCapacity - logLen, "%d,", fds[curfd].fd ); } #endif assert( curfd < nSockets ); ++curfd; } pthread_rwlock_unlock( &m_activeSocketsRWLock ); int nMillis = tmgr->GetPollTimeout(); #ifdef LOG_POLL logf( XW_LOGINFO, "polling %s nmillis=%d", log, nMillis ); #endif int nEvents = poll( fds, nSockets, nMillis ); #ifdef LOG_POLL logf( XW_LOGINFO, "back from poll: %d", nEvents ); #endif if ( m_timeToDie ) { break; } if ( nEvents == 0 ) { tmgr->FireElapsedTimers(); } else if ( nEvents < 0 ) { logf( XW_LOGERROR, "poll failed: errno: %s (%d)", strerror(errno), errno ); } if ( fds[0].revents != 0 ) { #ifdef LOG_POLL logf( XW_LOGINFO, "poll interrupted" ); #endif assert( fds[0].revents == POLLIN ); unsigned char byt; read( fds[0].fd, &byt, 1 ); --nEvents; } if ( nEvents > 0 ) { --nSockets; curfd = 1; int ii; for ( ii = 0; ii < nSockets && nEvents > 0; ++ii ) { if ( fds[curfd].revents != 0 ) { int socket = fds[curfd].fd; if ( !RemoveSocket( socket ) ) { /* no further processing if it's been removed while we've been sleeping in poll */ --nEvents; continue; } if ( 0 != (fds[curfd].revents & (POLLIN | POLLPRI)) ) { enqueue( socket, sinfos[curfd] ); } else { logf( XW_LOGERROR, "odd revents: %x", fds[curfd].revents ); EnqueueKill( socket, "error/hup in poll()" ); } --nEvents; } ++curfd; } assert( nEvents == 0 ); } } free( fds ); free( sinfos ); #ifdef LOG_POLL free( log ); #endif logf( XW_LOGINFO, "real_listener returning" ); return NULL; } /* real_listener */ /* static */ void* XWThreadPool::listener_main( void* closure ) { blockSignals(); XWThreadPool* me = (XWThreadPool*)closure; return me->real_listener(); } void XWThreadPool::enqueue( int socket, SockInfo si, QAction act ) { QueuePr pr = { act, socket, si }; MutexLock ml( &m_queueMutex ); m_queue.push_back( pr ); pthread_cond_signal( &m_queueCondVar ); } void XWThreadPool::grab_elem_locked( QueuePr* prp ) { bool found = false; prp->m_socket = -1; deque<QueuePr>::iterator iter; for ( iter = m_queue.begin(); !found && iter != m_queue.end(); ++iter ) { int socket = iter->m_socket; /* If NOT found */ if ( m_sockets_in_use.end() == m_sockets_in_use.find( socket ) ) { *prp = *iter; m_queue.erase( iter ); /* this was a double-free once! */ m_sockets_in_use.insert( socket ); found = true; } } print_in_use(); } /* grab_elem_locked */ 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 ); } print_in_use(); } void XWThreadPool::print_in_use( void ) { char buf[32] = {0}; int len = 0; set<int>::iterator iter; for ( iter = m_sockets_in_use.begin(); iter != m_sockets_in_use.end(); ++iter ) { len += snprintf( &buf[len], sizeof(buf)-len, "%d ", *iter ); } }