From c82dc43edc9fd913e8ddb20bebe778781ec0d6f7 Mon Sep 17 00:00:00 2001 From: Mike Buland Date: Tue, 11 Sep 2007 04:05:26 +0000 Subject: Everything seems to work with the new Bu::ItoServer class, it operates very, very similarly to the Bu::Server class, except that every incoming connection gets it's own thread. This functionality may have to be tuned later, to allow for maintaining a pool of connections as an option, but this is fine for now. --- src/itoserver.cpp | 178 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 178 insertions(+) create mode 100644 src/itoserver.cpp (limited to 'src/itoserver.cpp') diff --git a/src/itoserver.cpp b/src/itoserver.cpp new file mode 100644 index 0000000..0337057 --- /dev/null +++ b/src/itoserver.cpp @@ -0,0 +1,178 @@ +#include "bu/itoserver.h" +#include +#include "bu/serversocket.h" +#include "bu/client.h" +#include "bu/socket.h" +#include "osx_compatibility.h" + +Bu::ItoServer::ItoServer() : + nTimeoutSec( 1 ), + nTimeoutUSec( 0 ) +{ + FD_ZERO( &fdActive ); +} + +Bu::ItoServer::~ItoServer() +{ +} + +void Bu::ItoServer::addPort( int nPort, int nPoolSize ) +{ + ServerSocket *s = new ServerSocket( nPort, nPoolSize ); + int nSocket = s->getSocket(); + FD_SET( nSocket, &fdActive ); + hServers.insert( nSocket, s ); +} + +void Bu::ItoServer::addPort( const FString &sAddr, int nPort, int nPoolSize ) +{ + ServerSocket *s = new ServerSocket( sAddr, nPort, nPoolSize ); + int nSocket = s->getSocket(); + FD_SET( nSocket, &fdActive ); + hServers.insert( nSocket, s ); +} + +void Bu::ItoServer::setTimeout( int nTimeoutSec, int nTimeoutUSec ) +{ + this->nTimeoutSec = nTimeoutSec; + this->nTimeoutUSec = nTimeoutUSec; +} +/* +void Bu::ItoServer::scan() +{ + struct timeval xTimeout = { nTimeoutSec, nTimeoutUSec }; + + fd_set fdRead = fdActive; + fd_set fdWrite = fdActive; + fd_set fdException = fdActive; + + if( TEMP_FAILURE_RETRY( select( FD_SETSIZE, &fdRead, NULL, &fdException, &xTimeout ) ) < 0 ) + { + throw ExceptionBase("Error attempting to scan open connections."); + } + + for( int j = 0; j < FD_SETSIZE; j++ ) + { + if( FD_ISSET( j, &fdRead ) ) + { + if( hServers.has( j ) ) + { + ServerSocket *pSrv = hServers.get( j ); + addClient( pSrv->accept(), pSrv->getPort() ); + } + else + { + Client *pClient = hClients.get( j ); + pClient->processInput(); + if( !pClient->isOpen() ) + { + onClosedConnection( pClient ); + hClients.erase( j ); + FD_CLR( j, &fdActive ); + } + } + } + } + + // Now we just try to write all the pending data on all the sockets. + // this could be done better eventually, if we care about the socket + // wanting to accept writes (using a select). + for( ClientHash::iterator i = hClients.begin(); i != hClients.end(); i++ ) + { + (*i)->processOutput(); + } +} +*/ +void Bu::ItoServer::addClient( int nSocket, int nPort ) +{ + ItoClient *pC = new ItoClient( *this, nSocket, nPort, nTimeoutSec, + nTimeoutUSec ); + pC->start(); + +} + +void *Bu::ItoServer::run() +{ + for(;;) + { + struct timeval xTimeout = { nTimeoutSec, nTimeoutUSec }; + + fd_set fdRead = fdActive; + fd_set fdWrite = fdActive; + fd_set fdException = fdActive; + + if( TEMP_FAILURE_RETRY( select( FD_SETSIZE, &fdRead, NULL, &fdException, &xTimeout ) ) < 0 ) + { + throw ExceptionBase("Error attempting to scan open connections."); + } + + for( ServerHash::iterator i = hServers.begin(); i != hServers.end(); i++ ) + { + if( FD_ISSET( i.getKey(), &fdRead ) ) + { + ServerSocket *pSrv = i.getValue(); + addClient( pSrv->accept(), pSrv->getPort() ); + } + } + } + + return NULL; +} + +Bu::ItoServer::ItoClient::ItoClient( ItoServer &rSrv, int iSocket, int iPort, + int nTimeoutSec, int nTimeoutUSec ) : + rSrv( rSrv ), + iSocket( iSocket ), + iPort( iPort ), + nTimeoutSec( nTimeoutSec ), + nTimeoutUSec( nTimeoutUSec ) +{ + FD_ZERO( &fdActive ); + FD_SET( iSocket, &fdActive ); + + pClient = new Client( + new Bu::Socket( iSocket ) + ); + +} + +Bu::ItoServer::ItoClient::~ItoClient() +{ +} + +void *Bu::ItoServer::ItoClient::run() +{ + rSrv.onNewConnection( pClient, iPort ); + + for(;;) + { + struct timeval xTimeout = { nTimeoutSec, nTimeoutUSec }; + + fd_set fdRead = fdActive; + fd_set fdException = fdActive; + + if( TEMP_FAILURE_RETRY( select( FD_SETSIZE, &fdRead, NULL, &fdException, &xTimeout ) ) < 0 ) + { + throw ExceptionBase("Error attempting to scan open connections."); + } + + if( FD_ISSET( iSocket, &fdRead ) ) + { + pClient->processInput(); + if( !pClient->isOpen() ) + { + rSrv.onClosedConnection( pClient ); + + return NULL; + } + } + + // Now we just try to write all the pending data on the socket. + // this could be done better eventually, if we care about the socket + // wanting to accept writes (using a select). + pClient->processOutput(); + } + + return NULL; +} + -- cgit v1.2.3