764 lines
20 KiB
C++
764 lines
20 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Xbox console link
|
|
//
|
|
//=====================================================================================//
|
|
|
|
#include "xbox/xbox_console.h"
|
|
#include "xbox/xbox_vxconsole.h"
|
|
#include "tier0/threadtools.h"
|
|
#include "tier0/tslist.h"
|
|
#include "tier0/ICommandLine.h"
|
|
#include "tier0/memdbgon.h"
|
|
|
|
// all redirecting funneled here, stop redirecting in this module only
|
|
#undef OutputDebugStringA
|
|
|
|
#pragma comment( lib, "xbdm.lib" )
|
|
|
|
struct DebugString_t
|
|
{
|
|
unsigned int color;
|
|
char *pString;
|
|
};
|
|
|
|
#define XBX_DBGCOMMANDPREFIX "XCMD"
|
|
#define XBX_DBGRESPONSEPREFIX "XACK"
|
|
#define XBX_DBGPRINTPREFIX "XPRT"
|
|
#define XBX_DBGCOLORPREFIX "XCLR"
|
|
|
|
#define XBX_MAX_RCMDLENGTH 256
|
|
#define XBX_MAX_MESSAGE 2048
|
|
|
|
CThreadFastMutex g_xbx_dbgChannelMutex;
|
|
CThreadFastMutex g_xbx_dbgCommandHandlerMutex;
|
|
static char g_xbx_dbgRemoteBuf[XBX_MAX_RCMDLENGTH];
|
|
static HANDLE g_xbx_dbgValidEvent;
|
|
static HANDLE g_xbx_dbgCmdCompleteEvent;
|
|
bool g_xbx_bUseVXConsoleOutput = true;
|
|
bool g_xbx_bDoSyncOutput;
|
|
static ThreadHandle_t g_xbx_hDebugThread;
|
|
CTSQueue<DebugString_t> g_xbx_DebugStringQueue;
|
|
extern CInterlockedInt g_xbx_numProfileCounters;
|
|
extern unsigned int g_xbx_profileCounters[];
|
|
extern char g_xbx_profileName[];
|
|
int g_xbx_freeMemory;
|
|
|
|
_inline bool XBX_NoXBDM() { return false; }
|
|
|
|
static CXboxConsole XboxConsole;
|
|
|
|
DLL_EXPORT IXboxConsole *GetConsoleInterface()
|
|
{
|
|
return &XboxConsole;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Low level string output.
|
|
// Input string should be stack based, can get clobbered.
|
|
//-----------------------------------------------------------------------------
|
|
static void OutputStringToDevice( unsigned int color, char *pString, bool bRemoteValid )
|
|
{
|
|
if ( !bRemoteValid )
|
|
{
|
|
// local debug only
|
|
OutputDebugStringA( pString );
|
|
return;
|
|
}
|
|
|
|
// remote debug valid
|
|
// non pure colors don't translate well - find closest pure hue
|
|
unsigned int bestColor = color;
|
|
int r = ( bestColor & 0xFF );
|
|
int g = ( bestColor >> 8 ) & 0xFF;
|
|
int b = ( bestColor >> 16 ) & 0xFF;
|
|
if ( ( r && r != 255 ) || ( g && g != 255 ) || ( b && b != 255 ) )
|
|
{
|
|
int r0, g0, b0;
|
|
unsigned int minDist = 0xFFFFFFFF;
|
|
for ( int i=0; i<8; i++ )
|
|
{
|
|
r0 = g0 = b0 = 0;
|
|
if ( i&4 )
|
|
r0 = 255;
|
|
if ( i&2 )
|
|
g0 = 255;
|
|
if ( i&1 )
|
|
b0 = 255;
|
|
unsigned int d = ( r-r0 )*( r-r0 ) + ( g-g0 )*( g-g0 ) + ( b-b0 )*( b-b0 );
|
|
if ( minDist > d )
|
|
{
|
|
minDist = d;
|
|
bestColor = XMAKECOLOR( r0, g0, b0 );
|
|
}
|
|
}
|
|
}
|
|
|
|
// create color string
|
|
char colorString[16];
|
|
sprintf( colorString, XBX_DBGCOLORPREFIX "[%8.8x]", bestColor );
|
|
|
|
// chunk line out, for each cr
|
|
char strBuffer[XBX_MAX_RCMDLENGTH];
|
|
char *pStart = pString;
|
|
char *pEnd = pStart + strlen( pStart );
|
|
char *pNext;
|
|
while ( pStart < pEnd )
|
|
{
|
|
pNext = strchr( pStart, '\n' );
|
|
if ( !pNext )
|
|
pNext = pEnd;
|
|
else
|
|
*pNext = '\0';
|
|
|
|
int length = _snprintf( strBuffer, XBX_MAX_RCMDLENGTH, "%s!%s%s", XBX_DBGPRINTPREFIX, colorString, pStart );
|
|
if ( length == -1 )
|
|
{
|
|
strBuffer[sizeof( strBuffer )-1] = '\0';
|
|
}
|
|
// Send the string
|
|
DmSendNotificationString( strBuffer );
|
|
|
|
// advance past cr
|
|
pStart = pNext+1;
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// XBX_IsConsoleConnected
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
bool CXboxConsole::IsConsoleConnected()
|
|
{
|
|
bool bConnected;
|
|
|
|
if ( g_xbx_dbgValidEvent == NULL )
|
|
{
|
|
// init was never called
|
|
return false;
|
|
}
|
|
|
|
AUTO_LOCK_FM( g_xbx_dbgChannelMutex );
|
|
|
|
bConnected = ( WaitForSingleObject( g_xbx_dbgValidEvent, 0 ) == WAIT_OBJECT_0 );
|
|
|
|
return bConnected;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Output string to listening console. Queues output for slave thread.
|
|
// Needs to be lightweight.
|
|
//-----------------------------------------------------------------------------
|
|
void CXboxConsole::DebugString( unsigned int color, const char* pFormat, ... )
|
|
{
|
|
if ( XBX_NoXBDM() )
|
|
return;
|
|
|
|
va_list args;
|
|
char szStringBuffer[XBX_MAX_MESSAGE];
|
|
int length;
|
|
|
|
// resolve string
|
|
va_start( args, pFormat );
|
|
length = _vsnprintf( szStringBuffer, sizeof( szStringBuffer ), pFormat, args );
|
|
if ( length == -1 )
|
|
{
|
|
szStringBuffer[sizeof( szStringBuffer ) - 1] = '\0';
|
|
}
|
|
va_end( args );
|
|
|
|
if ( !g_xbx_bDoSyncOutput )
|
|
{
|
|
// queue string for delayed output
|
|
DebugString_t debugString;
|
|
debugString.color = color;
|
|
debugString.pString = strdup( szStringBuffer );
|
|
g_xbx_DebugStringQueue.PushItem( debugString );
|
|
}
|
|
else
|
|
{
|
|
bool bRemoteValid = g_xbx_bUseVXConsoleOutput && XBX_IsConsoleConnected();
|
|
OutputStringToDevice( color, szStringBuffer, bRemoteValid );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Waits for debug queue to drain.
|
|
//-----------------------------------------------------------------------------
|
|
void CXboxConsole::FlushDebugOutput()
|
|
{
|
|
while ( g_xbx_DebugStringQueue.Count() != 0 )
|
|
{
|
|
Sleep( 1 );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_strlen
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
int _xdbg_strlen( const CHAR* str )
|
|
{
|
|
const CHAR* strEnd = str;
|
|
|
|
while( *strEnd )
|
|
strEnd++;
|
|
|
|
return strEnd - str;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_tolower
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
inline CHAR _xdbg_tolower( CHAR ch )
|
|
{
|
|
if( ch >= 'A' && ch <= 'Z' )
|
|
return ch - ( 'A' - 'a' );
|
|
else
|
|
return ch;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_strnicmp
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
BOOL _xdbg_strnicmp( const CHAR* str1, const CHAR* str2, int n )
|
|
{
|
|
while ( ( _xdbg_tolower( *str1 ) == _xdbg_tolower( *str2 ) ) && *str1 && n > 0 )
|
|
{
|
|
--n;
|
|
++str1;
|
|
++str2;
|
|
}
|
|
return ( !n || _xdbg_tolower( *str1 ) == _xdbg_tolower( *str2 ) );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_strcpy
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
VOID _xdbg_strcpy( CHAR* strDest, const CHAR* strSrc )
|
|
{
|
|
while ( ( *strDest++ = *strSrc++ ) != 0 );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_strcpyn
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
VOID _xdbg_strcpyn( CHAR* strDest, const CHAR* strSrc, int numChars )
|
|
{
|
|
while ( numChars>0 && ( *strDest++ = *strSrc++ ) != 0 )
|
|
numChars--;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_gettoken
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
void _xdbg_gettoken( CHAR** tokenStream, CHAR* token, int tokenSize )
|
|
{
|
|
int c;
|
|
int len;
|
|
CHAR* data;
|
|
|
|
len = 0;
|
|
|
|
// skip prefix whitespace
|
|
data = *tokenStream;
|
|
while ( ( c = *data ) <= ' ' )
|
|
{
|
|
if ( !c )
|
|
goto cleanUp;
|
|
data++;
|
|
}
|
|
|
|
// parse a token
|
|
do
|
|
{
|
|
if ( len < tokenSize )
|
|
token[len++] = c;
|
|
|
|
data++;
|
|
c = *data;
|
|
} while ( c > ' ' );
|
|
|
|
if ( len >= tokenSize )
|
|
len = 0;
|
|
|
|
cleanUp:
|
|
token[len] = '\0';
|
|
*tokenStream = data;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_tokenize
|
|
//
|
|
// Critical section safe.
|
|
//-----------------------------------------------------------------------------
|
|
int _xdbg_tokenize( CHAR* tokenStream, CHAR** tokens, int maxTokens )
|
|
{
|
|
char token[64];
|
|
|
|
// tokenize stream into seperate tokens
|
|
int numTokens = 0;
|
|
while ( 1 )
|
|
{
|
|
tokens[numTokens++] = tokenStream;
|
|
if ( numTokens >= maxTokens )
|
|
break;
|
|
|
|
_xdbg_gettoken( &tokenStream, token, sizeof( token ) );
|
|
if ( !tokenStream[0] || !token[0] )
|
|
break;
|
|
|
|
*tokenStream = '\0';
|
|
tokenStream++;
|
|
}
|
|
return ( numTokens );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _xdbg_findtoken
|
|
//
|
|
// Critical section safe. Returns -1 if not found
|
|
//-----------------------------------------------------------------------------
|
|
int _xdbg_findtoken( CHAR** tokens, int numTokens, CHAR* token )
|
|
{
|
|
int i;
|
|
int len;
|
|
|
|
len = _xdbg_strlen( token );
|
|
for ( i=0; i<numTokens; i++ )
|
|
{
|
|
if ( _xdbg_strnicmp( tokens[i], token, len ) )
|
|
return i;
|
|
}
|
|
|
|
// not found
|
|
return -1;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// _DebugCommandHandler
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT __stdcall _DebugCommandHandler( const CHAR* strCommand, CHAR* strResponse, DWORD dwResponseLen, PDM_CMDCONT pdmcc )
|
|
{
|
|
CHAR buff[256];
|
|
CHAR* args[8];
|
|
int numArgs;
|
|
|
|
AUTO_LOCK_FM( g_xbx_dbgCommandHandlerMutex );
|
|
|
|
// skip over the command prefix and the exclamation mark
|
|
strCommand += _xdbg_strlen( XBX_DBGCOMMANDPREFIX ) + 1;
|
|
|
|
if ( strCommand[0] == '\0' )
|
|
{
|
|
// just a ping
|
|
goto cleanUp;
|
|
}
|
|
|
|
// get command and optional arguments
|
|
_xdbg_strcpyn( buff, strCommand, sizeof( buff ) );
|
|
numArgs = _xdbg_tokenize( buff, args, sizeof( args )/sizeof( CHAR* ) );
|
|
|
|
if ( _xdbg_strnicmp( args[0], "__connect__", 11 ) )
|
|
{
|
|
if ( numArgs > 1 && atoi( args[1] ) == VXCONSOLE_PROTOCOL_VERSION )
|
|
{
|
|
// initial connect - respond that we're connected
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Connected To Application.", dwResponseLen );
|
|
SetEvent( g_xbx_dbgValidEvent );
|
|
|
|
// notify convar system to send its commands
|
|
// allows vxconsole to re-connect during game
|
|
_xdbg_strcpy( g_xbx_dbgRemoteBuf, "getcvars" );
|
|
XBX_QueueEvent( XEV_REMOTECMD, ( int )g_xbx_dbgRemoteBuf, 0, 0 );
|
|
}
|
|
else
|
|
{
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Rejecting Connection: Wrong Protocol Version.", dwResponseLen );
|
|
}
|
|
goto cleanUp;
|
|
}
|
|
|
|
if ( _xdbg_strnicmp( args[0], "__disconnect__", 14 ) )
|
|
{
|
|
// respond that we're disconnected
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Disconnected.", dwResponseLen );
|
|
ResetEvent( g_xbx_dbgValidEvent );
|
|
goto cleanUp;
|
|
}
|
|
|
|
if ( _xdbg_strnicmp( args[0], "__complete__", 12 ) )
|
|
{
|
|
// remote server has finished command - respond to acknowledge
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
|
|
|
|
// set the complete event - allows expected synchronous calling mechanism
|
|
SetEvent( g_xbx_dbgCmdCompleteEvent );
|
|
goto cleanUp;
|
|
}
|
|
|
|
if ( _xdbg_strnicmp( args[0], "__memory__", 10 ) )
|
|
{
|
|
// get a current stat of available memory
|
|
MEMORYSTATUS stat;
|
|
GlobalMemoryStatus( &stat );
|
|
g_xbx_freeMemory = stat.dwAvailPhys;
|
|
|
|
if ( _xdbg_findtoken( args, numArgs, "quiet" ) > 0 )
|
|
{
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
|
|
}
|
|
else
|
|
{
|
|
// 32 MB is reserved and fixed by OS, so not reporting
|
|
_snprintf( strResponse, dwResponseLen, XBX_DBGRESPONSEPREFIX "Available: %.2f MB, Used: %.2f MB, Free: %.2f MB",
|
|
stat.dwTotalPhys/( 1024.0f*1024.0f ) - 32.0f,
|
|
( stat.dwTotalPhys - stat.dwAvailPhys )/( 1024.0f*1024.0f ) - 32.0f,
|
|
stat.dwAvailPhys/( 1024.0f*1024.0f ) );
|
|
}
|
|
goto cleanUp;
|
|
}
|
|
|
|
if ( g_xbx_dbgRemoteBuf[0] )
|
|
{
|
|
// previous command still pending
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "Cannot execute: Previous command still pending", dwResponseLen );
|
|
}
|
|
else
|
|
{
|
|
// add the command to the event queue to be processed by main app
|
|
_xdbg_strcpy( g_xbx_dbgRemoteBuf, strCommand );
|
|
XBX_QueueEvent( XEV_REMOTECMD, ( int )g_xbx_dbgRemoteBuf, 0, 0 );
|
|
_xdbg_strcpyn( strResponse, XBX_DBGRESPONSEPREFIX "OK", dwResponseLen );
|
|
}
|
|
|
|
cleanUp:
|
|
return XBDM_NOERR;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// XBX_SendRemoteCommand
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
void CXboxConsole::SendRemoteCommand( const char *pCommand, bool async )
|
|
{
|
|
char cmdString[XBX_MAX_RCMDLENGTH];
|
|
|
|
if ( XBX_NoXBDM() || !IsConsoleConnected() )
|
|
return;
|
|
|
|
AUTO_LOCK_FM( g_xbx_dbgChannelMutex );
|
|
|
|
_snprintf( cmdString, sizeof( cmdString ), "%s!%s", XBX_DBGCOMMANDPREFIX, pCommand );
|
|
HRESULT hr = DmSendNotificationString( cmdString );
|
|
if ( FAILED( hr ) )
|
|
{
|
|
XBX_Error( "XBX_SendRemoteCommand: failed on %s", cmdString );
|
|
}
|
|
|
|
// wait for command completion
|
|
if ( !async )
|
|
{
|
|
DWORD timeout;
|
|
if ( !strnicmp( pCommand, "Assert()", 8 ) )
|
|
{
|
|
// the assert is waiting for user to make selection
|
|
timeout = INFINITE;
|
|
}
|
|
else
|
|
{
|
|
// no vxconsole operation should take this long
|
|
timeout = 15000;
|
|
}
|
|
|
|
if ( WaitForSingleObject( g_xbx_dbgCmdCompleteEvent, timeout ) == WAIT_TIMEOUT )
|
|
{
|
|
// we have no choice but to dump core
|
|
DmCrashDump( false );
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Handle delayed VXConsole transactions
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
static unsigned _DebugThreadFunc( void *pParam )
|
|
{
|
|
while ( 1 )
|
|
{
|
|
Sleep( 10 );
|
|
|
|
if ( !g_xbx_DebugStringQueue.Count() && !g_xbx_numProfileCounters && !g_xbx_freeMemory )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ( g_xbx_numProfileCounters )
|
|
{
|
|
// build and send asynchronously
|
|
char dbgCommand[XBX_MAX_RCMDLENGTH];
|
|
_snprintf( dbgCommand, sizeof( dbgCommand ), "SetProfileData() %s 0x%8.8x", g_xbx_profileName, g_xbx_profileCounters );
|
|
XBX_SendRemoteCommand( dbgCommand, true );
|
|
|
|
// mark as sent
|
|
g_xbx_numProfileCounters = 0;
|
|
}
|
|
|
|
if ( g_xbx_freeMemory )
|
|
{
|
|
// build and send asynchronously
|
|
char dbgCommand[XBX_MAX_RCMDLENGTH];
|
|
_snprintf( dbgCommand, sizeof( dbgCommand ), "FreeMemory() 0x%8.8x", g_xbx_freeMemory );
|
|
XBX_SendRemoteCommand( dbgCommand, true );
|
|
|
|
// mark as sent
|
|
g_xbx_freeMemory = 0;
|
|
}
|
|
|
|
bool bRemoteValid = g_xbx_bUseVXConsoleOutput && XBX_IsConsoleConnected();
|
|
while ( 1 )
|
|
{
|
|
DebugString_t debugString;
|
|
if ( !g_xbx_DebugStringQueue.PopItem( &debugString ) )
|
|
{
|
|
break;
|
|
}
|
|
|
|
OutputStringToDevice( debugString.color, debugString.pString, bRemoteValid );
|
|
free( debugString.pString );
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// XBX_InitConsoleMonitor
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
void CXboxConsole::InitConsoleMonitor( bool bWaitForConnect )
|
|
{
|
|
if ( XBX_NoXBDM() )
|
|
return;
|
|
|
|
// create our events
|
|
g_xbx_dbgValidEvent = CreateEvent( XBOX_DONTCARE, TRUE, FALSE, NULL );
|
|
g_xbx_dbgCmdCompleteEvent = CreateEvent( XBOX_DONTCARE, FALSE, FALSE, NULL );
|
|
|
|
// register our command handler with the debug monitor
|
|
HRESULT hr = DmRegisterCommandProcessor( XBX_DBGCOMMANDPREFIX, _DebugCommandHandler );
|
|
if ( FAILED( hr ) )
|
|
{
|
|
XBX_Error( "XBX_InitConsoleMonitor: failed to register command processor" );
|
|
}
|
|
|
|
// user can have output bypass slave thread
|
|
g_xbx_bDoSyncOutput = CommandLine()->FindParm( "-syncoutput" ) != 0;
|
|
|
|
// create a slave thread to do delayed VXConsole transactions
|
|
ThreadId_t threadID;
|
|
g_xbx_hDebugThread = CreateSimpleThread( _DebugThreadFunc, NULL, &threadID, 16*1024 );
|
|
ThreadSetDebugName( threadID, "DebugThread" );
|
|
ThreadSetAffinity( g_xbx_hDebugThread, XBOX_PROCESSOR_5 );
|
|
|
|
if ( bWaitForConnect )
|
|
{
|
|
XBX_DebugString( XBX_CLR_DEFAULT, "Waiting For VXConsole Connection...\n" );
|
|
WaitForSingleObject( g_xbx_dbgValidEvent, INFINITE );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Sends a disconnect signal to possibly attached VXConsole.
|
|
//-----------------------------------------------------------------------------
|
|
void CXboxConsole::DisconnectConsoleMonitor()
|
|
{
|
|
if ( XBX_NoXBDM() )
|
|
return;
|
|
|
|
// caller is trying to safely stop vxconsole traffic, disconnect must be synchronous
|
|
XBX_SendRemoteCommand( "Disconnect()", false );
|
|
}
|
|
|
|
bool CXboxConsole::GetXboxName( char *pName, unsigned *pLength )
|
|
{
|
|
return ( DmGetXboxName( pName, (DWORD *)pLength ) == XBDM_NOERR );
|
|
}
|
|
|
|
void CXboxConsole::CrashDump( bool b )
|
|
{
|
|
DmCrashDump(b);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Walk to a specific module and dump size info
|
|
//-----------------------------------------------------------------------------
|
|
int CXboxConsole::DumpModuleSize( const char *pName )
|
|
{
|
|
HRESULT error;
|
|
PDM_WALK_MODULES pWalkMod = NULL;
|
|
DMN_MODLOAD modLoad;
|
|
int size = 0;
|
|
|
|
// iterate and find match
|
|
do
|
|
{
|
|
error = DmWalkLoadedModules( &pWalkMod, &modLoad );
|
|
if ( XBDM_NOERR == error && !stricmp( modLoad.Name, pName ) )
|
|
{
|
|
Msg( "0x%8.8x, %5.2f MB, %s\n", modLoad.BaseAddress, modLoad.Size/( 1024.0f*1024.0f ), modLoad.Name );
|
|
size = modLoad.Size;
|
|
error = XBDM_ENDOFLIST;
|
|
}
|
|
}
|
|
while ( XBDM_NOERR == error );
|
|
DmCloseLoadedModules( pWalkMod );
|
|
|
|
if ( error != XBDM_ENDOFLIST )
|
|
{
|
|
Warning( "DmWalkLoadedModules() failed.\n" );
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// 360 spew sizes of dll modules
|
|
//-----------------------------------------------------------------------------
|
|
char const* HACK_stristr( char const* pStr, char const* pSearch ) // hack because moved code from above vstdlib
|
|
{
|
|
AssertValidStringPtr(pStr);
|
|
AssertValidStringPtr(pSearch);
|
|
|
|
if (!pStr || !pSearch)
|
|
return 0;
|
|
|
|
char const* pLetter = pStr;
|
|
|
|
// Check the entire string
|
|
while (*pLetter != 0)
|
|
{
|
|
// Skip over non-matches
|
|
if (tolower((unsigned char)*pLetter) == tolower((unsigned char)*pSearch))
|
|
{
|
|
// Check for match
|
|
char const* pMatch = pLetter + 1;
|
|
char const* pTest = pSearch + 1;
|
|
while (*pTest != 0)
|
|
{
|
|
// We've run off the end; don't bother.
|
|
if (*pMatch == 0)
|
|
return 0;
|
|
|
|
if (tolower((unsigned char)*pMatch) != tolower((unsigned char)*pTest))
|
|
break;
|
|
|
|
++pMatch;
|
|
++pTest;
|
|
}
|
|
|
|
// Found a match!
|
|
if (*pTest == 0)
|
|
return pLetter;
|
|
}
|
|
|
|
++pLetter;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void CXboxConsole::DumpDllInfo( const char *pBasePath )
|
|
{
|
|
// Directories containing dlls
|
|
static char *dllDirs[] =
|
|
{
|
|
"bin",
|
|
"hl2\\bin",
|
|
"tf\\bin",
|
|
"portal\\bin",
|
|
"episodic\\bin"
|
|
};
|
|
|
|
char binPath[MAX_PATH];
|
|
char dllPath[MAX_PATH];
|
|
char searchPath[MAX_PATH];
|
|
|
|
HMODULE hModule;
|
|
WIN32_FIND_DATA wfd;
|
|
HANDLE hFind;
|
|
|
|
Msg( "Dumping Module Sizes...\n" );
|
|
|
|
for ( int i = 0; i < ARRAYSIZE( dllDirs ); ++i )
|
|
{
|
|
int totalSize = 0;
|
|
|
|
_snprintf( binPath, sizeof( binPath ), "%s\\%s", pBasePath, dllDirs[i] );
|
|
_snprintf( searchPath, sizeof( binPath ), "%s\\*.dll", binPath );
|
|
|
|
// show the directory we're searching
|
|
Msg( "\nDirectory: %s\n\n", binPath );
|
|
|
|
// Start the find and check for failure.
|
|
hFind = FindFirstFile( searchPath, &wfd );
|
|
if ( INVALID_HANDLE_VALUE == hFind )
|
|
{
|
|
Warning( "No Files Found.\n" );
|
|
}
|
|
else
|
|
{
|
|
// Load and unload each dll individually.
|
|
do
|
|
{
|
|
if ( !HACK_stristr( wfd.cFileName, "_360.dll" ) )
|
|
{
|
|
// exclude explicit pc dlls
|
|
// FindFirstFile does not support a spec mask of *_360.dll on the Xbox HDD
|
|
continue;
|
|
}
|
|
|
|
_snprintf( dllPath, sizeof( dllPath ), "%s\\%s", binPath, wfd.cFileName );
|
|
hModule = LoadLibrary( dllPath );
|
|
if ( hModule )
|
|
{
|
|
totalSize += DumpModuleSize( wfd.cFileName );
|
|
FreeLibrary( hModule );
|
|
}
|
|
else
|
|
{
|
|
Warning( "Failed to load: %s\n", dllPath );
|
|
}
|
|
}
|
|
while( FindNextFile( hFind, &wfd ) );
|
|
|
|
FindClose( hFind );
|
|
|
|
Msg( "Total Size: %.2f MB\n", totalSize/( 1024.0f*1024.0f ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
void CXboxConsole::OutputDebugString( const char *p )
|
|
{
|
|
::OutputDebugStringA( p );
|
|
}
|
|
|
|
bool CXboxConsole::IsDebuggerPresent()
|
|
{
|
|
return ( DmIsDebuggerPresent() != 0 );
|
|
}
|