/** @file * * VBox frontends: Qt GUI ("VirtualBox"): * Various COM definitions and COM wrapper class declarations * * This header is used in conjunction with the header generated from * XIDL expressed interface definitions to provide cross-platform Qt-based * interface wrapper classes. */ /* * Copyright (C) 2006-2007 innotek GmbH * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file 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, * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE * distribution. VirtualBox OSE is distributed in the hope that it will * be useful, but WITHOUT ANY WARRANTY of any kind. * * If you received this file as part of a commercial VirtualBox * distribution, then only the terms of your commercial VirtualBox * license agreement apply instead of the previous paragraph. */ #ifndef __COMDefs_h__ #define __COMDefs_h__ /* Both VBox/com/assert.h and qglobal.h contain a definition of ASSERT. * Either of them can be already included here, so try to shut them up. */ #undef ASSERT #include #undef ASSERT #include #include #include #include // for auto_copy_ptr /* * Additional COM / XPCOM defines and includes */ #define IN_BSTRPARAM INPTR BSTR #define IN_GUIDPARAM INPTR GUIDPARAM #if !defined (VBOX_WITH_XPCOM) #else /* !defined (VBOX_WITH_XPCOM) */ #include #include #include class XPCOMEventQSocketListener; #endif /* !defined (VBOX_WITH_XPCOM) */ /* VirtualBox interfaces declarations */ #if !defined (VBOX_WITH_XPCOM) #include #else /* !defined (VBOX_WITH_XPCOM) */ #include #endif /* !defined (VBOX_WITH_XPCOM) */ #include "VBoxDefs.h" ///////////////////////////////////////////////////////////////////////////// class CVirtualBoxErrorInfo; /** Represents extended error information */ class COMErrorInfo { public: COMErrorInfo() : mIsNull (true) , mIsBasicAvailable (false), mIsFullAvailable (false) , mResultCode (S_OK) {} COMErrorInfo (const CVirtualBoxErrorInfo &info) { init (info); } /* the default copy ctor and assignment op are ok */ bool isNull() const { return mIsNull; } bool isBasicAvailable() const { return mIsBasicAvailable; } bool isFullAvailable() const { return mIsFullAvailable; } HRESULT resultCode() const { return mResultCode; } QUuid interfaceID() const { return mInterfaceID; } QString component() const { return mComponent; } QString text() const { return mText; } const COMErrorInfo *next() const { return mNext.get(); } QString interfaceName() const { return mInterfaceName; } QUuid calleeIID() const { return mCalleeIID; } QString calleeName() const { return mCalleeName; } private: void init (const CVirtualBoxErrorInfo &info); void fetchFromCurrentThread (IUnknown *callee, const GUID *calleeIID); static QString getInterfaceNameFromIID (const QUuid &id); bool mIsNull : 1; bool mIsBasicAvailable : 1; bool mIsFullAvailable : 1; HRESULT mResultCode; QUuid mInterfaceID; QString mComponent; QString mText; cppx::auto_copy_ptr mNext; QString mInterfaceName; QUuid mCalleeIID; QString mCalleeName; friend class COMBaseWithEI; }; ///////////////////////////////////////////////////////////////////////////// /** * Base COM class the CInterface template and all wrapper classes are derived * from. Provides common functionality for all COM wrappers. */ class COMBase { public: static HRESULT initializeCOM(); static HRESULT cleanupCOM(); #if !defined (VBOX_WITH_XPCOM) /** Converts a GUID value to QUuid */ static QUuid toQUuid (const GUID &id) { return QUuid (id.Data1, id.Data2, id.Data3, id.Data4[0], id.Data4[1], id.Data4[2], id.Data4[3], id.Data4[4], id.Data4[5], id.Data4[6], id.Data4[7]); } #else /* !defined (VBOX_WITH_XPCOM) */ /** Converts a GUID value to QUuid */ static QUuid toQUuid (const nsID &id) { return QUuid (id.m0, id.m1, id.m2, id.m3[0], id.m3[1], id.m3[2], id.m3[3], id.m3[4], id.m3[5], id.m3[6], id.m3[7]); } #endif /* !defined (VBOX_WITH_XPCOM) */ /** * Returns the result code of the last interface method called * by the wrapper instance or the result of CInterface::createInstance() * operation. */ HRESULT lastRC() const { return mRC; } /** * Returns error info set by the last unsuccessfully invoked interface * method. Returned error info is useful only if CInterface::lastRC() * represents a failure (i.e. CInterface::isOk() is false). */ virtual COMErrorInfo errorInfo() const { return COMErrorInfo(); } protected: /* no arbitrary instance creations */ COMBase() : mRC (S_OK) {}; #if defined (VBOX_WITH_XPCOM) static XPCOMEventQSocketListener *sSocketListener; #endif /** Adapter to pass QString as input BSTR params */ class BSTRIn { public: BSTRIn (const QString &s) : bstr (SysAllocString ((const OLECHAR *) s.ucs2())) {} ~BSTRIn() { if (bstr) SysFreeString (bstr); } operator BSTR() const { return bstr; } private: BSTR bstr; }; /** Adapter to pass QString as output BSTR params */ class BSTROut { public: BSTROut (QString &s) : str (s), bstr (0) {} ~BSTROut() { if (bstr) { str = QString::fromUcs2 (bstr); SysFreeString (bstr); } } operator BSTR *() { return &bstr; } private: QString &str; BSTR bstr; }; /** Adapter to pass CEnums enums as output VirtualBox enum params (*_T) */ template class ENUMOut { public: ENUMOut (CE &e) : ce (e), ve ((VE) 0) {} ~ENUMOut() { ce = (CE) ve; } operator VE *() { return &ve; } private: CE &ce; VE ve; }; #if !defined (VBOX_WITH_XPCOM) /** Adapter to pass QUuid as input GUID params */ static GUID GUIDIn (const QUuid &uuid) { return uuid; } /** Adapter to pass QUuid as output GUID params */ class GUIDOut { public: GUIDOut (QUuid &id) : uuid (id) { ::memset (&guid, 0, sizeof (GUID)); } ~GUIDOut() { uuid = QUuid ( guid.Data1, guid.Data2, guid.Data3, guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3], guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]); } operator GUID *() { return &guid; } private: QUuid &uuid; GUID guid; }; #else /* !defined (VBOX_WITH_XPCOM) */ /** Adapter to pass QUuid as input GUID params */ static const nsID &GUIDIn (const QUuid &uuid) { return *(const nsID *) &uuid; } /** Adapter to pass QUuid as output GUID params */ class GUIDOut { public: GUIDOut (QUuid &id) : uuid (id), nsid (0) {} ~GUIDOut() { if (nsid) { uuid = QUuid ( nsid->m0, nsid->m1, nsid->m2, nsid->m3[0], nsid->m3[1], nsid->m3[2], nsid->m3[3], nsid->m3[4], nsid->m3[5], nsid->m3[6], nsid->m3[7]); nsMemory::Free (nsid); } } operator nsID **() { return &nsid; } private: QUuid &uuid; nsID *nsid; }; #endif /* !defined (VBOX_WITH_XPCOM) */ void fetchErrorInfo (IUnknown * /*callee*/, const GUID * /*calleeIID*/) const {} mutable HRESULT mRC; friend class COMErrorInfo; }; ///////////////////////////////////////////////////////////////////////////// /** * Alternative base class for the CInterface template that adds * the errorInfo() method for providing extended error info about * unsuccessful invocation of the last called interface method. */ class COMBaseWithEI : public COMBase { public: /** * Returns error info set by the last unsuccessfully invoked interface * method. Returned error info is useful only if CInterface::lastRC() * represents a failure (i.e. CInterface::isOk() is false). */ COMErrorInfo errorInfo() const { return mErrInfo; } protected: /* no arbitrary instance creations */ COMBaseWithEI() : COMBase () {}; void fetchErrorInfo (IUnknown *callee, const GUID *calleeIID) const { mErrInfo.fetchFromCurrentThread (callee, calleeIID); } mutable COMErrorInfo mErrInfo; }; ///////////////////////////////////////////////////////////////////////////// /** * Simple class that encapsulates the result code and COMErrorInfo. */ class COMResult { public: COMResult() : mRC (S_OK) {} /** Queries the current result code and error info from the given component */ COMResult (const COMBase &aComponent) { mErrInfo = aComponent.errorInfo(); mRC = aComponent.lastRC(); } /** Queries the current result code and error info from the given component */ COMResult &operator= (const COMBase &aComponent) { mErrInfo = aComponent.errorInfo(); mRC = aComponent.lastRC(); return *this; } bool isNull() const { return mErrInfo.isNull(); } bool isOk() const { return mErrInfo.isNull() && SUCCEEDED (mRC); } COMErrorInfo errorInfo() const { return mErrInfo; } HRESULT rc() const { return mRC; } private: COMErrorInfo mErrInfo; HRESULT mRC; }; ///////////////////////////////////////////////////////////////////////////// class CUnknown; /** * Wrapper template class for all interfaces. * * All interface methods named as they are in the original, i.e. starting * with the capital letter. All utility non-interface methods are named * starting with the small letter. Utility methods should be not normally * called by the end-user client application. * * @param I interface class (i.e. IUnknown/nsISupports derivant) * @param B base class, either COMBase (by default) or COMBaseWithEI */ template class CInterface : public B { public: typedef B Base; typedef I Iface; /* constructors & destructor */ CInterface() : mIface (NULL) {} CInterface (const CInterface &that) : B (that), mIface (that.mIface) { addref (mIface); } CInterface (const CUnknown &that); CInterface (I *i) : mIface (i) { addref (mIface); } virtual ~CInterface() { release (mIface); } /* utility methods */ void createInstance (const CLSID &clsid) { AssertMsg (!mIface, ("Instance is already non-NULL\n")); if (!mIface) { #if !defined (VBOX_WITH_XPCOM) B::mRC = CoCreateInstance (clsid, NULL, CLSCTX_ALL, _ATL_IIDOF (I), (void **) &mIface); #else /* !defined (VBOX_WITH_XPCOM) */ nsCOMPtr manager; B::mRC = NS_GetComponentManager (getter_AddRefs (manager)); if (SUCCEEDED (B::mRC)) B::mRC = manager->CreateInstance (clsid, nsnull, NS_GET_IID (I), (void **) &mIface); #endif /* !defined (VBOX_WITH_XPCOM) */ /* fetch error info, but don't assert if it's missing -- many other * reasons can lead to an error (w/o providing error info), not only * the instance initialization code (that should always provide it) */ B::fetchErrorInfo (NULL, NULL); } } void attach (I *i) { /* be aware of self (from COM point of view) assignment */ I *old_iface = mIface; mIface = i; addref (mIface); release (old_iface); B::mRC = S_OK; }; void attachUnknown (IUnknown *i) { /* be aware of self (from COM point of view) assignment */ I *old_iface = mIface; mIface = NULL; B::mRC = S_OK; if (i) #if !defined (VBOX_WITH_XPCOM) B::mRC = i->QueryInterface (_ATL_IIDOF (I), (void **) &mIface); #else /* !defined (VBOX_WITH_XPCOM) */ B::mRC = i->QueryInterface (NS_GET_IID (I), (void **) &mIface); #endif /* !defined (VBOX_WITH_XPCOM) */ release (old_iface); }; void detach() { release (mIface); mIface = NULL; } bool isNull() const { return mIface == NULL; } bool isOk() const { return !isNull() && SUCCEEDED (B::mRC); } /* utility operators */ CInterface &operator= (const CInterface &that) { attach (that.mIface); B::operator= (that); return *this; } I *iface() const { return mIface; } bool operator== (const CInterface &that) const { return mIface == that.mIface; } bool operator!= (const CInterface &that) const { return mIface != that.mIface; } CInterface &operator= (const CUnknown &that); protected: static void addref (I *i) { if (i) i->AddRef(); } static void release (I *i) { if (i) i->Release(); } mutable I *mIface; }; ///////////////////////////////////////////////////////////////////////////// class CUnknown : public CInterface { public: CUnknown() : CInterface () {} template explicit CUnknown (const C &that) { mIface = NULL; if (that.mIface) #if !defined (VBOX_WITH_XPCOM) mRC = that.mIface->QueryInterface (_ATL_IIDOF (IUnknown), (void**) &mIface); #else /* !defined (VBOX_WITH_XPCOM) */ mRC = that.mIface->QueryInterface (NS_GET_IID (IUnknown), (void**) &mIface); #endif /* !defined (VBOX_WITH_XPCOM) */ if (SUCCEEDED (mRC)) { mRC = that.lastRC(); mErrInfo = that.errorInfo(); } } /* specialization for CUnknown */ CUnknown (const CUnknown &that) : CInterface () { mIface = that.mIface; addref (mIface); COMBaseWithEI::operator= (that); } template CUnknown &operator= (const C &that) { /* be aware of self (from COM point of view) assignment */ IUnknown *old_iface = mIface; mIface = NULL; mRC = S_OK; #if !defined (VBOX_WITH_XPCOM) if (that.mIface) mRC = that.mIface->QueryInterface (_ATL_IIDOF (IUnknown), (void**) &mIface); #else /* !defined (VBOX_WITH_XPCOM) */ if (that.mIface) mRC = that.mIface->QueryInterface (NS_GET_IID (IUnknown), (void**) &mIface); #endif /* !defined (VBOX_WITH_XPCOM) */ if (SUCCEEDED (mRC)) { mRC = that.lastRC(); mErrInfo = that.errorInfo(); } release (old_iface); return *this; } /* specialization for CUnknown */ CUnknown &operator= (const CUnknown &that) { attach (that.mIface); COMBaseWithEI::operator= (that); return *this; } /* @internal Used in wrappers. */ IUnknown *&ifaceRef() { return mIface; }; }; /* inlined CInterface methods that use CUnknown */ template inline CInterface ::CInterface (const CUnknown &that) : mIface (NULL) { attachUnknown (that.iface()); if (SUCCEEDED (B::mRC)) B::operator= ((B &) that); } template inline CInterface &CInterface ::operator =(const CUnknown &that) { attachUnknown (that.iface()); if (SUCCEEDED (B::mRC)) B::operator= ((B &) that); return *this; } ///////////////////////////////////////////////////////////////////////////// /* include the generated header containing concrete wrapper definitions */ #include "COMWrappers.h" #endif // __COMDefs_h__