|
|
//----------------------------------------------------------------------------
//
// cppdbg.cpp
//
// C++-only debugging support.
//
// Copyright (C) Microsoft Corporation, 1997.
//
//----------------------------------------------------------------------------
#include "pch.cpp"
#pragma hdrstop
#if DBG
#include "cppdbg.hpp"
#ifdef _ALPHA_
// On Alpha va_list is a structure so it's not compatible with NULL.
static va_list NULLVA; #else
#define NULLVA NULL
#endif
static DebugModuleFlags g_FailureFlags[] = { DBG_DECLARE_MODFLAG(DBG_FAILURE, BREAK), DBG_DECLARE_MODFLAG(DBG_FAILURE, OUTPUT), DBG_DECLARE_MODFLAG(DBG_FAILURE, PROMPT), DBG_DECLARE_MODFLAG(DBG_FAILURE, FILENAME_ONLY), 0, NULL, };
static DebugModuleFlags g_OutputFlags[] = { DBG_DECLARE_MODFLAG(DBG_OUTPUT, SUPPRESS), DBG_DECLARE_MODFLAG(DBG_OUTPUT, ALL_MATCH), 0, NULL, };
static char *g_pFlagNames[] = { "AssertFlags", "HrFlags", "OutputFlags", "OutputMask", "UserFlags" };
//----------------------------------------------------------------------------
//
// DebugModule::DebugModule
//
//----------------------------------------------------------------------------
DebugModule::DebugModule(char *pModule, char *pPrefix, DebugModuleFlags *pOutputMasks, UINT uOutputMask, DebugModuleFlags *pUserFlags, UINT uUserFlags) { m_pModule = pModule; m_iModuleStartCol = strlen(m_pModule) + 2; m_pPrefix = pPrefix;
m_pModFlags[DBG_ASSERT_FLAGS] = g_FailureFlags; m_pModFlags[DBG_HR_FLAGS] = g_FailureFlags; m_pModFlags[DBG_OUTPUT_FLAGS] = g_OutputFlags; m_pModFlags[DBG_OUTPUT_MASK] = pOutputMasks; m_pModFlags[DBG_USER_FLAGS] = pUserFlags;
m_uFlags[DBG_ASSERT_FLAGS] = DBG_FAILURE_OUTPUT | DBG_FAILURE_BREAK | DBG_FAILURE_FILENAME_ONLY; m_uFlags[DBG_HR_FLAGS] = DBG_FAILURE_OUTPUT | DBG_FAILURE_FILENAME_ONLY; m_uFlags[DBG_OUTPUT_FLAGS] = 0; m_uFlags[DBG_OUTPUT_MASK] = uOutputMask; m_uFlags[DBG_USER_FLAGS] = uUserFlags; ReadReg(); }
//----------------------------------------------------------------------------
//
// DebugModule::OutVa
//
// Base debug output method.
//
//----------------------------------------------------------------------------
void DebugModule::OutVa(UINT uMask, char *pFmt, va_list Args) { if (m_uFlags[DBG_OUTPUT_FLAGS] & DBG_OUTPUT_SUPPRESS) { return; } if ((uMask & DBG_MASK_NO_PREFIX) == 0) { OutputDebugStringA(m_pModule); OutputDebugStringA(": "); }
char chMsg[1024];
_vsnprintf(chMsg, sizeof(chMsg), pFmt, Args); OutputDebugStringA(chMsg); }
//----------------------------------------------------------------------------
//
// DebugModule::Out
//
// Always-output debug output method.
//
//----------------------------------------------------------------------------
void DebugModule::Out(char *pFmt, ...) { va_list Args;
va_start(Args, pFmt); OutVa(0, pFmt, Args); va_end(Args); }
//----------------------------------------------------------------------------
//
// DebugModule::AssertFailedVa
//
// Handles assertion failure output and interface.
//
//----------------------------------------------------------------------------
void DebugModule::AssertFailedVa(char *pFmt, va_list Args, BOOL bNewLine) { if (m_uFlags[DBG_ASSERT_FLAGS] & DBG_FAILURE_OUTPUT) { if (OutPathFile("Assertion failed", m_uFlags[DBG_ASSERT_FLAGS])) { OutVa(DBG_MASK_NO_PREFIX, ":\n ", NULLVA); } else { OutVa(DBG_MASK_NO_PREFIX, ": ", NULLVA); } OutVa(DBG_MASK_NO_PREFIX, pFmt, Args); if (bNewLine) { OutVa(DBG_MASK_NO_PREFIX, "\n", NULLVA); } }
if (m_uFlags[DBG_ASSERT_FLAGS] & DBG_FAILURE_BREAK) { #if DBG
DebugBreak(); #endif
} else if (m_uFlags[DBG_ASSERT_FLAGS] & DBG_FAILURE_PROMPT) { Prompt(NULL); } }
//----------------------------------------------------------------------------
//
// DebugModule::AssertFailed
//
// Handles simple expression assertion failures.
//
//----------------------------------------------------------------------------
void DebugModule::AssertFailed(char *pExp) { AssertFailedVa(pExp, NULLVA, TRUE); }
//----------------------------------------------------------------------------
//
// DebugModule::AssertFailedMsg
//
// Handles assertion failures with arbitrary debug output.
//
//----------------------------------------------------------------------------
void DebugModule::AssertFailedMsg(char *pFmt, ...) { va_list Args;
va_start(Args, pFmt); AssertFailedVa(pFmt, Args, FALSE); va_end(Args); }
//----------------------------------------------------------------------------
//
// DebugModule::HrFailure
//
// Handles HRESULT failures.
//
//----------------------------------------------------------------------------
void DebugModule::HrFailure(HRESULT hr, char *pPrefix) { if (m_uFlags[DBG_HR_FLAGS] & DBG_FAILURE_OUTPUT) { OutPathFile(pPrefix, m_uFlags[DBG_HR_FLAGS]); OutMask(DBG_MASK_FORCE_CONT, ": %s\n", HrString(hr)); }
if (m_uFlags[DBG_HR_FLAGS] & DBG_FAILURE_BREAK) { #if DBG
DebugBreak(); #endif
} else if (m_uFlags[DBG_HR_FLAGS] & DBG_FAILURE_PROMPT) { Prompt(NULL); } }
//----------------------------------------------------------------------------
//
// DebugModule::HrStmtFailed
//
// Handles statement-style HRESULT failures.
//
//----------------------------------------------------------------------------
void DebugModule::HrStmtFailed(HRESULT hr) { HrFailure(hr, "HR test fail"); }
//----------------------------------------------------------------------------
//
// DebugModule::ReturnHr
//
// Handles expression-style HRESULT failures.
//
//----------------------------------------------------------------------------
HRESULT DebugModule::HrExpFailed(HRESULT hr) { HrFailure(hr, "HR expr fail"); return hr; }
//----------------------------------------------------------------------------
//
// DebugModule::Prompt
//
// Allows control over debug options via interactive input.
//
//----------------------------------------------------------------------------
void DebugModule::Prompt(char *pFmt, ...) { va_list Args;
if (pFmt != NULL) { va_start(Args, pFmt); OutVa(0, pFmt, Args); va_end(Args); }
#ifdef WINNT
char szInput[512]; char *pIdx; int iIdx; static char szFlagCommands[] = "ahomu";
for (;;) { ULONG uLen; uLen = DbgPrompt("[bgaAFhHmMoOrRuU] ", szInput, sizeof(szInput) - 1); if (uLen < 2) { Out("DbgPrompt failed\n"); #if DBG
DebugBreak(); #endif
return; }
// ATTENTION - Currently DbgPrompt returns a length that is two
// greater than the actual number of characters. Presumably this
// is an artifact of the Unicode/ANSI conversion and should
// really only be one greater, so attempt to handle both.
uLen -= 2; if (szInput[uLen] != 0) { uLen++; szInput[uLen] = 0; }
if (uLen < 1) { Out("Empty command ignored\n"); continue; } switch(szInput[0]) { case 'b': #if DBG
DebugBreak(); #endif
break; case 'g': return; case 'r': WriteReg(); break; case 'R': ReadReg(); break;
case 'a': case 'A': case 'h': case 'H': case 'm': case 'M': case 'o': case 'O': case 'u': case 'U': char chLower;
if (szInput[0] >= 'A' && szInput[0] <= 'Z') { chLower = szInput[0] - 'A' + 'a'; } else { chLower = szInput[0]; }
pIdx = strchr(szFlagCommands, chLower); if (pIdx == NULL) { // Should never happen.
break; }
iIdx = (int)((ULONG_PTR)(pIdx - szFlagCommands)); if (szInput[0] == chLower) { // Set.
m_uFlags[iIdx] = ParseUint(szInput + 1, m_pModFlags[iIdx]); }
// Set or Get.
OutUint(g_pFlagNames[iIdx], m_pModFlags[iIdx], m_uFlags[iIdx]); break; case 'F': if (uLen < 2) { Out("'F' must be followed by a flag group specifier\n"); break; } pIdx = strchr(szFlagCommands, szInput[1]); if (pIdx == NULL) { Out("Unknown flag group '%c'\n", szInput[1]); } else { iIdx = (int)((ULONG_PTR)(pIdx - szFlagCommands)); ShowFlags(g_pFlagNames[iIdx], m_pModFlags[iIdx]); } break;
default: Out("Unknown command '%c'\n", szInput[0]); break; } } #else
OutUint("OutputMask", m_pModFlags[DBG_OUTPUT_MASK], m_uFlags[DBG_OUTPUT_MASK]); Out("Prompt not available\n"); #if DBG
DebugBreak(); #endif
#endif
}
//----------------------------------------------------------------------------
//
// DebugModule::OpenDebugKey
//
// Opens the Direct3D\Debug\m_pModule key.
//
//----------------------------------------------------------------------------
HKEY DebugModule::OpenDebugKey(void) { HKEY hKey; char szKeyName[128];
strcpy(szKeyName, "Software\\Microsoft\\Direct3D\\Debug\\"); strcat(szKeyName, m_pModule); if (RegOpenKeyExA(HKEY_LOCAL_MACHINE, szKeyName, 0, KEY_ALL_ACCESS, &hKey) != ERROR_SUCCESS) { return NULL; } else { return hKey; } }
//----------------------------------------------------------------------------
//
// DebugModule::GetRegUint
//
// Gets a UINT value from the given key.
//
//----------------------------------------------------------------------------
UINT DebugModule::GetRegUint(HKEY hKey, char *pValue, UINT uDefault) { DWORD dwType, dwSize; DWORD dwVal;
dwSize = sizeof(dwVal); if (RegQueryValueExA(hKey, pValue, NULL, &dwType, (BYTE *)&dwVal, &dwSize) != ERROR_SUCCESS || dwType != REG_DWORD) { return uDefault; } else { return (UINT)dwVal; } }
//----------------------------------------------------------------------------
//
// DebugModule::SetRegUint
//
// Sets a UINT value for the given key.
//
//----------------------------------------------------------------------------
BOOL DebugModule::SetRegUint(HKEY hKey, char *pValue, UINT uValue) { return RegSetValueExA(hKey, pValue, NULL, REG_DWORD, (BYTE *)&uValue, sizeof(uValue)) == ERROR_SUCCESS; }
//----------------------------------------------------------------------------
//
// DebugModule::ReadReg
//
// Reads settings from the registry.
//
//----------------------------------------------------------------------------
void DebugModule::ReadReg(void) { HKEY hKey;
hKey = OpenDebugKey(); if (hKey != NULL) { int iIdx;
for (iIdx = 0; iIdx < DBG_FLAGS_COUNT; iIdx++) { m_uFlags[iIdx] = GetRegUint(hKey, g_pFlagNames[iIdx], m_uFlags[iIdx]); } RegCloseKey(hKey); } }
//----------------------------------------------------------------------------
//
// DebugModule::WriteReg
//
// Writes values to the registry.
//
//----------------------------------------------------------------------------
void DebugModule::WriteReg(void) { HKEY hKey;
hKey = OpenDebugKey(); if (hKey != NULL) { int iIdx;
for (iIdx = 0; iIdx < DBG_FLAGS_COUNT; iIdx++) { if (!SetRegUint(hKey, g_pFlagNames[iIdx], m_uFlags[iIdx])) { OutputDebugStringA("Error writing registry information\n"); } } RegCloseKey(hKey); } }
//----------------------------------------------------------------------------
//
// DebugModule::ParseUint
//
// Parses a string for a numeric value or a set of flag strings.
//
//----------------------------------------------------------------------------
UINT DebugModule::ParseUint(char *pString, DebugModuleFlags *pFlags) { UINT uVal;
uVal = 0;
for (;;) { while (*pString != 0 && (*pString == ' ' || *pString == '\t')) { pString++; }
if (*pString == 0) { break; }
char *pEnd; int iStepAfter;
pEnd = pString; while (*pEnd != 0 && *pEnd != ' ' && *pEnd != '\t') { pEnd++; } iStepAfter = *pEnd != 0 ? 1 : 0; *pEnd = 0; if (*pString >= '0' && *pString <= '9') { uVal |= strtoul(pString, &pString, 0); if (*pString != 0 && *pString != ' ' && *pString != '\t') { Out("Unrecognized characters '%s' after number\n", pString); } } else if (pFlags != NULL) { DebugModuleFlags *pFlag; for (pFlag = pFlags; pFlag->uFlag != 0; pFlag++) { if (!_stricmp(pString, pFlag->pName)) { break; } }
if (pFlag->uFlag == 0) { Out("Unrecognized flag string '%s'\n", pString); } else { uVal |= pFlag->uFlag; } } else { Out("No flag definitions, unable to convert '%s'\n", pString); }
pString = pEnd + iStepAfter; }
return uVal; }
//----------------------------------------------------------------------------
//
// DebugModule::OutUint
//
// Displays a UINT as a set of flag strings.
//
//----------------------------------------------------------------------------
void DebugModule::OutUint(char *pName, DebugModuleFlags *pFlags, UINT uValue) { if (pFlags == NULL || uValue == 0) { Out("%s: 0x%08X\n", pName, uValue); return; }
Out("%s:", pName); m_iStartCol = m_iModuleStartCol + strlen(pName) + 1; m_iCol = m_iStartCol; while (uValue != 0) { DebugModuleFlags *pFlag;
for (pFlag = pFlags; pFlag->uFlag != 0; pFlag++) { if ((pFlag->uFlag & uValue) == pFlag->uFlag) { AdvanceCols(strlen(pFlag->pName) + 1); OutMask(DBG_MASK_FORCE_CONT, " %s", pFlag->pName); uValue &= ~pFlag->uFlag; break; } }
if (pFlag->uFlag == 0) { AdvanceCols(11); OutMask(DBG_MASK_FORCE_CONT, " 0x%X", uValue); uValue = 0; } }
OutVa(DBG_MASK_NO_PREFIX, "\n", NULLVA); }
//----------------------------------------------------------------------------
//
// DebugModule::AdvanceCols
//
// Determines if there's enough space on the current line for
// the given number of columns. If not, a new line is started.
//
//----------------------------------------------------------------------------
void DebugModule::AdvanceCols(int iCols) { static char szSpaces[] = " ";
m_iCol += iCols; if (m_iCol >= 79) { int iSpace; OutVa(DBG_MASK_NO_PREFIX, "\n", NULLVA); // Force a prefix to be printed to start the line.
Out(""); m_iCol = m_iModuleStartCol; while (m_iCol < m_iStartCol) { iSpace = (int)min(sizeof(szSpaces) - 1, m_iStartCol - m_iCol); OutMask(DBG_MASK_FORCE_CONT, "%.*s", iSpace, szSpaces); m_iCol += iSpace; } } }
//----------------------------------------------------------------------------
//
// DebugModule::ShowFlags
//
// Shows the given flag set.
//
//----------------------------------------------------------------------------
void DebugModule::ShowFlags(char *pName, DebugModuleFlags *pFlags) { DebugModuleFlags *pFlag;
Out("%s:\n", pName); if (pFlags == NULL) { Out(" None defined\n"); } else { for (pFlag = pFlags; pFlag->uFlag != 0; pFlag++) { Out(" 0x%08X - %s\n", pFlag->uFlag, pFlag->pName); } } }
//----------------------------------------------------------------------------
//
// DebugModule::PathFile
//
// Returns the trailing filename component or NULL if the path is
// only a filename.
//
//----------------------------------------------------------------------------
char *DebugModule::PathFile(char *pPath) { char *pFile, *pSlash, *pBack, *pColon; pBack = strrchr(pPath, '\\'); pSlash = strrchr(pPath, '/'); pColon = strrchr(pPath, ':');
pFile = pBack; if (pSlash > pFile) { pFile = pSlash; } if (pColon > pFile) { pFile = pColon; }
return pFile != NULL ? pFile + 1 : NULL; }
//----------------------------------------------------------------------------
//
// DebugModule::OutPathFile
//
// Outputs the given string plus a path and filename.
// Returns whether the full path was output or not.
//
//----------------------------------------------------------------------------
BOOL DebugModule::OutPathFile(char *pPrefix, UINT uFailureFlags) { char *pFile; if (uFailureFlags & DBG_FAILURE_FILENAME_ONLY) { pFile = PathFile(m_pFile); } else { pFile = NULL; } if (pFile == NULL) { Out("%s %s(%d)", pPrefix, m_pFile, m_iLine); return TRUE; } else { Out("%s <>\\%s(%d)", pPrefix, pFile, m_iLine); return FALSE; } }
//----------------------------------------------------------------------------
//
// Global debug module.
//
//----------------------------------------------------------------------------
DBG_DECLARE_ONCE(Global, G, NULL, 0, NULL, 0);
#endif // #if DBG
|