VirtualBox

source: vbox/trunk/src/VBox/Main/src-all/VirtualBoxBase.cpp@ 74930

Last change on this file since 74930 was 72938, checked in by vboxsync, 7 years ago

Main: VirtualBoxBase::setErrorInternal: Provide default error text for VBOX_E_PASSWORD_INCORRECT like we do for the other VBOX_E_XXXX status codes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 28.0 KB
Line 
1/* $Id: VirtualBoxBase.cpp 72938 2018-07-06 21:12:49Z vboxsync $ */
2
3/** @file
4 *
5 * VirtualBox COM base classes implementation
6 */
7
8/*
9 * Copyright (C) 2006-2017 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20#include <iprt/semaphore.h>
21#include <iprt/asm.h>
22#include <iprt/cpp/exception.h>
23
24#include <typeinfo>
25
26#if !defined(VBOX_WITH_XPCOM)
27#include <iprt/win/windows.h>
28#include <dbghelp.h>
29#else /* !defined(VBOX_WITH_XPCOM) */
30/// @todo remove when VirtualBoxErrorInfo goes away from here
31#include <nsIServiceManager.h>
32#include <nsIExceptionService.h>
33#endif /* !defined(VBOX_WITH_XPCOM) */
34
35#include "VirtualBoxBase.h"
36#include "AutoCaller.h"
37#include "VirtualBoxErrorInfoImpl.h"
38#include "Logging.h"
39#include "Global.h"
40
41#include "VBox/com/ErrorInfo.h"
42#include "VBox/com/MultiResult.h"
43
44////////////////////////////////////////////////////////////////////////////////
45//
46// VirtualBoxBase
47//
48////////////////////////////////////////////////////////////////////////////////
49
50CLASSFACTORY_STAT g_aClassFactoryStats[CLASSFACTORYSTATS_MAX] =
51{
52 { "--- totals ---", 0 },
53 { NULL, 0 }
54};
55
56RWLockHandle *g_pClassFactoryStatsLock = NULL;
57
58
59VirtualBoxBase::VirtualBoxBase() :
60 mState(this),
61 iFactoryStat(~0U)
62{
63 mObjectLock = NULL;
64
65 if (!g_pClassFactoryStatsLock)
66 {
67 RWLockHandle *lock = new RWLockHandle(LOCKCLASS_OBJECTSTATE);
68 if (!ASMAtomicCmpXchgPtr(&g_pClassFactoryStatsLock, lock, NULL))
69 delete lock;
70 }
71 Assert(g_pClassFactoryStatsLock);
72}
73
74VirtualBoxBase::~VirtualBoxBase()
75{
76 Assert(iFactoryStat == ~0U);
77 if (mObjectLock)
78 delete mObjectLock;
79}
80
81HRESULT VirtualBoxBase::BaseFinalConstruct()
82{
83 Assert(iFactoryStat == ~0U);
84 if (g_pClassFactoryStatsLock)
85 {
86 AutoWriteLock alock(g_pClassFactoryStatsLock COMMA_LOCKVAL_SRC_POS);
87 g_aClassFactoryStats[0].current++;
88 g_aClassFactoryStats[0].overall++;
89 const char *pszName = getComponentName();
90 uint32_t i = 1;
91 while (i < CLASSFACTORYSTATS_MAX && g_aClassFactoryStats[i].psz)
92 {
93 if (g_aClassFactoryStats[i].psz == pszName)
94 break;
95 i++;
96 }
97 if (i < CLASSFACTORYSTATS_MAX)
98 {
99 if (!g_aClassFactoryStats[i].psz)
100 {
101 g_aClassFactoryStats[i].psz = pszName;
102 g_aClassFactoryStats[i].current = 0;
103 g_aClassFactoryStats[i].overall = 0;
104 }
105 iFactoryStat = i;
106 g_aClassFactoryStats[i].current++;
107 g_aClassFactoryStats[i].overall++;
108 }
109 else
110 AssertMsg(i < CLASSFACTORYSTATS_MAX, ("%u exhausts size of factory housekeeping array\n", i));
111 }
112 else
113 Assert(g_pClassFactoryStatsLock);
114
115#ifdef RT_OS_WINDOWS
116 return CoCreateFreeThreadedMarshaler(this, //GetControllingUnknown(),
117 m_pUnkMarshaler.asOutParam());
118#else
119 return S_OK;
120#endif
121}
122
123void VirtualBoxBase::BaseFinalRelease()
124{
125 if (g_pClassFactoryStatsLock)
126 {
127 AutoWriteLock alock(g_pClassFactoryStatsLock COMMA_LOCKVAL_SRC_POS);
128 g_aClassFactoryStats[0].current--;
129 const char *pszName = getComponentName();
130 if (iFactoryStat < CLASSFACTORYSTATS_MAX)
131 {
132 if (g_aClassFactoryStats[iFactoryStat].psz == pszName)
133 {
134 g_aClassFactoryStats[iFactoryStat].current--;
135 iFactoryStat = ~0U;
136 }
137 else
138 AssertMsgFailed(("could not find factory housekeeping array entry for %s (index %u contains %s)\n", pszName, iFactoryStat, g_aClassFactoryStats[iFactoryStat].psz));
139 }
140 else
141 AssertMsgFailed(("factory housekeeping array corruption, index %u is too large\n", iFactoryStat));
142 }
143 else
144 Assert(g_pClassFactoryStatsLock);
145
146#ifdef RT_OS_WINDOWS
147 m_pUnkMarshaler.setNull();
148#endif
149}
150
151void APIDumpComponentFactoryStats()
152{
153 if (g_pClassFactoryStatsLock)
154 {
155 AutoReadLock alock(g_pClassFactoryStatsLock COMMA_LOCKVAL_SRC_POS);
156 for (uint32_t i = 0; i < CLASSFACTORYSTATS_MAX && g_aClassFactoryStats[i].psz; i++)
157 LogRel(("CFS: component %-30s current %-10u total %-10u\n",
158 g_aClassFactoryStats[i].psz, g_aClassFactoryStats[i].current,
159 g_aClassFactoryStats[i].overall));
160 }
161 else
162 Assert(g_pClassFactoryStatsLock);
163}
164
165/**
166 * This virtual method returns an RWLockHandle that can be used to
167 * protect instance data. This RWLockHandle is generally referred to
168 * as the "object lock"; its locking class (for lock order validation)
169 * must be returned by another virtual method, getLockingClass(), which
170 * by default returns LOCKCLASS_OTHEROBJECT but is overridden by several
171 * subclasses such as VirtualBox, Host, Machine and others.
172 *
173 * On the first call this method lazily creates the RWLockHandle.
174 *
175 * @return
176 */
177/* virtual */
178RWLockHandle *VirtualBoxBase::lockHandle() const
179{
180 /* lazy initialization */
181 if (RT_LIKELY(mObjectLock))
182 return mObjectLock;
183
184 AssertCompile(sizeof(RWLockHandle *) == sizeof(void *));
185
186 // getLockingClass() is overridden by many subclasses to return
187 // one of the locking classes listed at the top of AutoLock.h
188 RWLockHandle *objLock = new RWLockHandle(getLockingClass());
189 if (!ASMAtomicCmpXchgPtr(&mObjectLock, objLock, NULL))
190 {
191 delete objLock;
192 objLock = ASMAtomicReadPtrT(&mObjectLock, RWLockHandle *);
193 }
194 return objLock;
195}
196
197/**
198 * Handles unexpected exceptions by turning them into COM errors in release
199 * builds or by hitting a breakpoint in the release builds.
200 *
201 * Usage pattern:
202 * @code
203 try
204 {
205 // ...
206 }
207 catch (LaLalA)
208 {
209 // ...
210 }
211 catch (...)
212 {
213 rc = VirtualBox::handleUnexpectedExceptions(this, RT_SRC_POS);
214 }
215 * @endcode
216 *
217 * @param aThis object where the exception happened
218 * @param SRC_POS "RT_SRC_POS" macro instantiation.
219 * */
220/* static */
221HRESULT VirtualBoxBase::handleUnexpectedExceptions(VirtualBoxBase *const aThis, RT_SRC_POS_DECL)
222{
223 try
224 {
225 /* re-throw the current exception */
226 throw;
227 }
228 catch (const RTCError &err) // includes all XML exceptions
229 {
230 return setErrorInternal(E_FAIL, aThis->getClassIID(), aThis->getComponentName(),
231 Utf8StrFmt(tr("%s.\n%s[%d] (%s)"),
232 err.what(),
233 pszFile, iLine, pszFunction).c_str(),
234 false /* aWarning */,
235 true /* aLogIt */);
236 }
237 catch (const std::exception &err)
238 {
239 return setErrorInternal(E_FAIL, aThis->getClassIID(), aThis->getComponentName(),
240 Utf8StrFmt(tr("Unexpected exception: %s [%s]\n%s[%d] (%s)"),
241 err.what(), typeid(err).name(),
242 pszFile, iLine, pszFunction).c_str(),
243 false /* aWarning */,
244 true /* aLogIt */);
245 }
246 catch (...)
247 {
248 return setErrorInternal(E_FAIL, aThis->getClassIID(), aThis->getComponentName(),
249 Utf8StrFmt(tr("Unknown exception\n%s[%d] (%s)"),
250 pszFile, iLine, pszFunction).c_str(),
251 false /* aWarning */,
252 true /* aLogIt */);
253 }
254
255#ifndef _MSC_VER /* (unreachable) */
256 /* should not get here */
257 AssertFailed();
258 return E_FAIL;
259#endif
260}
261
262/**
263 * Sets error info for the current thread. This is an internal function that
264 * gets eventually called by all public variants. If @a aWarning is
265 * @c true, then the highest (31) bit in the @a aResultCode value which
266 * indicates the error severity is reset to zero to make sure the receiver will
267 * recognize that the created error info object represents a warning rather
268 * than an error.
269 *
270 * @param aResultCode
271 * @param aIID
272 * @param pcszComponent
273 * @param aText
274 * @param aWarning
275 * @param aLogIt
276 * @param aResultDetail
277 */
278/* static */
279HRESULT VirtualBoxBase::setErrorInternal(HRESULT aResultCode,
280 const GUID &aIID,
281 const char *pcszComponent,
282 Utf8Str aText,
283 bool aWarning,
284 bool aLogIt,
285 LONG aResultDetail /* = 0*/)
286{
287 /* whether multi-error mode is turned on */
288 bool preserve = MultiResult::isMultiEnabled();
289
290 if (aLogIt)
291 LogRel(("%s [COM]: aRC=%Rhrc (%#08x) aIID={%RTuuid} aComponent={%s} aText={%s}, preserve=%RTbool aResultDetail=%d\n",
292 aWarning ? "WARNING" : "ERROR",
293 aResultCode,
294 aResultCode,
295 &aIID,
296 pcszComponent,
297 aText.c_str(),
298 preserve,
299 aResultDetail));
300
301 /* these are mandatory, others -- not */
302 AssertReturn((!aWarning && FAILED(aResultCode)) ||
303 (aWarning && aResultCode != S_OK),
304 E_FAIL);
305
306 /* reset the error severity bit if it's a warning */
307 if (aWarning)
308 aResultCode &= ~0x80000000;
309
310 HRESULT rc = S_OK;
311
312 if (aText.isEmpty())
313 {
314 /* Some default info */
315 switch (aResultCode)
316 {
317 case E_INVALIDARG: aText = "A parameter has an invalid value"; break;
318 case E_POINTER: aText = "A parameter is an invalid pointer"; break;
319 case E_UNEXPECTED: aText = "The result of the operation is unexpected"; break;
320 case E_ACCESSDENIED: aText = "The access to an object is not allowed"; break;
321 case E_OUTOFMEMORY: aText = "The allocation of new memory failed"; break;
322 case E_NOTIMPL: aText = "The requested operation is not implemented"; break;
323 case E_NOINTERFACE: aText = "The requested interface is not implemented"; break;
324 case E_FAIL: aText = "A general error occurred"; break;
325 case E_ABORT: aText = "The operation was canceled"; break;
326 case VBOX_E_OBJECT_NOT_FOUND: aText = "Object corresponding to the supplied arguments does not exist"; break;
327 case VBOX_E_INVALID_VM_STATE: aText = "Current virtual machine state prevents the operation"; break;
328 case VBOX_E_VM_ERROR: aText = "Virtual machine error occurred attempting the operation"; break;
329 case VBOX_E_FILE_ERROR: aText = "File not accessible or erroneous file contents"; break;
330 case VBOX_E_IPRT_ERROR: aText = "Runtime subsystem error"; break;
331 case VBOX_E_PDM_ERROR: aText = "Pluggable Device Manager error"; break;
332 case VBOX_E_INVALID_OBJECT_STATE: aText = "Current object state prohibits operation"; break;
333 case VBOX_E_HOST_ERROR: aText = "Host operating system related error"; break;
334 case VBOX_E_NOT_SUPPORTED: aText = "Requested operation is not supported"; break;
335 case VBOX_E_XML_ERROR: aText = "Invalid XML found"; break;
336 case VBOX_E_INVALID_SESSION_STATE: aText = "Current session state prohibits operation"; break;
337 case VBOX_E_OBJECT_IN_USE: aText = "Object being in use prohibits operation"; break;
338 case VBOX_E_PASSWORD_INCORRECT: aText = "Incorrect password provided"; break;
339 default: aText = "Unknown error"; break;
340 }
341 }
342
343 do
344 {
345 ComObjPtr<VirtualBoxErrorInfo> info;
346 rc = info.createObject();
347 if (FAILED(rc)) break;
348
349#if !defined(VBOX_WITH_XPCOM)
350
351 ComPtr<IVirtualBoxErrorInfo> curInfo;
352 if (preserve)
353 {
354 /* get the current error info if any */
355 ComPtr<IErrorInfo> err;
356 rc = ::GetErrorInfo(0, err.asOutParam());
357 if (FAILED(rc)) break;
358 rc = err.queryInterfaceTo(curInfo.asOutParam());
359 if (FAILED(rc))
360 {
361 /* create a IVirtualBoxErrorInfo wrapper for the native
362 * IErrorInfo object */
363 ComObjPtr<VirtualBoxErrorInfo> wrapper;
364 rc = wrapper.createObject();
365 if (SUCCEEDED(rc))
366 {
367 rc = wrapper->init(err);
368 if (SUCCEEDED(rc))
369 curInfo = wrapper;
370 }
371 }
372 }
373 /* On failure, curInfo will stay null */
374 Assert(SUCCEEDED(rc) || curInfo.isNull());
375
376 /* set the current error info and preserve the previous one if any */
377 rc = info->initEx(aResultCode, aResultDetail, aIID, pcszComponent, aText, curInfo);
378 if (FAILED(rc)) break;
379
380 ComPtr<IErrorInfo> err;
381 rc = info.queryInterfaceTo(err.asOutParam());
382 if (SUCCEEDED(rc))
383 rc = ::SetErrorInfo(0, err);
384
385#else // !defined(VBOX_WITH_XPCOM)
386
387 nsCOMPtr <nsIExceptionService> es;
388 es = do_GetService(NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
389 if (NS_SUCCEEDED(rc))
390 {
391 nsCOMPtr <nsIExceptionManager> em;
392 rc = es->GetCurrentExceptionManager(getter_AddRefs(em));
393 if (FAILED(rc)) break;
394
395 ComPtr<IVirtualBoxErrorInfo> curInfo;
396 if (preserve)
397 {
398 /* get the current error info if any */
399 ComPtr<nsIException> ex;
400 rc = em->GetCurrentException(ex.asOutParam());
401 if (FAILED(rc)) break;
402 rc = ex.queryInterfaceTo(curInfo.asOutParam());
403 if (FAILED(rc))
404 {
405 /* create a IVirtualBoxErrorInfo wrapper for the native
406 * nsIException object */
407 ComObjPtr<VirtualBoxErrorInfo> wrapper;
408 rc = wrapper.createObject();
409 if (SUCCEEDED(rc))
410 {
411 rc = wrapper->init(ex);
412 if (SUCCEEDED(rc))
413 curInfo = wrapper;
414 }
415 }
416 }
417 /* On failure, curInfo will stay null */
418 Assert(SUCCEEDED(rc) || curInfo.isNull());
419
420 /* set the current error info and preserve the previous one if any */
421 rc = info->initEx(aResultCode, aResultDetail, aIID, pcszComponent, Bstr(aText), curInfo);
422 if (FAILED(rc)) break;
423
424 ComPtr<nsIException> ex;
425 rc = info.queryInterfaceTo(ex.asOutParam());
426 if (SUCCEEDED(rc))
427 rc = em->SetCurrentException(ex);
428 }
429 else if (rc == NS_ERROR_UNEXPECTED)
430 {
431 /*
432 * It is possible that setError() is being called by the object
433 * after the XPCOM shutdown sequence has been initiated
434 * (for example, when XPCOM releases all instances it internally
435 * references, which can cause object's FinalConstruct() and then
436 * uninit()). In this case, do_GetService() above will return
437 * NS_ERROR_UNEXPECTED and it doesn't actually make sense to
438 * set the exception (nobody will be able to read it).
439 */
440 Log1WarningFunc(("Will not set an exception because nsIExceptionService is not available (NS_ERROR_UNEXPECTED). XPCOM is being shutdown?\n"));
441 rc = NS_OK;
442 }
443
444#endif // !defined(VBOX_WITH_XPCOM)
445 }
446 while (0);
447
448 AssertComRC(rc);
449
450 return SUCCEEDED(rc) ? aResultCode : rc;
451}
452
453/**
454 * Shortcut instance method to calling the static setErrorInternal with the
455 * class interface ID and component name inserted correctly. This uses the
456 * virtual getClassIID() and getComponentName() methods which are automatically
457 * defined by the VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT macro.
458 * @param aResultCode
459 * @return
460 */
461HRESULT VirtualBoxBase::setError(HRESULT aResultCode)
462{
463 return setErrorInternal(aResultCode,
464 this->getClassIID(),
465 this->getComponentName(),
466 "",
467 false /* aWarning */,
468 true /* aLogIt */);
469}
470
471/**
472 * Shortcut instance method to calling the static setErrorInternal with the
473 * class interface ID and component name inserted correctly. This uses the
474 * virtual getClassIID() and getComponentName() methods which are automatically
475 * defined by the VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT macro.
476 * @param aResultCode
477 * @param pcsz
478 * @return
479 */
480HRESULT VirtualBoxBase::setError(HRESULT aResultCode, const char *pcsz, ...)
481{
482 va_list args;
483 va_start(args, pcsz);
484 HRESULT rc = setErrorInternal(aResultCode,
485 this->getClassIID(),
486 this->getComponentName(),
487 Utf8Str(pcsz, args),
488 false /* aWarning */,
489 true /* aLogIt */);
490 va_end(args);
491 return rc;
492}
493
494/**
495 * Shortcut instance method to calling the static setErrorInternal with the
496 * class interface ID and component name inserted correctly. This uses the
497 * virtual getClassIID() and getComponentName() methods which are automatically
498 * defined by the VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT macro.
499 * @param ei
500 * @return
501 */
502HRESULT VirtualBoxBase::setError(const com::ErrorInfo &ei)
503{
504 /* whether multi-error mode is turned on */
505 bool preserve = MultiResult::isMultiEnabled();
506
507 HRESULT rc = S_OK;
508
509 do
510 {
511 ComObjPtr<VirtualBoxErrorInfo> info;
512 rc = info.createObject();
513 if (FAILED(rc)) break;
514
515#if !defined(VBOX_WITH_XPCOM)
516
517 ComPtr<IVirtualBoxErrorInfo> curInfo;
518 if (preserve)
519 {
520 /* get the current error info if any */
521 ComPtr<IErrorInfo> err;
522 rc = ::GetErrorInfo(0, err.asOutParam());
523 if (FAILED(rc)) break;
524 rc = err.queryInterfaceTo(curInfo.asOutParam());
525 if (FAILED(rc))
526 {
527 /* create a IVirtualBoxErrorInfo wrapper for the native
528 * IErrorInfo object */
529 ComObjPtr<VirtualBoxErrorInfo> wrapper;
530 rc = wrapper.createObject();
531 if (SUCCEEDED(rc))
532 {
533 rc = wrapper->init(err);
534 if (SUCCEEDED(rc))
535 curInfo = wrapper;
536 }
537 }
538 }
539 /* On failure, curInfo will stay null */
540 Assert(SUCCEEDED(rc) || curInfo.isNull());
541
542 /* set the current error info and preserve the previous one if any */
543 rc = info->init(ei, curInfo);
544 if (FAILED(rc)) break;
545
546 ComPtr<IErrorInfo> err;
547 rc = info.queryInterfaceTo(err.asOutParam());
548 if (SUCCEEDED(rc))
549 rc = ::SetErrorInfo(0, err);
550
551#else // !defined(VBOX_WITH_XPCOM)
552
553 nsCOMPtr <nsIExceptionService> es;
554 es = do_GetService(NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
555 if (NS_SUCCEEDED(rc))
556 {
557 nsCOMPtr <nsIExceptionManager> em;
558 rc = es->GetCurrentExceptionManager(getter_AddRefs(em));
559 if (FAILED(rc)) break;
560
561 ComPtr<IVirtualBoxErrorInfo> curInfo;
562 if (preserve)
563 {
564 /* get the current error info if any */
565 ComPtr<nsIException> ex;
566 rc = em->GetCurrentException(ex.asOutParam());
567 if (FAILED(rc)) break;
568 rc = ex.queryInterfaceTo(curInfo.asOutParam());
569 if (FAILED(rc))
570 {
571 /* create a IVirtualBoxErrorInfo wrapper for the native
572 * nsIException object */
573 ComObjPtr<VirtualBoxErrorInfo> wrapper;
574 rc = wrapper.createObject();
575 if (SUCCEEDED(rc))
576 {
577 rc = wrapper->init(ex);
578 if (SUCCEEDED(rc))
579 curInfo = wrapper;
580 }
581 }
582 }
583 /* On failure, curInfo will stay null */
584 Assert(SUCCEEDED(rc) || curInfo.isNull());
585
586 /* set the current error info and preserve the previous one if any */
587 rc = info->init(ei, curInfo);
588 if (FAILED(rc)) break;
589
590 ComPtr<nsIException> ex;
591 rc = info.queryInterfaceTo(ex.asOutParam());
592 if (SUCCEEDED(rc))
593 rc = em->SetCurrentException(ex);
594 }
595 else if (rc == NS_ERROR_UNEXPECTED)
596 {
597 /*
598 * It is possible that setError() is being called by the object
599 * after the XPCOM shutdown sequence has been initiated
600 * (for example, when XPCOM releases all instances it internally
601 * references, which can cause object's FinalConstruct() and then
602 * uninit()). In this case, do_GetService() above will return
603 * NS_ERROR_UNEXPECTED and it doesn't actually make sense to
604 * set the exception (nobody will be able to read it).
605 */
606 Log1WarningFunc(("Will not set an exception because nsIExceptionService is not available (NS_ERROR_UNEXPECTED). XPCOM is being shutdown?\n"));
607 rc = NS_OK;
608 }
609
610#endif // !defined(VBOX_WITH_XPCOM)
611 }
612 while (0);
613
614 AssertComRC(rc);
615
616 return SUCCEEDED(rc) ? ei.getResultCode() : rc;
617}
618
619/**
620 * Converts the VBox status code a COM one and sets the error info.
621 *
622 * The VBox status code is made available to the API user via
623 * IVirtualBoxErrorInfo::resultDetail attribute.
624 *
625 * @param vrc The VBox status code.
626 * @return COM status code appropriate for @a vrc.
627 *
628 * @sa VirtualBoxBase::setError(HRESULT)
629 */
630HRESULT VirtualBoxBase::setErrorVrc(int vrc)
631{
632 return setErrorInternal(Global::vboxStatusCodeToCOM(vrc),
633 this->getClassIID(),
634 this->getComponentName(),
635 Utf8StrFmt("%Rrc", vrc),
636 false /* aWarning */,
637 true /* aLogIt */,
638 vrc /* aResultDetail */);
639}
640
641/**
642 * Converts the VBox status code a COM one and sets the error info.
643 *
644 * @param vrc The VBox status code.
645 * @param pcszMsgFmt Error message format string.
646 * @param ... Argument specified in the @a pcszMsgFmt
647 * @return COM status code appropriate for @a vrc.
648 *
649 * @sa VirtualBoxBase::setError(HRESULT, const char *, ...)
650 */
651HRESULT VirtualBoxBase::setErrorVrc(int vrc, const char *pcszMsgFmt, ...)
652{
653 va_list va;
654 va_start(va, pcszMsgFmt);
655 HRESULT hrc = setErrorInternal(Global::vboxStatusCodeToCOM(vrc),
656 this->getClassIID(),
657 this->getComponentName(),
658 Utf8Str(pcszMsgFmt, va),
659 false /* aWarning */,
660 true /* aLogIt */,
661 vrc /* aResultDetail */);
662 va_end(va);
663 return hrc;
664}
665
666/**
667 * Sets error info with both a COM status and an VBox status code.
668 *
669 * The VBox status code is made available to the API user via
670 * IVirtualBoxErrorInfo::resultDetail attribute.
671 *
672 * @param hrc The COM status code to return.
673 * @param vrc The VBox status code.
674 * @return Most likely @a hrc, see setErrorInternal.
675 *
676 * @sa VirtualBoxBase::setError(HRESULT)
677 */
678HRESULT VirtualBoxBase::setErrorBoth(HRESULT hrc, int vrc)
679{
680 return setErrorInternal(hrc,
681 this->getClassIID(),
682 this->getComponentName(),
683 Utf8StrFmt("%Rrc", vrc),
684 false /* aWarning */,
685 true /* aLogIt */,
686 vrc /* aResultDetail */);
687}
688
689/**
690 * Sets error info with a message and both a COM status and an VBox status code.
691 *
692 * The VBox status code is made available to the API user via
693 * IVirtualBoxErrorInfo::resultDetail attribute.
694 *
695 * @param hrc The COM status code to return.
696 * @param vrc The VBox status code.
697 * @param pcszMsgFmt Error message format string.
698 * @param ... Argument specified in the @a pcszMsgFmt
699 * @return Most likely @a hrc, see setErrorInternal.
700 *
701 * @sa VirtualBoxBase::setError(HRESULT, const char *, ...)
702 */
703HRESULT VirtualBoxBase::setErrorBoth(HRESULT hrc, int vrc, const char *pcszMsgFmt, ...)
704{
705 va_list va;
706 va_start(va, pcszMsgFmt);
707 hrc = setErrorInternal(hrc,
708 this->getClassIID(),
709 this->getComponentName(),
710 Utf8Str(pcszMsgFmt, va),
711 false /* aWarning */,
712 true /* aLogIt */,
713 vrc /* aResultDetail */);
714 va_end(va);
715 return hrc;
716}
717
718/**
719 * Like setError(), but sets the "warning" bit in the call to setErrorInternal().
720 * @param aResultCode
721 * @param pcsz
722 * @return
723 */
724HRESULT VirtualBoxBase::setWarning(HRESULT aResultCode, const char *pcsz, ...)
725{
726 va_list args;
727 va_start(args, pcsz);
728 HRESULT rc = setErrorInternal(aResultCode,
729 this->getClassIID(),
730 this->getComponentName(),
731 Utf8Str(pcsz, args),
732 true /* aWarning */,
733 true /* aLogIt */);
734 va_end(args);
735 return rc;
736}
737
738/**
739 * Like setError(), but disables the "log" flag in the call to setErrorInternal().
740 * @param aResultCode
741 * @param pcsz
742 * @return
743 */
744HRESULT VirtualBoxBase::setErrorNoLog(HRESULT aResultCode, const char *pcsz, ...)
745{
746 va_list args;
747 va_start(args, pcsz);
748 HRESULT rc = setErrorInternal(aResultCode,
749 this->getClassIID(),
750 this->getComponentName(),
751 Utf8Str(pcsz, args),
752 false /* aWarning */,
753 false /* aLogIt */);
754 va_end(args);
755 return rc;
756}
757
758/**
759 * Clear the current error information.
760 */
761/*static*/
762void VirtualBoxBase::clearError(void)
763{
764#if !defined(VBOX_WITH_XPCOM)
765 ::SetErrorInfo(0, NULL);
766#else
767 HRESULT rc = S_OK;
768 nsCOMPtr <nsIExceptionService> es;
769 es = do_GetService(NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
770 if (NS_SUCCEEDED(rc))
771 {
772 nsCOMPtr <nsIExceptionManager> em;
773 rc = es->GetCurrentExceptionManager(getter_AddRefs(em));
774 if (SUCCEEDED(rc))
775 em->SetCurrentException(NULL);
776 }
777#endif
778}
779
780
781////////////////////////////////////////////////////////////////////////////////
782//
783// MultiResult methods
784//
785////////////////////////////////////////////////////////////////////////////////
786
787RTTLS MultiResult::sCounter = NIL_RTTLS;
788
789/*static*/
790void MultiResult::incCounter()
791{
792 if (sCounter == NIL_RTTLS)
793 {
794 sCounter = RTTlsAlloc();
795 AssertReturnVoid(sCounter != NIL_RTTLS);
796 }
797
798 uintptr_t counter = (uintptr_t)RTTlsGet(sCounter);
799 ++counter;
800 RTTlsSet(sCounter, (void*)counter);
801}
802
803/*static*/
804void MultiResult::decCounter()
805{
806 uintptr_t counter = (uintptr_t)RTTlsGet(sCounter);
807 AssertReturnVoid(counter != 0);
808 --counter;
809 RTTlsSet(sCounter, (void*)counter);
810}
811
812/*static*/
813bool MultiResult::isMultiEnabled()
814{
815 if (sCounter == NIL_RTTLS)
816 return false;
817
818 return ((uintptr_t)RTTlsGet(MultiResult::sCounter)) > 0;
819}
820
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