VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/include/COMDefs.h@ 6645

Last change on this file since 6645 was 5999, checked in by vboxsync, 17 years ago

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.7 KB
Line 
1/** @file
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * Various COM definitions and COM wrapper class declarations
5 *
6 * This header is used in conjunction with the header generated from
7 * XIDL expressed interface definitions to provide cross-platform Qt-based
8 * interface wrapper classes.
9 */
10
11/*
12 * Copyright (C) 2006-2007 innotek GmbH
13 *
14 * This file is part of VirtualBox Open Source Edition (OSE), as
15 * available from http://www.virtualbox.org. This file is free software;
16 * you can redistribute it and/or modify it under the terms of the GNU
17 * General Public License (GPL) as published by the Free Software
18 * Foundation, in version 2 as it comes in the "COPYING" file of the
19 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
20 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
21 */
22
23#ifndef __COMDefs_h__
24#define __COMDefs_h__
25
26
27/* Both VBox/com/assert.h and qglobal.h contain a definition of ASSERT.
28 * Either of them can be already included here, so try to shut them up. */
29#undef ASSERT
30
31#include <VBox/com/com.h>
32#include <VBox/com/assert.h>
33
34#undef ASSERT
35
36#include <qglobal.h>
37#include <qstring.h>
38#include <quuid.h>
39
40#include <iprt/memory> // for auto_copy_ptr
41
42/*
43 * Additional COM / XPCOM defines and includes
44 */
45
46#define IN_BSTRPARAM INPTR BSTR
47#define IN_GUIDPARAM INPTR GUIDPARAM
48
49#if !defined (VBOX_WITH_XPCOM)
50
51#else /* !defined (VBOX_WITH_XPCOM) */
52
53#include <nsXPCOM.h>
54#include <nsMemory.h>
55#include <nsIComponentManager.h>
56
57class XPCOMEventQSocketListener;
58
59#endif /* !defined (VBOX_WITH_XPCOM) */
60
61
62/* VirtualBox interfaces declarations */
63#if !defined (VBOX_WITH_XPCOM)
64 #include <VirtualBox.h>
65#else /* !defined (VBOX_WITH_XPCOM) */
66 #include <VirtualBox_XPCOM.h>
67#endif /* !defined (VBOX_WITH_XPCOM) */
68
69#include "VBoxDefs.h"
70
71
72/////////////////////////////////////////////////////////////////////////////
73
74class CVirtualBoxErrorInfo;
75
76/** Represents extended error information */
77class COMErrorInfo
78{
79public:
80
81 COMErrorInfo()
82 : mIsNull (true)
83 , mIsBasicAvailable (false), mIsFullAvailable (false)
84 , mResultCode (S_OK) {}
85
86 COMErrorInfo (const CVirtualBoxErrorInfo &info) { init (info); }
87
88 /* the default copy ctor and assignment op are ok */
89
90 bool isNull() const { return mIsNull; }
91
92 bool isBasicAvailable() const { return mIsBasicAvailable; }
93 bool isFullAvailable() const { return mIsFullAvailable; }
94
95 HRESULT resultCode() const { return mResultCode; }
96 QUuid interfaceID() const { return mInterfaceID; }
97 QString component() const { return mComponent; }
98 QString text() const { return mText; }
99
100 const COMErrorInfo *next() const { return mNext.get(); }
101
102 QString interfaceName() const { return mInterfaceName; }
103 QUuid calleeIID() const { return mCalleeIID; }
104 QString calleeName() const { return mCalleeName; }
105
106private:
107
108 void init (const CVirtualBoxErrorInfo &info);
109 void fetchFromCurrentThread (IUnknown *callee, const GUID *calleeIID);
110
111 static QString getInterfaceNameFromIID (const QUuid &id);
112
113 bool mIsNull : 1;
114 bool mIsBasicAvailable : 1;
115 bool mIsFullAvailable : 1;
116
117 HRESULT mResultCode;
118 QUuid mInterfaceID;
119 QString mComponent;
120 QString mText;
121
122 cppx::auto_copy_ptr <COMErrorInfo> mNext;
123
124 QString mInterfaceName;
125 QUuid mCalleeIID;
126 QString mCalleeName;
127
128 friend class COMBaseWithEI;
129};
130
131/////////////////////////////////////////////////////////////////////////////
132
133/**
134 * Base COM class the CInterface template and all wrapper classes are derived
135 * from. Provides common functionality for all COM wrappers.
136 */
137class COMBase
138{
139public:
140
141 static HRESULT initializeCOM();
142 static HRESULT cleanupCOM();
143
144#if !defined (VBOX_WITH_XPCOM)
145
146 /** Converts a GUID value to QUuid */
147 static QUuid toQUuid (const GUID &id)
148 {
149 return QUuid (id.Data1, id.Data2, id.Data3,
150 id.Data4[0], id.Data4[1], id.Data4[2], id.Data4[3],
151 id.Data4[4], id.Data4[5], id.Data4[6], id.Data4[7]);
152 }
153
154#else /* !defined (VBOX_WITH_XPCOM) */
155
156 /** Converts a GUID value to QUuid */
157 static QUuid toQUuid (const nsID &id)
158 {
159 return QUuid (id.m0, id.m1, id.m2,
160 id.m3[0], id.m3[1], id.m3[2], id.m3[3],
161 id.m3[4], id.m3[5], id.m3[6], id.m3[7]);
162 }
163
164#endif /* !defined (VBOX_WITH_XPCOM) */
165
166 /**
167 * Returns the result code of the last interface method called
168 * by the wrapper instance or the result of CInterface::createInstance()
169 * operation.
170 */
171 HRESULT lastRC() const { return mRC; }
172
173 /**
174 * Returns error info set by the last unsuccessfully invoked interface
175 * method. Returned error info is useful only if CInterface::lastRC()
176 * represents a failure (i.e. CInterface::isOk() is false).
177 */
178 virtual COMErrorInfo errorInfo() const { return COMErrorInfo(); }
179
180protected:
181
182 /* no arbitrary instance creations */
183 COMBase() : mRC (S_OK) {};
184
185#if defined (VBOX_WITH_XPCOM)
186 static XPCOMEventQSocketListener *sSocketListener;
187#endif
188
189 /** Adapter to pass QString as input BSTR params */
190 class BSTRIn
191 {
192 public:
193
194 BSTRIn (const QString &s) : bstr (SysAllocString ((const OLECHAR *) s.ucs2())) {}
195
196 ~BSTRIn()
197 {
198 if (bstr)
199 SysFreeString (bstr);
200 }
201
202 operator BSTR() const { return bstr; }
203
204 private:
205
206 BSTR bstr;
207 };
208
209 /** Adapter to pass QString as output BSTR params */
210 class BSTROut
211 {
212 public:
213
214 BSTROut (QString &s) : str (s), bstr (0) {}
215
216 ~BSTROut()
217 {
218 if (bstr) {
219 str = QString::fromUcs2 (bstr);
220 SysFreeString (bstr);
221 }
222 }
223
224 operator BSTR *() { return &bstr; }
225
226 private:
227
228 QString &str;
229 BSTR bstr;
230 };
231
232 /** Adapter to pass CEnums enums as output VirtualBox enum params (*_T) */
233 template <typename CE, typename VE>
234 class ENUMOut
235 {
236 public:
237
238 ENUMOut (CE &e) : ce (e), ve ((VE) 0) {}
239 ~ENUMOut() { ce = (CE) ve; }
240 operator VE *() { return &ve; }
241
242 private:
243
244 CE &ce;
245 VE ve;
246 };
247
248#if !defined (VBOX_WITH_XPCOM)
249
250 /** Adapter to pass QUuid as input GUID params */
251 static GUID GUIDIn (const QUuid &uuid) { return uuid; }
252
253 /** Adapter to pass QUuid as output GUID params */
254 class GUIDOut
255 {
256 public:
257
258 GUIDOut (QUuid &id) : uuid (id)
259 {
260 ::memset (&guid, 0, sizeof (GUID));
261 }
262
263 ~GUIDOut()
264 {
265 uuid = QUuid (
266 guid.Data1, guid.Data2, guid.Data3,
267 guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3],
268 guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]);
269 }
270
271 operator GUID *() { return &guid; }
272
273 private:
274
275 QUuid &uuid;
276 GUID guid;
277 };
278
279#else /* !defined (VBOX_WITH_XPCOM) */
280
281 /** Adapter to pass QUuid as input GUID params */
282 static const nsID &GUIDIn (const QUuid &uuid)
283 {
284 return *(const nsID *) &uuid;
285 }
286
287 /** Adapter to pass QUuid as output GUID params */
288 class GUIDOut
289 {
290 public:
291
292 GUIDOut (QUuid &id) : uuid (id), nsid (0) {}
293
294 ~GUIDOut()
295 {
296 if (nsid)
297 {
298 uuid = QUuid (
299 nsid->m0, nsid->m1, nsid->m2,
300 nsid->m3[0], nsid->m3[1], nsid->m3[2], nsid->m3[3],
301 nsid->m3[4], nsid->m3[5], nsid->m3[6], nsid->m3[7]);
302 nsMemory::Free (nsid);
303 }
304 }
305
306 operator nsID **() { return &nsid; }
307
308 private:
309
310 QUuid &uuid;
311 nsID *nsid;
312 };
313
314#endif /* !defined (VBOX_WITH_XPCOM) */
315
316 void fetchErrorInfo (IUnknown * /*callee*/, const GUID * /*calleeIID*/) const {}
317
318 mutable HRESULT mRC;
319
320 friend class COMErrorInfo;
321};
322
323/////////////////////////////////////////////////////////////////////////////
324
325/**
326 * Alternative base class for the CInterface template that adds
327 * the errorInfo() method for providing extended error info about
328 * unsuccessful invocation of the last called interface method.
329 */
330class COMBaseWithEI : public COMBase
331{
332public:
333
334 /**
335 * Returns error info set by the last unsuccessfully invoked interface
336 * method. Returned error info is useful only if CInterface::lastRC()
337 * represents a failure (i.e. CInterface::isOk() is false).
338 */
339 COMErrorInfo errorInfo() const { return mErrInfo; }
340
341protected:
342
343 /* no arbitrary instance creations */
344 COMBaseWithEI() : COMBase () {};
345
346 void fetchErrorInfo (IUnknown *callee, const GUID *calleeIID) const
347 {
348 mErrInfo.fetchFromCurrentThread (callee, calleeIID);
349 }
350
351 mutable COMErrorInfo mErrInfo;
352};
353
354/////////////////////////////////////////////////////////////////////////////
355
356/**
357 * Simple class that encapsulates the result code and COMErrorInfo.
358 */
359class COMResult
360{
361public:
362
363 COMResult() : mRC (S_OK) {}
364
365 /** Queries the current result code and error info from the given component */
366 COMResult (const COMBase &aComponent)
367 {
368 mErrInfo = aComponent.errorInfo();
369 mRC = aComponent.lastRC();
370 }
371
372 /** Queries the current result code and error info from the given component */
373 COMResult &operator= (const COMBase &aComponent)
374 {
375 mErrInfo = aComponent.errorInfo();
376 mRC = aComponent.lastRC();
377 return *this;
378 }
379
380 bool isNull() const { return mErrInfo.isNull(); }
381 bool isOk() const { return mErrInfo.isNull() && SUCCEEDED (mRC); }
382
383 COMErrorInfo errorInfo() const { return mErrInfo; }
384 HRESULT rc() const { return mRC; }
385
386private:
387
388 COMErrorInfo mErrInfo;
389 HRESULT mRC;
390};
391
392/////////////////////////////////////////////////////////////////////////////
393
394class CUnknown;
395
396/**
397 * Wrapper template class for all interfaces.
398 *
399 * All interface methods named as they are in the original, i.e. starting
400 * with the capital letter. All utility non-interface methods are named
401 * starting with the small letter. Utility methods should be not normally
402 * called by the end-user client application.
403 *
404 * @param I interface class (i.e. IUnknown/nsISupports derivant)
405 * @param B base class, either COMBase (by default) or COMBaseWithEI
406 */
407template <class I, class B = COMBase>
408class CInterface : public B
409{
410public:
411
412 typedef B Base;
413 typedef I Iface;
414
415 /* constructors & destructor */
416
417 CInterface() : mIface (NULL) {}
418
419 CInterface (const CInterface &that) : B (that), mIface (that.mIface)
420 {
421 addref (mIface);
422 }
423
424 CInterface (const CUnknown &that);
425
426 CInterface (I *i) : mIface (i) { addref (mIface); }
427
428 virtual ~CInterface() { release (mIface); }
429
430 /* utility methods */
431
432 void createInstance (const CLSID &clsid)
433 {
434 AssertMsg (!mIface, ("Instance is already non-NULL\n"));
435 if (!mIface)
436 {
437#if !defined (VBOX_WITH_XPCOM)
438
439 B::mRC = CoCreateInstance (clsid, NULL, CLSCTX_ALL,
440 _ATL_IIDOF (I), (void **) &mIface);
441
442#else /* !defined (VBOX_WITH_XPCOM) */
443
444 nsCOMPtr <nsIComponentManager> manager;
445 B::mRC = NS_GetComponentManager (getter_AddRefs (manager));
446 if (SUCCEEDED (B::mRC))
447 B::mRC = manager->CreateInstance (clsid, nsnull, NS_GET_IID (I),
448 (void **) &mIface);
449
450#endif /* !defined (VBOX_WITH_XPCOM) */
451
452 /* fetch error info, but don't assert if it's missing -- many other
453 * reasons can lead to an error (w/o providing error info), not only
454 * the instance initialization code (that should always provide it) */
455 B::fetchErrorInfo (NULL, NULL);
456 }
457 }
458
459 void attach (I *i)
460 {
461 /* be aware of self (from COM point of view) assignment */
462 I *old_iface = mIface;
463 mIface = i;
464 addref (mIface);
465 release (old_iface);
466 B::mRC = S_OK;
467 };
468
469 void attachUnknown (IUnknown *i)
470 {
471 /* be aware of self (from COM point of view) assignment */
472 I *old_iface = mIface;
473 mIface = NULL;
474 B::mRC = S_OK;
475 if (i)
476#if !defined (VBOX_WITH_XPCOM)
477 B::mRC = i->QueryInterface (_ATL_IIDOF (I), (void **) &mIface);
478#else /* !defined (VBOX_WITH_XPCOM) */
479 B::mRC = i->QueryInterface (NS_GET_IID (I), (void **) &mIface);
480#endif /* !defined (VBOX_WITH_XPCOM) */
481 release (old_iface);
482 };
483
484 void detach() { release (mIface); mIface = NULL; }
485
486 bool isNull() const { return mIface == NULL; }
487
488 bool isOk() const { return !isNull() && SUCCEEDED (B::mRC); }
489
490 /* utility operators */
491
492 CInterface &operator= (const CInterface &that)
493 {
494 attach (that.mIface);
495 B::operator= (that);
496 return *this;
497 }
498
499 I *iface() const { return mIface; }
500
501 bool operator== (const CInterface &that) const { return mIface == that.mIface; }
502 bool operator!= (const CInterface &that) const { return mIface != that.mIface; }
503
504 CInterface &operator= (const CUnknown &that);
505
506protected:
507
508 static void addref (I *i) { if (i) i->AddRef(); }
509 static void release (I *i) { if (i) i->Release(); }
510
511 mutable I *mIface;
512};
513
514/////////////////////////////////////////////////////////////////////////////
515
516class CUnknown : public CInterface <IUnknown, COMBaseWithEI>
517{
518public:
519
520 CUnknown() : CInterface <IUnknown, COMBaseWithEI> () {}
521
522 template <class C>
523 explicit CUnknown (const C &that)
524 {
525 mIface = NULL;
526 if (that.mIface)
527#if !defined (VBOX_WITH_XPCOM)
528 mRC = that.mIface->QueryInterface (_ATL_IIDOF (IUnknown), (void**) &mIface);
529#else /* !defined (VBOX_WITH_XPCOM) */
530 mRC = that.mIface->QueryInterface (NS_GET_IID (IUnknown), (void**) &mIface);
531#endif /* !defined (VBOX_WITH_XPCOM) */
532 if (SUCCEEDED (mRC))
533 {
534 mRC = that.lastRC();
535 mErrInfo = that.errorInfo();
536 }
537 }
538
539 /* specialization for CUnknown */
540 CUnknown (const CUnknown &that) : CInterface <IUnknown, COMBaseWithEI> ()
541 {
542 mIface = that.mIface;
543 addref (mIface);
544 COMBaseWithEI::operator= (that);
545 }
546
547 template <class C>
548 CUnknown &operator= (const C &that)
549 {
550 /* be aware of self (from COM point of view) assignment */
551 IUnknown *old_iface = mIface;
552 mIface = NULL;
553 mRC = S_OK;
554#if !defined (VBOX_WITH_XPCOM)
555 if (that.mIface)
556 mRC = that.mIface->QueryInterface (_ATL_IIDOF (IUnknown), (void**) &mIface);
557#else /* !defined (VBOX_WITH_XPCOM) */
558 if (that.mIface)
559 mRC = that.mIface->QueryInterface (NS_GET_IID (IUnknown), (void**) &mIface);
560#endif /* !defined (VBOX_WITH_XPCOM) */
561 if (SUCCEEDED (mRC))
562 {
563 mRC = that.lastRC();
564 mErrInfo = that.errorInfo();
565 }
566 release (old_iface);
567 return *this;
568 }
569
570 /* specialization for CUnknown */
571 CUnknown &operator= (const CUnknown &that)
572 {
573 attach (that.mIface);
574 COMBaseWithEI::operator= (that);
575 return *this;
576 }
577
578 /* @internal Used in wrappers. */
579 IUnknown *&ifaceRef() { return mIface; };
580};
581
582/* inlined CInterface methods that use CUnknown */
583
584template <class I, class B>
585inline CInterface <I, B>::CInterface (const CUnknown &that)
586 : mIface (NULL)
587{
588 attachUnknown (that.iface());
589 if (SUCCEEDED (B::mRC))
590 B::operator= ((B &) that);
591}
592
593template <class I, class B>
594inline CInterface <I, B> &CInterface <I, B>::operator =(const CUnknown &that)
595{
596 attachUnknown (that.iface());
597 if (SUCCEEDED (B::mRC))
598 B::operator= ((B &) that);
599 return *this;
600}
601
602/////////////////////////////////////////////////////////////////////////////
603
604/* include the generated header containing concrete wrapper definitions */
605#include "COMWrappers.h"
606
607#endif // __COMDefs_h__
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette