mirror of
https://github.com/s1lentq/ReGameDLL_CS.git
synced 2025-01-26 21:48:05 +03:00
Improve dbg code to work Assertion properly
This commit is contained in:
parent
806f5651de
commit
f97c9d9f46
1
.gitignore
vendored
1
.gitignore
vendored
@ -2,6 +2,7 @@
|
|||||||
*.bat
|
*.bat
|
||||||
*.log
|
*.log
|
||||||
*.lnk
|
*.lnk
|
||||||
|
*.aps
|
||||||
**/msvc/Debug*
|
**/msvc/Debug*
|
||||||
**/msvc/Release*
|
**/msvc/Release*
|
||||||
**/msvc/Tests
|
**/msvc/Tests
|
||||||
|
@ -161,6 +161,9 @@ set(SHARED_SRCS
|
|||||||
"public/FileSystem.cpp"
|
"public/FileSystem.cpp"
|
||||||
"public/interface.cpp"
|
"public/interface.cpp"
|
||||||
"public/MemPool.cpp"
|
"public/MemPool.cpp"
|
||||||
|
"public/MemPool.cpp"
|
||||||
|
"public/tier0/dbg.cpp"
|
||||||
|
"public/tier0/platform_posix.cpp"
|
||||||
)
|
)
|
||||||
|
|
||||||
set(GAMEDLL_SRCS
|
set(GAMEDLL_SRCS
|
||||||
|
@ -228,8 +228,13 @@ void GameDLL_SwapTeams_f()
|
|||||||
|
|
||||||
#endif // REGAMEDLL_ADD
|
#endif // REGAMEDLL_ADD
|
||||||
|
|
||||||
|
SpewRetval_t GameDLL_SpewHandler(SpewType_t spewType, int level, const char *pMsg);
|
||||||
|
|
||||||
void EXT_FUNC GameDLLInit()
|
void EXT_FUNC GameDLLInit()
|
||||||
{
|
{
|
||||||
|
// By default, direct dbg reporting...
|
||||||
|
SpewOutputFunc(GameDLL_SpewHandler);
|
||||||
|
|
||||||
g_pskill = CVAR_GET_POINTER("skill");
|
g_pskill = CVAR_GET_POINTER("skill");
|
||||||
g_psv_gravity = CVAR_GET_POINTER("sv_gravity");
|
g_psv_gravity = CVAR_GET_POINTER("sv_gravity");
|
||||||
g_psv_aim = CVAR_GET_POINTER("sv_aim");
|
g_psv_aim = CVAR_GET_POINTER("sv_aim");
|
||||||
@ -455,3 +460,28 @@ void EXT_FUNC GameDLLInit()
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SpewRetval_t GameDLL_SpewHandler(SpewType_t spewType, int level, const char *pMsg)
|
||||||
|
{
|
||||||
|
bool bSpewPrint = (CVAR_GET_FLOAT("developer") >= level);
|
||||||
|
switch (spewType)
|
||||||
|
{
|
||||||
|
case SPEW_LOG:
|
||||||
|
case SPEW_MESSAGE:
|
||||||
|
if (bSpewPrint) UTIL_ServerPrint("%s", pMsg);
|
||||||
|
break;
|
||||||
|
case SPEW_WARNING:
|
||||||
|
if (bSpewPrint) UTIL_ServerPrint("Warning: %s", pMsg);
|
||||||
|
break;
|
||||||
|
case SPEW_ERROR:
|
||||||
|
Sys_Error("%s", pMsg);
|
||||||
|
return SPEW_ABORT; // fatal error, terminate it!
|
||||||
|
case SPEW_ASSERT:
|
||||||
|
UTIL_ServerPrint("Assert: %s", pMsg);
|
||||||
|
return SPEW_DEBUGGER; // assert always tries to debugger break
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SPEW_CONTINUE; // spew handled, continue on
|
||||||
|
}
|
||||||
|
@ -16,22 +16,7 @@ C_DLLEXPORT void WINAPI GiveFnptrsToDll(enginefuncs_t *pEnginefuncsTable, global
|
|||||||
Regamedll_Game_Init();
|
Regamedll_Game_Init();
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef _WIN32
|
#if defined(_LINUX)
|
||||||
|
|
||||||
// DLL entry point
|
|
||||||
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
|
|
||||||
{
|
|
||||||
if (fdwReason == DLL_PROCESS_ATTACH)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
else if (fdwReason == DLL_PROCESS_DETACH)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
#else // _WIN32
|
|
||||||
|
|
||||||
void __attribute__((constructor)) DllMainLoad()
|
void __attribute__((constructor)) DllMainLoad()
|
||||||
{
|
{
|
||||||
@ -41,4 +26,4 @@ void __attribute__((destructor)) DllMainUnload()
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // _WIN32
|
#endif // _LINUX
|
||||||
|
@ -1819,10 +1819,11 @@ void NORETURN Sys_Error(const char *error, ...)
|
|||||||
|
|
||||||
CONSOLE_ECHO("FATAL ERROR (shutting down): %s\n", text);
|
CONSOLE_ECHO("FATAL ERROR (shutting down): %s\n", text);
|
||||||
|
|
||||||
//TerminateProcess(GetCurrentProcess(), 1);
|
#if defined(_WIN32)
|
||||||
int *null = 0;
|
MessageBoxA(NULL, text, "Fatal error", MB_ICONERROR | MB_OK);
|
||||||
*null = 0;
|
#endif
|
||||||
exit(-1);
|
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
int UTIL_CountPlayersInBrushVolume(bool bOnlyAlive, CBaseEntity *pBrushEntity, int &playersInCount, int &playersOutCount, CPlayerInVolumeAdapter *pAdapter)
|
int UTIL_CountPlayersInBrushVolume(bool bOnlyAlive, CBaseEntity *pBrushEntity, int &playersInCount, int &playersOutCount, CPlayerInVolumeAdapter *pAdapter)
|
||||||
|
@ -756,9 +756,8 @@ int Q_UnicodeConvertT(const SrcType *pIn, int nInChars, DstType *pOut, int nOutB
|
|||||||
|
|
||||||
if (bErr)
|
if (bErr)
|
||||||
{
|
{
|
||||||
#ifdef _DEBUG
|
DbgAssertMsg(!(ePolicy & _STRINGCONVERTFLAG_ASSERT), "invalid Unicode byte sequence");
|
||||||
AssertMsg(!(ePolicy & _STRINGCONVERTFLAG_ASSERT), "invalid Unicode byte sequence");
|
|
||||||
#endif
|
|
||||||
if (ePolicy & _STRINGCONVERTFLAG_SKIP)
|
if (ePolicy & _STRINGCONVERTFLAG_SKIP)
|
||||||
{
|
{
|
||||||
nOut -= EncodeDstLen(uVal);
|
nOut -= EncodeDstLen(uVal);
|
||||||
@ -793,9 +792,8 @@ int Q_UnicodeConvertT(const SrcType *pIn, int nInChars, DstType *pOut, int nOutB
|
|||||||
nOut += EncodeDst(uVal, pOut + nOut);
|
nOut += EncodeDst(uVal, pOut + nOut);
|
||||||
if (bErr)
|
if (bErr)
|
||||||
{
|
{
|
||||||
#ifdef _DEBUG
|
DbgAssertMsg(!(ePolicy & _STRINGCONVERTFLAG_ASSERT), "invalid Unicode byte sequence");
|
||||||
AssertMsg(!(ePolicy & _STRINGCONVERTFLAG_ASSERT), "invalid Unicode byte sequence");
|
|
||||||
#endif
|
|
||||||
if (ePolicy & _STRINGCONVERTFLAG_SKIP)
|
if (ePolicy & _STRINGCONVERTFLAG_SKIP)
|
||||||
{
|
{
|
||||||
nOut -= EncodeDstLen(uVal);
|
nOut -= EncodeDstLen(uVal);
|
||||||
|
@ -540,6 +540,7 @@
|
|||||||
<ClCompile Include="..\public\FileSystem.cpp" />
|
<ClCompile Include="..\public\FileSystem.cpp" />
|
||||||
<ClCompile Include="..\public\interface.cpp" />
|
<ClCompile Include="..\public\interface.cpp" />
|
||||||
<ClCompile Include="..\public\MemPool.cpp" />
|
<ClCompile Include="..\public\MemPool.cpp" />
|
||||||
|
<ClCompile Include="..\public\tier0\assert_dialog.cpp" />
|
||||||
<ClCompile Include="..\public\tier0\dbg.cpp">
|
<ClCompile Include="..\public\tier0\dbg.cpp">
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
</ExcludedFromBuild>
|
</ExcludedFromBuild>
|
||||||
@ -548,6 +549,7 @@
|
|||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">true</ExcludedFromBuild>
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release Play|Win32'">true</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Release Play|Win32'">true</ExcludedFromBuild>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\public\tier0\platform_win32.cpp" />
|
||||||
<ClCompile Include="..\public\utlsymbol.cpp">
|
<ClCompile Include="..\public\utlsymbol.cpp">
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</ExcludedFromBuild>
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug Play|Win32'">true</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='Debug Play|Win32'">true</ExcludedFromBuild>
|
||||||
@ -787,6 +789,7 @@
|
|||||||
<ClInclude Include="..\public\regamedll\regamedll_api.h" />
|
<ClInclude Include="..\public\regamedll\regamedll_api.h" />
|
||||||
<ClInclude Include="..\public\tier0\dbg.h" />
|
<ClInclude Include="..\public\tier0\dbg.h" />
|
||||||
<ClInclude Include="..\public\tier0\platform.h" />
|
<ClInclude Include="..\public\tier0\platform.h" />
|
||||||
|
<ClInclude Include="..\public\tier0\resource.h" />
|
||||||
<ClInclude Include="..\public\utlarray.h" />
|
<ClInclude Include="..\public\utlarray.h" />
|
||||||
<ClInclude Include="..\public\utlmemory.h" />
|
<ClInclude Include="..\public\utlmemory.h" />
|
||||||
<ClInclude Include="..\public\utlrbtree.h" />
|
<ClInclude Include="..\public\utlrbtree.h" />
|
||||||
@ -810,6 +813,9 @@
|
|||||||
<UseLibraryDependencyInputs>false</UseLibraryDependencyInputs>
|
<UseLibraryDependencyInputs>false</UseLibraryDependencyInputs>
|
||||||
</ProjectReference>
|
</ProjectReference>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ResourceCompile Include="..\public\tier0\assert_dialog.rc" />
|
||||||
|
</ItemGroup>
|
||||||
<PropertyGroup Label="Globals">
|
<PropertyGroup Label="Globals">
|
||||||
<ProjectGuid>{70A2B904-B7DB-4C48-8DE0-AF567360D572}</ProjectGuid>
|
<ProjectGuid>{70A2B904-B7DB-4C48-8DE0-AF567360D572}</ProjectGuid>
|
||||||
<RootNamespace>ReGameDLL</RootNamespace>
|
<RootNamespace>ReGameDLL</RootNamespace>
|
||||||
|
@ -555,6 +555,12 @@
|
|||||||
<ClCompile Include="..\public\FileSystem.cpp">
|
<ClCompile Include="..\public\FileSystem.cpp">
|
||||||
<Filter>public</Filter>
|
<Filter>public</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\public\tier0\platform_win32.cpp">
|
||||||
|
<Filter>public\tier0</Filter>
|
||||||
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\public\tier0\assert_dialog.cpp">
|
||||||
|
<Filter>public\tier0</Filter>
|
||||||
|
</ClCompile>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ClInclude Include="..\version\version.h">
|
<ClInclude Include="..\version\version.h">
|
||||||
@ -1055,5 +1061,13 @@
|
|||||||
<ClInclude Include="..\public\utlarray.h">
|
<ClInclude Include="..\public\utlarray.h">
|
||||||
<Filter>public</Filter>
|
<Filter>public</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
<ClInclude Include="..\public\tier0\resource.h">
|
||||||
|
<Filter>public\tier0</Filter>
|
||||||
|
</ClInclude>
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ResourceCompile Include="..\public\tier0\assert_dialog.rc">
|
||||||
|
<Filter>public\tier0</Filter>
|
||||||
|
</ResourceCompile>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
</Project>
|
302
regamedll/public/tier0/assert_dialog.cpp
Normal file
302
regamedll/public/tier0/assert_dialog.cpp
Normal file
@ -0,0 +1,302 @@
|
|||||||
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||||||
|
//
|
||||||
|
// Purpose:
|
||||||
|
//
|
||||||
|
// $NoKeywords: $
|
||||||
|
//
|
||||||
|
//===========================================================================//
|
||||||
|
|
||||||
|
#include "precompiled.h"
|
||||||
|
#include "resource.h"
|
||||||
|
|
||||||
|
class CDialogInitInfo
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
const char *m_pFilename;
|
||||||
|
int m_iLine;
|
||||||
|
const char *m_pExpression;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CAssertDisable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
char m_Filename[MAX_OSPATH];
|
||||||
|
|
||||||
|
// If these are not -1, then this CAssertDisable only disables asserts on lines between
|
||||||
|
// these values (inclusive).
|
||||||
|
int m_LineMin;
|
||||||
|
int m_LineMax;
|
||||||
|
|
||||||
|
// Decremented each time we hit this assert and ignore it, until it's 0.
|
||||||
|
// Then the CAssertDisable is removed.
|
||||||
|
// If this is -1, then we always ignore this assert.
|
||||||
|
int m_nIgnoreTimes;
|
||||||
|
|
||||||
|
CAssertDisable *m_pNext;
|
||||||
|
};
|
||||||
|
|
||||||
|
static CDialogInitInfo g_Info{};
|
||||||
|
static bool g_bAssertsEnabled = true;
|
||||||
|
static CAssertDisable *g_pAssertDisables = nullptr;
|
||||||
|
|
||||||
|
// Set to true if they want to break in the debugger
|
||||||
|
static bool g_bBreak = false;
|
||||||
|
|
||||||
|
// Internal functions
|
||||||
|
static HINSTANCE g_hTier0Instance = nullptr;
|
||||||
|
BOOL WINAPI DllMain(
|
||||||
|
HINSTANCE hinstDLL, // handle to the DLL module
|
||||||
|
DWORD fdwReason, // reason for calling function
|
||||||
|
LPVOID lpvReserved // reserved
|
||||||
|
)
|
||||||
|
{
|
||||||
|
g_hTier0Instance = hinstDLL;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool IsDebugBreakEnabled()
|
||||||
|
{
|
||||||
|
static bool bResult = (Q_strstr(Plat_GetCommandLine(), "-debugbreak") != nullptr);
|
||||||
|
return bResult;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool AreAssertsDisabled()
|
||||||
|
{
|
||||||
|
static bool bResult = (Q_strstr(Plat_GetCommandLine(), "-noassert") != nullptr);
|
||||||
|
return bResult;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool AreAssertsEnabledInFileLine(const char *pFilename, int iLine)
|
||||||
|
{
|
||||||
|
CAssertDisable **pPrev = &g_pAssertDisables;
|
||||||
|
CAssertDisable *pNext = nullptr;
|
||||||
|
|
||||||
|
for (CAssertDisable *pCur = g_pAssertDisables; pCur; pCur = pNext)
|
||||||
|
{
|
||||||
|
pNext = pCur->m_pNext;
|
||||||
|
|
||||||
|
if (Q_stricmp(pFilename, pCur->m_Filename) == 0)
|
||||||
|
{
|
||||||
|
// Are asserts disabled in the whole file?
|
||||||
|
bool bAssertsEnabled = true;
|
||||||
|
if (pCur->m_LineMin == -1 && pCur->m_LineMax == -1)
|
||||||
|
bAssertsEnabled = false;
|
||||||
|
|
||||||
|
// Are asserts disabled on the specified line?
|
||||||
|
if (iLine >= pCur->m_LineMin && iLine <= pCur->m_LineMax)
|
||||||
|
bAssertsEnabled = false;
|
||||||
|
|
||||||
|
if (!bAssertsEnabled)
|
||||||
|
{
|
||||||
|
// If this assert is only disabled for the next N times, then countdown..
|
||||||
|
if (pCur->m_nIgnoreTimes > 0)
|
||||||
|
{
|
||||||
|
pCur->m_nIgnoreTimes--;
|
||||||
|
|
||||||
|
if (pCur->m_nIgnoreTimes == 0)
|
||||||
|
{
|
||||||
|
// Remove this one from the list.
|
||||||
|
*pPrev = pNext;
|
||||||
|
delete pCur;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pPrev = &pCur->m_pNext;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
CAssertDisable *CreateNewAssertDisable(const char *pFilename)
|
||||||
|
{
|
||||||
|
CAssertDisable *pDisable = new CAssertDisable;
|
||||||
|
pDisable->m_pNext = g_pAssertDisables;
|
||||||
|
g_pAssertDisables = pDisable;
|
||||||
|
|
||||||
|
pDisable->m_LineMin = pDisable->m_LineMax = -1;
|
||||||
|
pDisable->m_nIgnoreTimes = -1;
|
||||||
|
|
||||||
|
Q_strlcpy(pDisable->m_Filename, g_Info.m_pFilename);
|
||||||
|
|
||||||
|
return pDisable;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IgnoreAssertsInCurrentFile()
|
||||||
|
{
|
||||||
|
CreateNewAssertDisable(g_Info.m_pFilename);
|
||||||
|
}
|
||||||
|
|
||||||
|
CAssertDisable *IgnoreAssertsNearby(int nRange)
|
||||||
|
{
|
||||||
|
CAssertDisable *pDisable = CreateNewAssertDisable(g_Info.m_pFilename);
|
||||||
|
pDisable->m_LineMin = g_Info.m_iLine - nRange;
|
||||||
|
pDisable->m_LineMax = g_Info.m_iLine - nRange;
|
||||||
|
return pDisable;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int g_iLastLineRange = 5;
|
||||||
|
static int g_nLastIgnoreNumTimes = 1;
|
||||||
|
|
||||||
|
int CALLBACK AssertDialogProc(
|
||||||
|
HWND hDlg, // handle to dialog box
|
||||||
|
UINT uMsg, // message
|
||||||
|
WPARAM wParam, // first message parameter
|
||||||
|
LPARAM lParam // second message parameter
|
||||||
|
)
|
||||||
|
{
|
||||||
|
switch (uMsg)
|
||||||
|
{
|
||||||
|
case WM_INITDIALOG:
|
||||||
|
{
|
||||||
|
SetDlgItemText(hDlg, IDC_ASSERT_MSG_CTRL, g_Info.m_pExpression);
|
||||||
|
SetDlgItemText(hDlg, IDC_FILENAME_CONTROL, g_Info.m_pFilename);
|
||||||
|
|
||||||
|
SetDlgItemInt(hDlg, IDC_LINE_CONTROL, g_Info.m_iLine, false);
|
||||||
|
SetDlgItemInt(hDlg, IDC_IGNORE_NUMLINES, g_iLastLineRange, false);
|
||||||
|
SetDlgItemInt(hDlg, IDC_IGNORE_NUMTIMES, g_nLastIgnoreNumTimes, false);
|
||||||
|
|
||||||
|
// Center the dialog.
|
||||||
|
RECT rcDlg, rcDesktop;
|
||||||
|
GetWindowRect(hDlg, &rcDlg);
|
||||||
|
GetWindowRect(GetDesktopWindow(), &rcDesktop);
|
||||||
|
SetWindowPos(hDlg, HWND_TOP,
|
||||||
|
((rcDesktop.right - rcDesktop.left) - (rcDlg.right - rcDlg.left)) / 2,
|
||||||
|
((rcDesktop.bottom - rcDesktop.top) - (rcDlg.bottom - rcDlg.top)) / 2,
|
||||||
|
0, 0,
|
||||||
|
SWP_NOSIZE
|
||||||
|
);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
case WM_COMMAND:
|
||||||
|
{
|
||||||
|
switch (LOWORD(wParam))
|
||||||
|
{
|
||||||
|
// Ignore this assert N times
|
||||||
|
case IDC_IGNORE_THIS:
|
||||||
|
{
|
||||||
|
BOOL bTranslated = false;
|
||||||
|
UINT value = GetDlgItemInt(hDlg, IDC_IGNORE_NUMTIMES, &bTranslated, false);
|
||||||
|
if (bTranslated && value > 1)
|
||||||
|
{
|
||||||
|
CAssertDisable *pDisable = IgnoreAssertsNearby(0);
|
||||||
|
pDisable->m_nIgnoreTimes = value - 1;
|
||||||
|
g_nLastIgnoreNumTimes = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
case IDC_IGNORE_NEARBY:
|
||||||
|
{
|
||||||
|
BOOL bTranslated = false;
|
||||||
|
UINT value = GetDlgItemInt(hDlg, IDC_IGNORE_NUMLINES, &bTranslated, false);
|
||||||
|
if (!bTranslated || value < 1)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
IgnoreAssertsNearby(value);
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
case IDC_IGNORE_FILE:
|
||||||
|
IgnoreAssertsInCurrentFile();
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
// Always ignore this assert
|
||||||
|
case IDC_IGNORE_ALWAYS:
|
||||||
|
IgnoreAssertsNearby(0);
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
case IDC_IGNORE_ALL:
|
||||||
|
g_bAssertsEnabled = false;
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
case IDC_BREAK:
|
||||||
|
g_bBreak = true;
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
case WM_KEYDOWN:
|
||||||
|
{
|
||||||
|
// Escape?
|
||||||
|
if (wParam == 2)
|
||||||
|
{
|
||||||
|
// Ignore this assert
|
||||||
|
EndDialog(hDlg, 0);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HWND g_hBestParentWindow = nullptr;
|
||||||
|
|
||||||
|
static BOOL CALLBACK ParentWindowEnumProc(
|
||||||
|
HWND hWnd, // handle to parent window
|
||||||
|
LPARAM lParam // application-defined value
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (IsWindowVisible(hWnd))
|
||||||
|
{
|
||||||
|
DWORD procID;
|
||||||
|
GetWindowThreadProcessId(hWnd, &procID);
|
||||||
|
if (procID == (DWORD)lParam)
|
||||||
|
{
|
||||||
|
g_hBestParentWindow = hWnd;
|
||||||
|
return FALSE; // don't iterate any more.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HWND FindLikelyParentWindow()
|
||||||
|
{
|
||||||
|
// Enumerate top-level windows and take the first visible one with our processID.
|
||||||
|
g_hBestParentWindow = nullptr;
|
||||||
|
EnumWindows(ParentWindowEnumProc, GetCurrentProcessId());
|
||||||
|
return g_hBestParentWindow;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DoNewAssertDialog(const char *pFilename, int line, const char *pExpression)
|
||||||
|
{
|
||||||
|
if (AreAssertsDisabled())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// If they have the old mode enabled (always break immediately), then just break right into
|
||||||
|
// the debugger like we used to do.
|
||||||
|
if (IsDebugBreakEnabled())
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// Have ALL Asserts been disabled?
|
||||||
|
if (!g_bAssertsEnabled)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Has this specific Assert been disabled?
|
||||||
|
if (!AreAssertsEnabledInFileLine(pFilename, line))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Now create the dialog.
|
||||||
|
g_Info.m_pFilename = pFilename;
|
||||||
|
g_Info.m_iLine = line;
|
||||||
|
g_Info.m_pExpression = pExpression;
|
||||||
|
|
||||||
|
g_bBreak = false;
|
||||||
|
|
||||||
|
HWND hParentWindow = FindLikelyParentWindow();
|
||||||
|
DialogBox(g_hTier0Instance, MAKEINTRESOURCE(IDD_ASSERT_DIALOG), hParentWindow, AssertDialogProc);
|
||||||
|
|
||||||
|
return g_bBreak;
|
||||||
|
}
|
117
regamedll/public/tier0/assert_dialog.rc
Normal file
117
regamedll/public/tier0/assert_dialog.rc
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
//Microsoft Developer Studio generated resource script.
|
||||||
|
//
|
||||||
|
#include "resource.h"
|
||||||
|
|
||||||
|
#define APSTUDIO_READONLY_SYMBOLS
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// Generated from the TEXTINCLUDE 2 resource.
|
||||||
|
//
|
||||||
|
#include "winres.h"
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
#undef APSTUDIO_READONLY_SYMBOLS
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// English (U.S.) resources
|
||||||
|
|
||||||
|
#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)
|
||||||
|
#ifdef _WIN32
|
||||||
|
LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US
|
||||||
|
#pragma code_page(1252)
|
||||||
|
#endif //_WIN32
|
||||||
|
|
||||||
|
#ifdef APSTUDIO_INVOKED
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// TEXTINCLUDE
|
||||||
|
//
|
||||||
|
|
||||||
|
1 TEXTINCLUDE DISCARDABLE
|
||||||
|
BEGIN
|
||||||
|
"resource.h\0"
|
||||||
|
END
|
||||||
|
|
||||||
|
2 TEXTINCLUDE DISCARDABLE
|
||||||
|
BEGIN
|
||||||
|
"#include ""afxres.h""\r\n"
|
||||||
|
"\0"
|
||||||
|
END
|
||||||
|
|
||||||
|
3 TEXTINCLUDE DISCARDABLE
|
||||||
|
BEGIN
|
||||||
|
"\r\n"
|
||||||
|
"\0"
|
||||||
|
END
|
||||||
|
|
||||||
|
#endif // APSTUDIO_INVOKED
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// Dialog
|
||||||
|
//
|
||||||
|
|
||||||
|
IDD_ASSERT_DIALOG DIALOG DISCARDABLE 0, 0, 268, 158
|
||||||
|
STYLE DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU
|
||||||
|
CAPTION "Assert"
|
||||||
|
FONT 8, "MS Sans Serif"
|
||||||
|
BEGIN
|
||||||
|
LTEXT "File:",IDC_NOID,7,7,23,9
|
||||||
|
LTEXT "c:/gsclient/src/blah.cpp",IDC_FILENAME_CONTROL,36,7,217,8
|
||||||
|
LTEXT "Line:",IDC_NOID,7,18,23,9
|
||||||
|
LTEXT "45",IDC_LINE_CONTROL,36,18,217,8
|
||||||
|
LTEXT "Assert:",IDC_NOID,7,29,23,9
|
||||||
|
CONTROL "ASSERT MESSAGE",IDC_ASSERT_MSG_CTRL,"Static",
|
||||||
|
SS_LEFTNOWORDWRAP | SS_NOPREFIX | WS_GROUP,36,29,217,9
|
||||||
|
DEFPUSHBUTTON "&Break in Debugger",IDC_BREAK,7,49,98,14
|
||||||
|
PUSHBUTTON "&Ignore This Assert",IDC_IGNORE_THIS,7,66,98,14
|
||||||
|
EDITTEXT IDC_IGNORE_NUMTIMES,110,66,24,14,ES_AUTOHSCROLL |
|
||||||
|
ES_NUMBER
|
||||||
|
LTEXT "time(s).",IDC_NOID,138,68,23,8
|
||||||
|
PUSHBUTTON "Always Ignore &This Assert",IDC_IGNORE_ALWAYS,7,84,98,
|
||||||
|
14
|
||||||
|
PUSHBUTTON "Ignore &Nearby Asserts",IDC_IGNORE_NEARBY,7,102,98,14
|
||||||
|
LTEXT "within",IDC_NOID,109,105,19,8
|
||||||
|
EDITTEXT IDC_IGNORE_NUMLINES,131,102,40,14,ES_AUTOHSCROLL |
|
||||||
|
ES_NUMBER
|
||||||
|
LTEXT "lines.",IDC_NOID,175,105,17,8
|
||||||
|
PUSHBUTTON "Ignore Asserts in This &File",IDC_IGNORE_FILE,7,120,98,
|
||||||
|
14
|
||||||
|
PUSHBUTTON "Ignore &All Asserts",IDC_IGNORE_ALL,7,137,98,14
|
||||||
|
END
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// DESIGNINFO
|
||||||
|
//
|
||||||
|
|
||||||
|
#ifdef APSTUDIO_INVOKED
|
||||||
|
GUIDELINES DESIGNINFO DISCARDABLE
|
||||||
|
BEGIN
|
||||||
|
IDD_ASSERT_DIALOG, DIALOG
|
||||||
|
BEGIN
|
||||||
|
LEFTMARGIN, 7
|
||||||
|
RIGHTMARGIN, 261
|
||||||
|
TOPMARGIN, 7
|
||||||
|
BOTTOMMARGIN, 151
|
||||||
|
END
|
||||||
|
END
|
||||||
|
#endif // APSTUDIO_INVOKED
|
||||||
|
|
||||||
|
#endif // English (U.S.) resources
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef APSTUDIO_INVOKED
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// Generated from the TEXTINCLUDE 3 resource.
|
||||||
|
//
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
#endif // not APSTUDIO_INVOKED
|
||||||
|
|
@ -1,50 +1,17 @@
|
|||||||
/*
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||||||
*
|
//
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
// Purpose:
|
||||||
* under the terms of the GNU General Public License as published by the
|
//
|
||||||
* Free Software Foundation; either version 2 of the License, or (at
|
// $NoKeywords: $
|
||||||
* 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
|
|
||||||
*
|
|
||||||
* In addition, as a special exception, the author gives permission to
|
|
||||||
* link the code of this program with the Half-Life Game Engine ("HL
|
|
||||||
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
|
||||||
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
|
||||||
* respects for all of the code used other than the HL Engine and MODs
|
|
||||||
* from Valve. If you modify this file, you may extend this exception
|
|
||||||
* to your version of the file, but you are not obligated to do so. If
|
|
||||||
* you do not wish to do so, delete this exception statement from your
|
|
||||||
* version.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "precompiled.h"
|
#include "precompiled.h"
|
||||||
|
|
||||||
// Internal structures
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
MAX_GROUP_NAME_LENGTH = 48
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SpewGroup_t
|
|
||||||
{
|
|
||||||
char m_GroupName[MAX_GROUP_NAME_LENGTH];
|
|
||||||
int m_Level;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// Templates to assist in validating pointers:
|
// Templates to assist in validating pointers:
|
||||||
void _AssertValidReadPtr(void *ptr, int count)
|
void _AssertValidReadPtr(void *ptr, int count)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32)
|
||||||
Assert(!IsBadReadPtr(ptr, count));
|
Assert(!IsBadReadPtr(ptr, count));
|
||||||
#else
|
#else
|
||||||
Assert(ptr);
|
Assert(ptr);
|
||||||
@ -54,7 +21,7 @@ void _AssertValidReadPtr(void *ptr, int count)
|
|||||||
|
|
||||||
void _AssertValidWritePtr(void *ptr, int count)
|
void _AssertValidWritePtr(void *ptr, int count)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32)
|
||||||
Assert(!IsBadWritePtr(ptr, count));
|
Assert(!IsBadWritePtr(ptr, count));
|
||||||
#else
|
#else
|
||||||
Assert(ptr);
|
Assert(ptr);
|
||||||
@ -63,26 +30,29 @@ void _AssertValidWritePtr(void *ptr, int count)
|
|||||||
|
|
||||||
void _AssertValidReadWritePtr(void *ptr, int count)
|
void _AssertValidReadWritePtr(void *ptr, int count)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32)
|
||||||
Assert(!(IsBadWritePtr(ptr, count) || IsBadReadPtr(ptr, count)));
|
Assert(!(IsBadWritePtr(ptr, count) || IsBadReadPtr(ptr, count)));
|
||||||
#else
|
#else
|
||||||
Assert(ptr);
|
Assert(ptr);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined(DBGFLAG_ASSERT)
|
||||||
void AssertValidStringPtr(const char *ptr, int maxchar)
|
void AssertValidStringPtr(const char *ptr, int maxchar)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32)
|
||||||
Assert(!IsBadStringPtr(ptr, maxchar));
|
Assert(!IsBadStringPtr(ptr, maxchar));
|
||||||
#else
|
#else
|
||||||
Assert(ptr);
|
Assert(ptr);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
#endif // DBGFLAG_ASSERT
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
SpewRetval_t DefaultSpewFunc(SpewType_t type, const char *pMsg)
|
SpewRetval_t DefaultSpewFunc(SpewType_t type, int level, const char *pMsg)
|
||||||
{
|
{
|
||||||
printf("%s", pMsg);
|
printf("%s", pMsg);
|
||||||
|
|
||||||
if (type == SPEW_ASSERT)
|
if (type == SPEW_ASSERT)
|
||||||
return SPEW_DEBUGGER;
|
return SPEW_DEBUGGER;
|
||||||
else if (type == SPEW_ERROR)
|
else if (type == SPEW_ERROR)
|
||||||
@ -93,13 +63,10 @@ SpewRetval_t DefaultSpewFunc(SpewType_t type, const char *pMsg)
|
|||||||
|
|
||||||
static SpewOutputFunc_t s_SpewOutputFunc = DefaultSpewFunc;
|
static SpewOutputFunc_t s_SpewOutputFunc = DefaultSpewFunc;
|
||||||
|
|
||||||
static const char *s_pFileName;
|
static const char *s_pMessage = nullptr;
|
||||||
static int s_Line;
|
static const char *s_pFileName = nullptr;
|
||||||
static SpewType_t s_SpewType;
|
static int s_Line = 0;
|
||||||
|
static SpewType_t s_SpewType = SPEW_MESSAGE;
|
||||||
static SpewGroup_t *s_pSpewGroups = 0;
|
|
||||||
static int s_GroupCount = 0;
|
|
||||||
static int s_DefaultLevel = 0;
|
|
||||||
|
|
||||||
// Spew output management.
|
// Spew output management.
|
||||||
void SpewOutputFunc(SpewOutputFunc_t func)
|
void SpewOutputFunc(SpewOutputFunc_t func)
|
||||||
@ -119,54 +86,92 @@ SpewOutputFunc_t GetSpewOutputFunc()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void _ExitOnFatalAssert()
|
||||||
|
{
|
||||||
|
Msg("Fatal assert failed: %s, file %s line %d. Application exiting.\n", s_pMessage, s_pFileName, s_Line);
|
||||||
|
|
||||||
|
#if defined(WIN32)
|
||||||
|
TerminateProcess(GetCurrentProcess(), EXIT_FAILURE); // die, die RIGHT NOW! (don't call exit() so destructors will not get run)
|
||||||
|
#else
|
||||||
|
exit(EXIT_FAILURE); // forcefully shutdown of the process without destructors running
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
// Spew functions
|
// Spew functions
|
||||||
void _SpewInfo(SpewType_t type, const char *pFile, int line)
|
void _SpewInfo(SpewType_t type, const char *pFile, int line)
|
||||||
{
|
{
|
||||||
// Only grab the file name. Ignore the path.
|
// Only grab the file name. Ignore the path
|
||||||
const char *pSlash = strrchr(pFile, '\\');
|
const char *pSlash = Q_strrchr(pFile, '\\');
|
||||||
const char *pSlash2 = strrchr(pFile, '/');
|
const char *pSlash2 = Q_strrchr(pFile, '/');
|
||||||
|
|
||||||
if (pSlash < pSlash2) pSlash = pSlash2;
|
if (pSlash < pSlash2) pSlash = pSlash2;
|
||||||
|
|
||||||
s_pFileName = pSlash ? pSlash + 1 : pFile;
|
s_pFileName = pSlash ? pSlash + 1 : pFile;
|
||||||
s_Line = line;
|
s_Line = line;
|
||||||
s_SpewType = type;
|
s_SpewType = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
SpewRetval_t _SpewMessage(SpewType_t spewType, const char *pMsgFormat, va_list args)
|
SpewRetval_t _SpewMessageV(SpewType_t spewType, int level, const char *pMsgFormat, va_list args)
|
||||||
{
|
{
|
||||||
char pTempBuffer[1024];
|
if (level < 0) level = DBG_DEFAULT_LEVEL;
|
||||||
|
|
||||||
|
static char szTempBuffer[4096]{};
|
||||||
|
szTempBuffer[0] = '\0';
|
||||||
|
s_pMessage = szTempBuffer;
|
||||||
|
|
||||||
|
// check that we won't artifically truncate the string
|
||||||
|
assert(Q_strlen(pMsgFormat) < sizeof(szTempBuffer));
|
||||||
|
|
||||||
// Printf the file and line for warning + assert only...
|
// Printf the file and line for warning + assert only...
|
||||||
int len = 0;
|
int len = 0;
|
||||||
if ((spewType == SPEW_ASSERT))
|
if (spewType == SPEW_ASSERT)
|
||||||
{
|
{
|
||||||
len = sprintf(pTempBuffer, "%s (%d) : ", s_pFileName, s_Line);
|
len = Q_snprintf(szTempBuffer, sizeof(szTempBuffer), "%s (%d) : ", s_pFileName, s_Line);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (len == -1)
|
||||||
|
{
|
||||||
|
return SPEW_ABORT;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the message....
|
// Create the message....
|
||||||
len += vsprintf(&pTempBuffer[len], pMsgFormat, args);
|
len += Q_vsnprintf(&szTempBuffer[len], sizeof(szTempBuffer) - len, pMsgFormat, args);
|
||||||
|
|
||||||
|
// Use normal assert here; to avoid recursion
|
||||||
|
assert(len < sizeof(szTempBuffer));
|
||||||
|
|
||||||
// Add \n for warning and assert
|
// Add \n for warning and assert
|
||||||
if ((spewType == SPEW_ASSERT))
|
if ((spewType == SPEW_ASSERT))
|
||||||
{
|
{
|
||||||
len += sprintf(&pTempBuffer[len], "\n");
|
len += Q_snprintf(&szTempBuffer[len], sizeof(szTempBuffer) - len, "\n");
|
||||||
|
Plat_OutputDebugString(szTempBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(len < 1024); // use normal assert here; to avoid recursion.
|
// use normal assert here; to avoid recursion
|
||||||
|
assert((size_t)len < (sizeof(szTempBuffer) / sizeof(szTempBuffer[0]) - 1));
|
||||||
assert(s_SpewOutputFunc);
|
assert(s_SpewOutputFunc);
|
||||||
|
|
||||||
// direct it to the appropriate target(s)
|
// direct it to the appropriate target(s)
|
||||||
SpewRetval_t ret = s_SpewOutputFunc(spewType, pTempBuffer);
|
SpewRetval_t ret = s_SpewOutputFunc(spewType, level, szTempBuffer);
|
||||||
switch (ret)
|
switch (ret)
|
||||||
{
|
{
|
||||||
// Put the break into the macro so it would occur in the right place
|
// Put the break into the macro so it would occur in the right place
|
||||||
//case SPEW_DEBUGGER:
|
case SPEW_DEBUGGER:
|
||||||
// DebuggerBreak();
|
{
|
||||||
// break;
|
if (spewType != SPEW_ASSERT)
|
||||||
|
DebuggerBreakIfDebugging();
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
case SPEW_ABORT:
|
case SPEW_ABORT:
|
||||||
// MessageBox(nullptr, "Error in _SpewMessage", "Error", MB_OK);
|
{
|
||||||
exit(0);
|
#if defined(WIN32)
|
||||||
|
TerminateProcess(GetCurrentProcess(), EXIT_FAILURE); // die, die RIGHT NOW! (don't call exit() so destructors will not get run)
|
||||||
|
#else
|
||||||
|
exit(EXIT_FAILURE); // forcefully shutdown of the process without destructors running
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -174,81 +179,52 @@ SpewRetval_t _SpewMessage(SpewType_t spewType, const char *pMsgFormat, va_list a
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
SpewRetval_t _SpewMessage(const char *pMsgFormat, ...)
|
#if defined(_WIN32)
|
||||||
|
// Returns true if they want to break in the debugger
|
||||||
|
bool DoNewAssertDialog(const char *pFile, int line, const char *pExpression);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
bool _SpewAssertDialog()
|
||||||
{
|
{
|
||||||
|
#if defined(_WIN32)
|
||||||
|
return DoNewAssertDialog(s_pFileName, s_Line, s_pMessage);
|
||||||
|
#else
|
||||||
|
return false;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
SpewRetval_t _SpewAssert(const char *pFile, int line, int level, const char *pMsgFormat, ...)
|
||||||
|
{
|
||||||
|
_SpewInfo(SPEW_ASSERT, pFile, line);
|
||||||
|
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, pMsgFormat);
|
va_start(args, pMsgFormat);
|
||||||
SpewRetval_t ret = _SpewMessage(s_SpewType, pMsgFormat, args);
|
SpewRetval_t ret = _SpewMessageV(s_SpewType, level, pMsgFormat, args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
SpewRetval_t _DSpewMessage(const char *pGroupName, int level, const char *pMsgFormat, ...)
|
void _Msg(int level, const char *pMsgFormat, ...)
|
||||||
{
|
|
||||||
if (!IsSpewActive(pGroupName, level))
|
|
||||||
return SPEW_CONTINUE;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
SpewRetval_t ret = _SpewMessage(s_SpewType, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Msg(const char *pMsgFormat, ...)
|
|
||||||
{
|
{
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, pMsgFormat);
|
va_start(args, pMsgFormat);
|
||||||
_SpewMessage(SPEW_MESSAGE, pMsgFormat, args);
|
_SpewMessageV(SPEW_MESSAGE, level, pMsgFormat, args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DMsg(const char *pGroupName, int level, const char *pMsgFormat, ...)
|
void _Warning(int level, const char *pMsgFormat, ...)
|
||||||
{
|
{
|
||||||
if (!IsSpewActive(pGroupName, level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, pMsgFormat);
|
va_start(args, pMsgFormat);
|
||||||
_SpewMessage(SPEW_MESSAGE, pMsgFormat, args);
|
_SpewMessageV(SPEW_WARNING, level, pMsgFormat, args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Warning(const char *pMsgFormat, ...)
|
void _Log(int level, const char *pMsgFormat, ...)
|
||||||
{
|
{
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, pMsgFormat);
|
va_start(args, pMsgFormat);
|
||||||
_SpewMessage(SPEW_WARNING, pMsgFormat, args);
|
_SpewMessageV(SPEW_LOG, level, pMsgFormat, args);
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DWarning(const char *pGroupName, int level, const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive(pGroupName, level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_WARNING, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Log(const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_LOG, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DLog(const char *pGroupName, int level, const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive(pGroupName, level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_LOG, pMsgFormat, args);
|
|
||||||
va_end(args);
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -256,146 +232,6 @@ void Error(const char *pMsgFormat, ...)
|
|||||||
{
|
{
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, pMsgFormat);
|
va_start(args, pMsgFormat);
|
||||||
_SpewMessage(SPEW_ERROR, pMsgFormat, args);
|
_SpewMessageV(SPEW_ERROR, DBG_DEFAULT_LEVEL, pMsgFormat, args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
// A couple of super-common dynamic spew messages, here for convenience
|
|
||||||
// These looked at the "developer" group, print if it's level 1 or higher
|
|
||||||
void DevMsg(int level, char const *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive("developer", level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_MESSAGE, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DevWarning(int level, const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive("developer", level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_WARNING, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DevLog(int level, const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive("developer", level))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_LOG, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DevMsg(const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
if (!IsSpewActive("developer", 1))
|
|
||||||
return;
|
|
||||||
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_MESSAGE, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DevWarning(const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_WARNING, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DevLog(const char *pMsgFormat, ...)
|
|
||||||
{
|
|
||||||
va_list args;
|
|
||||||
va_start(args, pMsgFormat);
|
|
||||||
_SpewMessage(SPEW_LOG, pMsgFormat, args);
|
|
||||||
va_end(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find a group, return true if found, false if not. Return in ind the
|
|
||||||
// index of the found group, or the index of the group right before where the
|
|
||||||
// group should be inserted into the list to maintain sorted order.
|
|
||||||
bool FindSpewGroup(const char *pGroupName, int *pInd)
|
|
||||||
{
|
|
||||||
int s = 0;
|
|
||||||
if (s_GroupCount)
|
|
||||||
{
|
|
||||||
int e = (int)(s_GroupCount - 1);
|
|
||||||
while (s <= e)
|
|
||||||
{
|
|
||||||
int m = (s + e) >> 1;
|
|
||||||
int cmp = Q_stricmp(pGroupName, s_pSpewGroups[m].m_GroupName);
|
|
||||||
if (!cmp)
|
|
||||||
{
|
|
||||||
*pInd = m;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cmp < 0)
|
|
||||||
e = m - 1;
|
|
||||||
else
|
|
||||||
s = m + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
*pInd = s;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sets the priority level for a spew group
|
|
||||||
void SpewActivate(const char *pGroupName, int level)
|
|
||||||
{
|
|
||||||
Assert(pGroupName);
|
|
||||||
|
|
||||||
// check for the default group first...
|
|
||||||
if ((pGroupName[0] == '*') && (pGroupName[1] == '\0'))
|
|
||||||
{
|
|
||||||
s_DefaultLevel = level;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Normal case, search in group list using binary search.
|
|
||||||
// If not found, grow the list of groups and insert it into the
|
|
||||||
// right place to maintain sorted order. Then set the level.
|
|
||||||
int ind;
|
|
||||||
if (!FindSpewGroup(pGroupName, &ind))
|
|
||||||
{
|
|
||||||
// not defined yet, insert an entry.
|
|
||||||
s_GroupCount++;
|
|
||||||
if (s_pSpewGroups)
|
|
||||||
{
|
|
||||||
s_pSpewGroups = (SpewGroup_t *)realloc(s_pSpewGroups, s_GroupCount * sizeof(SpewGroup_t));
|
|
||||||
|
|
||||||
// shift elements down to preserve order
|
|
||||||
int numToMove = s_GroupCount - ind - 1;
|
|
||||||
memmove(&s_pSpewGroups[ind + 1], &s_pSpewGroups[ind], numToMove * sizeof(SpewGroup_t));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
s_pSpewGroups = (SpewGroup_t *)malloc(s_GroupCount * sizeof(SpewGroup_t));
|
|
||||||
|
|
||||||
Assert(strlen(pGroupName) < MAX_GROUP_NAME_LENGTH);
|
|
||||||
strcpy(s_pSpewGroups[ind].m_GroupName, pGroupName);
|
|
||||||
}
|
|
||||||
s_pSpewGroups[ind].m_Level = level;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tests to see if a particular spew is active
|
|
||||||
bool IsSpewActive(const char *pGroupName, int level)
|
|
||||||
{
|
|
||||||
// If we don't find the spew group, use the default level.
|
|
||||||
int ind;
|
|
||||||
if (FindSpewGroup(pGroupName, &ind))
|
|
||||||
return s_pSpewGroups[ind].m_Level >= level;
|
|
||||||
else
|
|
||||||
return s_DefaultLevel >= level;
|
|
||||||
}
|
|
||||||
|
@ -1,30 +1,10 @@
|
|||||||
/*
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||||||
*
|
//
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
// Purpose:
|
||||||
* under the terms of the GNU General Public License as published by the
|
//
|
||||||
* Free Software Foundation; either version 2 of the License, or (at
|
// $NoKeywords: $
|
||||||
* 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
|
|
||||||
*
|
|
||||||
* In addition, as a special exception, the author gives permission to
|
|
||||||
* link the code of this program with the Half-Life Game Engine ("HL
|
|
||||||
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
|
||||||
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
|
||||||
* respects for all of the code used other than the HL Engine and MODs
|
|
||||||
* from Valve. If you modify this file, you may extend this exception
|
|
||||||
* to your version of the file, but you are not obligated to do so. If
|
|
||||||
* you do not wish to do so, delete this exception statement from your
|
|
||||||
* version.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
@ -34,16 +14,7 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
|
|
||||||
// dll export stuff
|
#define DBG_DEFAULT_LEVEL 0
|
||||||
#ifdef TIER0_DLL_EXPORT
|
|
||||||
#define DBG_INTERFACE DLL_EXPORT
|
|
||||||
#define DBG_OVERLOAD DLL_GLOBAL_EXPORT
|
|
||||||
#define DBG_CLASS DLL_CLASS_EXPORT
|
|
||||||
#else
|
|
||||||
#define DBG_INTERFACE DLL_IMPORT
|
|
||||||
#define DBG_OVERLOAD DLL_GLOBAL_IMPORT
|
|
||||||
#define DBG_CLASS DLL_CLASS_IMPORT
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Usage model for the Dbg library
|
// Usage model for the Dbg library
|
||||||
//
|
//
|
||||||
@ -87,22 +58,6 @@
|
|||||||
// Msg("Isn't this exciting %d?", 5);
|
// Msg("Isn't this exciting %d?", 5);
|
||||||
// Error("I'm just thrilled");
|
// Error("I'm just thrilled");
|
||||||
//
|
//
|
||||||
// Dynamic Spew messages
|
|
||||||
//
|
|
||||||
// It is possible to dynamically turn spew on and off. Dynamic spew is
|
|
||||||
// identified by a spew group and priority level. To turn spew on for a
|
|
||||||
// particular spew group, use SpewActivate("group", level). This will
|
|
||||||
// cause all spew in that particular group with priority levels <= the
|
|
||||||
// level specified in the SpewActivate function to be printed. Use DSpew
|
|
||||||
// to perform the spew:
|
|
||||||
//
|
|
||||||
// DWarning("group", level, "Oh I feel even yummier!\n");
|
|
||||||
//
|
|
||||||
// Priority level 0 means that the spew will *always* be printed, and group
|
|
||||||
// '*' is the default spew group. If a DWarning is encountered using a group
|
|
||||||
// whose priority has not been set, it will use the priority of the default
|
|
||||||
// group. The priority of the default group is initially set to 0.
|
|
||||||
//
|
|
||||||
// Spew output
|
// Spew output
|
||||||
//
|
//
|
||||||
// The output of the spew system can be redirected to an externally-supplied
|
// The output of the spew system can be redirected to an externally-supplied
|
||||||
@ -131,12 +86,6 @@
|
|||||||
// }
|
// }
|
||||||
// );
|
// );
|
||||||
//
|
//
|
||||||
// Code can be activated based on the dynamic spew groups also. Use
|
|
||||||
//
|
|
||||||
// DBG_DCODE("group", level,
|
|
||||||
// { int x = 5; ++x; }
|
|
||||||
// );
|
|
||||||
//
|
|
||||||
// 3. Breaking into the debugger.
|
// 3. Breaking into the debugger.
|
||||||
//
|
//
|
||||||
// To cause an unconditional break into the debugger in debug builds only, use DBG_BREAK
|
// To cause an unconditional break into the debugger in debug builds only, use DBG_BREAK
|
||||||
@ -170,7 +119,7 @@ enum SpewRetval_t
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Type of externally defined function used to display debug spew
|
// Type of externally defined function used to display debug spew
|
||||||
typedef SpewRetval_t (*SpewOutputFunc_t)(SpewType_t spewType, const char *pMsg);
|
typedef SpewRetval_t (*SpewOutputFunc_t)(SpewType_t spewType, int level, const char *pMsg);
|
||||||
|
|
||||||
// Used to redirect spew output
|
// Used to redirect spew output
|
||||||
void SpewOutputFunc(SpewOutputFunc_t func);
|
void SpewOutputFunc(SpewOutputFunc_t func);
|
||||||
@ -178,134 +127,220 @@ void SpewOutputFunc(SpewOutputFunc_t func);
|
|||||||
// Used ot get the current spew output function
|
// Used ot get the current spew output function
|
||||||
SpewOutputFunc_t GetSpewOutputFunc();
|
SpewOutputFunc_t GetSpewOutputFunc();
|
||||||
|
|
||||||
// Used to manage spew groups and subgroups
|
|
||||||
void SpewActivate(const char *pGroupName, int level);
|
|
||||||
bool IsSpewActive(const char *pGroupName, int level);
|
|
||||||
|
|
||||||
// Used to display messages, should never be called directly.
|
// Used to display messages, should never be called directly.
|
||||||
void _SpewInfo(SpewType_t type, const char *pFile, int line);
|
SpewRetval_t _SpewAssert(const char *pFile, int line, int level, const char *pMsg, ...);
|
||||||
SpewRetval_t _SpewMessage(const char *pMsg, ...);
|
void _ExitOnFatalAssert();
|
||||||
SpewRetval_t _DSpewMessage(const char *pGroupName, int level, const char *pMsg, ...);
|
bool _SpewAssertDialog();
|
||||||
|
|
||||||
// Used to define macros, never use these directly.
|
inline bool ShouldUseNewAssertDialog()
|
||||||
#define _Assert(_exp) \
|
{
|
||||||
do { \
|
#if defined(DBGFLAG_ASSERTDLG)
|
||||||
if (!(_exp)) \
|
return true; // always show an assert dialog
|
||||||
{ \
|
#else
|
||||||
_SpewInfo(SPEW_ASSERT, __FILE__, __LINE__); \
|
return Plat_IsInDebugSession(); // only show an assert dialog if the process is being debugged
|
||||||
if (_SpewMessage("Assertion Failed: " #_exp) == SPEW_DEBUGGER) \
|
#endif // DBGFLAG_ASSERTDLG
|
||||||
{ \
|
}
|
||||||
DebuggerBreak(); \
|
|
||||||
} \
|
#define _AssertMsg(_exp, _msg, _executeExp, _bFatal) \
|
||||||
} \
|
do { \
|
||||||
|
if (!(_exp)) \
|
||||||
|
{ \
|
||||||
|
SpewRetval_t ret = _SpewAssert(__FILE__, __LINE__, DBG_DEFAULT_LEVEL, _msg); \
|
||||||
|
if (ret == SPEW_DEBUGGER) \
|
||||||
|
{ \
|
||||||
|
if (!ShouldUseNewAssertDialog() || _SpewAssertDialog()) \
|
||||||
|
DebuggerBreakIfDebugging(); \
|
||||||
|
} \
|
||||||
|
_executeExp; \
|
||||||
|
if (_bFatal) \
|
||||||
|
_ExitOnFatalAssert(); \
|
||||||
|
} \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define _AssertMsg(_exp, _msg) \
|
#define _AssertMsgWarn(_exp, _msg) \
|
||||||
do { \
|
if (!(_exp)) \
|
||||||
if (!(_exp)) \
|
{ \
|
||||||
{ \
|
Warning("%s (%d) : " _msg, __FILE__, __LINE__); \
|
||||||
_SpewInfo(SPEW_ASSERT, __FILE__, __LINE__); \
|
} \
|
||||||
if (_SpewMessage(_msg) == SPEW_DEBUGGER) \
|
|
||||||
{ \
|
|
||||||
DebuggerBreak(); \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define _AssertFunc(_exp, _f) \
|
#define _AssertMsgOnce(_exp, _msg, _bFatal) \
|
||||||
do { \
|
do { \
|
||||||
if (!(_exp)) \
|
static bool fAsserted = false; \
|
||||||
{ \
|
if (!fAsserted && !(_exp)) \
|
||||||
_SpewInfo(SPEW_ASSERT, __FILE__, __LINE__); \
|
{ \
|
||||||
SpewRetval_t ret = _SpewMessage("Assertion Failed!" #_exp); \
|
fAsserted = true; \
|
||||||
_f; \
|
_AssertMsg(_exp, _msg, (fAsserted = true), _bFatal); \
|
||||||
if (ret == SPEW_DEBUGGER) \
|
} \
|
||||||
{ \
|
|
||||||
DebuggerBreak(); \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define _AssertEquals(_exp, _expectedValue) \
|
|
||||||
do { \
|
|
||||||
if ((_exp) != (_expectedValue)) \
|
|
||||||
{ \
|
|
||||||
_SpewInfo(SPEW_ASSERT, __FILE__, __LINE__); \
|
|
||||||
SpewRetval_t ret = _SpewMessage("Expected %d but got %d!", (_expectedValue), (_exp)); \
|
|
||||||
if (ret == SPEW_DEBUGGER) \
|
|
||||||
{ \
|
|
||||||
DebuggerBreak(); \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define _AssertFloatEquals(_exp, _expectedValue, _tol) \
|
|
||||||
do { \
|
|
||||||
if (fabs((_exp) - (_expectedValue)) > (_tol)) \
|
|
||||||
{ \
|
|
||||||
_SpewInfo(SPEW_ASSERT, __FILE__, __LINE__); \
|
|
||||||
SpewRetval_t ret = _SpewMessage("Expected %f but got %f!", (_expectedValue), (_exp)); \
|
|
||||||
if (ret == SPEW_DEBUGGER) \
|
|
||||||
{ \
|
|
||||||
DebuggerBreak(); \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
// Spew macros...
|
// Spew macros...
|
||||||
#ifdef _DEBUG
|
// AssertFatal macros
|
||||||
|
// AssertFatal is used to detect an unrecoverable error condition.
|
||||||
|
// If enabled, it may display an assert dialog (if DBGFLAG_ASSERTDLG is turned on or running under the debugger),
|
||||||
|
// and always terminates the application
|
||||||
|
|
||||||
#define Assert(_exp) _Assert(_exp)
|
#if defined(DBGFLAG_ASSERTFATAL)
|
||||||
#define AssertMsg(_exp, _msg) _AssertMsg(_exp, _msg)
|
|
||||||
#define AssertFunc(_exp, _f) _AssertFunc(_exp, _f)
|
|
||||||
#define AssertEquals(_exp, _expectedValue) _AssertEquals(_exp, _expectedValue)
|
|
||||||
#define AssertFloatEquals(_exp, _expectedValue, _tol) _AssertFloatEquals(_exp, _expectedValue, _tol)
|
|
||||||
#define Verify(_exp) _Assert(_exp)
|
|
||||||
|
|
||||||
#define AssertMsg1(_exp, _msg, a1) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1))
|
#define AssertWarn(_exp) _AssertMsgWarn(_exp, "Assertion Failed: " #_exp)
|
||||||
#define AssertMsg2(_exp, _msg, a1, a2) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2))
|
#define AssertFatal(_exp) _AssertMsg(_exp, "Assertion Failed: " #_exp, ((void)0), true)
|
||||||
#define AssertMsg3(_exp, _msg, a1, a2, a3) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3))
|
#define AssertFatalOnce(_exp) _AssertMsgOnce(_exp, "Assertion Failed: " #_exp, true)
|
||||||
#define AssertMsg4(_exp, _msg, a1, a2, a3, a4) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4))
|
#define AssertFatalMsg(_exp, _msg) _AssertMsg(_exp, _msg, ((void)0), true)
|
||||||
#define AssertMsg5(_exp, _msg, a1, a2, a3, a4, a5) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5))
|
#define AssertFatalMsgOnce(_exp, _msg) _AssertMsgOnce(_exp, _msg, true)
|
||||||
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6))
|
#define AssertFatalFunc(_exp, _f) _AssertMsg(_exp, "Assertion Failed: " #_exp, _f, true)
|
||||||
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6))
|
#define AssertFatalEquals(_exp, _expectedValue) AssertFatalMsg2((_exp) == (_expectedValue), "Expected %d but got %d!", (_expectedValue), (_exp))
|
||||||
#define AssertMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7))
|
#define AssertFatalFloatEquals(_exp, _expectedValue, _tol) AssertFatalMsg2(fabs((_exp) - (_expectedValue)) <= (_tol), "Expected %f but got %f!", (_expectedValue), (_exp))
|
||||||
#define AssertMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8))
|
#define VerifyFatal(_exp) AssertFatal(_exp)
|
||||||
#define AssertMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) _AssertMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8, a9))
|
#define VerifyEqualsFatal(_exp, _expectedValue) AssertFatalEquals(_exp, _expectedValue)
|
||||||
|
|
||||||
#else // _DEBUG
|
#if defined(_DEBUG)
|
||||||
|
#define DbgVerifyFatal(_exp) AssertFatal(_exp)
|
||||||
|
#else
|
||||||
|
#define DbgVerifyFatal(_exp) ((void)0)
|
||||||
|
#endif
|
||||||
|
|
||||||
#define Assert(_exp) ((void)0)
|
#define AssertFatalMsg1(_exp, _msg, a1) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1).ToString())
|
||||||
#define AssertMsg(_exp, _msg) ((void)0)
|
#define AssertFatalMsg2(_exp, _msg, a1, a2) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2).ToString())
|
||||||
#define AssertFunc(_exp, _f) ((void)0)
|
#define AssertFatalMsg3(_exp, _msg, a1, a2, a3) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3).ToString())
|
||||||
#define AssertEquals(_exp, _expectedValue) ((void)0)
|
#define AssertFatalMsg4(_exp, _msg, a1, a2, a3, a4) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4).ToString())
|
||||||
#define AssertFloatEquals(_exp, _expectedValue, _tol) ((void)0)
|
#define AssertFatalMsg5(_exp, _msg, a1, a2, a3, a4, a5) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5).ToString())
|
||||||
#define Verify(_exp) (_exp)
|
#define AssertFatalMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6).ToString())
|
||||||
|
#define AssertFatalMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7).ToString())
|
||||||
|
#define AssertFatalMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8).ToString())
|
||||||
|
#define AssertFatalMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) AssertFatalMsg(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8, a9).ToString())
|
||||||
|
|
||||||
#define AssertMsg1(_exp, _msg, a1) ((void)0)
|
#else // DBGFLAG_ASSERTFATAL
|
||||||
#define AssertMsg2(_exp, _msg, a1, a2) ((void)0)
|
|
||||||
#define AssertMsg3(_exp, _msg, a1, a2, a3) ((void)0)
|
|
||||||
#define AssertMsg4(_exp, _msg, a1, a2, a3, a4) ((void)0)
|
|
||||||
#define AssertMsg5(_exp, _msg, a1, a2, a3, a4, a5) ((void)0)
|
|
||||||
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) ((void)0)
|
|
||||||
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) ((void)0)
|
|
||||||
#define AssertMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) ((void)0)
|
|
||||||
#define AssertMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) ((void)0)
|
|
||||||
#define AssertMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) ((void)0)
|
|
||||||
|
|
||||||
#endif // _DEBUG
|
#define AssertWarn(_exp) ((void)0)
|
||||||
|
#define AssertFatal(_exp) ((void)0)
|
||||||
|
#define AssertFatalOnce(_exp) ((void)0)
|
||||||
|
#define AssertFatalMsg(_exp, _msg) ((void)0)
|
||||||
|
#define AssertFatalMsgOnce(_exp, _msg) ((void)0)
|
||||||
|
#define AssertFatalFunc(_exp, _f) ((void)0)
|
||||||
|
#define AssertFatalEquals(_exp, _expectedValue) ((void)0)
|
||||||
|
#define AssertFatalFloatEquals(_exp, _expectedValue, _tol) ((void)0)
|
||||||
|
#define VerifyFatal(_exp) (_exp)
|
||||||
|
#define VerifyEqualsFatal(_exp, _expectedValue) (_exp)
|
||||||
|
|
||||||
|
#define DbgVerifyFatal(_exp) (_exp)
|
||||||
|
|
||||||
|
#define AssertFatalMsg1(_exp, _msg, a1) ((void)0)
|
||||||
|
#define AssertFatalMsg2(_exp, _msg, a1, a2) ((void)0)
|
||||||
|
#define AssertFatalMsg3(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define AssertFatalMsg4(_exp, _msg, a1, a2, a3, a4) ((void)0)
|
||||||
|
#define AssertFatalMsg5(_exp, _msg, a1, a2, a3, a4, a5) ((void)0)
|
||||||
|
#define AssertFatalMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) ((void)0)
|
||||||
|
#define AssertFatalMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) ((void)0)
|
||||||
|
#define AssertFatalMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) ((void)0)
|
||||||
|
#define AssertFatalMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) ((void)0)
|
||||||
|
|
||||||
|
#endif // DBGFLAG_ASSERTFATAL
|
||||||
|
|
||||||
|
#if defined(DBGFLAG_ASSERT)
|
||||||
|
|
||||||
|
#define Assert(_exp) _AssertMsg(_exp, "Assertion Failed: " #_exp, ((void)0), false)
|
||||||
|
#define AssertMsg_(_exp, _msg) _AssertMsg(_exp, _msg, ((void)0), false)
|
||||||
|
#define AssertOnce(_exp) _AssertMsgOnce(_exp, "Assertion Failed: " #_exp, false)
|
||||||
|
#define AssertMsgOnce(_exp, _msg) _AssertMsgOnce(_exp, _msg, false)
|
||||||
|
#define AssertFunc(_exp, _f) _AssertMsg(_exp, "Assertion Failed: " #_exp, _f, false)
|
||||||
|
#define AssertEquals(_exp, _expectedValue) AssertMsg2((_exp) == (_expectedValue), "Expected %d but got %d!", (_expectedValue), (_exp))
|
||||||
|
#define AssertFloatEquals(_exp, _expectedValue, _tol) AssertMsg2(fabs((_exp) - (_expectedValue)) <= (_tol), "Expected %f but got %f!", (_expectedValue), (_exp))
|
||||||
|
#define Verify(_exp) (_exp)
|
||||||
|
#define VerifyEquals(_exp, _expectedValue) AssertEquals(_exp, _expectedValue)
|
||||||
|
|
||||||
|
#if defined(_DEBUG)
|
||||||
|
#define DbgVerify(_exp) (_exp)
|
||||||
|
#define DbgAssert(_exp) Assert(_exp)
|
||||||
|
#define DbgAssertMsg(_exp, _msg) AssertMsg(_exp, _msg)
|
||||||
|
#define DbgAssertMsg1(_exp, _msg, a1) AssertMsg1(_exp, _msg, a1)
|
||||||
|
#define DbgAssertMsg2(_exp, _msg, a1, a2) AssertMsg2(_exp, _msg, a1, a2)
|
||||||
|
#define DbgAssertMsg3(_exp, _msg, a1, a2, a3) AssertMsg3(_exp, _msg, a1, a2, a3)
|
||||||
|
#define DbgAssertMsg4(_exp, _msg, a1, a2, a3) AssertMsg4(_exp, _msg, a1, a2, a3, a4)
|
||||||
|
#define DbgAssertMsg5(_exp, _msg, a1, a2, a3) AssertMsg5(_exp, _msg, a1, a2, a3, a4, a5)
|
||||||
|
#define DbgAssertMsg6(_exp, _msg, a1, a2, a3) AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6)
|
||||||
|
#define DbgAssertMsg7(_exp, _msg, a1, a2, a3) AssertMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7)
|
||||||
|
#define DbgAssertMsg8(_exp, _msg, a1, a2, a3) AssertMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8)
|
||||||
|
#define DbgAssertMsg9(_exp, _msg, a1, a2, a3) AssertMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9)
|
||||||
|
#else
|
||||||
|
#define DbgVerify(_exp) ((void)0)
|
||||||
|
#define DbgAssert(_exp) ((void)0)
|
||||||
|
#define DbgAssertMsg(_exp, _msg) ((void)0)
|
||||||
|
#define DbgAssertMsg1(_exp, _msg, a1) ((void)0)
|
||||||
|
#define DbgAssertMsg2(_exp, _msg, a1, a2) ((void)0)
|
||||||
|
#define DbgAssertMsg3(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg4(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg5(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg6(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg7(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg8(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg9(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define AssertMsg(_exp, _msg) AssertMsg_(_exp, _msg)
|
||||||
|
#define AssertMsg1(_exp, _msg, a1) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1).ToString())
|
||||||
|
#define AssertMsg2(_exp, _msg, a1, a2) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2).ToString())
|
||||||
|
#define AssertMsg3(_exp, _msg, a1, a2, a3) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3).ToString())
|
||||||
|
#define AssertMsg4(_exp, _msg, a1, a2, a3, a4) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4).ToString())
|
||||||
|
#define AssertMsg5(_exp, _msg, a1, a2, a3, a4, a5) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5).ToString())
|
||||||
|
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6).ToString())
|
||||||
|
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6).ToString())
|
||||||
|
#define AssertMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7).ToString())
|
||||||
|
#define AssertMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8).ToString())
|
||||||
|
#define AssertMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) AssertMsg_(_exp, CDbgFmtMsg(_msg, a1, a2, a3, a4, a5, a6, a7, a8, a9).ToString())
|
||||||
|
|
||||||
|
#else // DBGFLAG_ASSERT
|
||||||
|
|
||||||
|
#define Assert(_exp) ((void)0)
|
||||||
|
#define AssertMsg(_exp, _msg) ((void)0)
|
||||||
|
#define AssertOnce(_exp) ((void)0)
|
||||||
|
#define AssertMsgOnce(_exp, _msg) ((void)0)
|
||||||
|
#define AssertFunc(_exp, _f) ((void)0)
|
||||||
|
#define AssertEquals(_exp, _expectedValue) ((void)0)
|
||||||
|
#define AssertFloatEquals(_exp, _expectedValue, _tol) ((void)0)
|
||||||
|
|
||||||
|
#define Verify(_exp) ((void)0)
|
||||||
|
#define VerifyEquals(_exp, _expectedValue) ((void)0)
|
||||||
|
|
||||||
|
#define DbgVerify(_exp) ((void)0)
|
||||||
|
#define DbgAssert(_exp) ((void)0)
|
||||||
|
#define DbgAssertMsg(_exp, _msg) ((void)0)
|
||||||
|
#define DbgAssertMsg1(_exp, _msg, a1) ((void)0)
|
||||||
|
#define DbgAssertMsg2(_exp, _msg, a1, a2) ((void)0)
|
||||||
|
#define DbgAssertMsg3(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg4(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg5(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg6(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg7(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg8(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define DbgAssertMsg9(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
|
||||||
|
#define AssertMsg1(_exp, _msg, a1) ((void)0)
|
||||||
|
#define AssertMsg2(_exp, _msg, a1, a2) ((void)0)
|
||||||
|
#define AssertMsg3(_exp, _msg, a1, a2, a3) ((void)0)
|
||||||
|
#define AssertMsg4(_exp, _msg, a1, a2, a3, a4) ((void)0)
|
||||||
|
#define AssertMsg5(_exp, _msg, a1, a2, a3, a4, a5) ((void)0)
|
||||||
|
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) ((void)0)
|
||||||
|
#define AssertMsg6(_exp, _msg, a1, a2, a3, a4, a5, a6) ((void)0)
|
||||||
|
#define AssertMsg7(_exp, _msg, a1, a2, a3, a4, a5, a6, a7) ((void)0)
|
||||||
|
#define AssertMsg8(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8) ((void)0)
|
||||||
|
#define AssertMsg9(_exp, _msg, a1, a2, a3, a4, a5, a6, a7, a8, a9) ((void)0)
|
||||||
|
|
||||||
|
#endif // DBGFLAG_ASSERT
|
||||||
|
|
||||||
|
// default level versions (level 0)
|
||||||
|
#define Msg(...) _Msg (0, __VA_ARGS__)
|
||||||
|
#define Warning(...) _Warning(0, __VA_ARGS__)
|
||||||
|
#define Log(...) _Log (0, __VA_ARGS__)
|
||||||
|
|
||||||
|
// These looked at the "developer" functions
|
||||||
|
#define DevMsg(...) _Msg (1, __VA_ARGS__)
|
||||||
|
#define DevWarning(...) _Warning(1, __VA_ARGS__)
|
||||||
|
#define DevLog(...) _Log (1, __VA_ARGS__)
|
||||||
|
|
||||||
// These are always compiled in
|
// These are always compiled in
|
||||||
void Msg(const char *pMsg, ...);
|
void _Msg (int level, const char *pMsg, ...);
|
||||||
void DMsg(const char *pGroupName, int level, const char *pMsg, ...);
|
void _Warning(int level, const char *pMsg, ...);
|
||||||
|
void _Log (int level, const char *pMsg, ...);
|
||||||
void Warning(const char *pMsg, ...);
|
void Error (const char *pMsg, ...);
|
||||||
void DWarning(const char *pGroupName, int level, const char *pMsg, ...);
|
|
||||||
|
|
||||||
void Log(const char *pMsg, ...);
|
|
||||||
void DLog(const char *pGroupName, int level, const char *pMsg, ...);
|
|
||||||
|
|
||||||
void Error(const char *pMsg, ...);
|
|
||||||
|
|
||||||
// You can use this macro like a runtime assert macro.
|
// You can use this macro like a runtime assert macro.
|
||||||
// If the condition fails, then Error is called with the message. This macro is called
|
// If the condition fails, then Error is called with the message. This macro is called
|
||||||
@ -320,57 +355,39 @@ void Error(const char *pMsg, ...);
|
|||||||
Error msg; \
|
Error msg; \
|
||||||
}
|
}
|
||||||
|
|
||||||
// A couple of super-common dynamic spew messages, here for convenience
|
|
||||||
// These looked at the "developer" group
|
|
||||||
void DevMsg(int level, char const* pMsg, ...);
|
|
||||||
void DevWarning(int level, const char *pMsg, ...);
|
|
||||||
void DevLog(int level, const char *pMsg, ...);
|
|
||||||
|
|
||||||
// default level versions (level 1)
|
|
||||||
void DevMsg(char const* pMsg, ...);
|
|
||||||
void DevWarning(const char *pMsg, ...);
|
|
||||||
void DevLog(const char *pMsg, ...);
|
|
||||||
|
|
||||||
// Code macros, debugger interface
|
// Code macros, debugger interface
|
||||||
#ifdef _DEBUG
|
#if defined(_DEBUG)
|
||||||
#define DBG_CODE(_code) if (0) ; else { _code }
|
#define DBG_CODE(_code) if (0) ; else { _code }
|
||||||
#define DBG_DCODE(_g, _l, _code) if (IsSpewActive(_g, _l)) { _code } else {}
|
#define DBG_DCODE(_g, _l, _code) if (IsSpewActive(_g, _l)) { _code } else {}
|
||||||
#define DBG_BREAK() DebuggerBreak()
|
#define DBG_BREAK() DebuggerBreakIfDebugging()
|
||||||
#else // _DEBUG
|
#else // _DEBUG
|
||||||
#define DBG_CODE(_code) ((void)0)
|
#define DBG_CODE(_code) ((void)0)
|
||||||
#define DBG_DCODE(_g, _l, _code) ((void)0)
|
#define DBG_DCODE(_g, _l, _code) ((void)0)
|
||||||
#define DBG_BREAK() ((void)0)
|
#define DBG_BREAK() ((void)0)
|
||||||
#endif // _DEBUG
|
#endif // _DEBUG
|
||||||
|
|
||||||
// Macro to assist in asserting constant invariants during compilation
|
|
||||||
#define UID_PREFIX generated_id_
|
|
||||||
#define UID_CAT1(a, c) a ## c
|
|
||||||
#define UID_CAT2(a, c) UID_CAT1(a,c)
|
|
||||||
#define UNIQUE_ID UID_CAT2(UID_PREFIX, __LINE__)
|
|
||||||
|
|
||||||
#ifdef _DEBUG
|
|
||||||
#define COMPILE_TIME_ASSERT(pred) switch(0){case 0:case pred:;}
|
|
||||||
#define ASSERT_INVARIANT(pred) static void UNIQUE_ID() { COMPILE_TIME_ASSERT(pred) }
|
|
||||||
#else
|
|
||||||
#define COMPILE_TIME_ASSERT(pred)
|
|
||||||
#define ASSERT_INVARIANT(pred)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Templates to assist in validating pointers:
|
// Templates to assist in validating pointers:
|
||||||
// Have to use these stubs so we don't have to include windows.h here.
|
// Have to use these stubs so we don't have to include windows.h here.
|
||||||
void _AssertValidReadPtr(void *ptr, int count = 1);
|
void _AssertValidReadPtr(void *ptr, int count = 1);
|
||||||
void _AssertValidWritePtr(void *ptr, int count = 1);
|
void _AssertValidWritePtr(void *ptr, int count = 1);
|
||||||
void _AssertValidReadWritePtr(void *ptr, int count = 1);
|
void _AssertValidReadWritePtr(void *ptr, int count = 1);
|
||||||
|
|
||||||
void AssertValidStringPtr(const char *ptr, int maxchar = 0xFFFFFF);
|
#if defined(DBGFLAG_ASSERT)
|
||||||
template<class T> inline void AssertValidReadPtr(T *ptr, int count = 1) { _AssertValidReadPtr((void *)ptr, count); }
|
void AssertValidStringPtr(const char *ptr, int maxchar = 0xFFFFFF);
|
||||||
template<class T> inline void AssertValidWritePtr(T *ptr, int count = 1) { _AssertValidWritePtr((void *)ptr, count); }
|
template <class T> inline void AssertValidReadPtr(T *ptr, int count = 1) { _AssertValidReadPtr((void *)ptr, count); }
|
||||||
template<class T> inline void AssertValidReadWritePtr(T *ptr, int count = 1) { _AssertValidReadWritePtr((void *)ptr, count); }
|
template <class T> inline void AssertValidWritePtr(T *ptr, int count = 1) { _AssertValidWritePtr((void *)ptr, count); }
|
||||||
|
template <class T> inline void AssertValidReadWritePtr(T *ptr, int count = 1) { _AssertValidReadWritePtr((void *)ptr, count); }
|
||||||
#define AssertValidThis() AssertValidReadWritePtr(this, sizeof(*this))
|
#define AssertValidThis() AssertValidReadWritePtr(this, sizeof(*this))
|
||||||
|
#else
|
||||||
|
#define AssertValidStringPtr(...) ((void)0)
|
||||||
|
#define AssertValidReadPtr(...) ((void)0)
|
||||||
|
#define AssertValidWritePtr(...) ((void)0)
|
||||||
|
#define AssertValidReadWritePtr(...) ((void)0)
|
||||||
|
#define AssertValidThis() ((void)0)
|
||||||
|
#endif // #if defined(DBGFLAG_ASSERT)
|
||||||
|
|
||||||
// Macro to protect functions that are not reentrant
|
// Macro to protect functions that are not reentrant
|
||||||
#ifdef _DEBUG
|
#if defined(_DEBUG)
|
||||||
class CReentryGuard
|
class CReentryGuard
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
@ -406,6 +423,7 @@ public:
|
|||||||
va_start(arg_ptr, pszFormat);
|
va_start(arg_ptr, pszFormat);
|
||||||
_vsnprintf(m_szBuf, sizeof(m_szBuf) - 1, pszFormat, arg_ptr);
|
_vsnprintf(m_szBuf, sizeof(m_szBuf) - 1, pszFormat, arg_ptr);
|
||||||
va_end(arg_ptr);
|
va_end(arg_ptr);
|
||||||
|
|
||||||
m_szBuf[sizeof(m_szBuf) - 1] = '\0';
|
m_szBuf[sizeof(m_szBuf) - 1] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -414,14 +432,11 @@ public:
|
|||||||
return m_szBuf;
|
return m_szBuf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *ToString() const
|
||||||
|
{
|
||||||
|
return m_szBuf;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
char m_szBuf[256];
|
char m_szBuf[256];
|
||||||
};
|
};
|
||||||
|
|
||||||
// Embed debug info in each file.
|
|
||||||
//#ifdef _WIN32
|
|
||||||
//#ifdef _DEBUG
|
|
||||||
//#pragma comment(compiler)
|
|
||||||
//#pragma comment(exestr,"*** DEBUG file detected, Last Compile: " __DATE__ ", " __TIME__ " ***")
|
|
||||||
//#endif
|
|
||||||
//#endif
|
|
||||||
|
@ -41,7 +41,14 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Used to step into the debugger
|
// Used to step into the debugger
|
||||||
#define DebuggerBreak() __asm { int 3 }
|
#if defined(__GNUC__) && !defined(__clang__)
|
||||||
|
#define DebuggerBreak() __asm__ __volatile__("int3;")
|
||||||
|
#else
|
||||||
|
#define DebuggerBreak() __asm { int 3 }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define DebuggerBreakIfDebugging() if (Plat_IsInDebugSession()) { DebuggerBreak(); }
|
||||||
|
#define DebuggerSegFault() { volatile int *null = 0; *null = 0; }
|
||||||
|
|
||||||
// C functions for external declarations that call the appropriate C++ methods
|
// C functions for external declarations that call the appropriate C++ methods
|
||||||
#ifndef EXPORT
|
#ifndef EXPORT
|
||||||
@ -105,6 +112,37 @@
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
//
|
||||||
|
// Macro to assist in asserting constant invariants during compilation
|
||||||
|
|
||||||
|
#define STRINGIFY_INTERNAL(x) #x
|
||||||
|
#define STRINGIFY(x) STRINGIFY_INTERNAL(x)
|
||||||
|
|
||||||
|
// This implementation of compile time assert has zero cost (so it can safely be
|
||||||
|
// included in release builds) and can be used at file scope or function scope.
|
||||||
|
#define COMPILE_TIME_ASSERT(pred) static_assert(pred, "Compile time assert constraint is not true: " #pred)
|
||||||
|
|
||||||
|
// ASSERT_INVARIANT used to be needed in order to allow COMPILE_TIME_ASSERTs at global
|
||||||
|
// scope. However the new COMPILE_TIME_ASSERT macro supports that by default.
|
||||||
|
#define ASSERT_INVARIANT(pred) COMPILE_TIME_ASSERT(pred)
|
||||||
|
|
||||||
|
// Macro to assist in asserting constant invariants during compilation
|
||||||
|
//
|
||||||
|
// If available use static_assert instead of weird language tricks. This
|
||||||
|
// leads to much more readable messages when compile time assert constraints
|
||||||
|
// are violated.
|
||||||
|
#if !defined(OSX) && (_MSC_VER > 1500 || __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))
|
||||||
|
#define PLAT_COMPILE_TIME_ASSERT(pred) static_assert(pred, "Compile time assert constraint is not true: " #pred)
|
||||||
|
#else
|
||||||
|
#define PLAT_COMPILE_TIME_ASSERT(pred) typedef int UNIQUE_ID[ (pred) ? 1 : -1]
|
||||||
|
#endif
|
||||||
|
|
||||||
|
bool Plat_IsInDebugSession();
|
||||||
|
void Plat_DebugString(const char *psz);
|
||||||
|
void Plat_OutputDebugString(const char *psz);
|
||||||
|
void Plat_OutputDebugStringRaw(const char *psz);
|
||||||
|
const char *Plat_GetCommandLine();
|
||||||
|
|
||||||
// Methods to invoke the constructor, copy constructor, and destructor
|
// Methods to invoke the constructor, copy constructor, and destructor
|
||||||
template <class T>
|
template <class T>
|
||||||
inline void Construct(T *pMemory)
|
inline void Construct(T *pMemory)
|
||||||
|
113
regamedll/public/tier0/platform_posix.cpp
Normal file
113
regamedll/public/tier0/platform_posix.cpp
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the author gives permission to
|
||||||
|
* link the code of this program with the Half-Life Game Engine ("HL
|
||||||
|
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
||||||
|
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
||||||
|
* respects for all of the code used other than the HL Engine and MODs
|
||||||
|
* from Valve. If you modify this file, you may extend this exception
|
||||||
|
* to your version of the file, but you are not obligated to do so. If
|
||||||
|
* you do not wish to do so, delete this exception statement from your
|
||||||
|
* version.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "precompiled.h"
|
||||||
|
|
||||||
|
bool Plat_IsInDebugSession()
|
||||||
|
{
|
||||||
|
#if defined(OSX)
|
||||||
|
int mib[4];
|
||||||
|
struct kinfo_proc info;
|
||||||
|
size_t size;
|
||||||
|
mib[0] = CTL_KERN;
|
||||||
|
mib[1] = KERN_PROC;
|
||||||
|
mib[2] = KERN_PROC_PID;
|
||||||
|
mib[3] = getpid();
|
||||||
|
size = sizeof(info);
|
||||||
|
info.kp_proc.p_flag = 0;
|
||||||
|
sysctl(mib, 4, &info, &size, NULL, 0);
|
||||||
|
bool result = ((info.kp_proc.p_flag & P_TRACED) == P_TRACED);
|
||||||
|
return result;
|
||||||
|
#elif defined(_LINUX)
|
||||||
|
char s[256];
|
||||||
|
snprintf(s, 256, "/proc/%d/cmdline", getppid());
|
||||||
|
FILE *fp = fopen(s, "r");
|
||||||
|
if (fp)
|
||||||
|
{
|
||||||
|
fread(s, 256, 1, fp);
|
||||||
|
fclose(fp);
|
||||||
|
return (0 == strncmp(s, "gdb", 3));
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_OutputDebugStringRaw(const char *psz)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s", psz);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_OutputDebugString(const char *psz)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s", psz);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_DebugString(const char *psz)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s", psz);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char g_CmdLine[2048]{};
|
||||||
|
|
||||||
|
const char *Plat_GetCommandLine()
|
||||||
|
{
|
||||||
|
#if defined(_LINUX)
|
||||||
|
static bool commandline_initialized = false;
|
||||||
|
if (!commandline_initialized)
|
||||||
|
{
|
||||||
|
commandline_initialized = true;
|
||||||
|
|
||||||
|
FILE *fp = fopen("/proc/self/cmdline", "rb");
|
||||||
|
if (fp)
|
||||||
|
{
|
||||||
|
size_t nCharRead = 0;
|
||||||
|
|
||||||
|
// -1 to leave room for the '\0'
|
||||||
|
nCharRead = fread(g_CmdLine, sizeof(g_CmdLine[0]), ARRAYSIZE(g_CmdLine) - 1, fp);
|
||||||
|
if (feof(fp) && !ferror(fp)) // Should have read the whole command line without error
|
||||||
|
{
|
||||||
|
Assert(nCharRead < ARRAYSIZE(g_CmdLine));
|
||||||
|
|
||||||
|
for (int i = 0; i < nCharRead; i++)
|
||||||
|
{
|
||||||
|
if (!g_CmdLine[i])
|
||||||
|
g_CmdLine[i] = ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
g_CmdLine[nCharRead] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
Assert(g_CmdLine[0]);
|
||||||
|
}
|
||||||
|
#endif // LINUX
|
||||||
|
|
||||||
|
return g_CmdLine;
|
||||||
|
}
|
57
regamedll/public/tier0/platform_win32.cpp
Normal file
57
regamedll/public/tier0/platform_win32.cpp
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* In addition, as a special exception, the author gives permission to
|
||||||
|
* link the code of this program with the Half-Life Game Engine ("HL
|
||||||
|
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
||||||
|
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
||||||
|
* respects for all of the code used other than the HL Engine and MODs
|
||||||
|
* from Valve. If you modify this file, you may extend this exception
|
||||||
|
* to your version of the file, but you are not obligated to do so. If
|
||||||
|
* you do not wish to do so, delete this exception statement from your
|
||||||
|
* version.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "precompiled.h"
|
||||||
|
|
||||||
|
bool Plat_IsInDebugSession()
|
||||||
|
{
|
||||||
|
return IsDebuggerPresent() != FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_OutputDebugStringRaw(const char *psz)
|
||||||
|
{
|
||||||
|
OutputDebugString(psz);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_OutputDebugString(const char *psz)
|
||||||
|
{
|
||||||
|
static char buf[4096];
|
||||||
|
int len = Q_snprintf(buf, sizeof(buf), "%s", psz);
|
||||||
|
Assert(len > 0);
|
||||||
|
OutputDebugString(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Plat_DebugString(const char *psz)
|
||||||
|
{
|
||||||
|
Plat_OutputDebugString(psz);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *Plat_GetCommandLine()
|
||||||
|
{
|
||||||
|
return GetCommandLineA();
|
||||||
|
}
|
44
regamedll/public/tier0/resource.h
Normal file
44
regamedll/public/tier0/resource.h
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* Copyright (c) 1996-2002, Valve LLC. All rights reserved.
|
||||||
|
*
|
||||||
|
* This product contains software technology licensed from Id
|
||||||
|
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
|
||||||
|
* All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Use, distribution, and modification of this source code and/or resulting
|
||||||
|
* object code is restricted to non-commercial enhancements to products from
|
||||||
|
* Valve LLC. All other use, distribution, or modification is prohibited
|
||||||
|
* without written permission from Valve LLC.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
//{{NO_DEPENDENCIES}}
|
||||||
|
// Microsoft Developer Studio generated include file.
|
||||||
|
// Used by assert_dialog.rc
|
||||||
|
//
|
||||||
|
|
||||||
|
#define IDD_ASSERT_DIALOG 101
|
||||||
|
#define IDC_FILENAME_CONTROL 1000
|
||||||
|
#define IDC_LINE_CONTROL 1001
|
||||||
|
#define IDC_IGNORE_FILE 1002
|
||||||
|
#define IDC_IGNORE_NEARBY 1003
|
||||||
|
#define IDC_IGNORE_NUMLINES 1004
|
||||||
|
#define IDC_IGNORE_THIS 1005
|
||||||
|
#define IDC_BREAK 1006
|
||||||
|
#define IDC_IGNORE_ALL 1008
|
||||||
|
#define IDC_IGNORE_ALWAYS 1009
|
||||||
|
#define IDC_IGNORE_NUMTIMES 1010
|
||||||
|
#define IDC_ASSERT_MSG_CTRL 1011
|
||||||
|
#define IDC_NOID -1
|
||||||
|
|
||||||
|
// Next default values for new objects
|
||||||
|
//
|
||||||
|
#ifdef APSTUDIO_INVOKED
|
||||||
|
#ifndef APSTUDIO_READONLY_SYMBOLS
|
||||||
|
#define _APS_NEXT_RESOURCE_VALUE 103
|
||||||
|
#define _APS_NEXT_COMMAND_VALUE 40001
|
||||||
|
#define _APS_NEXT_CONTROL_VALUE 1005
|
||||||
|
#define _APS_NEXT_SYMED_VALUE 101
|
||||||
|
#endif
|
||||||
|
#endif
|
Loading…
x
Reference in New Issue
Block a user