Windows2003-3790/admin/snapin/corecopy/comptr.h
2020-09-30 16:53:55 +02:00

1294 lines
31 KiB
C++

#ifndef COMPTR_H
#define COMPTR_H
#if _MSC_VER >= 1100
#pragma warning(disable:4800)
#include <comdef.h>
#define CIP_RETYPEDEF(I) typedef I##Ptr I##CIP;
#define CIP_TYPEDEF(I) _COM_SMARTPTR_TYPEDEF(I, IID_##I); CIP_RETYPEDEF(I);
#define DEFINE_CIP(x)\
CIP_TYPEDEF(x)
#define DECLARE_CIP(x) DEFINE_CIP(x) x##CIP
CIP_RETYPEDEF(IUnknown);
CIP_RETYPEDEF(IDataObject);
CIP_RETYPEDEF(IStorage);
CIP_RETYPEDEF(IStream);
CIP_RETYPEDEF(IPersistStorage);
CIP_RETYPEDEF(IPersistStream);
CIP_RETYPEDEF(IPersistStreamInit);
CIP_RETYPEDEF(IDispatch);
#else // _MSC_VER < 1100
#define USE_OLD_COMPILER (_MSC_VER<1100)
#define USE_INTERMEDIATE_COMPILER (USE_OLD_COMPILER && (_MSC_VER>1020))
// This avoids "warning C4290: C++ Exception Specification ignored"
// JonN 12/16/96
#pragma warning(4:4290)
#ifndef BOOL_H
#include <bool.h>
#endif
#ifndef __wtypes_h__
#include <wtypes.h>
#endif
template<typename _Interface, const IID* _IID/*=&__uuidof(_Interface)*/>
class CIID
// Provide Interface to IID association
{
public: typedef _Interface Interface;
public: static _Interface* GetInterfacePtr() throw()
{
return NULL;
}
public: static _Interface& GetInterface() throw()
{
return *GetInterfacePtr();
}
public: static const IID& GetIID() throw()
{
return *_IID;
}
}; // class CIID
template<typename _CIID> class CIP
{
#if USE_OLD_COMPILER
private: class _IUnknown: public IUnknown {};
// Unique type used to provide for operations between different pointer
// types.
#endif // USE_OLD_COMPILER
// Declare interface type so that the type may be available outside
// the scope of this template.
public: typedef _CIID ThisCIID;
public: typedef _CIID::Interface Interface;
public: static const IID& GetIID() throw()
// When the compiler supports references in template params,
// _CLSID will be changed to a reference. To avoid conversion
// difficulties this function should be used to obtain the
// CLSID.
{
return ThisCIID::GetIID();
}
//REVIEW: add support for assignment of nonpointer interfaces
// i.e. IUnknown, instead of simple IUnknown*
public: CIP() throw()
// Construct empty in preperation for assignment.
: _pInterface(NULL)
{
}
public: CIP(int null) throw()
// This constructor is provided to allow NULL assignment. It will assert
// if any value other than null is assigned to the object.
: _pInterface(NULL)
{
ASSERT(!null);
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
template<typename _InterfacePtr> CIP(_InterfacePtr p) throw()
// Queries for this interface.
#else
public: CIP(_IUnknown& p) throw()
: _pInterface(NULL)
{
if (&p)
{
const HRESULT hr = _QueryInterface(&p);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
}
else _pInterface = NULL;
}
public: CIP(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
: _pInterface(NULL)
{
if (p)
{
const HRESULT hr = _QueryInterface(p);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
}
else _pInterface = NULL;
}
public: CIP(const CIP& cp) throw()
// Copy the pointer and AddRef().
: _pInterface(cp._pInterface)
{
_AddRef();
}
public: CIP(Interface* pInterface) throw()
// Saves the interface
: _pInterface(pInterface)
{
_AddRef();
}
public: CIP(Interface* pInterface, bool bAddRef) throw()
// Copies the pointer. If bAddRef is TRUE, the interface will
// be AddRef()ed.
: _pInterface(pInterface)
{
if (bAddRef)
{
ASSERT(!pInterface);
if (pInterface)
_AddRef();
}
}
public: CIP(const CLSID& clsid, DWORD dwClsContext = CLSCTX_ALL) explicit throw()
// Calls CoCreateClass with the provided CLSID.
: _pInterface(NULL)
{
const HRESULT hr = CreateInstance(clsid, dwClsContext);
ASSERT(SUCCEEDED(hr));
}
public: CIP(LPOLESTR str, DWORD dwClsContext = CLSCTX_ALL) explicit throw()
// Calls CoCreateClass with the provided CLSID retrieved from
// the string.
: _pInterface(NULL)
{
const HRESULT hr = CreateInstance(str, dwClsContext);
ASSERT(SUCCEEDED(hr));
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> CIP& operator=(_InterfacePtr& p) throw()
// Queries for interface.
#else
public: CIP& operator=(_IUnknown& p) throw()
{
return operator=(static_cast<IUnknown*>(&p));
}
public: CIP& operator=(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
const HRESULT hr = _QueryInterface(p);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
return *this;
}
public: CIP& operator=(Interface* pInterface) throw()
// Saves the interface.
{
if (_pInterface != pInterface)
{
Interface* pOldInterface = _pInterface;
_pInterface = pInterface;
_AddRef();
if (pOldInterface)
pOldInterface->Release();
}
return *this;
}
public: CIP& operator=(const CIP& cp) throw()
// Copies and AddRef()'s the interface.
{
return operator=(cp._pInterface);
}
public: CIP& operator=(int null) throw()
// This operator is provided to permit the assignment of NULL to the class.
// It will assert if any value other than NULL is assigned to it.
{
ASSERT(!null);
return operator=(reinterpret_cast<Interface*>(NULL));
}
public: ~CIP() throw()
// If we still have an interface then Release() it. The interface
// may be NULL if Detach() has previosly been called, or if it was
// never set.
{
_Release();
}
public: void Attach(Interface* pInterface) throw()
// Saves/sets the interface without AddRef()ing. This call
// will release any previously aquired interface.
{
_Release();
_pInterface = pInterface;
}
public: void Attach(Interface* pInterface, bool bAddRef) throw()
// Saves/sets the interface only AddRef()ing if bAddRef is TRUE.
// This call will release any previously aquired interface.
{
_Release();
_pInterface = pInterface;
if (bAddRef)
{
ASSERT(pInterface);
if (pInterface)
pInterface->AddRef();
}
}
public: Interface* Detach() throw()
// Simply NULL the interface pointer so that it isn't Released()'ed.
{
Interface* const old=_pInterface;
_pInterface = NULL;
return old;
}
public: operator Interface*() const throw()
// Return the interface. This value may be NULL
{
return _pInterface;
}
public: Interface& operator*() const throw()
// Allows an instance of this class to act as though it were the
// actual interface. Also provides minimal assertion verification.
{
ASSERT(_pInterface);
return *_pInterface;
}
public: Interface** operator&() throw()
// Returns the address of the interface pointer contained in this
// class. This is useful when using the COM/OLE interfaces to create
// this interface.
{
_Release();
_pInterface = NULL;
return &_pInterface;
}
public: Interface* operator->() const throw()
// Allows this class to be used as the interface itself.
// Also provides simple assertion verification.
{
ASSERT(_pInterface);
return _pInterface;
}
public: operator bool() const throw()
// This operator is provided so that simple boolean expressions will
// work. For example: "if (p) ...".
// Returns TRUE if the pointer is not NULL.
{
return _pInterface;
}
public: bool operator!() throw()
// Returns TRUE if the interface is NULL.
// This operator will be removed when support for type bool
// is added to the compiler.
{
return !_pInterface;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator==(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator==(_IUnknown& p) throw()
{
return operator==(static_cast<IUnknown*>(&p));
}
public: bool operator==(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return !_CompareUnknown(p);
}
public: bool operator==(Interface* p) throw()
// Compare with other interface
{
return (_pInterface == p) ? true : !_CompareUnknown(p);
}
public: bool operator==(CIP& p) throw()
// Compares 2 CIPs
{
return operator==(p._pInterface);
}
public: bool operator==(int null) throw()
// For comparison to NULL
{
ASSERT(!null);
return !_pInterface;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator!=(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator!=(_IUnknown& p) throw()
{
return operator!=(static_cast<IUnknown*>(&p));
}
public: bool operator!=(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return _CompareUnknown(p);
}
public: bool operator!=(Interface* p) throw()
// Compare with other interface
{
return (_pInterface!=p)?true:_CompareUnknown(p);
}
public: bool operator!=(CIP& p) throw()
// Compares 2 CIPs
{
return operator!=(p._pInterface);
}
public: bool operator!=(int null) throw()
// For comparison to NULL
{
ASSERT(!null);
return _pInterface;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator<(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator<(_IUnknown& p) throw()
{
return operator<(static_cast<IUnknown*>(&p));
}
public: bool operator<(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return _CompareUnknown(p)<0;
}
public: bool operator<(Interface* p) throw()
// Compare with other interface
{
return (_pInterface<p) ? true : _CompareUnknown(p) < 0;
}
public: bool operator<(CIP& p) throw()
// Compares 2 CIPs
{
return operator<(p._pInterface);
}
public: bool operator<(int null) throw()
// For comparison with NULL
{
ASSERT(!null);
return _pInterface<NULL;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator>(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator>(_IUnknown& p) throw()
{
return operator>(static_cast<IUnknown*>(&p));
}
public: bool operator>(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return _CompareUnknown(p) > 0;
}
public: bool operator>(Interface* p) throw()
// Compare with other interface
{
return (_pInterface>p) ? true : _CompareUnknown(p) > 0;
}
public: bool operator>(CIP& p) throw()
// Compares 2 CIPs
{
return operator>(p._pInterface);
}
public: bool operator>(int null) throw()
// For comparison with NULL
{
ASSERT(!null);
return _pInterface > NULL;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator<=(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator<=(_IUnknown& p) throw()
{
return operator<=(static_cast<IUnknown*>(&p));
}
public: bool operator<=(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return _CompareUnknown(p)<=0;
}
public: bool operator<=(Interface* p) throw()
// Compare with other interface
{
return (_pInterface<=p) ? true : _CompareUnknown(p) <= 0;
}
public: bool operator<=(CIP& p) throw()
// Compares 2 CIPs
{
return operator<=(p._pInterface);
}
public: bool operator<=(int null) throw()
// For comparison with NULL
{
ASSERT(!null);
return _pInterface <= NULL;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfacePtr> bool operator>=(_InterfacePtr p) throw()
// Compare to pointers
#else
public: bool operator>=(_IUnknown& p) throw()
{
return operator>=(static_cast<IUnknown*>(&p));
}
public: bool operator>=(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
return _CompareUnknown(p) >= 0;
}
public: bool operator>=(Interface* p) throw()
// Compare with other interface
{
return (_pInterface>=p) ? true : _CompareUnknown(p) >= 0;
}
public: bool operator>=(CIP& p) throw()
// Compares 2 CIPs
{
return operator>=(p._pInterface);
}
public: bool operator>=(int null) throw()
// For comparison with NULL
{
ASSERT(!null);
return _pInterface >= NULL;
}
#if USE_OLD_COMPILER
public: operator _IUnknown&() const throw()
// Provided for casts between different pointer types.
{
return *reinterpret_cast<_IUnknown*>(static_cast<IUnknown*>(_pInterface));
}
#endif // USE_OLD_COMPILER
public: void Release() throw()
// Provides assertion verified, Release()ing of this interface.
{
ASSERT(_pInterface);
if (_pInterface)
{
_pInterface->Release();
_pInterface = NULL;
}
}
public: void AddRef() throw()
// Provides assertion verified AddRef()ing of this interface.
{
ASSERT(_pInterface);
if (_pInterface)
_pInterface->AddRef();
}
public: Interface* GetInterfacePtr() const throw()
// Another way to get the interface pointer without casting.
{
return _pInterface;
}
public: HRESULT CreateInstance(
const CLSID& clsid, DWORD dwClsContext=CLSCTX_ALL) throw()
// Loads an interface for the provided CLSID.
// Returns an HRESULT. Any previous interface is released.
{
_Release();
const HRESULT hr = CoCreateInstance(clsid, NULL, dwClsContext,
GetIID(), reinterpret_cast<void**>(&_pInterface));
ASSERT(SUCCEEDED(hr));
return hr;
}
public: HRESULT CreateInstance(
LPOLESTR clsidString, DWORD dwClsContext=CLSCTX_ALL) throw()
// Creates the class specified by clsidString. clsidString may
// contain a class id, or a prog id string.
{
// ISSUE-2002/03/29-JonN Should handle NULL case
ASSERT(clsidString);
CLSID clsid;
HRESULT hr;
if (clsidString[0] == '{')
hr = CLSIDFromString(clsidString, &clsid);
else
hr = CLSIDFromProgID(clsidString, &clsid);
ASSERT(SUCCEEDED(hr));
if (FAILED(hr))
return hr;
return CreateInstance(clsid, dwClsContext);
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfaceType> HRESULT QueryInterface(const IID& iid, _InterfaceType*& p) throw()
// Perfoms the QI for the specified IID and returns it in p.
// As with all QIs, the interface will be AddRef'd.
#else
public: HRESULT QueryInterface(const IID& iid, IUnknown*& p) throw()
#endif // !USE_OLD_COMPILER
{
return _pInterface ?
_pInterface->QueryInterface(iid, reinterpret_cast<void**>(&p)) :
E_NOINTERFACE;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfaceType> HRESULT QueryInterface(const IID& iid, _InterfaceType** p) throw()
// Perfoms the QI for the specified IID and returns it in p.
// As with all QIs, the interface will be AddRef'd.
#else
public: HRESULT QueryInterface(const IID& iid, IUnknown** p) throw()
#endif // !USE_OLD_COMPILER
{
return QueryInterface(iid, *p);
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
public: template<typename _InterfaceType> _InterfaceType* QueryInterface(const IID& iid) throw()
// Perfoms the QI for the specified IID and returns it.
// As with all QIs, the interface will be AddRef'd.
#else
public: IUnknown* QueryInterface(const IID& iid) throw()
#endif // !USE_OLD_COMPILER
{
#if USE_OLD_COMPILER
typedef IUnknown _InterfaceType;
#endif // USE_OLD_COMPILER
_InterfaceType* pInterface;
QueryInterface(iid, pInterface);
return pInterface;
}
private: Interface* _pInterface;
// The Interface.
private: void _Release() throw()
// Releases only if the interface is not null.
// The interface is not set to NULL.
{
if (_pInterface)
_pInterface->Release();
}
private: void _AddRef() throw()
// AddRefs only if the interface is not NULL
{
if (_pInterface)
_pInterface->AddRef();
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
private: template<typename _InterfacePtr> HRESULT _QueryInterface(_InterfacePtr p) throw()
// Performs a QI on pUnknown for the interface type returned
// for this class. The interface is stored. If pUnknown is
// NULL, or the QI fails, E_NOINTERFACE is returned and
// _pInterface is set to NULL.
#else
private: HRESULT _QueryInterface(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
if (!p) // Can't QI NULL
{
operator=(static_cast<Interface*>(NULL));
return E_NOINTERFACE;
}
// Query for this interface
Interface* pInterface;
const HRESULT hr = p->QueryInterface(GetIID(),
reinterpret_cast<void**>(&pInterface));
if (FAILED(hr))
{
// If failed intialize interface to NULL and return HRESULT.
Attach(NULL);
return hr;
}
// Save the interface without AddRef()ing.
Attach(pInterface);
return hr;
}
#if !USE_OLD_COMPILER //REVIEW: remove after v5
private: template<typename _InterfacePtr> int _CompareUnknown(_InterfacePtr& p) throw()
// Compares the provided pointer with this by obtaining IUnknown interfaces
// for each pointer and then returning the difference.
#else
private: int _CompareUnknown(IUnknown* p) throw()
#endif // !USE_OLD_COMPILER
{
IUnknown* pu1;
if (_pInterface)
{
const HRESULT hr = QueryInterface(IID_IUnknown, pu1);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
if (pu1)
pu1->Release();
}
else pu1=NULL;
IUnknown* pu2;
if (p)
{
const HRESULT hr = p->QueryInterface(IID_IUnknown, reinterpret_cast<void**>(&pu2));
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
if (pu2)
pu2->Release();
}
else pu2 = NULL;
return pu1 - pu2;
}
}; // class CIP
// Reverse comparison operators for CIP
template<typename _Interface> bool operator==(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p == NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator==(_Interface* i, CIP<_InterfacePtr>& p)
{
return p == i;
}
template<typename _Interface> bool operator!=(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p != NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator!=(_Interface* i, CIP<_InterfacePtr>& p)
{
return p != i;
}
template<typename _Interface> bool operator<(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p < NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator<(_Interface* i, CIP<_InterfacePtr>& p)
{
return p < i;
}
template<typename _Interface> bool operator>(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p > NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator>(_Interface* i, CIP<_InterfacePtr>& p)
{
return p > i;
}
template<typename _Interface> bool operator<=(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p <= NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator<=(_Interface* i, CIP<_InterfacePtr>& p)
{
return p <= i;
}
template<typename _Interface> bool operator>=(int null, CIP<_Interface>& p)
{
ASSERT(!null);
return p >= NULL;
}
template<typename _Interface, typename _InterfacePtr> bool operator>=(_Interface* i, CIP<_InterfacePtr>& p)
{
return p >= i;
}
#define DEFINE_CIP(x)\
typedef CIID<x, &IID_##x> x##IID;\
typedef CIP<x##IID> x##CIP;
#define DECLARE_CIP(x) DEFINE_CIP(x) x##CIP
DEFINE_CIP(IUnknown);
#if USE_OLD_COMPILER
#if USE_INTERMEDIATE_COMPILER
template<>
#endif
class CIP<IUnknownIID>
{
private:
#if USE_OLD_COMPILER
// Unique type used to provide for operations between different pointer
// types.
class _IUnknown: public IUnknown {};
#endif // USE_OLD_COMPILER
public:
// Declare interface type so that the type may be available outside
// the scope of this template.
typedef IUnknownIID ThisCIID;
typedef IUnknown Interface;
// When the compiler supports references in template params,
// _CLSID will be changed to a reference. To avoid conversion
// difficulties this function should be used to obtain the
// CLSID.
static const IID& GetIID() throw()
{
return ThisCIID::GetIID();
}
// Construct empty in preperation for assignment.
CIP() throw()
: _pInterface(NULL)
{
}
// This constructor is provided to allow NULL assignment. It will assert
// if any value other than null is assigned to the object.
CIP(int null) throw()
: _pInterface(NULL)
{
ASSERT(!null);
}
CIP(_IUnknown& p) throw()
: _pInterface(NULL)
{
if (&p)
{
const HRESULT hr=_QueryInterface(&p);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
}
else _pInterface=NULL;
}
// Copy the pointer and AddRef().
CIP(const CIP& cp) throw()
: _pInterface(cp._pInterface)
{
_AddRef();
}
// Saves the interface
CIP(Interface* pInterface) throw()
: _pInterface(pInterface)
{
_AddRef();
}
// Copies the pointer. If bAddRef is TRUE, the interface will
// be AddRef()ed.
CIP(Interface* pInterface, bool bAddRef) throw()
: _pInterface(pInterface)
{
if (bAddRef)
{
ASSERT(!pInterface);
_AddRef();
}
}
// Calls CoCreateClass with the provided CLSID.
CIP(const CLSID& clsid, DWORD dwClsContext = CLSCTX_ALL) explicit throw()
: _pInterface(NULL)
{
const HRESULT hr = CreateInstance(clsid, dwClsContext);
ASSERT(SUCCEEDED(hr));
}
// Calls CoCreateClass with the provided CLSID retrieved from
// the string.
CIP(LPOLESTR str, DWORD dwClsContext = CLSCTX_ALL) explicit throw()
: _pInterface(NULL)
{
const HRESULT hr = CreateInstance(str, dwClsContext);
ASSERT(SUCCEEDED(hr));
}
CIP& operator=(_IUnknown& p) throw()
{
return operator=(static_cast<IUnknown*>(&p));
}
// Saves the interface.
CIP& operator=(Interface* pInterface) throw()
{
if (_pInterface != pInterface)
{
Interface* pOldInterface = _pInterface;
_pInterface = pInterface;
_AddRef();
if (pOldInterface)
pOldInterface->Release();
}
return *this;
}
// Copies and AddRef()'s the interface.
CIP& operator=(const CIP& cp) throw()
{
return operator=(cp._pInterface);
}
// This operator is provided to permit the assignment of NULL to the class.
// It will assert if any value other than NULL is assigned to it.
CIP& operator=(int null) throw()
{
ASSERT(!null);
return operator=(reinterpret_cast<Interface*>(NULL));
}
// If we still have an interface then Release() it. The interface
// may be NULL if Detach() has previosly been called, or if it was
// never set.
~CIP() throw()
{
_Release();
}
// Saves/sets the interface without AddRef()ing. This call
// will release any previously aquired interface.
void Attach(Interface* pInterface) throw()
{
_Release();
_pInterface = pInterface;
}
// Saves/sets the interface only AddRef()ing if bAddRef is TRUE.
// This call will release any previously aquired interface.
void Attach(Interface* pInterface, bool bAddRef) throw()
{
_Release();
_pInterface = pInterface;
if (bAddRef)
{
ASSERT(pInterface);
if (pInterface)
pInterface->AddRef();
}
}
// Simply NULL the interface pointer so that it isn't Released()'ed.
IUnknown* Detach() throw()
{
ASSERT(_pInterface);
IUnknown* const old = _pInterface;
_pInterface = NULL;
return old;
}
// Return the interface. This value may be NULL
operator Interface*() const throw()
{
return _pInterface;
}
// Queries for the unknown and return it
// Provides minimal level assertion before use.
operator Interface&() const throw()
{
ASSERT(_pInterface);
return *_pInterface;
}
// Allows an instance of this class to act as though it were the
// actual interface. Also provides minimal assertion verification.
Interface& operator*() const throw()
{
ASSERT(_pInterface);
return *_pInterface;
}
// Returns the address of the interface pointer contained in this
// class. This is useful when using the COM/OLE interfaces to create
// this interface.
Interface** operator&() throw()
{
_Release();
_pInterface = NULL;
return &_pInterface;
}
// Allows this class to be used as the interface itself.
// Also provides simple assertion verification.
Interface* operator->() const throw()
{
ASSERT(_pInterface);
return _pInterface;
}
// This operator is provided so that simple boolean expressions will
// work. For example: "if (p) ...".
// Returns TRUE if the pointer is not NULL.
operator bool() const throw()
{
return _pInterface;
}
// Returns TRUE if the interface is NULL.
// This operator will be removed when support for type bool
// is added to the compiler.
bool operator!() throw()
{
return !_pInterface;
}
bool operator==(_IUnknown& p) throw()
{
return operator==(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator==(Interface* p) throw()
{
return (_pInterface==p)?true:!_CompareUnknown(p);
}
// Compares 2 CIPs
bool operator==(CIP& p) throw()
{
return operator==(p._pInterface);
}
// For comparison to NULL
bool operator==(int null) throw()
{
ASSERT(!null);
return !_pInterface;
}
bool operator!=(_IUnknown& p) throw()
{
return operator!=(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator!=(Interface* p) throw()
{
return (_pInterface!=p)?true:_CompareUnknown(p);
}
// Compares 2 CIPs
bool operator!=(CIP& p) throw()
{
return operator!=(p._pInterface);
}
// For comparison to NULL
bool operator!=(int null) throw()
{
ASSERT(!null);
return _pInterface;
}
bool operator<(_IUnknown& p) throw()
{
return operator<(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator<(Interface* p) throw()
{
return (_pInterface<p)?true:_CompareUnknown(p)<0;
}
// Compares 2 CIPs
bool operator<(CIP& p) throw()
{
return operator<(p._pInterface);
}
// For comparison with NULL
bool operator<(int null) throw()
{
ASSERT(!null);
return _pInterface<NULL;
}
bool operator>(_IUnknown& p) throw()
{
return operator>(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator>(Interface* p) throw()
{
return (_pInterface>p)?true:_CompareUnknown(p)>0;
}
// Compares 2 CIPs
bool operator>(CIP& p) throw()
{
return operator>(p._pInterface);
}
// For comparison with NULL
bool operator>(int null) throw()
{
ASSERT(!null);
return _pInterface>NULL;
}
bool operator<=(_IUnknown& p) throw()
{
return operator<=(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator<=(Interface* p) throw()
{
return (_pInterface<=p)?true:_CompareUnknown(p)<=0;
}
// Compares 2 CIPs
bool operator<=(CIP& p) throw()
{
return operator<=(p._pInterface);
}
// For comparison with NULL
bool operator<=(int null) throw()
{
ASSERT(!null);
return _pInterface<=NULL;
}
bool operator>=(_IUnknown& p) throw()
{
return operator>=(static_cast<IUnknown*>(&p));
}
// Compare with other interface
bool operator>=(Interface* p) throw()
{
return (_pInterface>=p)?true:_CompareUnknown(p)>=0;
}
// Compares 2 CIPs
bool operator>=(CIP& p) throw()
{
return operator>=(p._pInterface);
}
// For comparison with NULL
bool operator>=(int null) throw()
{
ASSERT(!null);
return _pInterface>=NULL;
}
// Provided for casts between different pointer types.
operator _IUnknown&() const throw()
{
return *reinterpret_cast<_IUnknown*>(static_cast<IUnknown*>(_pInterface));
}
// Provides assertion verified, Release()ing of this interface.
void Release() throw()
{
ASSERT(_pInterface);
if (_pInterface)
{
_pInterface->Release();
_pInterface = NULL;
}
}
// Provides assertion verified AddRef()ing of this interface.
void AddRef() throw()
{
ASSERT(_pInterface);
if (_pInterface)
_pInterface->AddRef();
}
// Another way to get the interface pointer without casting.
Interface* GetInterfacePtr() const throw()
{
return _pInterface;
}
// Loads an interface for the provided CLSID.
// Returns an HRESULT. Any previous interface is released.
HRESULT CreateInstance(
const CLSID& clsid, DWORD dwClsContext=CLSCTX_ALL) throw()
{
_Release();
const HRESULT hr = CoCreateInstance(clsid, NULL, dwClsContext,
GetIID(), reinterpret_cast<void**>(&_pInterface));
ASSERT(SUCCEEDED(hr));
return hr;
}
// Creates the class specified by clsidString. clsidString may
// contain a class id, or a prog id string.
HRESULT CreateInstance(
LPOLESTR clsidString, DWORD dwClsContext=CLSCTX_ALL) throw()
{
// ISSUE-2002/03/29-JonN Should handle NULL case
ASSERT(clsidString);
CLSID clsid;
HRESULT hr;
if (clsidString[0] == '{')
hr = CLSIDFromString(clsidString, &clsid);
else
hr = CLSIDFromProgID(clsidString, &clsid);
ASSERT(SUCCEEDED(hr));
if (FAILED(hr))
return hr;
return CreateInstance(clsid, dwClsContext);
}
HRESULT QueryInterface(const IID& iid, IUnknown*& p) throw()
{
return _pInterface ?
_pInterface->QueryInterface(iid, reinterpret_cast<void**>(&p)) :
E_NOINTERFACE;
}
HRESULT QueryInterface(const IID& iid, IUnknown** p) throw()
{
return QueryInterface(iid, *p);
}
// Perfoms the QI for the specified IID and returns it.
// As with all QIs, the interface will be AddRef'd.
IUnknown* QueryInterface(const IID& iid) throw()
{
typedef IUnknown _InterfaceType;
_InterfaceType* pInterface;
QueryInterface(iid, pInterface);
return pInterface;
}
private:
// The Interface.
Interface* _pInterface;
// Releases only if the interface is not null.
// The interface is not set to NULL.
void _Release() throw()
{
if (_pInterface)
_pInterface->Release();
}
// AddRefs only if the interface is not NULL
void _AddRef() throw()
{
if (_pInterface)
_pInterface->AddRef();
}
// Performs a QI on pUnknown for the interface type returned
// for this class. The interface is stored. If pUnknown is
// NULL, or the QI fails, E_NOINTERFACE is returned and
// _pInterface is set to NULL.
HRESULT _QueryInterface(IUnknown* p) throw()
{
if (!p) // Can't QI NULL
{
operator=(static_cast<Interface*>(NULL));
return E_NOINTERFACE;
}
// Query for this interface
Interface* pInterface;
const HRESULT hr = p->QueryInterface(GetIID(),
reinterpret_cast<void**>(&pInterface));
if (FAILED(hr))
{
// If failed intialize interface to NULL and return HRESULT.
Attach(NULL);
return hr;
}
// Save the interface without AddRef()ing.
Attach(pInterface);
return hr;
}
// Compares the provided pointer with this by obtaining IUnknown interfaces
// for each pointer and then returning the difference.
int _CompareUnknown(IUnknown* p) throw()
{
IUnknown* pu1;
if (_pInterface)
{
const HRESULT hr=QueryInterface(IID_IUnknown, pu1);
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
if (pu1)
pu1->Release();
}
else pu1=NULL;
IUnknown* pu2;
if (p)
{
const HRESULT hr=p->QueryInterface(IID_IUnknown, reinterpret_cast<void**>(&pu2));
ASSERT(SUCCEEDED(hr) || hr == E_NOINTERFACE);
if (pu2)
pu2->Release();
}
else pu2=NULL;
return pu1-pu2;
}
}; // class CIP
#endif // USE_OLD_COMPILER
#endif // _MSC_VER < 1100
#endif // COMPTR_H