You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
303 lines
11 KiB
303 lines
11 KiB
//=--------------------------------------------------------------------------=
|
|
// Macros.h
|
|
//=--------------------------------------------------------------------------=
|
|
// Copyright 1997 Microsoft Corporation. All Rights Reserved.
|
|
//
|
|
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
|
|
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
|
|
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
|
|
// PARTICULAR PURPOSE.
|
|
//=--------------------------------------------------------------------------=
|
|
// Handy macros like the ones we use in the VB code base.
|
|
//=--------------------------------------------------------------------------=
|
|
#ifndef _MACROS_H_
|
|
|
|
#include <globals.h>
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Debugging Heap Memory Leaks:
|
|
// Macros and definitions
|
|
//---------------------------------------------------------------------------
|
|
#ifdef DEBUG
|
|
typedef char * LPSZ;
|
|
#define NUM_INST_TABLE_ENTRIES 1024
|
|
#define Deb_FILELINEPROTO , LPSTR lpszFile, UINT line
|
|
#define Deb_FILELINECALL , __FILE__, __LINE__
|
|
#define Deb_FILELINEPASS , lpszFile, line
|
|
#else // DEBUG
|
|
#define Deb_FILELINEPROTO
|
|
#define Deb_FILELINECALL
|
|
#define Deb_FILELINEPASS
|
|
#endif // DEBUG
|
|
|
|
|
|
// Function prototypes for the actual implementations of the debug heap wrapper functions.
|
|
#ifdef DEBUG
|
|
LPVOID CtlHeapAllocImpl(HANDLE g_hHeap, DWORD dwFlags, DWORD dwBytes Deb_FILELINEPROTO);
|
|
LPVOID CtlHeapReAllocImpl(HANDLE g_hHeap, DWORD dwFlags, LPVOID lpvMem, DWORD dwBytes Deb_FILELINEPROTO);
|
|
BOOL CtlHeapFreeImpl(HANDLE g_hHeap, DWORD dwFlags, LPVOID lpvMem);
|
|
extern VOID CheckForLeaks(VOID);
|
|
inline UINT HashInst(VOID * pv) { return ((UINT) ((ULONG)pv >> 4)) % NUM_INST_TABLE_ENTRIES; } // Hashing function
|
|
#endif // DEBUG
|
|
|
|
|
|
#define OleAlloc(dwBytes) CoTaskMemAlloc(dwBytes)
|
|
#define OleReAlloc(lpvMem, dwBytes) CoTaskMemReAlloc(lpvMem, dwBytes)
|
|
#define OleFree(lpvMem) CoTaskMemFree(lpvMem)
|
|
#define New new (g_hHeap Deb_FILELINECALL)
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
// Macros for our memory leak detection
|
|
|
|
#ifdef DEBUG
|
|
// Use these functions to allocate memory from the global heap.
|
|
#define CtlHeapAlloc(g_hHeap, dwFlags, dwBytes) CtlHeapAllocImpl(g_hHeap, dwFlags, dwBytes Deb_FILELINECALL)
|
|
#define CtlHeapReAlloc(g_hHeap, dwFlags, lpvMem, dwBytes) CtlHeapReAllocImpl(g_hHeap, dwFlags, lpvMem, dwBytes Deb_FILELINECALL)
|
|
#define CtlHeapFree(g_hHeap, dwFlags, lpvMem) CtlHeapFreeImpl(g_hHeap, dwFlags, lpvMem)
|
|
#define CtlAlloc(dwBytes) CtlHeapAllocImpl(g_hHeap, 0, dwBytes Deb_FILELINECALL)
|
|
#define CtlAllocZero(dwBytes) CtlHeapAllocImpl(g_hHeap, HEAP_ZERO_MEMORY, dwBytes Deb_FILELINECALL)
|
|
#define CtlReAlloc(lpvMem, dwBytes) CtlHeapReAllocImpl(g_hHeap, 0, lpvMem, dwBytes Deb_FILELINECALL)
|
|
#define CtlReAllocZero(lpvMem, dwBytes) CtlHeapReAllocImpl(g_hHeap, HEAP_ZERO_MEMORY, lpvMem, dwBytes Deb_FILELINECALL)
|
|
#define CtlFree(lpvMem) CtlHeapFreeImpl(g_hHeap, 0, lpvMem)
|
|
#define NewCtlHeapAlloc(g_hHeap, dwFlags, dwBytes) CtlHeapAllocImpl(g_hHeap, dwFlags, dwBytes Deb_FILELINEPASS)
|
|
|
|
#else
|
|
// In retail on Win32 we map directly to the Win32 Heap API
|
|
#define CtlHeapAlloc(g_hHeap, dwFlags, dwBytes) HeapAlloc(g_hHeap, dwFlags, dwBytes)
|
|
#define CtlHeapReAlloc(g_hHeap, dwFlags, lpvMem, dwBytes) HeapReAlloc(g_hHeap, dwFlags, lpvMem, dwBytes)
|
|
#define CtlHeapFree(g_hHeap, dwFlags, lpvMem) HeapFree(g_hHeap, dwFlags, lpvMem)
|
|
#define CtlAlloc(dwBytes) HeapAlloc(g_hHeap, 0, dwBytes)
|
|
#define CtlAllocZero(dwBytes) HeapAlloc(g_hHeap, HEAP_ZERO_MEMORY, dwBytes)
|
|
#define CtlReAlloc(lpvMem, dwBytes) HeapReAlloc(g_hHeap, 0, lpvMem, dwBytes)
|
|
#define CtlReAllocZero(lpvMem, dwBytes) HeapReAlloc(g_hHeap, HEAP_ZERO_MEMORY, lpvMem, dwBytes)
|
|
#define CtlFree(lpvMem) HeapFree(g_hHeap, 0, lpvMem)
|
|
#define NewCtlHeapAlloc(g_hHeap, dwFlags, dwBytes) HeapAlloc(g_hHeap, dwFlags, dwBytes)
|
|
#endif // DEBUG
|
|
|
|
// Macros for header files
|
|
// SZTHISFILE cannot be defined in header files. These macros avoid its re-definition
|
|
#ifdef DEBUG
|
|
#define CtlHeapAlloc_Header_Util(g_hHeap, dwFlags, dwBytes) CtlHeapAllocImpl(g_hHeap, dwFlags, dwBytes, __FILE__, __LINE__);
|
|
#else
|
|
#define CtlHeapAlloc_Header_Util(g_hHeap, dwFlags, dwBytes) HeapAlloc (g_hHeap, dwFlags, dwBytes);
|
|
#endif // DEBUG
|
|
|
|
|
|
//=---------------------------------------------------------------------------=
|
|
// class CtlNewDelete
|
|
//
|
|
// This class MUST be inherited by any class in the CTLS Tree that wants
|
|
// to use "new" or "delete" to allocate/free.
|
|
//
|
|
// This class has no data members or virtual functions, so it does not
|
|
// change the size of any instances of classes which inherit from it.
|
|
//=---------------------------------------------------------------------------=
|
|
class CtlNewDelete
|
|
{
|
|
public:
|
|
inline void * _cdecl operator new (size_t size, HANDLE g_hHeap Deb_FILELINEPROTO);
|
|
inline void _cdecl operator delete (LPVOID pv, HANDLE g_hHeap Deb_FILELINEPROTO);
|
|
inline void _cdecl operator delete (LPVOID pv);
|
|
};
|
|
|
|
|
|
//=---------------------------------------------------------------------------=
|
|
// CtlNewDelete::operator new
|
|
//=---------------------------------------------------------------------------=
|
|
// Parameters:
|
|
// size_t - [in] what size do we alloc
|
|
// g_hHeap - [in] our global heap
|
|
// lpszFile - [in] what file are we allocating from
|
|
// line - [in] what line # do we allocate from
|
|
//
|
|
// Output:
|
|
// VOID * - new memory.
|
|
//
|
|
// Notes:
|
|
//
|
|
// We don't need to worry about ENTERCRITICALSECTION1 here.
|
|
// New is either called by the c run-time or after
|
|
// g_hHeap has been initialized in DllMain PROCESS_ATTACH.
|
|
// In either case this call is synchronized.
|
|
// If we try putting ENTERCRITICALSECTION1 here, we will
|
|
// blow up if the c run-time is attempting to initialize
|
|
// our static objects such as objects w/ global constructors.
|
|
inline void * _cdecl CtlNewDelete::operator new (size_t size, HANDLE g_hHeap Deb_FILELINEPROTO)
|
|
{
|
|
if (!g_hHeap)
|
|
{
|
|
g_hHeap = GetProcessHeap();
|
|
return g_hHeap ? NewCtlHeapAlloc(g_hHeap, 0, size) : NULL;
|
|
}
|
|
|
|
return NewCtlHeapAlloc(g_hHeap, 0, size);
|
|
}
|
|
|
|
//=---------------------------------------------------------------------------=
|
|
// CtlNewDelete::operator delete
|
|
//=---------------------------------------------------------------------------=
|
|
// retail case just uses win32 Local* heap mgmt functions
|
|
//
|
|
// Parameters:
|
|
// void * - [in] free me!
|
|
//
|
|
// Notes:
|
|
//
|
|
inline void _cdecl CtlNewDelete::operator delete ( void *ptr, HANDLE g_hHeap Deb_FILELINEPROTO)
|
|
{
|
|
if (ptr)
|
|
CtlHeapFree(g_hHeap, 0, ptr);
|
|
}
|
|
inline void _cdecl CtlNewDelete::operator delete ( void *ptr)
|
|
{
|
|
if (ptr)
|
|
CtlHeapFree(g_hHeap, 0, ptr);
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Convert C's BOOL to Basic's BOOL
|
|
//---------------------------------------------------------------------------
|
|
#define BASICBOOLOF(f) ((f) ? -1 : 0 )
|
|
#define FMAKEBOOL(f) (!!(f))
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Code macros
|
|
//---------------------------------------------------------------------------
|
|
#define SWAP(type, a, b) { type _z_=(a); (a)=(b); (b)=_z_; }
|
|
|
|
#if 0
|
|
#define loop while(1) // "loop" keyword for infinite loops
|
|
#endif // 0
|
|
|
|
// "scope" keyword for { } that are used just to introduce a new name scope.
|
|
// It's disconcerting to see { } without some keyword in front of the {...
|
|
#define scope
|
|
|
|
|
|
#define ADDREF(PUNK) \
|
|
{if (PUNK) (PUNK)->AddRef();}
|
|
|
|
#ifndef RELEASE
|
|
#define RELEASE(PUNK) \
|
|
{if (PUNK) {LPUNKNOWN punkXXX = (PUNK); (PUNK) = NULL; punkXXX->Release();}}
|
|
#endif //RELEASE
|
|
|
|
// In some multiple inheritance cases you need to dis-ambiguate which IUnknown implementation to use
|
|
#define RELEASETYPE(PUNK,TYPE) \
|
|
{if (PUNK) {LPUNKNOWN punkXXX = (TYPE *)(PUNK); (PUNK) = NULL; punkXXX->Release();}}
|
|
|
|
#define FREESTRING(bstrVal) \
|
|
{if((bstrVal) != NULL) {SysFreeString((bstrVal)); (bstrVal) = NULL; }}
|
|
|
|
//---------------------------------------------------------------------
|
|
// Debug macros
|
|
//---------------------------------------------------------------------
|
|
#if DEBUG
|
|
void _DebugPrintf(char* pszFormat, ...);
|
|
void _DebugPrintIf(BOOL fPrint, char* pszFormat, ...);
|
|
|
|
#define DebugPrintf _DebugPrintf
|
|
#define DebugPrintIf _DebugPrintIf
|
|
|
|
#else // DEBUG || DEBUG_OUTPUT_ON
|
|
|
|
inline void _DebugNop(...) {}
|
|
|
|
#define DebugPrintf 1 ? (void)0 : _DebugNop
|
|
#define DebugPrintIf 1 ? (void)0 : _DebugNop
|
|
#define DebugMessageBox 1 ? (void)0 : _DebugNop
|
|
|
|
#endif // DEBUG
|
|
|
|
//---------------------------------------------------------------------
|
|
// Error handling macros
|
|
//---------------------------------------------------------------------
|
|
|
|
#ifdef DEBUG
|
|
extern HRESULT HrDebugTraceReturn(HRESULT hr, char *szFile, int iLine);
|
|
#define RRETURN(hr) return HrDebugTraceReturn(hr, _szThisFile, __LINE__)
|
|
#else
|
|
#define RRETURN(hr) return (hr)
|
|
#endif //DEBUG
|
|
|
|
|
|
// FAILEDHR : Same as FAILED(hr), but prints a debug message if the test failed.
|
|
#if DEBUG
|
|
#define FAILEDHR(HR) _FAILEDHR(HR, _szThisFile, __LINE__)
|
|
inline BOOL _FAILEDHR(HRESULT hr, char* pszFile, int iLine)
|
|
{
|
|
if (FAILED(hr))
|
|
HrDebugTraceReturn(hr, pszFile, iLine);
|
|
return FAILED(hr);
|
|
}
|
|
#else
|
|
#define FAILEDHR(HR) FAILED(HR)
|
|
#endif
|
|
|
|
// SUCCEEDEDHR : Same as SUCCEEDED(hr), but prints a debug message if the test failed.
|
|
#define SUCCEEDEDHR(HR) (!FAILEDHR(HR))
|
|
|
|
// Print a debug message if FAILED(hr).
|
|
#if DEBUG
|
|
#define CHECKHR(HR) _CHECKHR(HR, _szThisFile, __LINE__)
|
|
inline void _CHECKHR(HRESULT hr, char* pszFile, int iLine)
|
|
{
|
|
if (FAILED(hr))
|
|
HrDebugTraceReturn(hr, pszFile, iLine);
|
|
}
|
|
#else
|
|
#define CHECKHR(HR) HR
|
|
#endif
|
|
|
|
#define IfErrGoto(EXPR, LABEL) \
|
|
{ err = (EXPR); if (err) goto LABEL; }
|
|
|
|
#define IfErrRet(EXPR) \
|
|
{ err = (EXPR); if (err) return err; };
|
|
|
|
#define IfErrGo(EXPR) IfErrGoto(EXPR, Error)
|
|
|
|
|
|
#define IfFailGoto(EXPR, LABEL) \
|
|
{ hr = (EXPR); if(FAILEDHR(hr)) goto LABEL; }
|
|
|
|
#ifndef IfFailRet
|
|
#define IfFailRet(EXPR) \
|
|
{ hr = (EXPR); if(FAILED(hr)) RRETURN(hr); }
|
|
#endif // IfFailRet
|
|
|
|
#define IfFailGo(EXPR) IfFailGoto(EXPR, Error)
|
|
|
|
|
|
#define IfFalseGoto(EXPR, HR, LABEL) \
|
|
{ if(!(EXPR)) { hr = (HR); goto LABEL; } }
|
|
|
|
|
|
#define IfFalseRet(EXPR, HR) \
|
|
{ if(!(EXPR)) RRETURN(HR); }
|
|
|
|
#define IfFalseGo(EXPR, HR) IfFalseGoto(EXPR, HR, Error)
|
|
|
|
|
|
#if DEBUG
|
|
#define CHECKRESULT(x) ASSERT((x)==NOERROR,"");
|
|
#else // DEBUG
|
|
#define CHECKRESULT(x) (x)
|
|
#endif // DEBUG
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
// STATICF is for static functions. In retail we disable this in order to
|
|
// do better function reordering via the linker.
|
|
//---------------------------------------------------------------------------
|
|
#if !defined(STATICF)
|
|
#ifdef DEBUG
|
|
#define STATICF static
|
|
#else // DEBUG
|
|
#define STATICF
|
|
#endif // DEBUG
|
|
#endif
|
|
|
|
|
|
#define _MACROS_H_
|
|
#endif // _MACROS_H_
|