VirtualBox

source: vbox/trunk/src/VBox/Main/include/VirtualBoxBase.h@ 40532

Last change on this file since 40532 was 40432, checked in by vboxsync, 13 years ago

Main/VirtualBox+Machine+Medium: redesign the medium registry save logic, as the local lists in the threads caused inconsistencies with concurrent operations and the return of spurious errors

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 36.6 KB
Line 
1/** @file
2 * VirtualBox COM base classes definition
3 */
4
5/*
6 * Copyright (C) 2006-2012 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#ifndef ____H_VIRTUALBOXBASEIMPL
18#define ____H_VIRTUALBOXBASEIMPL
19
20#include <iprt/cdefs.h>
21#include <iprt/thread.h>
22
23#include <list>
24#include <map>
25
26#include "VBox/com/AutoLock.h"
27#include "VBox/com/string.h"
28#include "VBox/com/Guid.h"
29
30#include "VBox/com/VirtualBox.h"
31
32// avoid including VBox/settings.h and VBox/xml.h;
33// only declare the classes
34namespace xml
35{
36class File;
37}
38
39namespace com
40{
41class ErrorInfo;
42}
43
44using namespace com;
45using namespace util;
46
47class AutoInitSpan;
48class AutoUninitSpan;
49
50class VirtualBox;
51class Machine;
52class Medium;
53class Host;
54typedef std::list< ComObjPtr<Medium> > MediaList;
55
56////////////////////////////////////////////////////////////////////////////////
57//
58// COM helpers
59//
60////////////////////////////////////////////////////////////////////////////////
61
62#if !defined (VBOX_WITH_XPCOM)
63
64#include <atlcom.h>
65
66/* use a special version of the singleton class factory,
67 * see KB811591 in msdn for more info. */
68
69#undef DECLARE_CLASSFACTORY_SINGLETON
70#define DECLARE_CLASSFACTORY_SINGLETON(obj) DECLARE_CLASSFACTORY_EX(CMyComClassFactorySingleton<obj>)
71
72template <class T>
73class CMyComClassFactorySingleton : public CComClassFactory
74{
75public:
76 CMyComClassFactorySingleton() : m_hrCreate(S_OK){}
77 virtual ~CMyComClassFactorySingleton(){}
78 // IClassFactory
79 STDMETHOD(CreateInstance)(LPUNKNOWN pUnkOuter, REFIID riid, void** ppvObj)
80 {
81 HRESULT hRes = E_POINTER;
82 if (ppvObj != NULL)
83 {
84 *ppvObj = NULL;
85 // Aggregation is not supported in singleton objects.
86 ATLASSERT(pUnkOuter == NULL);
87 if (pUnkOuter != NULL)
88 hRes = CLASS_E_NOAGGREGATION;
89 else
90 {
91 if (m_hrCreate == S_OK && m_spObj == NULL)
92 {
93 Lock();
94 __try
95 {
96 // Fix: The following If statement was moved inside the __try statement.
97 // Did another thread arrive here first?
98 if (m_hrCreate == S_OK && m_spObj == NULL)
99 {
100 // lock the module to indicate activity
101 // (necessary for the monitor shutdown thread to correctly
102 // terminate the module in case when CreateInstance() fails)
103 _pAtlModule->Lock();
104 CComObjectCached<T> *p;
105 m_hrCreate = CComObjectCached<T>::CreateInstance(&p);
106 if (SUCCEEDED(m_hrCreate))
107 {
108 m_hrCreate = p->QueryInterface(IID_IUnknown, (void**)&m_spObj);
109 if (FAILED(m_hrCreate))
110 {
111 delete p;
112 }
113 }
114 _pAtlModule->Unlock();
115 }
116 }
117 __finally
118 {
119 Unlock();
120 }
121 }
122 if (m_hrCreate == S_OK)
123 {
124 hRes = m_spObj->QueryInterface(riid, ppvObj);
125 }
126 else
127 {
128 hRes = m_hrCreate;
129 }
130 }
131 }
132 return hRes;
133 }
134 HRESULT m_hrCreate;
135 CComPtr<IUnknown> m_spObj;
136};
137
138#endif /* !defined (VBOX_WITH_XPCOM) */
139
140////////////////////////////////////////////////////////////////////////////////
141//
142// Macros
143//
144////////////////////////////////////////////////////////////////////////////////
145
146/**
147 * Special version of the Assert macro to be used within VirtualBoxBase
148 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
149 *
150 * In the debug build, this macro is equivalent to Assert.
151 * In the release build, this macro uses |setError(E_FAIL, ...)| to set the
152 * error info from the asserted expression.
153 *
154 * @see VirtualBoxSupportErrorInfoImpl::setError
155 *
156 * @param expr Expression which should be true.
157 */
158#if defined (DEBUG)
159#define ComAssert(expr) Assert(expr)
160#else
161#define ComAssert(expr) \
162 do { \
163 if (RT_UNLIKELY(!(expr))) \
164 setError(E_FAIL, \
165 "Assertion failed: [%s] at '%s' (%d) in %s.\nPlease contact the product vendor!", \
166 #expr, __FILE__, __LINE__, __PRETTY_FUNCTION__); \
167 } while (0)
168#endif
169
170/**
171 * Special version of the AssertFailed macro to be used within VirtualBoxBase
172 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
173 *
174 * In the debug build, this macro is equivalent to AssertFailed.
175 * In the release build, this macro uses |setError(E_FAIL, ...)| to set the
176 * error info from the asserted expression.
177 *
178 * @see VirtualBoxSupportErrorInfoImpl::setError
179 *
180 */
181#if defined (DEBUG)
182#define ComAssertFailed() AssertFailed()
183#else
184#define ComAssertFailed() \
185 do { \
186 setError(E_FAIL, \
187 "Assertion failed: at '%s' (%d) in %s.\nPlease contact the product vendor!", \
188 __FILE__, __LINE__, __PRETTY_FUNCTION__); \
189 } while (0)
190#endif
191
192/**
193 * Special version of the AssertMsg macro to be used within VirtualBoxBase
194 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
195 *
196 * See ComAssert for more info.
197 *
198 * @param expr Expression which should be true.
199 * @param a printf argument list (in parenthesis).
200 */
201#if defined (DEBUG)
202#define ComAssertMsg(expr, a) AssertMsg(expr, a)
203#else
204#define ComAssertMsg(expr, a) \
205 do { \
206 if (RT_UNLIKELY(!(expr))) \
207 setError(E_FAIL, \
208 "Assertion failed: [%s] at '%s' (%d) in %s.\n%s.\nPlease contact the product vendor!", \
209 #expr, __FILE__, __LINE__, __PRETTY_FUNCTION__, Utf8StrFmt a .c_str()); \
210 } while (0)
211#endif
212
213/**
214 * Special version of the AssertRC macro to be used within VirtualBoxBase
215 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
216 *
217 * See ComAssert for more info.
218 *
219 * @param vrc VBox status code.
220 */
221#if defined (DEBUG)
222#define ComAssertRC(vrc) AssertRC(vrc)
223#else
224#define ComAssertRC(vrc) ComAssertMsgRC(vrc, ("%Rra", vrc))
225#endif
226
227/**
228 * Special version of the AssertMsgRC macro to be used within VirtualBoxBase
229 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
230 *
231 * See ComAssert for more info.
232 *
233 * @param vrc VBox status code.
234 * @param msg printf argument list (in parenthesis).
235 */
236#if defined (DEBUG)
237#define ComAssertMsgRC(vrc, msg) AssertMsgRC(vrc, msg)
238#else
239#define ComAssertMsgRC(vrc, msg) ComAssertMsg(RT_SUCCESS(vrc), msg)
240#endif
241
242/**
243 * Special version of the AssertComRC macro to be used within VirtualBoxBase
244 * subclasses that also inherit the VirtualBoxSupportErrorInfoImpl template.
245 *
246 * See ComAssert for more info.
247 *
248 * @param rc COM result code
249 */
250#if defined (DEBUG)
251#define ComAssertComRC(rc) AssertComRC(rc)
252#else
253#define ComAssertComRC(rc) ComAssertMsg(SUCCEEDED(rc), ("COM RC = %Rhrc (0x%08X)", (rc), (rc)))
254#endif
255
256
257/** Special version of ComAssert that returns ret if expr fails */
258#define ComAssertRet(expr, ret) \
259 do { ComAssert(expr); if (!(expr)) return (ret); } while (0)
260/** Special version of ComAssertMsg that returns ret if expr fails */
261#define ComAssertMsgRet(expr, a, ret) \
262 do { ComAssertMsg(expr, a); if (!(expr)) return (ret); } while (0)
263/** Special version of ComAssertRC that returns ret if vrc does not succeed */
264#define ComAssertRCRet(vrc, ret) \
265 do { ComAssertRC(vrc); if (!RT_SUCCESS(vrc)) return (ret); } while (0)
266/** Special version of ComAssertComRC that returns ret if rc does not succeed */
267#define ComAssertComRCRet(rc, ret) \
268 do { ComAssertComRC(rc); if (!SUCCEEDED(rc)) return (ret); } while (0)
269/** Special version of ComAssertComRC that returns rc if rc does not succeed */
270#define ComAssertComRCRetRC(rc) \
271 do { ComAssertComRC(rc); if (!SUCCEEDED(rc)) return (rc); } while (0)
272/** Special version of ComAssert that returns ret */
273#define ComAssertFailedRet(ret) \
274 if (1) { ComAssertFailed(); { return (ret); } } else do {} while (0)
275
276
277/** Special version of ComAssert that evaluates eval and breaks if expr fails */
278#define ComAssertBreak(expr, eval) \
279 if (1) { ComAssert(expr); if (!(expr)) { eval; break; } } else do {} while (0)
280/** Special version of ComAssertMsg that evaluates eval and breaks if expr fails */
281#define ComAssertMsgBreak(expr, a, eval) \
282 if (1) { ComAssertMsg(expr, a); if (!(expr)) { eval; break; } } else do {} while (0)
283/** Special version of ComAssertRC that evaluates eval and breaks if vrc does not succeed */
284#define ComAssertRCBreak(vrc, eval) \
285 if (1) { ComAssertRC(vrc); if (!RT_SUCCESS(vrc)) { eval; break; } } else do {} while (0)
286/** Special version of ComAssertFailed that evaluates eval and breaks */
287#define ComAssertFailedBreak(eval) \
288 if (1) { ComAssertFailed(); { eval; break; } } else do {} while (0)
289/** Special version of ComAssertMsgFailed that evaluates eval and breaks */
290#define ComAssertMsgFailedBreak(msg, eval) \
291 if (1) { ComAssertMsgFailed (msg); { eval; break; } } else do {} while (0)
292/** Special version of ComAssertComRC that evaluates eval and breaks if rc does not succeed */
293#define ComAssertComRCBreak(rc, eval) \
294 if (1) { ComAssertComRC(rc); if (!SUCCEEDED(rc)) { eval; break; } } else do {} while (0)
295/** Special version of ComAssertComRC that just breaks if rc does not succeed */
296#define ComAssertComRCBreakRC(rc) \
297 if (1) { ComAssertComRC(rc); if (!SUCCEEDED(rc)) { break; } } else do {} while (0)
298
299
300/** Special version of ComAssert that evaluates eval and throws it if expr fails */
301#define ComAssertThrow(expr, eval) \
302 if (1) { ComAssert(expr); if (!(expr)) { throw (eval); } } else do {} while (0)
303/** Special version of ComAssertRC that evaluates eval and throws it if vrc does not succeed */
304#define ComAssertRCThrow(vrc, eval) \
305 if (1) { ComAssertRC(vrc); if (!RT_SUCCESS(vrc)) { throw (eval); } } else do {} while (0)
306/** Special version of ComAssertComRC that evaluates eval and throws it if rc does not succeed */
307#define ComAssertComRCThrow(rc, eval) \
308 if (1) { ComAssertComRC(rc); if (!SUCCEEDED(rc)) { throw (eval); } } else do {} while (0)
309/** Special version of ComAssertComRC that just throws rc if rc does not succeed */
310#define ComAssertComRCThrowRC(rc) \
311 if (1) { ComAssertComRC(rc); if (!SUCCEEDED(rc)) { throw rc; } } else do {} while (0)
312/** Special version of ComAssert that throws eval */
313#define ComAssertFailedThrow(eval) \
314 if (1) { ComAssertFailed(); { throw (eval); } } else do {} while (0)
315
316////////////////////////////////////////////////////////////////////////////////
317
318/**
319 * Checks that the pointer argument is not NULL and returns E_INVALIDARG +
320 * extended error info on failure.
321 * @param arg Input pointer-type argument (strings, interface pointers...)
322 */
323#define CheckComArgNotNull(arg) \
324 do { \
325 if (RT_UNLIKELY((arg) == NULL)) \
326 return setError(E_INVALIDARG, tr("Argument %s is NULL"), #arg); \
327 } while (0)
328
329/**
330 * Checks that the pointer argument is a valid pointer or NULL and returns
331 * E_INVALIDARG + extended error info on failure.
332 * @param arg Input pointer-type argument (strings, interface pointers...)
333 */
334#define CheckComArgMaybeNull(arg) \
335 do { \
336 if (RT_UNLIKELY(!RT_VALID_PTR(arg) && (arg) != NULL)) \
337 return setError(E_INVALIDARG, tr("Argument %s is an invalid pointer"), #arg); \
338 } while (0)
339
340/**
341 * Checks that safe array argument is not NULL and returns E_INVALIDARG +
342 * extended error info on failure.
343 * @param arg Input safe array argument (strings, interface pointers...)
344 */
345#define CheckComArgSafeArrayNotNull(arg) \
346 do { \
347 if (RT_UNLIKELY(ComSafeArrayInIsNull(arg))) \
348 return setError(E_INVALIDARG, tr("Argument %s is NULL"), #arg); \
349 } while (0)
350
351/**
352 * Checks that a string input argument is valid (not NULL or obviously invalid
353 * pointer), returning E_INVALIDARG + extended error info if invalid.
354 * @param a_bstrIn Input string argument (IN_BSTR).
355 */
356#define CheckComArgStr(a_bstrIn) \
357 do { \
358 IN_BSTR const bstrInCheck = (a_bstrIn); /* type check */ \
359 if (RT_UNLIKELY(!RT_VALID_PTR(bstrInCheck))) \
360 return setError(E_INVALIDARG, tr("Argument %s is an invalid pointer"), #a_bstrIn); \
361 } while (0)
362/**
363 * Checks that the string argument is not a NULL, a invalid pointer or an empty
364 * string, returning E_INVALIDARG + extended error info on failure.
365 * @param a_bstrIn Input string argument (BSTR etc.).
366 */
367#define CheckComArgStrNotEmptyOrNull(a_bstrIn) \
368 do { \
369 IN_BSTR const bstrInCheck = (a_bstrIn); /* type check */ \
370 if (RT_UNLIKELY(!RT_VALID_PTR(bstrInCheck) || *(bstrInCheck) == '\0')) \
371 return setError(E_INVALIDARG, tr("Argument %s is empty or an invalid pointer"), #a_bstrIn); \
372 } while (0)
373
374/**
375 * Converts the Guid input argument (string) to a Guid object, returns with
376 * E_INVALIDARG and error message on failure.
377 *
378 * @param a_Arg Argument.
379 * @param a_GuidVar The Guid variable name.
380 */
381#define CheckComArgGuid(a_Arg, a_GuidVar) \
382 do { \
383 Guid tmpGuid(a_Arg); \
384 (a_GuidVar) = tmpGuid; \
385 if (RT_UNLIKELY((a_GuidVar).isEmpty())) \
386 return setError(E_INVALIDARG, \
387 tr("GUID argument %s is not valid (\"%ls\")"), #a_Arg, Bstr(a_Arg).raw()); \
388 } while (0)
389
390/**
391 * Checks that the given expression (that must involve the argument) is true and
392 * returns E_INVALIDARG + extended error info on failure.
393 * @param arg Argument.
394 * @param expr Expression to evaluate.
395 */
396#define CheckComArgExpr(arg, expr) \
397 do { \
398 if (RT_UNLIKELY(!(expr))) \
399 return setError(E_INVALIDARG, \
400 tr("Argument %s is invalid (must be %s)"), #arg, #expr); \
401 } while (0)
402
403/**
404 * Checks that the given expression (that must involve the argument) is true and
405 * returns E_INVALIDARG + extended error info on failure. The error message must
406 * be customized.
407 * @param arg Argument.
408 * @param expr Expression to evaluate.
409 * @param msg Parenthesized printf-like expression (must start with a verb,
410 * like "must be one of...", "is not within...").
411 */
412#define CheckComArgExprMsg(arg, expr, msg) \
413 do { \
414 if (RT_UNLIKELY(!(expr))) \
415 return setError(E_INVALIDARG, tr ("Argument %s %s"), \
416 #arg, Utf8StrFmt msg .c_str()); \
417 } while (0)
418
419/**
420 * Checks that the given pointer to an output argument is valid and returns
421 * E_POINTER + extended error info otherwise.
422 * @param arg Pointer argument.
423 */
424#define CheckComArgOutPointerValid(arg) \
425 do { \
426 if (RT_UNLIKELY(!VALID_PTR(arg))) \
427 return setError(E_POINTER, \
428 tr("Output argument %s points to invalid memory location (%p)"), \
429 #arg, (void *) (arg)); \
430 } while (0)
431
432/**
433 * Checks that the given pointer to an output safe array argument is valid and
434 * returns E_POINTER + extended error info otherwise.
435 * @param arg Safe array argument.
436 */
437#define CheckComArgOutSafeArrayPointerValid(arg) \
438 do { \
439 if (RT_UNLIKELY(ComSafeArrayOutIsNull(arg))) \
440 return setError(E_POINTER, \
441 tr("Output argument %s points to invalid memory location (%p)"), \
442 #arg, (void*)(arg)); \
443 } while (0)
444
445/**
446 * Sets the extended error info and returns E_NOTIMPL.
447 */
448#define ReturnComNotImplemented() \
449 do { \
450 return setError(E_NOTIMPL, tr("Method %s is not implemented"), __FUNCTION__); \
451 } while (0)
452
453/**
454 * Declares an empty constructor and destructor for the given class.
455 * This is useful to prevent the compiler from generating the default
456 * ctor and dtor, which in turn allows to use forward class statements
457 * (instead of including their header files) when declaring data members of
458 * non-fundamental types with constructors (which are always called implicitly
459 * by constructors and by the destructor of the class).
460 *
461 * This macro is to be placed within (the public section of) the class
462 * declaration. Its counterpart, DEFINE_EMPTY_CTOR_DTOR, must be placed
463 * somewhere in one of the translation units (usually .cpp source files).
464 *
465 * @param cls class to declare a ctor and dtor for
466 */
467#define DECLARE_EMPTY_CTOR_DTOR(cls) cls(); ~cls();
468
469/**
470 * Defines an empty constructor and destructor for the given class.
471 * See DECLARE_EMPTY_CTOR_DTOR for more info.
472 */
473#define DEFINE_EMPTY_CTOR_DTOR(cls) \
474 cls::cls() { /*empty*/ } \
475 cls::~cls() { /*empty*/ }
476
477/**
478 * A variant of 'throw' that hits a debug breakpoint first to make
479 * finding the actual thrower possible.
480 */
481#ifdef DEBUG
482#define DebugBreakThrow(a) \
483 do { \
484 RTAssertDebugBreak(); \
485 throw (a); \
486} while (0)
487#else
488#define DebugBreakThrow(a) throw (a)
489#endif
490
491/**
492 * Parent class of VirtualBoxBase which enables translation support (which
493 * Main doesn't have yet, but this provides the tr() function which will one
494 * day provide translations).
495 *
496 * This class sits in between Lockable and VirtualBoxBase only for the one
497 * reason that the USBProxyService wants translation support but is not
498 * implemented as a COM object, which VirtualBoxBase implies.
499 */
500class ATL_NO_VTABLE VirtualBoxTranslatable
501 : public Lockable
502{
503public:
504
505 /**
506 * Placeholder method with which translations can one day be implemented
507 * in Main. This gets called by the tr() function.
508 * @param context
509 * @param pcszSourceText
510 * @param comment
511 * @return
512 */
513 static const char *translate(const char *context,
514 const char *pcszSourceText,
515 const char *comment = 0)
516 {
517 NOREF(context);
518 NOREF(comment);
519 return pcszSourceText;
520 }
521
522 /**
523 * Translates the given text string by calling translate() and passing
524 * the name of the C class as the first argument ("context of
525 * translation"). See VirtualBoxBase::translate() for more info.
526 *
527 * @param aSourceText String to translate.
528 * @param aComment Comment to the string to resolve possible
529 * ambiguities (NULL means no comment).
530 *
531 * @return Translated version of the source string in UTF-8 encoding, or
532 * the source string itself if the translation is not found in the
533 * specified context.
534 */
535 inline static const char *tr(const char *pcszSourceText,
536 const char *aComment = NULL)
537 {
538 return VirtualBoxTranslatable::translate(NULL, // getComponentName(), eventually
539 pcszSourceText,
540 aComment);
541 }
542};
543
544////////////////////////////////////////////////////////////////////////////////
545//
546// VirtualBoxBase
547//
548////////////////////////////////////////////////////////////////////////////////
549
550#define VIRTUALBOXBASE_ADD_VIRTUAL_COMPONENT_METHODS(cls, iface) \
551 virtual const IID& getClassIID() const \
552 { \
553 return cls::getStaticClassIID(); \
554 } \
555 static const IID& getStaticClassIID() \
556 { \
557 return COM_IIDOF(iface); \
558 } \
559 virtual const char* getComponentName() const \
560 { \
561 return cls::getStaticComponentName(); \
562 } \
563 static const char* getStaticComponentName() \
564 { \
565 return #cls; \
566 }
567
568/**
569 * VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT:
570 * This macro must be used once in the declaration of any class derived
571 * from VirtualBoxBase. It implements the pure virtual getClassIID() and
572 * getComponentName() methods. If this macro is not present, instances
573 * of a class derived from VirtualBoxBase cannot be instantiated.
574 *
575 * @param X The class name, e.g. "Class".
576 * @param IX The interface name which this class implements, e.g. "IClass".
577 */
578#ifdef VBOX_WITH_XPCOM
579 #define VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT(cls, iface) \
580 VIRTUALBOXBASE_ADD_VIRTUAL_COMPONENT_METHODS(cls, iface)
581#else // #ifdef VBOX_WITH_XPCOM
582 #define VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT(cls, iface) \
583 VIRTUALBOXBASE_ADD_VIRTUAL_COMPONENT_METHODS(cls, iface) \
584 STDMETHOD(InterfaceSupportsErrorInfo)(REFIID riid) \
585 { \
586 const _ATL_INTMAP_ENTRY* pEntries = cls::_GetEntries(); \
587 Assert(pEntries); \
588 if (!pEntries) \
589 return S_FALSE; \
590 BOOL bSupports = FALSE; \
591 BOOL bISupportErrorInfoFound = FALSE; \
592 while (pEntries->pFunc != NULL && !bSupports) \
593 { \
594 if (!bISupportErrorInfoFound) \
595 bISupportErrorInfoFound = InlineIsEqualGUID(*(pEntries->piid), IID_ISupportErrorInfo); \
596 else \
597 bSupports = InlineIsEqualGUID(*(pEntries->piid), riid); \
598 pEntries++; \
599 } \
600 Assert(bISupportErrorInfoFound); \
601 return bSupports ? S_OK : S_FALSE; \
602 }
603#endif // #ifdef VBOX_WITH_XPCOM
604
605/**
606 * Abstract base class for all component classes implementing COM
607 * interfaces of the VirtualBox COM library.
608 *
609 * Declares functionality that should be available in all components.
610 *
611 * Among the basic functionality implemented by this class is the primary object
612 * state that indicates if the object is ready to serve the calls, and if not,
613 * what stage it is currently at. Here is the primary state diagram:
614 *
615 * +-------------------------------------------------------+
616 * | |
617 * | (InitFailed) -----------------------+ |
618 * | ^ | |
619 * v | v |
620 * [*] ---> NotReady ----> (InInit) -----> Ready -----> (InUninit) ----+
621 * ^ |
622 * | v
623 * | Limited
624 * | |
625 * +-------+
626 *
627 * The object is fully operational only when its state is Ready. The Limited
628 * state means that only some vital part of the object is operational, and it
629 * requires some sort of reinitialization to become fully operational. The
630 * NotReady state means the object is basically dead: it either was not yet
631 * initialized after creation at all, or was uninitialized and is waiting to be
632 * destroyed when the last reference to it is released. All other states are
633 * transitional.
634 *
635 * The NotReady->InInit->Ready, NotReady->InInit->Limited and
636 * NotReady->InInit->InitFailed transition is done by the AutoInitSpan smart
637 * class.
638 *
639 * The Limited->InInit->Ready, Limited->InInit->Limited and
640 * Limited->InInit->InitFailed transition is done by the AutoReinitSpan smart
641 * class.
642 *
643 * The Ready->InUninit->NotReady and InitFailed->InUninit->NotReady
644 * transitions are done by the AutoUninitSpan smart class.
645 *
646 * In order to maintain the primary state integrity and declared functionality
647 * all subclasses must:
648 *
649 * 1) Use the above Auto*Span classes to perform state transitions. See the
650 * individual class descriptions for details.
651 *
652 * 2) All public methods of subclasses (i.e. all methods that can be called
653 * directly, not only from within other methods of the subclass) must have a
654 * standard prolog as described in the AutoCaller and AutoLimitedCaller
655 * documentation. Alternatively, they must use addCaller()/releaseCaller()
656 * directly (and therefore have both the prolog and the epilog), but this is
657 * not recommended.
658 */
659class ATL_NO_VTABLE VirtualBoxBase
660 : public VirtualBoxTranslatable,
661 public CComObjectRootEx<CComMultiThreadModel>
662#if !defined (VBOX_WITH_XPCOM)
663 , public ISupportErrorInfo
664#endif
665{
666protected:
667#ifdef RT_OS_WINDOWS
668 CComPtr <IUnknown> m_pUnkMarshaler;
669#endif
670
671 HRESULT BaseFinalConstruct()
672 {
673#ifdef RT_OS_WINDOWS
674 return CoCreateFreeThreadedMarshaler(this, //GetControllingUnknown(),
675 &m_pUnkMarshaler.p);
676#else
677 return S_OK;
678#endif
679 }
680
681 void BaseFinalRelease()
682 {
683#ifdef RT_OS_WINDOWS
684 m_pUnkMarshaler.Release();
685#endif
686 }
687
688
689public:
690 enum State { NotReady, Ready, InInit, InUninit, InitFailed, Limited };
691
692 VirtualBoxBase();
693 virtual ~VirtualBoxBase();
694
695 /**
696 * Uninitialization method.
697 *
698 * Must be called by all final implementations (component classes) when the
699 * last reference to the object is released, before calling the destructor.
700 *
701 * @note Never call this method the AutoCaller scope or after the
702 * #addCaller() call not paired by #releaseCaller() because it is a
703 * guaranteed deadlock. See AutoUninitSpan for details.
704 */
705 virtual void uninit()
706 { }
707
708 virtual HRESULT addCaller(State *aState = NULL,
709 bool aLimited = false);
710 virtual void releaseCaller();
711
712 /**
713 * Adds a limited caller. This method is equivalent to doing
714 * <tt>addCaller (aState, true)</tt>, but it is preferred because provides
715 * better self-descriptiveness. See #addCaller() for more info.
716 */
717 HRESULT addLimitedCaller(State *aState = NULL)
718 {
719 return addCaller(aState, true /* aLimited */);
720 }
721
722 /**
723 * Pure virtual method for simple run-time type identification without
724 * having to enable C++ RTTI.
725 *
726 * This *must* be implemented by every subclass deriving from VirtualBoxBase;
727 * use the VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT macro to do that most easily.
728 */
729 virtual const IID& getClassIID() const = 0;
730
731 /**
732 * Pure virtual method for simple run-time type identification without
733 * having to enable C++ RTTI.
734 *
735 * This *must* be implemented by every subclass deriving from VirtualBoxBase;
736 * use the VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT macro to do that most easily.
737 */
738 virtual const char* getComponentName() const = 0;
739
740 /**
741 * Virtual method which determines the locking class to be used for validating
742 * lock order with the standard member lock handle. This method is overridden
743 * in a number of subclasses.
744 */
745 virtual VBoxLockingClass getLockingClass() const
746 {
747 return LOCKCLASS_OTHEROBJECT;
748 }
749
750 virtual RWLockHandle *lockHandle() const;
751
752 /**
753 * Returns a lock handle used to protect the primary state fields (used by
754 * #addCaller(), AutoInitSpan, AutoUninitSpan, etc.). Only intended to be
755 * used for similar purposes in subclasses. WARNING: NO any other locks may
756 * be requested while holding this lock!
757 */
758 WriteLockHandle *stateLockHandle() { return &mStateLock; }
759
760 static HRESULT setErrorInternal(HRESULT aResultCode,
761 const GUID &aIID,
762 const char *aComponent,
763 Utf8Str aText,
764 bool aWarning,
765 bool aLogIt);
766 static void clearError(void);
767
768 HRESULT setError(HRESULT aResultCode);
769 HRESULT setError(HRESULT aResultCode, const char *pcsz, ...);
770 HRESULT setError(const ErrorInfo &ei);
771 HRESULT setWarning(HRESULT aResultCode, const char *pcsz, ...);
772 HRESULT setErrorNoLog(HRESULT aResultCode, const char *pcsz, ...);
773
774
775 /** Initialize COM for a new thread. */
776 static HRESULT initializeComForThread(void)
777 {
778#ifndef VBOX_WITH_XPCOM
779 HRESULT hrc = CoInitializeEx(NULL, COINIT_MULTITHREADED | COINIT_DISABLE_OLE1DDE | COINIT_SPEED_OVER_MEMORY);
780 AssertComRCReturn(hrc, hrc);
781#endif
782 return S_OK;
783 }
784
785 /** Uninitializes COM for a dying thread. */
786 static void uninitializeComForThread(void)
787 {
788#ifndef VBOX_WITH_XPCOM
789 CoUninitialize();
790#endif
791 }
792
793
794private:
795
796 void setState(State aState)
797 {
798 Assert(mState != aState);
799 mState = aState;
800 mStateChangeThread = RTThreadSelf();
801 }
802
803 /** Primary state of this object */
804 State mState;
805 /** Thread that caused the last state change */
806 RTTHREAD mStateChangeThread;
807 /** Total number of active calls to this object */
808 unsigned mCallers;
809 /** Posted when the number of callers drops to zero */
810 RTSEMEVENT mZeroCallersSem;
811 /** Posted when the object goes from InInit/InUninit to some other state */
812 RTSEMEVENTMULTI mInitUninitSem;
813 /** Number of threads waiting for mInitUninitDoneSem */
814 unsigned mInitUninitWaiters;
815
816 /** Protects access to state related data members */
817 WriteLockHandle mStateLock;
818
819 /** User-level object lock for subclasses */
820 mutable RWLockHandle *mObjectLock;
821
822 friend class AutoInitSpan;
823 friend class AutoReinitSpan;
824 friend class AutoUninitSpan;
825};
826
827/**
828 * Dummy macro that is used to shut down Qt's lupdate tool warnings in some
829 * situations. This macro needs to be present inside (better at the very
830 * beginning) of the declaration of the class that inherits from
831 * VirtualBoxSupportTranslation template, to make lupdate happy.
832 */
833#define Q_OBJECT
834
835////////////////////////////////////////////////////////////////////////////////
836
837////////////////////////////////////////////////////////////////////////////////
838
839
840/**
841 * Simple template that manages data structure allocation/deallocation
842 * and supports data pointer sharing (the instance that shares the pointer is
843 * not responsible for memory deallocation as opposed to the instance that
844 * owns it).
845 */
846template <class D>
847class Shareable
848{
849public:
850
851 Shareable() : mData (NULL), mIsShared(FALSE) {}
852 ~Shareable() { free(); }
853
854 void allocate() { attach(new D); }
855
856 virtual void free() {
857 if (mData) {
858 if (!mIsShared)
859 delete mData;
860 mData = NULL;
861 mIsShared = false;
862 }
863 }
864
865 void attach(D *d) {
866 AssertMsg(d, ("new data must not be NULL"));
867 if (d && mData != d) {
868 if (mData && !mIsShared)
869 delete mData;
870 mData = d;
871 mIsShared = false;
872 }
873 }
874
875 void attach(Shareable &d) {
876 AssertMsg(
877 d.mData == mData || !d.mIsShared,
878 ("new data must not be shared")
879 );
880 if (this != &d && !d.mIsShared) {
881 attach(d.mData);
882 d.mIsShared = true;
883 }
884 }
885
886 void share(D *d) {
887 AssertMsg(d, ("new data must not be NULL"));
888 if (mData != d) {
889 if (mData && !mIsShared)
890 delete mData;
891 mData = d;
892 mIsShared = true;
893 }
894 }
895
896 void share(const Shareable &d) { share(d.mData); }
897
898 void attachCopy(const D *d) {
899 AssertMsg(d, ("data to copy must not be NULL"));
900 if (d)
901 attach(new D(*d));
902 }
903
904 void attachCopy(const Shareable &d) {
905 attachCopy(d.mData);
906 }
907
908 virtual D *detach() {
909 D *d = mData;
910 mData = NULL;
911 mIsShared = false;
912 return d;
913 }
914
915 D *data() const {
916 return mData;
917 }
918
919 D *operator->() const {
920 AssertMsg(mData, ("data must not be NULL"));
921 return mData;
922 }
923
924 bool isNull() const { return mData == NULL; }
925 bool operator!() const { return isNull(); }
926
927 bool isShared() const { return mIsShared; }
928
929protected:
930
931 D *mData;
932 bool mIsShared;
933};
934
935/**
936 * Simple template that enhances Shareable<> and supports data
937 * backup/rollback/commit (using the copy constructor of the managed data
938 * structure).
939 */
940template<class D>
941class Backupable : public Shareable<D>
942{
943public:
944
945 Backupable() : Shareable<D> (), mBackupData(NULL) {}
946
947 void free()
948 {
949 AssertMsg(this->mData || !mBackupData, ("backup must be NULL if data is NULL"));
950 rollback();
951 Shareable<D>::free();
952 }
953
954 D *detach()
955 {
956 AssertMsg(this->mData || !mBackupData, ("backup must be NULL if data is NULL"));
957 rollback();
958 return Shareable<D>::detach();
959 }
960
961 void share(const Backupable &d)
962 {
963 AssertMsg(!d.isBackedUp(), ("data to share must not be backed up"));
964 if (!d.isBackedUp())
965 Shareable<D>::share(d.mData);
966 }
967
968 /**
969 * Stores the current data pointer in the backup area, allocates new data
970 * using the copy constructor on current data and makes new data active.
971 *
972 * @deprecated Use backupEx to avoid throwing wild out-of-memory exceptions.
973 */
974 void backup()
975 {
976 AssertMsg(this->mData, ("data must not be NULL"));
977 if (this->mData && !mBackupData)
978 {
979 D *pNewData = new D(*this->mData);
980 mBackupData = this->mData;
981 this->mData = pNewData;
982 }
983 }
984
985 /**
986 * Stores the current data pointer in the backup area, allocates new data
987 * using the copy constructor on current data and makes new data active.
988 *
989 * @returns S_OK, E_OUTOFMEMORY or E_FAIL (internal error).
990 */
991 HRESULT backupEx()
992 {
993 AssertMsgReturn(this->mData, ("data must not be NULL"), E_FAIL);
994 if (this->mData && !mBackupData)
995 {
996 try
997 {
998 D *pNewData = new D(*this->mData);
999 mBackupData = this->mData;
1000 this->mData = pNewData;
1001 }
1002 catch (std::bad_alloc &)
1003 {
1004 return E_OUTOFMEMORY;
1005 }
1006 }
1007 return S_OK;
1008 }
1009
1010 /**
1011 * Deletes new data created by #backup() and restores previous data pointer
1012 * stored in the backup area, making it active again.
1013 */
1014 void rollback()
1015 {
1016 if (this->mData && mBackupData)
1017 {
1018 delete this->mData;
1019 this->mData = mBackupData;
1020 mBackupData = NULL;
1021 }
1022 }
1023
1024 /**
1025 * Commits current changes by deleting backed up data and clearing up the
1026 * backup area. The new data pointer created by #backup() remains active
1027 * and becomes the only managed pointer.
1028 *
1029 * This method is much faster than #commitCopy() (just a single pointer
1030 * assignment operation), but makes the previous data pointer invalid
1031 * (because it is freed). For this reason, this method must not be
1032 * used if it's possible that data managed by this instance is shared with
1033 * some other Shareable instance. See #commitCopy().
1034 */
1035 void commit()
1036 {
1037 if (this->mData && mBackupData)
1038 {
1039 if (!this->mIsShared)
1040 delete mBackupData;
1041 mBackupData = NULL;
1042 this->mIsShared = false;
1043 }
1044 }
1045
1046 /**
1047 * Commits current changes by assigning new data to the previous data
1048 * pointer stored in the backup area using the assignment operator.
1049 * New data is deleted, the backup area is cleared and the previous data
1050 * pointer becomes active and the only managed pointer.
1051 *
1052 * This method is slower than #commit(), but it keeps the previous data
1053 * pointer valid (i.e. new data is copied to the same memory location).
1054 * For that reason it's safe to use this method on instances that share
1055 * managed data with other Shareable instances.
1056 */
1057 void commitCopy()
1058 {
1059 if (this->mData && mBackupData)
1060 {
1061 *mBackupData = *(this->mData);
1062 delete this->mData;
1063 this->mData = mBackupData;
1064 mBackupData = NULL;
1065 }
1066 }
1067
1068 void assignCopy(const D *pData)
1069 {
1070 AssertMsg(this->mData, ("data must not be NULL"));
1071 AssertMsg(pData, ("data to copy must not be NULL"));
1072 if (this->mData && pData)
1073 {
1074 if (!mBackupData)
1075 {
1076 D *pNewData = new D(*pData);
1077 mBackupData = this->mData;
1078 this->mData = pNewData;
1079 }
1080 else
1081 *this->mData = *pData;
1082 }
1083 }
1084
1085 void assignCopy(const Backupable &d)
1086 {
1087 assignCopy(d.mData);
1088 }
1089
1090 bool isBackedUp() const
1091 {
1092 return mBackupData != NULL;
1093 }
1094
1095 D *backedUpData() const
1096 {
1097 return mBackupData;
1098 }
1099
1100protected:
1101
1102 D *mBackupData;
1103};
1104
1105#endif // !____H_VIRTUALBOXBASEIMPL
1106
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