VirtualBox

source: vbox/trunk/src/VBox/Main/ProgressImpl.cpp@ 226

Last change on this file since 226 was 1, checked in by vboxsync, 55 years ago

import

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 37.1 KB
Line 
1/** @file
2 *
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22#if !defined (__WIN__)
23#include <nsIServiceManager.h>
24#include <nsIExceptionService.h>
25#include <nsCOMPtr.h>
26#endif // !defined (__WIN__)
27
28#include "ProgressImpl.h"
29#include "VirtualBoxImpl.h"
30#include "VirtualBoxErrorInfoImpl.h"
31
32#include "Logging.h"
33
34#include <iprt/time.h>
35#include <iprt/semaphore.h>
36
37#include <VBox/err.h>
38
39////////////////////////////////////////////////////////////////////////////////
40// ProgressBase class
41////////////////////////////////////////////////////////////////////////////////
42
43// constructor / destructor
44////////////////////////////////////////////////////////////////////////////////
45
46/** Subclasses must call this method from their FinalConstruct() implementations */
47HRESULT ProgressBase::FinalConstruct()
48{
49 mCancelable = FALSE;
50 mCompleted = FALSE;
51 mCanceled = FALSE;
52 mResultCode = S_OK;
53 mOperationCount = 0;
54 mOperation = 0;
55 mOperationPercent = 0;
56
57 return S_OK;
58}
59
60// public initializer/uninitializer for internal purposes only
61////////////////////////////////////////////////////////////////////////////////
62
63/**
64 * Initializes the progress base object.
65 *
66 * Subclasses should call this or any other #init() method from their
67 * init() implementations.
68 *
69 * @param aParent
70 * Parent object (only for server-side Progress objects)
71 * @param aInitiator
72 * Initiator of the task (for server-side objects
73 * can be NULL which means initiator = parent, otherwise
74 * must not be NULL)
75 * @param aDescription
76 * Task description
77 * @param aID
78 * Address of result GUID structure (optional)
79 *
80 * @note
81 * This method doesn't do |isReady()| check and doesn't call
82 * |setReady (true)| on success!
83 * @note
84 * This method must be called from under the object's lock!
85 */
86HRESULT ProgressBase::protectedInit (
87#if !defined (VBOX_COM_INPROC)
88 VirtualBox *aParent,
89#endif
90 IUnknown *aInitiator,
91 const BSTR aDescription, GUIDPARAMOUT aId /* = NULL */)
92{
93#if !defined (VBOX_COM_INPROC)
94 ComAssertRet (aParent, E_POINTER);
95#else
96 ComAssertRet (aInitiator, E_POINTER);
97#endif
98
99 ComAssertRet (aDescription, E_INVALIDARG);
100
101#if !defined (VBOX_COM_INPROC)
102 mParent = aParent;
103#endif
104
105#if !defined (VBOX_COM_INPROC)
106 // assign (and therefore addref) initiator only if it is not VirtualBox
107 // (to avoid cycling); otherwise mInitiator will remain null which means
108 // that it is the same as the parent
109 if (aInitiator && !mParent.equalsTo (aInitiator))
110 mInitiator = aInitiator;
111#else
112 mInitiator = aInitiator;
113#endif
114
115 mDescription = aDescription;
116
117 mId.create();
118 if (aId)
119 mId.cloneTo (aId);
120
121#if !defined (VBOX_COM_INPROC)
122 // add to the global colleciton of progess operations
123 mParent->addProgress (this);
124 // cause #uninit() to be called automatically upon VirtualBox uninit
125 mParent->addDependentChild (this);
126#endif
127
128 return S_OK;
129}
130
131/**
132 * Initializes the progress base object.
133 * This is a special initializator for progress objects that are combined
134 * within a CombinedProgress instance, so it doesn't require the parent,
135 * initiator, description and doesn't create an ID. Note that calling respective
136 * getter methods on an object initialized with this constructor will hit an
137 * assertion.
138 *
139 * Subclasses should call this or any other #init() method from their
140 * init() implementations.
141 *
142 * @note
143 * This method doesn't do |isReady()| check and doesn't call
144 * |setReady (true)| on success!
145 * @note
146 * This method must be called from under the object's lock!
147 */
148HRESULT ProgressBase::protectedInit()
149{
150 return S_OK;
151}
152
153/**
154 * Uninitializes the instance.
155 * Subclasses should call this from their uninit() implementations.
156 * The readiness flag must be true on input and will be set to false
157 * on output.
158 *
159 * @param alock this object's autolock (with lock level = 1!)
160 *
161 * @note
162 * Using mParent member after this method returns is forbidden.
163 */
164void ProgressBase::protectedUninit (AutoLock &alock)
165{
166 LogFlowMember (("ProgressBase::protectedUninit()\n"));
167
168 Assert (alock.belongsTo (this) && alock.level() == 1);
169 Assert (isReady());
170
171 /*
172 * release initiator
173 * (effective only if mInitiator has been assigned in init())
174 */
175 mInitiator.setNull();
176
177 setReady (false);
178
179#if !defined (VBOX_COM_INPROC)
180 if (mParent)
181 {
182 alock.leave();
183 mParent->removeDependentChild (this);
184 alock.enter();
185 }
186
187 mParent.setNull();
188#endif
189}
190
191// IProgress properties
192/////////////////////////////////////////////////////////////////////////////
193
194STDMETHODIMP ProgressBase::COMGETTER(Id) (GUIDPARAMOUT aId)
195{
196 if (!aId)
197 return E_POINTER;
198
199 AutoLock lock (this);
200 CHECK_READY();
201
202 ComAssertRet (!mId.isEmpty(), E_FAIL);
203
204 mId.cloneTo (aId);
205 return S_OK;
206}
207
208STDMETHODIMP ProgressBase::COMGETTER(Description) (BSTR *aDescription)
209{
210 if (!aDescription)
211 return E_POINTER;
212
213 AutoLock lock (this);
214 CHECK_READY();
215
216 ComAssertRet (!mDescription.isNull(), E_FAIL);
217
218 mDescription.cloneTo (aDescription);
219 return S_OK;
220}
221
222STDMETHODIMP ProgressBase::COMGETTER(Initiator) (IUnknown **aInitiator)
223{
224 if (!aInitiator)
225 return E_POINTER;
226
227 AutoLock lock(this);
228 CHECK_READY();
229
230#if !defined (VBOX_COM_INPROC)
231 ComAssertRet (!mInitiator.isNull() || !mParent.isNull(), E_FAIL);
232
233 if (mInitiator)
234 mInitiator.queryInterfaceTo (aInitiator);
235 else
236 mParent.queryInterfaceTo (aInitiator);
237#else
238 ComAssertRet (!mInitiator.isNull(), E_FAIL);
239
240 mInitiator.queryInterfaceTo (aInitiator);
241#endif
242
243 return S_OK;
244}
245
246STDMETHODIMP ProgressBase::COMGETTER(Cancelable) (BOOL *aCancelable)
247{
248 if (!aCancelable)
249 return E_POINTER;
250
251 AutoLock lock(this);
252 CHECK_READY();
253
254 *aCancelable = mCancelable;
255 return S_OK;
256}
257
258STDMETHODIMP ProgressBase::COMGETTER(Percent) (LONG *aPercent)
259{
260 if (!aPercent)
261 return E_POINTER;
262
263 AutoLock lock(this);
264 CHECK_READY();
265
266 if (mCompleted && SUCCEEDED (mResultCode))
267 *aPercent = 100;
268 else
269 {
270 // global percent = (100 / mOperationCount) * mOperation +
271 // ((100 / mOperationCount) / 100) * mOperationPercent
272 *aPercent = (100 * mOperation + mOperationPercent) / mOperationCount;
273 }
274
275 return S_OK;
276}
277
278STDMETHODIMP ProgressBase::COMGETTER(Completed) (BOOL *aCompleted)
279{
280 if (!aCompleted)
281 return E_POINTER;
282
283 AutoLock lock(this);
284 CHECK_READY();
285
286 *aCompleted = mCompleted;
287 return S_OK;
288}
289
290STDMETHODIMP ProgressBase::COMGETTER(Canceled) (BOOL *aCanceled)
291{
292 if (!aCanceled)
293 return E_POINTER;
294
295 AutoLock lock(this);
296 CHECK_READY();
297
298 *aCanceled = mCanceled;
299 return S_OK;
300}
301
302STDMETHODIMP ProgressBase::COMGETTER(ResultCode) (HRESULT *aResultCode)
303{
304 if (!aResultCode)
305 return E_POINTER;
306
307 AutoLock lock(this);
308 CHECK_READY();
309
310 if (!mCompleted)
311 return setError (E_FAIL,
312 tr ("Result code is not available, operation is still in progress"));
313
314 *aResultCode = mResultCode;
315 return S_OK;
316}
317
318STDMETHODIMP ProgressBase::COMGETTER(ErrorInfo) (IVirtualBoxErrorInfo **aErrorInfo)
319{
320 if (!aErrorInfo)
321 return E_POINTER;
322
323 AutoLock lock(this);
324 CHECK_READY();
325
326 if (!mCompleted)
327 return setError (E_FAIL,
328 tr ("Error info is not available, operation is still in progress"));
329
330 mErrorInfo.queryInterfaceTo (aErrorInfo);
331 return S_OK;
332}
333
334STDMETHODIMP ProgressBase::COMGETTER(OperationCount) (ULONG *aOperationCount)
335{
336 if (!aOperationCount)
337 return E_POINTER;
338
339 AutoLock lock(this);
340 CHECK_READY();
341
342 *aOperationCount = mOperationCount;
343 return S_OK;
344}
345
346STDMETHODIMP ProgressBase::COMGETTER(Operation) (ULONG *aOperation)
347{
348 if (!aOperation)
349 return E_POINTER;
350
351 AutoLock lock(this);
352 CHECK_READY();
353
354 *aOperation = mOperation;
355 return S_OK;
356}
357
358STDMETHODIMP ProgressBase::COMGETTER(OperationDescription) (BSTR *aOperationDescription)
359{
360 if (!aOperationDescription)
361 return E_POINTER;
362
363 AutoLock lock(this);
364 CHECK_READY();
365
366 mOperationDescription.cloneTo (aOperationDescription);
367 return S_OK;
368}
369
370STDMETHODIMP ProgressBase::COMGETTER(OperationPercent) (LONG *aOperationPercent)
371{
372 if (!aOperationPercent)
373 return E_POINTER;
374
375 AutoLock lock(this);
376 CHECK_READY();
377
378 if (mCompleted && SUCCEEDED (mResultCode))
379 *aOperationPercent = 100;
380 else
381 *aOperationPercent = mOperationPercent;
382 return S_OK;
383}
384
385////////////////////////////////////////////////////////////////////////////////
386// Progress class
387////////////////////////////////////////////////////////////////////////////////
388
389HRESULT Progress::FinalConstruct()
390{
391 HRESULT rc = ProgressBase::FinalConstruct();
392 if (FAILED (rc))
393 return rc;
394
395 mCompletedSem = NIL_RTSEMEVENT;
396 mWaitersCount = 0;
397
398 return S_OK;
399}
400
401void Progress::FinalRelease()
402{
403 uninit();
404}
405
406// public initializer/uninitializer for internal purposes only
407////////////////////////////////////////////////////////////////////////////////
408
409/**
410 * Initializes the progress object.
411 *
412 * @param aParent see ProgressBase::init()
413 * @param aInitiator see ProgressBase::init()
414 * @param aDescription see ProgressBase::init()
415 * @param aCancelable Flag whether the task maybe canceled
416 * @param aOperationCount Number of operations within this task (at least 1)
417 * @param aOperationDescription Description of the first operation
418 * @param aId see ProgressBase::init()
419 */
420HRESULT Progress::init (
421#if !defined (VBOX_COM_INPROC)
422 VirtualBox *aParent,
423#endif
424 IUnknown *aInitiator,
425 const BSTR aDescription, BOOL aCancelable,
426 ULONG aOperationCount, const BSTR aOperationDescription,
427 GUIDPARAMOUT aId /* = NULL */)
428{
429 LogFlowMember(("Progress::init(): aDescription={%ls}\n", aDescription));
430
431 ComAssertRet (aOperationDescription, E_INVALIDARG);
432 ComAssertRet (aOperationCount >= 1, E_INVALIDARG);
433
434 AutoLock lock(this);
435 ComAssertRet (!isReady(), E_UNEXPECTED);
436
437 HRESULT rc = S_OK;
438
439 do
440 {
441 rc = ProgressBase::protectedInit (
442#if !defined (VBOX_COM_INPROC)
443 aParent,
444#endif
445 aInitiator, aDescription, aId);
446 CheckComRCBreakRC (rc);
447
448 // set ready to let protectedUninit() be called on failure
449 setReady (true);
450
451 mCancelable = aCancelable;
452
453 mOperationCount = aOperationCount;
454 mOperation = 0; // the first operation
455 mOperationDescription = aOperationDescription;
456
457 int vrc = RTSemEventMultiCreate (&mCompletedSem);
458 ComAssertRCBreak (vrc, rc = E_FAIL);
459
460 RTSemEventMultiReset (mCompletedSem);
461 }
462 while (0);
463
464 if (FAILED (rc))
465 uninit();
466
467 return rc;
468}
469
470HRESULT Progress::init (BOOL aCancelable, ULONG aOperationCount,
471 const BSTR aOperationDescription)
472{
473 LogFlowMember(("Progress::init(): <undescriptioned>\n"));
474
475 AutoLock lock(this);
476 ComAssertRet (!isReady(), E_UNEXPECTED);
477
478 HRESULT rc = S_OK;
479
480 do
481 {
482 rc = ProgressBase::protectedInit();
483 CheckComRCBreakRC (rc);
484
485 // set ready to let protectedUninit() be called on failure
486 setReady (true);
487
488 mCancelable = aCancelable;
489
490 mOperationCount = aOperationCount;
491 mOperation = 0; // the first operation
492 mOperationDescription = aOperationDescription;
493
494 int vrc = RTSemEventMultiCreate (&mCompletedSem);
495 ComAssertRCBreak (vrc, rc = E_FAIL);
496
497 RTSemEventMultiReset (mCompletedSem);
498 }
499 while (0);
500
501 if (FAILED (rc))
502 uninit();
503
504 return rc;
505}
506
507/**
508 * Uninitializes the instance and sets the ready flag to FALSE.
509 * Called either from FinalRelease() or by the parent when it gets destroyed.
510 */
511void Progress::uninit()
512{
513 LogFlowMember (("Progress::uninit()\n"));
514
515 AutoLock alock (this);
516
517 LogFlowMember (("Progress::uninit(): isReady=%d\n", isReady()));
518
519 if (!isReady())
520 return;
521
522 // wake up all threads still waiting by occasion
523 if (mWaitersCount > 0)
524 {
525 LogFlow (("WARNING: There are still %d threads waiting for '%ls' completion!\n",
526 mWaitersCount, mDescription.raw()));
527 RTSemEventMultiSignal (mCompletedSem);
528 }
529
530 RTSemEventMultiDestroy (mCompletedSem);
531
532 ProgressBase::protectedUninit (alock);
533}
534
535// IProgress properties
536/////////////////////////////////////////////////////////////////////////////
537
538// IProgress methods
539/////////////////////////////////////////////////////////////////////////////
540
541/**
542 * @note
543 * XPCOM: when this method is called not on the main XPCOM thread, it
544 * it simply blocks the thread until mCompletedSem is signalled. If the
545 * thread has its own event queue (hmm, what for?) that it must run, then
546 * calling this method will definitey freese event processing.
547 */
548STDMETHODIMP Progress::WaitForCompletion (LONG aTimeout)
549{
550 LogFlowMember(("Progress::WaitForCompletion: BEGIN: timeout=%d\n", aTimeout));
551
552 AutoLock lock(this);
553 CHECK_READY();
554
555 // if we're already completed, take a shortcut
556 if (!mCompleted)
557 {
558 RTTIMESPEC time;
559 RTTimeNow (&time);
560
561 int vrc = VINF_SUCCESS;
562 bool forever = aTimeout < 0;
563 int64_t timeLeft = aTimeout;
564 int64_t lastTime = RTTimeSpecGetMilli (&time);
565
566 while (!mCompleted && (forever || timeLeft > 0))
567 {
568 mWaitersCount ++;
569 lock.unlock();
570 int vrc = RTSemEventMultiWait (mCompletedSem,
571 forever ? RT_INDEFINITE_WAIT
572 : (unsigned) timeLeft);
573 lock.lock();
574 mWaitersCount --;
575
576 // the progress might have been uninitialized
577 if (!isReady())
578 break;
579
580 // the last waiter resets the semaphore
581 if (mWaitersCount == 0)
582 RTSemEventMultiReset (mCompletedSem);
583
584 if (VBOX_FAILURE (vrc) && vrc != VERR_TIMEOUT)
585 break;
586
587 if (!forever)
588 {
589 RTTimeNow (&time);
590 timeLeft -= RTTimeSpecGetMilli (&time) - lastTime;
591 lastTime = RTTimeSpecGetMilli (&time);
592 }
593 }
594
595 if (VBOX_FAILURE (vrc) && vrc != VERR_TIMEOUT)
596 return setError (E_FAIL,
597 tr ("Failed to wait for the task completion (%Vrc)"), vrc);
598 }
599
600 LogFlowMember(("Progress::WaitForCompletion: END\n"));
601 return S_OK;
602}
603
604/**
605 * @note
606 * XPCOM: when this method is called not on the main XPCOM thread, it
607 * it simply blocks the thread until mCompletedSem is signalled. If the
608 * thread has its own event queue (hmm, what for?) that it must run, then
609 * calling this method will definitey freese event processing.
610 */
611STDMETHODIMP Progress::WaitForOperationCompletion (ULONG aOperation, LONG aTimeout)
612{
613 LogFlowMember(("Progress::WaitForOperationCompletion: BEGIN: "
614 "operation=%d, timeout=%d\n", aOperation, aTimeout));
615
616 AutoLock lock(this);
617 CHECK_READY();
618
619 if (aOperation >= mOperationCount)
620 return setError (E_FAIL,
621 tr ("Operation number must be in range [0, %d]"), mOperation - 1);
622
623 // if we're already completed or if the given operation is already done,
624 // then take a shortcut
625 if (!mCompleted && aOperation >= mOperation)
626 {
627 RTTIMESPEC time;
628 RTTimeNow (&time);
629
630 int vrc = VINF_SUCCESS;
631 bool forever = aTimeout < 0;
632 int64_t timeLeft = aTimeout;
633 int64_t lastTime = RTTimeSpecGetMilli (&time);
634
635 while (!mCompleted && aOperation >= mOperation &&
636 (forever || timeLeft > 0))
637 {
638 mWaitersCount ++;
639 lock.unlock();
640 int vrc = RTSemEventMultiWait (mCompletedSem,
641 forever ? RT_INDEFINITE_WAIT
642 : (unsigned) timeLeft);
643 lock.lock();
644 mWaitersCount --;
645
646 // the progress might have been uninitialized
647 if (!isReady())
648 break;
649
650 // the last waiter resets the semaphore
651 if (mWaitersCount == 0)
652 RTSemEventMultiReset (mCompletedSem);
653
654 if (VBOX_FAILURE (vrc) && vrc != VERR_TIMEOUT)
655 break;
656
657 if (!forever)
658 {
659 RTTimeNow (&time);
660 timeLeft -= RTTimeSpecGetMilli (&time) - lastTime;
661 lastTime = RTTimeSpecGetMilli (&time);
662 }
663 }
664
665 if (VBOX_FAILURE (vrc) && vrc != VERR_TIMEOUT)
666 return setError (E_FAIL,
667 tr ("Failed to wait for the operation completion (%Vrc)"), vrc);
668 }
669
670 LogFlowMember(("Progress::WaitForOperationCompletion: END\n"));
671 return S_OK;
672}
673
674STDMETHODIMP Progress::Cancel()
675{
676 AutoLock lock(this);
677 CHECK_READY();
678
679 if (!mCancelable)
680 return setError (E_FAIL, tr ("Operation cannot be cancelled"));
681
682/// @todo (dmik): implement operation cancellation!
683// mCompleted = TRUE;
684// mCanceled = TRUE;
685// return S_OK;
686
687 ComAssertMsgFailed (("Not implemented!"));
688 return E_NOTIMPL;
689}
690
691// public methods only for internal purposes
692/////////////////////////////////////////////////////////////////////////////
693
694/**
695 * Updates the percentage value of the current operation.
696 *
697 * @param aPercent New percentage value of the operation in progress
698 * (in range [0, 100]).
699 */
700HRESULT Progress::notifyProgress (LONG aPercent)
701{
702 AutoLock lock (this);
703 AssertReturn (isReady(), E_UNEXPECTED);
704
705 AssertReturn (!mCompleted && !mCanceled, E_FAIL);
706 AssertReturn (aPercent >= 0 && aPercent <= 100, E_INVALIDARG);
707
708 mOperationPercent = aPercent;
709 return S_OK;
710}
711
712/**
713 * Signals that the current operation is successfully completed
714 * and advances to the next operation. The operation percentage is reset
715 * to 0.
716 *
717 * @param aOperationDescription Description of the next operation
718 *
719 * @note
720 * The current operation must not be the last one.
721 */
722HRESULT Progress::advanceOperation (const BSTR aOperationDescription)
723{
724 AssertReturn (aOperationDescription, E_INVALIDARG);
725
726 AutoLock lock (this);
727 AssertReturn (isReady(), E_UNEXPECTED);
728
729 AssertReturn (!mCompleted && !mCanceled, E_FAIL);
730 AssertReturn (mOperation + 1 < mOperationCount, E_FAIL);
731
732 mOperation ++;
733 mOperationDescription = aOperationDescription;
734 mOperationPercent = 0;
735
736 // wake up all waiting threads
737 if (mWaitersCount > 0)
738 RTSemEventMultiSignal (mCompletedSem);
739
740 return S_OK;
741}
742
743/**
744 * Marks the whole task as complete and sets the result code.
745 *
746 * If the result code indicates a failure (|FAILED (@a aResultCode)|)
747 * then this method will import the error info from the current
748 * thread and assign it to the errorInfo attribute (it will return an
749 * error if no info is available in such case).
750 *
751 * If the result code indicates a success (|SUCCEEDED (@a aResultCode)|)
752 * then the current operation is set to the last
753 *
754 * @param aResultCode Operation result code
755 */
756HRESULT Progress::notifyComplete (HRESULT aResultCode)
757{
758 AutoLock lock (this);
759 AssertReturn (isReady(), E_FAIL);
760
761 mCompleted = TRUE;
762 mResultCode = aResultCode;
763
764 HRESULT rc = S_OK;
765
766 if (FAILED (aResultCode))
767 {
768 /* try to import error info from the current thread */
769
770#if defined (__WIN__)
771
772 ComPtr <IErrorInfo> err;
773 rc = ::GetErrorInfo (0, err.asOutParam());
774 if (rc == S_OK && err)
775 {
776 rc = err.queryInterfaceTo (mErrorInfo.asOutParam());
777 if (SUCCEEDED (rc) && !mErrorInfo)
778 rc = E_FAIL;
779 }
780
781#else // !defined (__WIN__)
782
783 nsCOMPtr <nsIExceptionService> es;
784 es = do_GetService (NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
785 if (NS_SUCCEEDED (rc))
786 {
787 nsCOMPtr <nsIExceptionManager> em;
788 rc = es->GetCurrentExceptionManager (getter_AddRefs (em));
789 if (NS_SUCCEEDED (rc))
790 {
791 ComPtr <nsIException> ex;
792 rc = em->GetCurrentException (ex.asOutParam());
793 if (NS_SUCCEEDED (rc) && ex)
794 {
795 rc = ex.queryInterfaceTo (mErrorInfo.asOutParam());
796 if (NS_SUCCEEDED (rc) && !mErrorInfo)
797 rc = E_FAIL;
798 }
799 }
800 }
801#endif // !defined (__WIN__)
802
803 AssertMsg (rc == S_OK, ("Couldn't get error info (rc=%08X) while trying "
804 "to set a failed result (%08X)!\n", rc, aResultCode));
805 }
806 else
807 {
808 mOperation = mOperationCount - 1; /* last operation */
809 mOperationPercent = 100;
810 }
811
812#if !defined VBOX_COM_INPROC
813 /* remove from the global collection of pending progress operations */
814 if (mParent)
815 mParent->removeProgress (mId);
816#endif
817
818 /* wake up all waiting threads */
819 if (mWaitersCount > 0)
820 RTSemEventMultiSignal (mCompletedSem);
821
822 return rc;
823}
824
825/**
826 * Marks the operation as complete and attaches full error info.
827 * See VirtualBoxSupportErrorInfoImpl::setError(HRESULT, const GUID &, const wchar_t *, const char *, ...)
828 * for more info.
829 *
830 * @param aResultCode operation result (error) code, must not be S_OK
831 * @param aIID IID of the intrface that defines the error
832 * @param aComponent name of the component that generates the error
833 * @param aText error message (must not be null), an RTStrPrintf-like
834 * format string in UTF-8 encoding
835 * @param ... list of arguments for the format string
836 */
837HRESULT Progress::notifyComplete (HRESULT aResultCode, const GUID &aIID,
838 const Bstr &aComponent,
839 const char *aText, ...)
840{
841 AutoLock lock (this);
842 AssertReturn (isReady(), E_UNEXPECTED);
843
844 mCompleted = TRUE;
845 mResultCode = aResultCode;
846
847 AssertReturn (FAILED (aResultCode), E_FAIL);
848
849 ComObjPtr <VirtualBoxErrorInfo> errorInfo;
850 HRESULT rc = errorInfo.createObject();
851 AssertComRC (rc);
852 if (SUCCEEDED (rc))
853 {
854 va_list args;
855 va_start (args, aText);
856 Bstr bstrText = Utf8StrFmt (aText, args);
857 va_end (args);
858 errorInfo->init (aResultCode, aIID, aComponent, bstrText);
859 errorInfo.queryInterfaceTo (mErrorInfo.asOutParam());
860 }
861
862#if !defined VBOX_COM_INPROC
863 /* remove from the global collection of pending progress operations */
864 if (mParent)
865 mParent->removeProgress (mId);
866#endif
867
868 /* wake up all waiting threads */
869 if (mWaitersCount > 0)
870 RTSemEventMultiSignal (mCompletedSem);
871
872 return rc;
873}
874
875////////////////////////////////////////////////////////////////////////////////
876// CombinedProgress class
877////////////////////////////////////////////////////////////////////////////////
878
879HRESULT CombinedProgress::FinalConstruct()
880{
881 HRESULT rc = ProgressBase::FinalConstruct();
882 if (FAILED (rc))
883 return rc;
884
885 mProgress = 0;
886 mCompletedOperations = 0;
887
888 return S_OK;
889}
890
891void CombinedProgress::FinalRelease()
892{
893 uninit();
894}
895
896// public initializer/uninitializer for internal purposes only
897////////////////////////////////////////////////////////////////////////////////
898
899/**
900 * Initializes this object based on individual combined progresses.
901 * Must be called only from #init()!
902 *
903 * @param aParent see ProgressBase::init()
904 * @param aInitiator see ProgressBase::init()
905 * @param aDescription see ProgressBase::init()
906 * @param aId see ProgressBase::init()
907 */
908HRESULT CombinedProgress::protectedInit (
909#if !defined (VBOX_COM_INPROC)
910 VirtualBox *aParent,
911#endif
912 IUnknown *aInitiator,
913 const BSTR aDescription, GUIDPARAMOUT aId)
914{
915 LogFlowMember (("CombinedProgress::protectedInit(): "
916 "aDescription={%ls} mProgresses.size()=%d\n",
917 aDescription, mProgresses.size()));
918
919 HRESULT rc = S_OK;
920
921 do
922 {
923 rc = ProgressBase::protectedInit (
924#if !defined (VBOX_COM_INPROC)
925 aParent,
926#endif
927 aInitiator, aDescription, aId);
928 CheckComRCBreakRC (rc);
929
930 // set ready to let protectedUninit() be called on failure
931 setReady (true);
932
933 mProgress = 0; // the first object
934 mCompletedOperations = 0;
935
936 mCompleted = FALSE;
937 mCancelable = TRUE; // until any progress returns FALSE
938 mCanceled = FALSE;
939
940 mOperationCount = 0; // will be calculated later
941 mOperation = 0;
942 rc = mProgresses [0]->COMGETTER(OperationDescription) (
943 mOperationDescription.asOutParam());
944 CheckComRCBreakRC (rc);
945
946 for (size_t i = 0; i < mProgresses.size(); i ++)
947 {
948 if (mCancelable)
949 {
950 BOOL cancelable = FALSE;
951 rc = mProgresses [i]->COMGETTER(Cancelable) (&cancelable);
952 if (FAILED (rc))
953 return rc;
954 if (!cancelable)
955 mCancelable = FALSE;
956 }
957
958 {
959 ULONG opCount = 0;
960 rc = mProgresses [i]->COMGETTER(OperationCount) (&opCount);
961 if (FAILED (rc))
962 return rc;
963 mOperationCount += opCount;
964 }
965 }
966
967 rc = checkProgress();
968 CheckComRCBreakRC (rc);
969 }
970 while (0);
971
972 if (FAILED (rc))
973 uninit();
974
975 return rc;
976}
977
978/**
979 * Uninitializes the instance and sets the ready flag to FALSE.
980 * Called either from FinalRelease() or by the parent when it gets destroyed.
981 */
982void CombinedProgress::uninit()
983{
984 LogFlowMember (("CombinedProgress::uninit()\n"));
985
986 AutoLock alock (this);
987
988 LogFlowMember (("CombinedProgress::uninit(): isReady=%d\n", isReady()));
989
990 if (!isReady())
991 return;
992
993 mProgress = 0;
994 mProgresses.clear();
995
996 ProgressBase::protectedUninit (alock);
997}
998
999// IProgress properties
1000////////////////////////////////////////////////////////////////////////////////
1001
1002STDMETHODIMP CombinedProgress::COMGETTER(Percent) (LONG *aPercent)
1003{
1004 if (!aPercent)
1005 return E_POINTER;
1006
1007 AutoLock lock(this);
1008 CHECK_READY();
1009
1010 if (mCompleted && SUCCEEDED (mResultCode))
1011 *aPercent = 100;
1012 else
1013 {
1014 HRESULT rc = checkProgress();
1015 if (FAILED (rc))
1016 return rc;
1017
1018 // global percent = (100 / mOperationCount) * mOperation +
1019 // ((100 / mOperationCount) / 100) * mOperationPercent
1020 *aPercent = (100 * mOperation + mOperationPercent) / mOperationCount;
1021 }
1022
1023 return S_OK;
1024}
1025
1026STDMETHODIMP CombinedProgress::COMGETTER(Completed) (BOOL *aCompleted)
1027{
1028 if (!aCompleted)
1029 return E_POINTER;
1030
1031 AutoLock lock(this);
1032 CHECK_READY();
1033
1034 HRESULT rc = checkProgress();
1035 if (FAILED (rc))
1036 return rc;
1037
1038 return ProgressBase::COMGETTER(Completed) (aCompleted);
1039}
1040
1041STDMETHODIMP CombinedProgress::COMGETTER(Canceled) (BOOL *aCanceled)
1042{
1043 if (!aCanceled)
1044 return E_POINTER;
1045
1046 AutoLock lock(this);
1047 CHECK_READY();
1048
1049 HRESULT rc = checkProgress();
1050 if (FAILED (rc))
1051 return rc;
1052
1053 return ProgressBase::COMGETTER(Canceled) (aCanceled);
1054}
1055
1056STDMETHODIMP CombinedProgress::COMGETTER(ResultCode) (HRESULT *aResultCode)
1057{
1058 if (!aResultCode)
1059 return E_POINTER;
1060
1061 AutoLock lock(this);
1062 CHECK_READY();
1063
1064 HRESULT rc = checkProgress();
1065 if (FAILED (rc))
1066 return rc;
1067
1068 return ProgressBase::COMGETTER(ResultCode) (aResultCode);
1069}
1070
1071STDMETHODIMP CombinedProgress::COMGETTER(ErrorInfo) (IVirtualBoxErrorInfo **aErrorInfo)
1072{
1073 if (!aErrorInfo)
1074 return E_POINTER;
1075
1076 AutoLock lock(this);
1077 CHECK_READY();
1078
1079 HRESULT rc = checkProgress();
1080 if (FAILED (rc))
1081 return rc;
1082
1083 return ProgressBase::COMGETTER(ErrorInfo) (aErrorInfo);
1084}
1085
1086STDMETHODIMP CombinedProgress::COMGETTER(Operation) (ULONG *aOperation)
1087{
1088 if (!aOperation)
1089 return E_POINTER;
1090
1091 AutoLock lock(this);
1092 CHECK_READY();
1093
1094 HRESULT rc = checkProgress();
1095 if (FAILED (rc))
1096 return rc;
1097
1098 return ProgressBase::COMGETTER(Operation) (aOperation);
1099}
1100
1101STDMETHODIMP CombinedProgress::COMGETTER(OperationDescription) (BSTR *aOperationDescription)
1102{
1103 if (!aOperationDescription)
1104 return E_POINTER;
1105
1106 AutoLock lock(this);
1107 CHECK_READY();
1108
1109 HRESULT rc = checkProgress();
1110 if (FAILED (rc))
1111 return rc;
1112
1113 return ProgressBase::COMGETTER(OperationDescription) (aOperationDescription);
1114}
1115
1116STDMETHODIMP CombinedProgress::COMGETTER(OperationPercent) (LONG *aOperationPercent)
1117{
1118 if (!aOperationPercent)
1119 return E_POINTER;
1120
1121 AutoLock lock(this);
1122 CHECK_READY();
1123
1124 HRESULT rc = checkProgress();
1125 if (FAILED (rc))
1126 return rc;
1127
1128 return ProgressBase::COMGETTER(OperationPercent) (aOperationPercent);
1129}
1130
1131// IProgress methods
1132/////////////////////////////////////////////////////////////////////////////
1133
1134/**
1135 * @note
1136 * XPCOM: when this method is called not on the main XPCOM thread, it
1137 * it simply blocks the thread until mCompletedSem is signalled. If the
1138 * thread has its own event queue (hmm, what for?) that it must run, then
1139 * calling this method will definitey freese event processing.
1140 */
1141STDMETHODIMP CombinedProgress::WaitForCompletion (LONG aTimeout)
1142{
1143 LogFlowMember (("CombinedProgress::WaitForCompletion: BEGIN: timeout=%d\n",
1144 aTimeout));
1145
1146 AutoLock lock (this);
1147 CHECK_READY();
1148
1149 // if we're already completed, take a shortcut
1150 if (!mCompleted)
1151 {
1152 RTTIMESPEC time;
1153 RTTimeNow (&time);
1154
1155 HRESULT rc = S_OK;
1156 bool forever = aTimeout < 0;
1157 int64_t timeLeft = aTimeout;
1158 int64_t lastTime = RTTimeSpecGetMilli (&time);
1159
1160 while (!mCompleted && (forever || timeLeft > 0))
1161 {
1162 lock.unlock();
1163 rc = mProgresses.back()->WaitForCompletion (
1164 forever ? -1 : (LONG) timeLeft);
1165 lock.lock();
1166
1167 // the progress might have been uninitialized
1168 if (!isReady())
1169 break;
1170
1171 if (SUCCEEDED (rc))
1172 rc = checkProgress();
1173
1174 if (FAILED (rc))
1175 break;
1176
1177 if (!forever)
1178 {
1179 RTTimeNow (&time);
1180 timeLeft -= RTTimeSpecGetMilli (&time) - lastTime;
1181 lastTime = RTTimeSpecGetMilli (&time);
1182 }
1183 }
1184
1185 if (FAILED (rc))
1186 return rc;
1187 }
1188
1189 LogFlowMember(("CombinedProgress::WaitForCompletion: END\n"));
1190 return S_OK;
1191}
1192
1193/**
1194 * @note
1195 * XPCOM: when this method is called not on the main XPCOM thread, it
1196 * it simply blocks the thread until mCompletedSem is signalled. If the
1197 * thread has its own event queue (hmm, what for?) that it must run, then
1198 * calling this method will definitey freese event processing.
1199 */
1200STDMETHODIMP CombinedProgress::WaitForOperationCompletion (ULONG aOperation, LONG aTimeout)
1201{
1202 LogFlowMember(("CombinedProgress::WaitForOperationCompletion: BEGIN: "
1203 "operation=%d, timeout=%d\n", aOperation, aTimeout));
1204
1205 AutoLock lock(this);
1206 CHECK_READY();
1207
1208 if (aOperation >= mOperationCount)
1209 return setError (E_FAIL,
1210 tr ("Operation number must be in range [0, %d]"), mOperation - 1);
1211
1212 // if we're already completed or if the given operation is already done,
1213 // then take a shortcut
1214 if (!mCompleted && aOperation >= mOperation)
1215 {
1216 HRESULT rc = S_OK;
1217
1218 // find the right progress object to wait for
1219 size_t progress = mProgress;
1220 ULONG operation = 0, completedOps = mCompletedOperations;
1221 do
1222 {
1223 ULONG opCount = 0;
1224 rc = mProgresses [progress]->COMGETTER(OperationCount) (&opCount);
1225 if (FAILED (rc))
1226 return rc;
1227
1228 if (completedOps + opCount > aOperation)
1229 {
1230 // found the right progress object
1231 operation = aOperation - completedOps;
1232 break;
1233 }
1234
1235 completedOps += opCount;
1236 progress ++;
1237 ComAssertRet (progress < mProgresses.size(), E_FAIL);
1238 }
1239 while (1);
1240
1241 LogFlowMember (("CombinedProgress::WaitForOperationCompletion(): "
1242 "will wait for mProgresses [%d] (%d)\n",
1243 progress, operation));
1244
1245 RTTIMESPEC time;
1246 RTTimeNow (&time);
1247
1248 bool forever = aTimeout < 0;
1249 int64_t timeLeft = aTimeout;
1250 int64_t lastTime = RTTimeSpecGetMilli (&time);
1251
1252 while (!mCompleted && aOperation >= mOperation &&
1253 (forever || timeLeft > 0))
1254 {
1255 lock.unlock();
1256 // wait for the appropriate progress operation completion
1257 rc = mProgresses [progress]-> WaitForOperationCompletion (
1258 operation, forever ? -1 : (LONG) timeLeft);
1259 lock.lock();
1260
1261 // the progress might have been uninitialized
1262 if (!isReady())
1263 break;
1264
1265 if (SUCCEEDED (rc))
1266 rc = checkProgress();
1267
1268 if (FAILED (rc))
1269 break;
1270
1271 if (!forever)
1272 {
1273 RTTimeNow (&time);
1274 timeLeft -= RTTimeSpecGetMilli (&time) - lastTime;
1275 lastTime = RTTimeSpecGetMilli (&time);
1276 }
1277 }
1278
1279 if (FAILED (rc))
1280 return rc;
1281 }
1282
1283 LogFlowMember(("CombinedProgress::WaitForOperationCompletion: END\n"));
1284 return S_OK;
1285}
1286
1287STDMETHODIMP CombinedProgress::Cancel()
1288{
1289 AutoLock lock(this);
1290 CHECK_READY();
1291
1292 if (!mCancelable)
1293 return setError (E_FAIL, tr ("Operation cannot be cancelled"));
1294
1295/// @todo (dmik): implement operation cancellation!
1296// mCompleted = TRUE;
1297// mCanceled = TRUE;
1298// return S_OK;
1299
1300 return setError (E_NOTIMPL, ("Not implemented!"));
1301}
1302
1303// private methods
1304////////////////////////////////////////////////////////////////////////////////
1305
1306/**
1307 * Fetches the properties of the current progress object and, if it is
1308 * successfully completed, advances to the next uncompleted or unsucessfully
1309 * completed object in the vector of combined progress objects.
1310 *
1311 * @note Must be called from under the object's lock!
1312 */
1313HRESULT CombinedProgress::checkProgress()
1314{
1315 // do nothing if we're already marked ourselves as completed
1316 if (mCompleted)
1317 return S_OK;
1318
1319 ComAssertRet (mProgress < mProgresses.size(), E_FAIL);
1320
1321 ComPtr <IProgress> progress = mProgresses [mProgress];
1322 ComAssertRet (!progress.isNull(), E_FAIL);
1323
1324 HRESULT rc = S_OK;
1325 BOOL completed = FALSE;
1326
1327 do
1328 {
1329 rc = progress->COMGETTER(Completed) (&completed);
1330 if (FAILED (rc))
1331 return rc;
1332
1333 if (completed)
1334 {
1335 rc = progress->COMGETTER(Canceled) (&mCanceled);
1336 if (FAILED (rc))
1337 return rc;
1338
1339 rc = progress->COMGETTER(ResultCode) (&mResultCode);
1340 if (FAILED (rc))
1341 return rc;
1342
1343 if (FAILED (mResultCode))
1344 {
1345 rc = progress->COMGETTER(ErrorInfo) (mErrorInfo.asOutParam());
1346 if (FAILED (rc))
1347 return rc;
1348 }
1349
1350 if (FAILED (mResultCode) || mCanceled)
1351 {
1352 mCompleted = TRUE;
1353 }
1354 else
1355 {
1356 ULONG opCount = 0;
1357 rc = progress->COMGETTER(OperationCount) (&opCount);
1358 if (FAILED (rc))
1359 return rc;
1360
1361 mCompletedOperations += opCount;
1362 mProgress ++;
1363
1364 if (mProgress < mProgresses.size())
1365 progress = mProgresses [mProgress];
1366 else
1367 mCompleted = TRUE;
1368 }
1369 }
1370 }
1371 while (completed && !mCompleted);
1372
1373 rc = progress->COMGETTER(OperationPercent) (&mOperationPercent);
1374 if (SUCCEEDED (rc))
1375 {
1376 ULONG operation = 0;
1377 rc = progress->COMGETTER(Operation) (&operation);
1378 if (SUCCEEDED (rc) && mCompletedOperations + operation > mOperation)
1379 {
1380 mOperation = mCompletedOperations + operation;
1381 rc = progress->COMGETTER(OperationDescription) (
1382 mOperationDescription.asOutParam());
1383 }
1384 }
1385
1386 return rc;
1387}
1388
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