comip.h 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. /**
  2. * This file has no copyright assigned and is placed in the Public Domain.
  3. * This file is part of the mingw-w64 runtime package.
  4. * No warranty is given; refer to the file DISCLAIMER.PD within this package.
  5. */
  6. #ifndef _INC_COMIP
  7. #define _INC_COMIP
  8. #include <_mingw.h>
  9. #include <ole2.h>
  10. #include <malloc.h>
  11. #include <comutil.h>
  12. #ifdef __cplusplus
  13. #pragma push_macro("new")
  14. #undef new
  15. #include <new.h>
  16. class _com_error;
  17. #ifndef WINAPI
  18. #if defined(_ARM_)
  19. #define WINAPI
  20. #else
  21. #define WINAPI __stdcall
  22. #endif
  23. #endif
  24. void WINAPI _com_issue_error(HRESULT);
  25. struct IUnknown;
  26. template<typename _Interface,const IID *_IID >
  27. class _com_IIID {
  28. public:
  29. typedef _Interface Interface;
  30. static _Interface *GetInterfacePtr() throw() { return NULL; }
  31. static _Interface& GetInterface() throw() { return *GetInterfacePtr(); }
  32. static const IID& GetIID() throw() { return *_IID; }
  33. };
  34. /* This is needed for _COM_SMARTPTR_TYPEDEF using emulated __uuidof. Since we can't pass
  35. * IID as a template argument, it's passed as a wrapper function. */
  36. template<typename _Interface,const IID &(*iid_getter)() >
  37. class _com_IIID_getter {
  38. public:
  39. typedef _Interface Interface;
  40. static _Interface *GetInterfacePtr() throw() { return NULL; }
  41. static _Interface& GetInterface() throw() { return *GetInterfacePtr(); }
  42. static const IID& GetIID() throw() { return iid_getter(); }
  43. };
  44. template<typename _IIID> class _com_ptr_t {
  45. public:
  46. typedef _IIID ThisIIID;
  47. typedef typename _IIID::Interface Interface;
  48. static const IID& GetIID() throw() { return ThisIIID::GetIID(); }
  49. template<typename _OtherIID> _com_ptr_t(const _com_ptr_t<_OtherIID> &p) : m_pInterface(NULL) {
  50. HRESULT hr = _QueryInterface(p);
  51. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  52. }
  53. template<typename _InterfaceType> _com_ptr_t(_InterfaceType *p) : m_pInterface(NULL) {
  54. HRESULT hr = _QueryInterface(p);
  55. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  56. }
  57. _com_ptr_t(LPSTR str) { new(this) _com_ptr_t(static_cast<LPCSTR> (str),NULL); }
  58. _com_ptr_t(LPWSTR str) { new(this) _com_ptr_t(static_cast<LPCWSTR> (str),NULL); }
  59. explicit _com_ptr_t(_com_ptr_t *p) : m_pInterface(NULL) {
  60. if(!p) { _com_issue_error(E_POINTER); }
  61. else {
  62. m_pInterface = p->m_pInterface;
  63. AddRef();
  64. }
  65. }
  66. _com_ptr_t() throw() : m_pInterface(NULL) { }
  67. _com_ptr_t(int null) : m_pInterface(NULL) {
  68. if(null!=0) { _com_issue_error(E_POINTER); }
  69. }
  70. #ifdef _NATIVE_NULLPTR_SUPPORTED
  71. _com_ptr_t(decltype(nullptr)) : m_pInterface(NULL) {}
  72. #endif
  73. _com_ptr_t(const _com_ptr_t &cp) throw() : m_pInterface(cp.m_pInterface) { _AddRef(); }
  74. _com_ptr_t(Interface *pInterface) throw() : m_pInterface(pInterface) { _AddRef(); }
  75. _com_ptr_t(Interface *pInterface,bool fAddRef) throw() : m_pInterface(pInterface) {
  76. if(fAddRef) _AddRef();
  77. }
  78. _com_ptr_t(const _variant_t& varSrc) : m_pInterface(NULL) {
  79. HRESULT hr = QueryStdInterfaces(varSrc);
  80. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  81. }
  82. explicit _com_ptr_t(const CLSID &clsid,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) : m_pInterface(NULL) {
  83. HRESULT hr = CreateInstance(clsid,pOuter,dwClsContext);
  84. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  85. }
  86. explicit _com_ptr_t(LPCWSTR str,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) : m_pInterface(NULL) {
  87. HRESULT hr = CreateInstance(str,pOuter,dwClsContext);
  88. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  89. }
  90. explicit _com_ptr_t(LPCSTR str,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) : m_pInterface(NULL) {
  91. HRESULT hr = CreateInstance(str,pOuter,dwClsContext);
  92. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  93. }
  94. template<typename _OtherIID> _com_ptr_t &operator=(const _com_ptr_t<_OtherIID> &p) {
  95. HRESULT hr = _QueryInterface(p);
  96. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  97. return *this;
  98. }
  99. template<typename _InterfaceType> _com_ptr_t &operator=(_InterfaceType *p) {
  100. HRESULT hr = _QueryInterface(p);
  101. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  102. return *this;
  103. }
  104. _com_ptr_t &operator=(Interface *pInterface) throw() {
  105. if(m_pInterface!=pInterface) {
  106. Interface *pOldInterface = m_pInterface;
  107. m_pInterface = pInterface;
  108. _AddRef();
  109. if(pOldInterface!=NULL) pOldInterface->Release();
  110. }
  111. return *this;
  112. }
  113. _com_ptr_t &operator=(const _com_ptr_t &cp) throw() { return operator=(cp.m_pInterface); }
  114. _com_ptr_t &operator=(int null) {
  115. if(null!=0) { _com_issue_error(E_POINTER); }
  116. return operator=(reinterpret_cast<Interface*>(NULL));
  117. }
  118. _com_ptr_t &operator=(long long null) {
  119. if(null!=0) { _com_issue_error(E_POINTER); }
  120. return operator=(reinterpret_cast<Interface*>(NULL));
  121. }
  122. _com_ptr_t &operator=(const _variant_t& varSrc) {
  123. HRESULT hr = QueryStdInterfaces(varSrc);
  124. if(FAILED(hr) && (hr!=E_NOINTERFACE)) { _com_issue_error(hr); }
  125. return *this;
  126. }
  127. ~_com_ptr_t() throw() { _Release(); }
  128. void Attach(Interface *pInterface) throw() {
  129. _Release();
  130. m_pInterface = pInterface;
  131. }
  132. void Attach(Interface *pInterface,bool fAddRef) throw() {
  133. _Release();
  134. m_pInterface = pInterface;
  135. if(fAddRef) {
  136. if(!pInterface) { _com_issue_error(E_POINTER); }
  137. else pInterface->AddRef();
  138. }
  139. }
  140. Interface *Detach() throw() {
  141. Interface *const old = m_pInterface;
  142. m_pInterface = NULL;
  143. return old;
  144. }
  145. operator Interface*() const throw() { return m_pInterface; }
  146. operator Interface&() const {
  147. if(!m_pInterface) { _com_issue_error(E_POINTER); }
  148. return *m_pInterface;
  149. }
  150. Interface& operator*() const {
  151. if(!m_pInterface) { _com_issue_error(E_POINTER); }
  152. return *m_pInterface;
  153. }
  154. Interface **operator&() throw() {
  155. _Release();
  156. m_pInterface = NULL;
  157. return &m_pInterface;
  158. }
  159. Interface *operator->() const {
  160. if(!m_pInterface) { _com_issue_error(E_POINTER); }
  161. return m_pInterface;
  162. }
  163. operator bool() const throw() { return m_pInterface!=NULL; }
  164. template<typename _OtherIID> bool operator==(const _com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)==0; }
  165. template<typename _OtherIID> bool operator==(_com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)==0; }
  166. template<typename _InterfaceType> bool operator==(_InterfaceType *p) { return _CompareUnknown(p)==0; }
  167. bool operator==(Interface *p) { return (m_pInterface==p) ? true : _CompareUnknown(p)==0; }
  168. bool operator==(const _com_ptr_t &p) throw() { return operator==(p.m_pInterface); }
  169. bool operator==(_com_ptr_t &p) throw() { return operator==(p.m_pInterface); }
  170. bool operator==(int null) {
  171. if(null!=0) { _com_issue_error(E_POINTER); }
  172. return !m_pInterface;
  173. }
  174. bool operator==(long long null) {
  175. if(null) { _com_issue_error(E_POINTER); }
  176. return !m_pInterface;
  177. }
  178. template<typename _OtherIID> bool operator!=(const _com_ptr_t<_OtherIID> &p) { return !(operator==(p)); }
  179. template<typename _OtherIID> bool operator!=(_com_ptr_t<_OtherIID> &p) { return !(operator==(p)); }
  180. template<typename _InterfaceType> bool operator!=(_InterfaceType *p) { return !(operator==(p)); }
  181. bool operator!=(int null) { return !(operator==(null)); }
  182. bool operator!=(long long null) { return !(operator==(null)); }
  183. template<typename _OtherIID> bool operator<(const _com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)<0; }
  184. template<typename _OtherIID> bool operator<(_com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)<0; }
  185. template<typename _InterfaceType> bool operator<(_InterfaceType *p) { return _CompareUnknown(p)<0; }
  186. template<typename _OtherIID> bool operator>(const _com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)>0; }
  187. template<typename _OtherIID> bool operator>(_com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)>0; }
  188. template<typename _InterfaceType> bool operator>(_InterfaceType *p) { return _CompareUnknown(p)>0; }
  189. template<typename _OtherIID> bool operator<=(const _com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)<=0; }
  190. template<typename _OtherIID> bool operator<=(_com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)<=0; }
  191. template<typename _InterfaceType> bool operator<=(_InterfaceType *p) { return _CompareUnknown(p)<=0; }
  192. template<typename _OtherIID> bool operator>=(const _com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)>=0; }
  193. template<typename _OtherIID> bool operator>=(_com_ptr_t<_OtherIID> &p) { return _CompareUnknown(p)>=0; }
  194. template<typename _InterfaceType> bool operator>=(_InterfaceType *p) { return _CompareUnknown(p)>=0; }
  195. void Release() {
  196. if(!m_pInterface) { _com_issue_error(E_POINTER); }
  197. else {
  198. m_pInterface->Release();
  199. m_pInterface = NULL;
  200. }
  201. }
  202. void AddRef() {
  203. if(!m_pInterface) { _com_issue_error(E_POINTER); }
  204. else m_pInterface->AddRef();
  205. }
  206. Interface *GetInterfacePtr() const throw() { return m_pInterface; }
  207. Interface*& GetInterfacePtr() throw() { return m_pInterface; }
  208. HRESULT CreateInstance(const CLSID &rclsid,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) throw() {
  209. HRESULT hr;
  210. _Release();
  211. if(dwClsContext & (CLSCTX_LOCAL_SERVER | CLSCTX_REMOTE_SERVER)) {
  212. IUnknown *pIUnknown;
  213. hr = CoCreateInstance(rclsid,pOuter,dwClsContext,__uuidof(IUnknown),reinterpret_cast<void**>(&pIUnknown));
  214. if(SUCCEEDED(hr)) {
  215. hr = OleRun(pIUnknown);
  216. if(SUCCEEDED(hr)) hr = pIUnknown->QueryInterface(GetIID(),reinterpret_cast<void**>(&m_pInterface));
  217. pIUnknown->Release();
  218. }
  219. } else hr = CoCreateInstance(rclsid,pOuter,dwClsContext,GetIID(),reinterpret_cast<void**>(&m_pInterface));
  220. if(FAILED(hr)) m_pInterface = NULL;
  221. return hr;
  222. }
  223. HRESULT CreateInstance(LPCWSTR clsidString,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) throw() {
  224. if(!clsidString) return E_INVALIDARG;
  225. CLSID clsid;
  226. HRESULT hr;
  227. if(clsidString[0]==L'{') hr = CLSIDFromString(const_cast<LPWSTR> (clsidString),&clsid);
  228. else hr = CLSIDFromProgID(const_cast<LPWSTR> (clsidString),&clsid);
  229. if(FAILED(hr)) return hr;
  230. return CreateInstance(clsid,pOuter,dwClsContext);
  231. }
  232. HRESULT CreateInstance(LPCSTR clsidStringA,IUnknown *pOuter = NULL,DWORD dwClsContext = CLSCTX_ALL) throw() {
  233. if(!clsidStringA) return E_INVALIDARG;
  234. int size = lstrlenA(clsidStringA) + 1;
  235. int destSize = MultiByteToWideChar(CP_ACP,0,clsidStringA,size,NULL,0);
  236. if(destSize==0) return HRESULT_FROM_WIN32(GetLastError());
  237. LPWSTR clsidStringW;
  238. clsidStringW = static_cast<LPWSTR>(_malloca(destSize*sizeof(WCHAR)));
  239. if(!clsidStringW) return E_OUTOFMEMORY;
  240. if(MultiByteToWideChar(CP_ACP,0,clsidStringA,size,clsidStringW,destSize)==0) {
  241. _freea(clsidStringW);
  242. return HRESULT_FROM_WIN32(GetLastError());
  243. }
  244. HRESULT hr=CreateInstance(clsidStringW,pOuter,dwClsContext);
  245. _freea(clsidStringW);
  246. return hr;
  247. }
  248. HRESULT GetActiveObject(const CLSID &rclsid) throw() {
  249. _Release();
  250. IUnknown *pIUnknown;
  251. HRESULT hr = ::GetActiveObject(rclsid,NULL,&pIUnknown);
  252. if(SUCCEEDED(hr)) {
  253. hr = pIUnknown->QueryInterface(GetIID(),reinterpret_cast<void**>(&m_pInterface));
  254. pIUnknown->Release();
  255. }
  256. if(FAILED(hr)) m_pInterface = NULL;
  257. return hr;
  258. }
  259. HRESULT GetActiveObject(LPCWSTR clsidString) throw() {
  260. if(!clsidString) return E_INVALIDARG;
  261. CLSID clsid;
  262. HRESULT hr;
  263. if(clsidString[0]=='{') hr = CLSIDFromString(const_cast<LPWSTR> (clsidString),&clsid);
  264. else hr = CLSIDFromProgID(const_cast<LPWSTR> (clsidString),&clsid);
  265. if(FAILED(hr)) return hr;
  266. return GetActiveObject(clsid);
  267. }
  268. HRESULT GetActiveObject(LPCSTR clsidStringA) throw() {
  269. if(!clsidStringA) return E_INVALIDARG;
  270. int size = lstrlenA(clsidStringA) + 1;
  271. int destSize = MultiByteToWideChar(CP_ACP,0,clsidStringA,size,NULL,0);
  272. LPWSTR clsidStringW;
  273. try {
  274. clsidStringW = static_cast<LPWSTR>(_alloca(destSize*sizeof(WCHAR)));
  275. } catch (...) {
  276. clsidStringW = NULL;
  277. }
  278. if(!clsidStringW) return E_OUTOFMEMORY;
  279. if(MultiByteToWideChar(CP_ACP,0,clsidStringA,size,clsidStringW,destSize)==0) return HRESULT_FROM_WIN32(GetLastError());
  280. return GetActiveObject(clsidStringW);
  281. }
  282. template<typename _InterfaceType> HRESULT QueryInterface(const IID& iid,_InterfaceType*& p) throw () {
  283. if(m_pInterface!=NULL) return m_pInterface->QueryInterface(iid,reinterpret_cast<void**>(&p));
  284. return E_POINTER;
  285. }
  286. template<typename _InterfaceType> HRESULT QueryInterface(const IID& iid,_InterfaceType **p) throw() { return QueryInterface(iid,*p); }
  287. private:
  288. Interface *m_pInterface;
  289. void _Release() throw() {
  290. if(m_pInterface!=NULL) m_pInterface->Release();
  291. }
  292. void _AddRef() throw() {
  293. if(m_pInterface!=NULL) m_pInterface->AddRef();
  294. }
  295. template<typename _InterfacePtr> HRESULT _QueryInterface(_InterfacePtr p) throw() {
  296. HRESULT hr;
  297. if(p!=NULL) {
  298. Interface *pInterface;
  299. hr = p->QueryInterface(GetIID(),reinterpret_cast<void**>(&pInterface));
  300. Attach(SUCCEEDED(hr)? pInterface: NULL);
  301. } else {
  302. operator=(static_cast<Interface*>(NULL));
  303. hr = E_NOINTERFACE;
  304. }
  305. return hr;
  306. }
  307. template<typename _InterfacePtr> int _CompareUnknown(_InterfacePtr p) {
  308. IUnknown *pu1,*pu2;
  309. if(m_pInterface!=NULL) {
  310. HRESULT hr = m_pInterface->QueryInterface(__uuidof(IUnknown),reinterpret_cast<void**>(&pu1));
  311. if(FAILED(hr)) {
  312. _com_issue_error(hr);
  313. pu1 = NULL;
  314. } else pu1->Release();
  315. } else pu1 = NULL;
  316. if(p!=NULL) {
  317. HRESULT hr = p->QueryInterface(__uuidof(IUnknown),reinterpret_cast<void**>(&pu2));
  318. if(FAILED(hr)) {
  319. _com_issue_error(hr);
  320. pu2 = NULL;
  321. } else pu2->Release();
  322. } else pu2 = NULL;
  323. return pu1 - pu2;
  324. }
  325. HRESULT QueryStdInterfaces(const _variant_t& varSrc) throw() {
  326. if(V_VT(&varSrc)==VT_DISPATCH) return _QueryInterface(V_DISPATCH(&varSrc));
  327. if(V_VT(&varSrc)==VT_UNKNOWN) return _QueryInterface(V_UNKNOWN(&varSrc));
  328. VARIANT varDest;
  329. VariantInit(&varDest);
  330. HRESULT hr = VariantChangeType(&varDest,const_cast<VARIANT*>(static_cast<const VARIANT*>(&varSrc)),0,VT_DISPATCH);
  331. if(SUCCEEDED(hr)) hr = _QueryInterface(V_DISPATCH(&varSrc));
  332. if(hr==E_NOINTERFACE) {
  333. VariantInit(&varDest);
  334. hr = VariantChangeType(&varDest,const_cast<VARIANT*>(static_cast<const VARIANT*>(&varSrc)),0,VT_UNKNOWN);
  335. if(SUCCEEDED(hr)) hr = _QueryInterface(V_UNKNOWN(&varSrc));
  336. }
  337. VariantClear(&varDest);
  338. return hr;
  339. }
  340. };
  341. template<typename _InterfaceType> bool operator==(int null,_com_ptr_t<_InterfaceType> &p) {
  342. if(null!=0) { _com_issue_error(E_POINTER); }
  343. return !p;
  344. }
  345. template<typename _Interface,typename _InterfacePtr> bool operator==(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p==i; }
  346. template<typename _Interface> bool operator!=(int null,_com_ptr_t<_Interface> &p) {
  347. if(null!=0) { _com_issue_error(E_POINTER); }
  348. return p!=NULL;
  349. }
  350. template<typename _Interface,typename _InterfacePtr> bool operator!=(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p!=i; }
  351. template<typename _Interface> bool operator<(int null,_com_ptr_t<_Interface> &p) {
  352. if(null!=0) { _com_issue_error(E_POINTER); }
  353. return p>NULL;
  354. }
  355. template<typename _Interface,typename _InterfacePtr> bool operator<(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p>i; }
  356. template<typename _Interface> bool operator>(int null,_com_ptr_t<_Interface> &p) {
  357. if(null!=0) { _com_issue_error(E_POINTER); }
  358. return p<NULL;
  359. }
  360. template<typename _Interface,typename _InterfacePtr> bool operator>(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p<i; }
  361. template<typename _Interface> bool operator<=(int null,_com_ptr_t<_Interface> &p) {
  362. if(null!=0) { _com_issue_error(E_POINTER); }
  363. return p>=NULL;
  364. }
  365. template<typename _Interface,typename _InterfacePtr> bool operator<=(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p>=i; }
  366. template<typename _Interface> bool operator>=(int null,_com_ptr_t<_Interface> &p) {
  367. if(null!=0) { _com_issue_error(E_POINTER); }
  368. return p<=NULL;
  369. }
  370. template<typename _Interface,typename _InterfacePtr> bool operator>=(_Interface *i,_com_ptr_t<_InterfacePtr> &p) { return p<=i; }
  371. #pragma pop_macro("new")
  372. #endif /* __cplusplus */
  373. #endif /* _INC_COMIP */