csgo-2018-source/dedicated/sys_subproc.cpp
2021-07-24 21:11:47 -07:00

846 lines
20 KiB
C++

//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======//
//
// Purpose:
//
// $NoKeywords: $
//===========================================================================//
#ifdef _LINUX
// linux has a multi-processing forked server mode.
#include <stdio.h>
#include <stdlib.h>
#include <sys/poll.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <syscall.h>
#include <unistd.h>
#include <arpa/inet.h>
//#include <linux/tcp.h>
#include <netdb.h>
//#include <sys/param.h>
#include <sys/uio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include "isys.h"
#include "dedicated.h"
#include "engine_hlds_api.h"
#include "filesystem.h"
#include "tier0/dbg.h"
#include "tier1/strtools.h"
#include "tier0/icommandline.h"
#include "tier2/socketcreator.h"
#include "idedicatedexports.h"
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/ioctl.h>
#include "mathlib/expressioncalculator.h"
#define closesocket close
#define WSAGetLastError() errno
#define ioctlsocket ioctl
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
static netadr_t net_local_adr;
unsigned short NET_HostToNetShort( unsigned short us_in )
{
return htons( us_in );
}
unsigned short NET_NetToHostShort( unsigned short us_in )
{
return ntohs( us_in );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *s -
// *sadr -
// Output : bool NET_StringToSockaddr
//-----------------------------------------------------------------------------
bool NET_StringToSockaddr( const char *s, struct sockaddr *sadr )
{
char *colon;
char copy[128];
Q_memset (sadr, 0, sizeof(*sadr));
((struct sockaddr_in *)sadr)->sin_family = AF_INET;
((struct sockaddr_in *)sadr)->sin_port = 0;
Q_strncpy (copy, s, sizeof( copy ) );
// strip off a trailing :port if present
for (colon = copy ; *colon ; colon++)
{
if (*colon == ':')
{
*colon = 0;
((struct sockaddr_in *)sadr)->sin_port = NET_HostToNetShort((short)atoi(colon+1));
}
}
if (copy[0] >= '0' && copy[0] <= '9' && Q_strstr( copy, "." ) )
{
*(int *)&((struct sockaddr_in *)sadr)->sin_addr = inet_addr(copy);
}
else
{
struct hostent *h;
if ( (h = gethostbyname(copy)) == NULL )
return false;
*(int *)&((struct sockaddr_in *)sadr)->sin_addr = *(int *)h->h_addr_list[0];
}
return true;
}
/*
=============
NET_StringToAdr
localhost
idnewt
idnewt:28000
192.246.40.70
192.246.40.70:28000
=============
*/
bool NET_StringToAdr ( const char *s, netadr_t *a)
{
struct sockaddr saddr;
char address[128];
Q_strncpy( address, s, sizeof(address) );
if ( !Q_strncmp( address, "localhost", 10 ) || !Q_strncmp( address, "localhost:", 10 ) )
{
// subsitute 'localhost' with '127.0.0.1", both have 9 chars
// this way we can resolve 'localhost' without DNS and still keep the port
Q_memcpy( address, "127.0.0.1", 9 );
}
if ( !NET_StringToSockaddr (address, &saddr) )
return false;
a->SetFromSockadr( &saddr );
return true;
}
void NET_GetLocalAddress (void)
{
net_local_adr.Clear();
char buff[512];
gethostname( buff, sizeof(buff) ); // get own IP address
buff[sizeof(buff)-1] = 0; // Ensure that it doesn't overrun the buffer
NET_StringToAdr (buff, &net_local_adr);
}
#define MAX_STATUS_STRING_LENGTH 1024
#define MAX_INPUT_FROM_CHILD 2048
class CConnectedNetConsoleData
{
public:
int m_nCharsInCommandBuffer;
char m_pszInputCommandBuffer[MAX_INPUT_FROM_CHILD];
bool m_bAuthorized; // for password protection
CConnectedNetConsoleData( void )
{
m_nCharsInCommandBuffer = 0;
m_bAuthorized = false;
}
};
class CParentProcessNetConsoleMgr : public ISocketCreatorListener
{
public:
CSocketCreator m_Socket;
netadr_t m_Address;
char m_pPassword[256]; // if set
bool m_bPasswordProtected;
bool m_bActive;
bool ShouldAcceptSocket( SocketHandle_t hSocket, const netadr_t &netAdr )
{
return true;
}
void OnSocketAccepted( SocketHandle_t hSocket, const netadr_t &netAdr, void** ppData )
{
CConnectedNetConsoleData *pData = new CConnectedNetConsoleData;
if ( ! m_bPasswordProtected )
pData->m_bAuthorized = true; // no password, auto-auth
*ppData = pData;
}
void OnSocketClosed( SocketHandle_t hSocket, const netadr_t &netAdr, void* pData )
{
if ( pData )
free( pData );
}
void RunFrame( void );
CParentProcessNetConsoleMgr( void );
void HandleInputChars( char const *pIn, int recvLen, CConnectedNetConsoleData *pData, int idx );
void SendString( char const *pString, int idx = -1 ); // send a string to all sockets or just one
void Execute( CConnectedNetConsoleData *pData, int idx );
};
void CParentProcessNetConsoleMgr::SendString( char const *pString, int nidx )
{
m_Socket.RunFrame();
int nCount = m_Socket.GetAcceptedSocketCount();
if ( nCount )
{
// lets add the lf to any cr's
char *pTmp = (char * ) stackalloc( strlen( pString ) * 2 + 1 );
char *oString = pTmp;
char const *pIn = pString;
while ( *pIn )
{
if ( *pIn == '\n' )
*( oString++ ) = '\r';
*( oString++ ) = *( pIn++ );
}
*( oString++ ) = 0;
for ( int i = 0; i < nCount; i++ )
{
if ( ( nidx == -1 ) || ( i == nidx ) )
{
SocketHandle_t hSocket = m_Socket.GetAcceptedSocketHandle( i );
//const netadr_t& socketAdr = m_Socket.GetAcceptedSocketAddress( i );
CConnectedNetConsoleData *pData = ( CConnectedNetConsoleData * ) m_Socket.GetAcceptedSocketData( i );
if ( pData->m_bAuthorized ) // no output to un-authed net consoles
{
send( hSocket, pTmp, oString - pTmp - 1, MSG_NOSIGNAL );
}
}
}
}
}
CParentProcessNetConsoleMgr::CParentProcessNetConsoleMgr( void ) : m_Socket( this )
{
m_bActive = false;
m_bPasswordProtected = false;
int nPassword = CommandLine()->FindParm( "-netconpassword" );
if ( nPassword )
{
char const *pPassword = CommandLine()->GetParm( nPassword + 1 );
V_strncpy( m_pPassword, pPassword, sizeof( m_pPassword ) );
m_bPasswordProtected = true;
}
int nPort = CommandLine()->FindParm( "-netconport" );
if ( nPort )
{
NET_GetLocalAddress();
char const *pPortNum = CommandLine()->GetParm( nPort + 1 );
char newBuf[256];
V_strncpy( newBuf, pPortNum, sizeof( newBuf ) );
char *pReplace = V_strstr( newBuf, "##" );
if ( pReplace )
{
pReplace[0] = '0';
pReplace[1] = '0';
}
m_Address = net_local_adr;
int nPortNumber = EvaluateExpression( newBuf, -1 );
if ( nPortNumber > 0 )
{
m_Address.SetPort( nPortNumber );
m_bActive = true;
m_Socket.CreateListenSocket( m_Address, true );
}
}
}
void CParentProcessNetConsoleMgr::RunFrame( void )
{
// check for incoming data
if (! m_bActive )
return;
m_Socket.RunFrame();
int nCount = m_Socket.GetAcceptedSocketCount();
for ( int i = nCount - 1; i >= 0; i-- )
{
SocketHandle_t hSocket = m_Socket.GetAcceptedSocketHandle( i );
// const netadr_t& socketAdr = m_Socket.GetAcceptedSocketAddress( i );
CConnectedNetConsoleData *pData = ( CConnectedNetConsoleData * ) m_Socket.GetAcceptedSocketData( i );
char ch;
int pendingLen = recv( hSocket, &ch, sizeof(ch), MSG_PEEK );
if ( pendingLen == -1 && SocketWouldBlock() )
continue;
if ( pendingLen <= 0 ) // eof or error
{
m_Socket.CloseAcceptedSocket( i );
continue;
}
// find out how much we have to read
unsigned long readLen;
ioctlsocket( hSocket, FIONREAD, &readLen );
while( readLen > 0 )
{
char recvBuf[256];
int recvLen = recv( hSocket, recvBuf , MIN( sizeof( recvBuf ) , readLen ), 0 );
if ( recvLen == 0 ) // socket was closed
{
m_Socket.CloseAcceptedSocket( i );
break;
}
if ( recvLen < 0 && !SocketWouldBlock() )
{
break;
}
readLen -= recvLen;
// now, lets write what we've got into the command buffer
HandleInputChars( recvBuf, recvLen, pData, i );
}
}
}
void CParentProcessNetConsoleMgr::HandleInputChars( char const *pIn, int recvLen, CConnectedNetConsoleData *pData, int idx )
{
while( recvLen )
{
switch( *pIn )
{
case '\r':
case '\n':
{
if ( pData->m_nCharsInCommandBuffer )
{
pData->m_pszInputCommandBuffer[pData->m_nCharsInCommandBuffer] = 0;
Execute( pData, idx );
}
pData->m_nCharsInCommandBuffer = 0;
break;
}
default:
{
if ( pData->m_nCharsInCommandBuffer < MAX_INPUT_FROM_CHILD - 1 )
pData->m_pszInputCommandBuffer[pData->m_nCharsInCommandBuffer++] = *pIn;
break;
}
}
pIn++;
recvLen--;
}
}
struct CServerInstance
{
pid_t m_nPid;
int m_nSocketToChild; // "our" side of the socket connection
int m_nNumCharsInInputBuffer;
int m_nNumPlayers;
char m_pszStatus[MAX_STATUS_STRING_LENGTH];
char m_pszMapName[MAX_PATH];
char m_pszInputBuffer[MAX_INPUT_FROM_CHILD];
bool m_bRunning;
void ClearInputBuffer( void )
{
m_nNumCharsInInputBuffer = 0;
}
void ResetStatus( void )
{
m_pszMapName[0] = 0;
m_nNumPlayers = 0;
}
CServerInstance( void )
{
m_pszStatus[0] = 0; // clear status string
m_bRunning = false;
m_nSocketToChild = -1;
ClearInputBuffer();
ResetStatus();
}
void HandleSocketInput( void );
void ProcessInputFromChild( void );
};
#define MAX_CHILD_PROCESSSES 100
CParentProcessNetConsoleMgr *g_pParentProcessNetConsole;
int g_nNumChildInstances;
CServerInstance *g_pChildProcesses;
static bool s_bQuit = false;
static bool s_bDelayedQuit = false;
typedef void (*CMDFN)( char const *pArgs, int nIdx );
struct CommandDescriptor
{
char const *m_pCmdName;
CMDFN m_pCmdFn;
char const *m_pCmdHelp;
};
static char *va( char *format, ... )
{
va_list argptr;
static char string[8][512];
static int curstring = 0;
curstring = ( curstring + 1 ) % 8;
va_start (argptr, format);
Q_vsnprintf( string[curstring], sizeof( string[curstring] ), format, argptr );
va_end (argptr);
return string[curstring];
}
static void s_DoStatusCmd( char const *pArgs, int nConsoleIdx )
{
// print status
g_pParentProcessNetConsole->SendString( "#status\n", nConsoleIdx );
for( int i = 0; i < g_nNumChildInstances; i++ )
{
CServerInstance *pChild = g_pChildProcesses + i;
g_pParentProcessNetConsole->SendString( va( "child %d\n", i ), nConsoleIdx );
if ( pChild && ( pChild->m_nSocketToChild != -1 ) )
{
g_pParentProcessNetConsole->SendString( va( " pid : %d\n", i, pChild->m_nPid ), nConsoleIdx );
g_pParentProcessNetConsole->SendString( va( " map : %s\n", pChild->m_pszMapName ), nConsoleIdx );
g_pParentProcessNetConsole->SendString( va( " numplayers : %d\n", pChild->m_nNumPlayers ), nConsoleIdx );
}
}
g_pParentProcessNetConsole->SendString( "#end\n", nConsoleIdx );
}
static void s_DoQuit( char const *pArgs, int nConsoleIdx )
{
g_pParentProcessNetConsole->SendString( "Killing all children and exiting\n", nConsoleIdx );
for( int i = 0; i < g_nNumChildInstances; i++ )
{
CServerInstance *pChild = g_pChildProcesses + i;
if ( pChild && ( pChild->m_nSocketToChild != -1 ) )
{
g_pParentProcessNetConsole->SendString( va( "killing child %d\n", i ), nConsoleIdx );
kill( pChild->m_nPid, SIGKILL );
}
}
s_bQuit = true;
}
static void s_DoBroadCastCmd( char const *pArgs, int nConsoleIdx )
{
if ( ! pArgs )
{
g_pParentProcessNetConsole->SendString( "Format of command is \"broadcast <concommand>\"\n" );
}
else
{
for( int i = 0; i < g_nNumChildInstances; i++ )
{
CServerInstance *pChild = g_pChildProcesses + i;
if ( pChild && ( pChild->m_nSocketToChild != -1 ) )
{
send( pChild->m_nSocketToChild, pArgs, strlen( pArgs ), MSG_NOSIGNAL );
send( pChild->m_nSocketToChild, "\n", 1, MSG_NOSIGNAL );
}
}
}
}
static void s_DoShutdown( char const *pArgs, int nConsoleIdx )
{
s_bDelayedQuit = ! s_bDelayedQuit;
if ( s_bDelayedQuit )
{
g_pParentProcessNetConsole->SendString( "Server will shutdown when all games are finished and children have exited.\n" );
}
else
{
g_pParentProcessNetConsole->SendString( "Server shutdown cancelled.\n" );
}
for( int i = 0; i < g_nNumChildInstances; i++ )
{
CServerInstance *pChild = g_pChildProcesses + i;
if ( pChild && ( pChild->m_nSocketToChild != -1 ) )
{
if ( pChild->m_nNumPlayers == 0 )
{
kill( pChild->m_nPid, SIGKILL );
}
}
}
}
static void s_DoFind( char const *pArgs, int nConsoleIdx );
static CommandDescriptor s_CmdTable[]={
{ "status", s_DoStatusCmd, "List the status of all subprocesses." },
{ "broadcast", s_DoBroadCastCmd, "Send a command to all subprocesses." },
{ "find", s_DoFind, "find commands containing a string." },
{ "shutdown", s_DoShutdown, "Tell the server shutdown once all players have left. This is a toggle." },
{ "quit", s_DoQuit, "immediately shut down the server and all its child processes." },
};
static void s_DoFind( char const *pArgs, int nConsoleIdx )
{
for( int i = 0; i < ARRAYSIZE( s_CmdTable ); i++ )
{
if ( ( pArgs[0] == 0 ) || ( V_stristr( s_CmdTable[i].m_pCmdName, pArgs ) ) )
{
g_pParentProcessNetConsole->SendString( va( "%s:\t%s\n", s_CmdTable[i].m_pCmdName, s_CmdTable[i].m_pCmdHelp) , nConsoleIdx );
}
}
}
void CParentProcessNetConsoleMgr::Execute( CConnectedNetConsoleData *pData, int idx )
{
if ( memcmp( pData->m_pszInputCommandBuffer, "PASS ", 5 ) == 0 )
{
if ( V_strcmp( pData->m_pszInputCommandBuffer + 5, m_pPassword ) == 0 )
{
pData->m_bAuthorized = true;
}
else
{
// bad password
Warning( "Bad password attempt from net console\n" );
pData->m_bAuthorized = false;
}
}
else
{
if ( pData->m_bAuthorized )
{
char const *pCmd = pData->m_pszInputCommandBuffer;
pCmd += strspn( pCmd, " \t" );
char const *pArgs = strchr( pCmd, ' ' );
int nCmdLen;
if ( pArgs )
{
nCmdLen = pArgs - pCmd;
pArgs += strspn( pArgs, " \t" ); // skip to first char of first word
}
else
{
nCmdLen = strlen( pCmd );
pArgs = pCmd + strlen( pCmd ); // point at trailing 0 bytes
}
for( int i = 0; i < ARRAYSIZE( s_CmdTable ); i++ )
{
char const *pTblCmd = s_CmdTable[i].m_pCmdName;
if ( ( strlen( pTblCmd ) == nCmdLen ) &&
( memcmp( pTblCmd, pCmd, nCmdLen ) == 0 ) )
{
// found it
( *s_CmdTable[i].m_pCmdFn )( pArgs, idx );
break;
}
}
}
else
{
SendString( "This server is password protected. Enter PASS <passwd> for access\n", idx );
}
}
}
static void HandleDeadChildProcesses( void )
{
for(;;)
{
int nStatus;
pid_t nWait = waitpid( -1, &nStatus, WNOHANG );
if ( nWait > 0 )
{
// find the process that exited
CServerInstance *pFound = NULL;
int nFound = -1;
for( int i = 0; i < g_nNumChildInstances; i++ )
{
if ( g_pChildProcesses[i].m_nPid == nWait )
{
pFound = g_pChildProcesses + i;
nFound = i;
break;
}
}
if ( ! pFound )
{
Warning( "unknown child process %d exited?\n", nWait );
}
else
{
if ( WIFEXITED( nStatus ) )
{
Msg( "Child %d exited with status %d\n", nFound, WEXITSTATUS( nStatus ) );
}
if ( WIFSIGNALED( nStatus ) )
{
Msg( "Child %d aborted with signal %d\n", nFound, WTERMSIG( nStatus ) );
}
if ( WCOREDUMP( nStatus ) )
{
Msg( "Child wrote a core dump\n");
}
pFound->m_bRunning = false;
if ( pFound->m_nSocketToChild != -1 )
{
close( pFound->m_nSocketToChild );
pFound->m_nSocketToChild = -1;
}
}
}
else
{
break; // no dead children
}
}
}
#define MAX_ACTIVE_PARENT_NETCONSOLE_SOCKETS 20
static void ServiceChildProcesses( void )
{
// for any children that aren't running (or not running yet), start them
pollfd pollFds[MAX_CHILD_PROCESSSES + 1 + MAX_ACTIVE_PARENT_NETCONSOLE_SOCKETS ];
int nPoll = 0;
int nNumRunning = 0;
for( int i = 0; i < g_nNumChildInstances; i++ )
{
if ( g_pChildProcesses[i].m_bRunning == false )
{
if (! s_bDelayedQuit )
{
int nSockets[2];
int nRslt = socketpair( AF_UNIX, SOCK_STREAM, 0, nSockets );
if ( nRslt != 0 )
{
Error( "socket pair returned error errno = %d\n", errno );
}
pid_t nChild = fork();
if ( nChild == 0 ) // are we the forked child?
{
//ResetBaseTime(); // start plat_float time over at 0 for precision
PerformCommandLineSubstitutions( i + 1 );
close( nSockets[1] );
engine->SetSubProcessID( i + 1, nSockets[0] );
g_nSubProcessId = i + 1;
RunServer( true );
syscall( SYS_exit, 0 ); // we are not going to perform a normal c++ exit. We _dont_ want to run destructors, etc.
}
else
{
g_pChildProcesses[i].m_nPid = nChild;
g_pChildProcesses[i].m_pszStatus[0] = 0;
g_pChildProcesses[i].m_bRunning = true;
close( nSockets[0] );
g_pChildProcesses[i].m_nSocketToChild = nSockets[1];
}
}
}
else
{
nNumRunning++;
}
if ( g_pChildProcesses[i].m_nSocketToChild != -1 )
{
pollFds[nPoll].fd = g_pChildProcesses[i].m_nSocketToChild;
pollFds[nPoll].events = POLLIN | POLLERR | POLLHUP;
pollFds[nPoll].revents = 0;
nPoll++;
}
}
if ( s_bDelayedQuit && ( nNumRunning == 0 ) )
{
_exit( 0 );
}
// now, wait for activity on any of our sockets or stdin
// pollFds[nPoll].fd = STDIN_FILENO;
// pollFds[nPoll].events = POLLIN;
// pollFds[nPoll].revents = 0;
// nPoll++;
if ( g_pParentProcessNetConsole && ( g_pParentProcessNetConsole->m_bActive ) )
{
pollFds[nPoll].fd = g_pParentProcessNetConsole->m_Socket.m_hListenSocket;
pollFds[nPoll].events = POLLIN;
pollFds[nPoll].revents = 0;
nPoll++;
int nCount = g_pParentProcessNetConsole->m_Socket.GetAcceptedSocketCount();
for( int i = 0; ( i < nCount ) && ( nPoll < ARRAYSIZE( pollFds ) ); i++ )
{
SocketHandle_t hSocket = g_pParentProcessNetConsole->m_Socket.GetAcceptedSocketHandle( i );
pollFds[nPoll].fd = hSocket;
pollFds[nPoll].events = POLLIN;
pollFds[nPoll].revents = 0;
nPoll++;
}
}
int nPollResult = poll( pollFds, nPoll, 10 * 1000 ); // wait up to 10 seconds. Could wait forever, really
// check for activity on the sockets from our children
int np = 0;
for( int i = 0; i < g_nNumChildInstances; i++ )
{
if ( g_pChildProcesses[i].m_nSocketToChild != -1 )
{
if ( pollFds[np].revents & POLLIN ) // data ready to read?
{
g_pChildProcesses[i].HandleSocketInput();
}
np++;
}
}
// see if any children have exited
HandleDeadChildProcesses();
g_pParentProcessNetConsole->RunFrame();
}
void RunServerSubProcesses( int nNumChildren )
{
g_nNumChildInstances = nNumChildren;
g_pChildProcesses = new CServerInstance[g_nNumChildInstances];
g_pParentProcessNetConsole = new CParentProcessNetConsoleMgr;
while( ! s_bQuit )
{
ServiceChildProcesses();
}
_exit( 0 );
}
static bool DecodeParam( char const *pParamName, char const *pInput, char const **pOutPtr )
{
// if the left of the string matches pParamName, return the right of the string else return null
int nPLen = strlen( pParamName );
if ( memcmp( pParamName, pInput, nPLen ) == 0 )
{
*pOutPtr= pInput + nPLen;
}
else
{
*pOutPtr = NULL;
}
return ( *pOutPtr );
}
void CServerInstance::ProcessInputFromChild( void )
{
if ( m_pszInputBuffer[0] == '#' ) // spew?
{
puts( m_pszInputBuffer );
}
else
{
char *pSpace = strchr( m_pszInputBuffer, ' ' );
if ( pSpace )
{
*( pSpace++ ) = 0;
pSpace += strspn( pSpace, " \t" );
}
else
{
pSpace = m_pszInputBuffer + strlen( m_pszInputBuffer );
}
if ( !strcmp( m_pszInputBuffer, "status" ) )
{
CUtlStringList statusRecords;
V_SplitString( pSpace, ";", statusRecords );
for( int i = 0; i < statusRecords.Count(); i++ )
{
char const *pRecord = statusRecords[i];
char const *pParm;
if ( DecodeParam( "map=", pRecord, &pParm ) )
{
V_strncpy( m_pszMapName, pParm, sizeof( m_pszMapName ) );
}
else if ( DecodeParam( "players=", pRecord, &pParm ) )
{
m_nNumPlayers = atoi( pParm );
}
}
}
else
{
Warning("got unknown cmd %s args %s\n", m_pszInputBuffer, pSpace );
}
}
}
void CServerInstance::HandleSocketInput( void )
{
char *pDest = m_pszInputBuffer + m_nNumCharsInInputBuffer;
int nRead = recv( m_nSocketToChild, pDest, sizeof( m_pszInputBuffer ) - m_nNumCharsInInputBuffer, MSG_DONTWAIT );
if ( nRead > 0 )
{
m_nNumCharsInInputBuffer += nRead;
if ( m_pszInputBuffer[m_nNumCharsInInputBuffer - 1] == 0 )
{
ProcessInputFromChild();
m_nNumCharsInInputBuffer = 0;
}
if ( m_nNumCharsInInputBuffer == MAX_INPUT_FROM_CHILD )
m_nNumCharsInInputBuffer = 0;
}
}
#endif //linux