2020-09-30 16:53:55 +02:00

391 lines
19 KiB
C++

//____________________________________________________________________________
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1995 - 1996.
//
// File: macros.h
//
// Contents: Useful macros
//
// Macros: ARRAYLEN
//
// BREAK_ON_FAIL(hresult)
// BREAK_ON_FAIL(hresult)
//
// DECLARE_IUNKNOWN_METHODS
// DECLARE_STANDARD_IUNKNOWN
// IMPLEMENT_STANDARD_IUNKNOWN
//
// SAFE_RELEASE
//
// DECLARE_SAFE_INTERFACE_PTR_MEMBERS
//
// History: 6/3/1996 RaviR Created
// 7/23/1996 JonN Added exception handling macros
//
//____________________________________________________________________________
#ifndef _MACROS_H_
#define _MACROS_H_
//____________________________________________________________________________
//
// Macro: ARRAYLEN
//
// Purpose: To determine the length of an array.
//____________________________________________________________________________
//
#define ARRAYLEN(a) (sizeof(a) / sizeof((a)[0]))
//____________________________________________________________________________
//
// Macros: BREAK_ON_FAIL(hresult), BREAK_ON_ERROR(lastError)
//
// Purpose: To break out of a loop on error.
//____________________________________________________________________________
//
#define BREAK_ON_FAIL(hr) if (FAILED(hr)) { break; } else 1;
#define BREAK_ON_ERROR(lr) if (lr != ERROR_SUCCESS) { break; } else 1;
//____________________________________________________________________________
//
// Macros: DwordAlign(n)
//____________________________________________________________________________
//
#define DwordAlign(n) (((n) + 3) & ~3)
//____________________________________________________________________________
//
// Macros: SAFE_RELEASE
//____________________________________________________________________________
//
#ifndef SAFE_RELEASE
#define SAFE_RELEASE(punk) \
if (punk != NULL) \
{ \
punk##->Release(); \
punk = NULL; \
} \
else \
{ \
TRACE(_T("Release called on NULL interface ptr")); \
}
#endif // SAFE_RELEASE
//____________________________________________________________________________
//
// Macro: DECLARE_IUNKNOWN_METHODS
//
// Purpose: This declares the set of IUnknown methods and is for
// general-purpose use inside classes that inherit from IUnknown
//____________________________________________________________________________
//
#define DECLARE_IUNKNOWN_METHODS \
STDMETHOD(QueryInterface) (REFIID riid, LPVOID FAR* ppvObj); \
STDMETHOD_(ULONG,AddRef) (void); \
STDMETHOD_(ULONG,Release) (void)
//____________________________________________________________________________
//
// Macro: DECLARE_STANDARD_IUNKNOWN
//
// Purpose: This is for use in declaring non-aggregatable objects. It
// declares the IUnknown methods and reference counter, m_ulRefs.
// m_ulRefs should be initialized to 1 in the constructor of
// the object
//____________________________________________________________________________
//
#define DECLARE_STANDARD_IUNKNOWN \
DECLARE_IUNKNOWN_METHODS; \
ULONG m_ulRefs
//____________________________________________________________________________
//
// Macro: IMPLEMENT_STANDARD_IUNKNOWN
//
// Purpose: Partial implementaion of standard IUnknown.
//
// Note: This does NOT implement QueryInterface, which must be
// implemented by each object
//____________________________________________________________________________
//
#define IMPLEMENT_STANDARD_IUNKNOWN(cls) \
STDMETHODIMP_(ULONG) cls##::AddRef() \
{ return InterlockedIncrement((LONG*)&m_ulRefs); } \
STDMETHODIMP_(ULONG) cls##::Release() \
{ ULONG ulRet = InterlockedDecrement((LONG*)&m_ulRefs); \
if (0 == ulRet) { delete this; } \
return ulRet; }
// ISSUE-2002/04/01-JonN This is not used, remove it
//____________________________________________________________________________
//
// Macro: DECLARE_SAFE_INTERFACE_PTR_MEMBERS(cls, Interface, m_iptr)
//
// Purpose: Make the interface ptr 'm_iptr' of interface type 'Interface'
// a safe pointer for the given class 'cls', by adding methods and
// overloading operators to manipulate the pointer m_iptr.
//
// History: 6/3/1996 RaviR Created
//
// Notes: Adds safe interface pointer member functions to the given
// class for the given OLE interface. 'm_iptr' is the member
// variable name of the interface ptr in the given class.
//
// The Copy function creates a valid additional copy of
// the captured pointer (following the AddRef/Release protocol)
// so can be used to hand out copies from a safe pointer declared
// as a member of some other class.
//
// The 'Transfer' function transfers the interface pointer, and
// invalidates its member value (by setting it to NULL).
//
// To release the existing interface ptr and set it to a new
// instance use the 'Set' member fuction. This method takes a
// parameter which specifies whether the new pointer should be
// AddRef'd, defaulting to TRUE.
//
// The following methods manipulate the interface pointer with
// out following the AddRef/Release protocol: Transfer, Attach
// and Detach.
//____________________________________________________________________________
//
#define DECLARE_SAFE_INTERFACE_PTR_MEMBERS(cls, Interface, m_iptr) \
\
public: \
cls##(Interface * iptr=NULL, BOOL fInc=TRUE) : m_iptr(iptr) \
{ \
if (fInc && (m_iptr != NULL)) \
{ \
m_iptr->AddRef(); \
} \
} \
\
~##cls##() \
{ \
if (m_iptr != NULL) \
{ \
m_iptr->Release(); \
m_iptr = NULL; \
} \
} \
\
inline BOOL IsNull(void) \
{ \
return (m_iptr == NULL); \
} \
\
void Transfer(Interface **piptr) \
{ \
*piptr = m_iptr; \
m_iptr = NULL; \
} \
\
void Copy(Interface **piptr) \
{ \
*piptr = m_iptr; \
if (m_iptr != NULL) \
m_iptr->AddRef(); \
} \
\
void Set(Interface* iptr, BOOL fInc = TRUE) \
{ \
if (m_iptr) \
{ \
m_iptr->Release(); \
} \
m_iptr = iptr; \
if (fInc && m_iptr) \
{ \
m_iptr->AddRef(); \
} \
} \
\
void SafeRelease(void) \
{ \
if (m_iptr) \
{ \
m_iptr->Release(); \
m_iptr = NULL; \
} \
} \
\
void SimpleRelease(void) \
{ \
ASSERT(m_iptr != NULL); \
m_iptr->Release(); \
m_iptr = NULL; \
} \
\
void Attach(Interface* iptr) \
{ \
ASSERT(m_iptr == NULL); \
m_iptr = iptr; \
} \
\
void Detach(void) \
{ \
m_iptr = NULL; \
} \
\
Interface * operator-> () { return m_iptr; } \
Interface& operator * () { return *m_iptr; } \
operator Interface *() { return m_iptr; } \
\
Interface ** operator &() \
{ \
ASSERT(m_iptr == NULL); \
return &m_iptr; \
} \
\
Interface *Self(void) { return m_iptr; } \
\
private: \
void operator= (const cls &) {;} \
cls(const cls &){;}
//____________________________________________________________________________
//
// Macro: EXCEPTION HANDLING MACROS
//
// Purpose: Provide standard macros for exception-handling in
// OLE servers.
//
// History: 7/23/1996 JonN Created
//
// Notes: Declare USE_HANDLE_MACROS("Component name") in each source
// file before these are used.
//
// These macros can only be used in function calls which return
// type HRESULT.
//
// Bracket routines which can generate exceptions
// with STANDARD_TRY and STANDARD_CATCH.
//
// Where these routines are COM methods requiring MFC
// support, use MFC_TRY and MFC_CATCH instead.
//____________________________________________________________________________
//
#define USE_HANDLE_MACROS(component) \
static TCHAR* You_forgot_to_declare_USE_HANDLE_MACROS = _T(component);
#define STANDARD_TRY \
try {
#define MFC_TRY \
AFX_MANAGE_STATE(AfxGetStaticModuleState( )); \
STANDARD_TRY
// ISSUE-2002/04/01-JonN remove ENDMETHOD_READBLOCK
//
// CODEWORK don't quite have ENDMETHOD_READBLOCK working yet
//
#ifdef DEBUG
#define ENDMETHOD_STRING \
"%s: The unexpected error can be identified as \"%s\" context %n\n"
#define ENDMETHOD_READBLOCK \
{ \
TCHAR szError[MAX_PATH]; \
UINT nHelpContext = 0; \
if ( e->GetErrorMessage( szError, MAX_PATH, &nHelpContext ) ) \
{ \
TRACE( ENDMETHOD_STRING, \
You_forgot_to_declare_USE_HANDLE_MACROS, \
szError, \
nHelpContext ); \
} \
}
#else
#define ENDMETHOD_READBLOCK
#endif
#define ERRSTRING_MEMORY "%s: An out-of-memory error occurred\n"
#define ERRSTRING_FILE "%s: File error 0x%lx occurred on file \"%s\"\n"
#define ERRSTRING_OLE "%s: OLE error 0x%lx occurred\n"
#define ERRSTRING_UNEXPECTED "%s: An unexpected error occurred\n"
#define BADPARM_STRING "%s: Bad string parameter\n"
#define BADPARM_POINTER "%s: Bad pointer parameter\n"
#define TRACEERR(s) TRACE( s, You_forgot_to_declare_USE_HANDLE_MACROS )
#define TRACEERR1(s,a) TRACE( s, You_forgot_to_declare_USE_HANDLE_MACROS,a )
#define TRACEERR2(s,a,b) TRACE( s, You_forgot_to_declare_USE_HANDLE_MACROS,a,b )
// Note that it is important to use "e->Delete();" and not "delete e;"
#define STANDARD_CATCH \
} \
catch (CMemoryException* e) \
{ \
TRACEERR( ERRSTRING_MEMORY ); \
ASSERT( FALSE ); \
e->Delete(); \
return E_OUTOFMEMORY; \
} \
catch (COleException* e) \
{ \
HRESULT hr = (HRESULT)e->Process(e); \
TRACEERR1( ERRSTRING_OLE, hr ); \
ASSERT( FALSE ); \
e->Delete(); \
ASSERT( FAILED(hr) ); \
return hr; \
} \
catch (CFileException* e) \
{ \
HRESULT hr = (HRESULT)e->m_lOsError; \
TRACEERR2( ERRSTRING_FILE, hr, e->m_strFileName ); \
ASSERT( FALSE ); \
e->Delete(); \
ASSERT( FAILED(hr) ); \
return hr; \
} \
catch (CException* e) \
{ \
TRACEERR( ERRSTRING_UNEXPECTED ); \
ASSERT( FALSE ); \
e->Delete(); \
return E_UNEXPECTED; \
}
#define MFC_CATCH \
STANDARD_CATCH
#define TEST_STRING_PARAM(x) \
if ( (x) != NULL && !AfxIsValidString(x) ) { \
TRACEERR( BADPARM_STRING ); return E_POINTER; }
#define TEST_NONNULL_STRING_PARAM(x) \
if ( !AfxIsValidString(x) ) { \
TRACEERR( BADPARM_STRING ); return E_POINTER; }
#define TEST_NONNULL_PTR_PARAM(x) \
if ( (x) == NULL || IsBadWritePtr((x),sizeof(x)) ) { \
TRACEERR( BADPARM_POINTER ); return E_POINTER; }
#endif // _MACROS_H_