VirtualBox

source: vbox/trunk/src/VBox/Main/src-all/ProgressImpl.cpp@ 37608

Last change on this file since 37608 was 37069, checked in by vboxsync, 14 years ago

Main: add waitForAsyncProgressCompletion to IProgress

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 55.4 KB
Line 
1/* $Id: ProgressImpl.cpp 37069 2011-05-13 12:41:38Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox Progress COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2010 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include <iprt/types.h>
20
21#if defined(VBOX_WITH_XPCOM)
22#include <nsIServiceManager.h>
23#include <nsIExceptionService.h>
24#include <nsCOMPtr.h>
25#endif /* defined(VBOX_WITH_XPCOM) */
26
27#include "ProgressCombinedImpl.h"
28
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#include <iprt/cpp/utils.h>
37
38#include <VBox/err.h>
39
40////////////////////////////////////////////////////////////////////////////////
41// ProgressBase class
42////////////////////////////////////////////////////////////////////////////////
43
44// constructor / destructor
45////////////////////////////////////////////////////////////////////////////////
46
47ProgressBase::ProgressBase()
48#if !defined(VBOX_COM_INPROC)
49 : mParent(NULL)
50#endif
51{
52}
53
54ProgressBase::~ProgressBase()
55{
56}
57
58
59/**
60 * Subclasses must call this method from their FinalConstruct() implementations.
61 */
62HRESULT ProgressBase::FinalConstruct()
63{
64 mCancelable = FALSE;
65 mCompleted = FALSE;
66 mCanceled = FALSE;
67 mResultCode = S_OK;
68
69 m_cOperations
70 = m_ulTotalOperationsWeight
71 = m_ulOperationsCompletedWeight
72 = m_ulCurrentOperation
73 = m_ulCurrentOperationWeight
74 = m_ulOperationPercent
75 = m_cMsTimeout
76 = 0;
77
78 // get creation timestamp
79 m_ullTimestamp = RTTimeMilliTS();
80
81 m_pfnCancelCallback = NULL;
82 m_pvCancelUserArg = NULL;
83
84 return BaseFinalConstruct();
85}
86
87// protected initializer/uninitializer for internal purposes only
88////////////////////////////////////////////////////////////////////////////////
89
90/**
91 * Initializes the progress base object.
92 *
93 * Subclasses should call this or any other #protectedInit() method from their
94 * init() implementations.
95 *
96 * @param aAutoInitSpan AutoInitSpan object instantiated by a subclass.
97 * @param aParent Parent object (only for server-side Progress objects).
98 * @param aInitiator Initiator of the task (for server-side objects. Can be
99 * NULL which means initiator = parent, otherwise must not
100 * be NULL).
101 * @param aDescription ask description.
102 * @param aID Address of result GUID structure (optional).
103 *
104 * @return COM result indicator.
105 */
106HRESULT ProgressBase::protectedInit(AutoInitSpan &aAutoInitSpan,
107#if !defined(VBOX_COM_INPROC)
108 VirtualBox *aParent,
109#endif
110 IUnknown *aInitiator,
111 CBSTR aDescription,
112 OUT_GUID aId /* = NULL */)
113{
114 /* Guarantees subclasses call this method at the proper time */
115 NOREF(aAutoInitSpan);
116
117 AutoCaller autoCaller(this);
118 AssertReturn(autoCaller.state() == InInit, E_FAIL);
119
120#if !defined(VBOX_COM_INPROC)
121 AssertReturn(aParent, E_INVALIDARG);
122#else
123 AssertReturn(aInitiator, E_INVALIDARG);
124#endif
125
126 AssertReturn(aDescription, E_INVALIDARG);
127
128#if !defined(VBOX_COM_INPROC)
129 /* share parent weakly */
130 unconst(mParent) = aParent;
131#endif
132
133#if !defined(VBOX_COM_INPROC)
134 /* assign (and therefore addref) initiator only if it is not VirtualBox
135 * (to avoid cycling); otherwise mInitiator will remain null which means
136 * that it is the same as the parent */
137 if (aInitiator)
138 {
139 ComObjPtr<VirtualBox> pVirtualBox(mParent);
140 if (!(pVirtualBox == aInitiator))
141 unconst(mInitiator) = aInitiator;
142 }
143#else
144 unconst(mInitiator) = aInitiator;
145#endif
146
147 unconst(mId).create();
148 if (aId)
149 mId.cloneTo(aId);
150
151#if !defined(VBOX_COM_INPROC)
152 /* add to the global collection of progress operations (note: after
153 * creating mId) */
154 mParent->addProgress(this);
155#endif
156
157 unconst(mDescription) = aDescription;
158
159 return S_OK;
160}
161
162/**
163 * Initializes the progress base object.
164 *
165 * This is a special initializer that doesn't initialize any field. Used by one
166 * of the Progress::init() forms to create sub-progress operations combined
167 * together using a CombinedProgress instance, so it doesn't require the parent,
168 * initiator, description and doesn't create an ID.
169 *
170 * Subclasses should call this or any other #protectedInit() method from their
171 * init() implementations.
172 *
173 * @param aAutoInitSpan AutoInitSpan object instantiated by a subclass.
174 */
175HRESULT ProgressBase::protectedInit(AutoInitSpan &aAutoInitSpan)
176{
177 /* Guarantees subclasses call this method at the proper time */
178 NOREF(aAutoInitSpan);
179
180 return S_OK;
181}
182
183/**
184 * Uninitializes the instance.
185 *
186 * Subclasses should call this from their uninit() implementations.
187 *
188 * @param aAutoUninitSpan AutoUninitSpan object instantiated by a subclass.
189 *
190 * @note Using the mParent member after this method returns is forbidden.
191 */
192void ProgressBase::protectedUninit(AutoUninitSpan &aAutoUninitSpan)
193{
194 /* release initiator (effective only if mInitiator has been assigned in
195 * init()) */
196 unconst(mInitiator).setNull();
197
198#if !defined(VBOX_COM_INPROC)
199 if (mParent)
200 {
201 /* remove the added progress on failure to complete the initialization */
202 if (aAutoUninitSpan.initFailed() && !mId.isEmpty())
203 mParent->removeProgress(mId.ref());
204
205 unconst(mParent) = NULL;
206 }
207#endif
208}
209
210// IProgress properties
211/////////////////////////////////////////////////////////////////////////////
212
213STDMETHODIMP ProgressBase::COMGETTER(Id)(BSTR *aId)
214{
215 CheckComArgOutPointerValid(aId);
216
217 AutoCaller autoCaller(this);
218 if (FAILED(autoCaller.rc())) return autoCaller.rc();
219
220 /* mId is constant during life time, no need to lock */
221 mId.toUtf16().cloneTo(aId);
222
223 return S_OK;
224}
225
226STDMETHODIMP ProgressBase::COMGETTER(Description)(BSTR *aDescription)
227{
228 CheckComArgOutPointerValid(aDescription);
229
230 AutoCaller autoCaller(this);
231 if (FAILED(autoCaller.rc())) return autoCaller.rc();
232
233 /* mDescription is constant during life time, no need to lock */
234 mDescription.cloneTo(aDescription);
235
236 return S_OK;
237}
238
239STDMETHODIMP ProgressBase::COMGETTER(Initiator)(IUnknown **aInitiator)
240{
241 CheckComArgOutPointerValid(aInitiator);
242
243 AutoCaller autoCaller(this);
244 if (FAILED(autoCaller.rc())) return autoCaller.rc();
245
246 /* mInitiator/mParent are constant during life time, no need to lock */
247
248#if !defined(VBOX_COM_INPROC)
249 if (mInitiator)
250 mInitiator.queryInterfaceTo(aInitiator);
251 else
252 {
253 ComObjPtr<VirtualBox> pVirtualBox(mParent);
254 pVirtualBox.queryInterfaceTo(aInitiator);
255 }
256#else
257 mInitiator.queryInterfaceTo(aInitiator);
258#endif
259
260 return S_OK;
261}
262
263STDMETHODIMP ProgressBase::COMGETTER(Cancelable)(BOOL *aCancelable)
264{
265 CheckComArgOutPointerValid(aCancelable);
266
267 AutoCaller autoCaller(this);
268 if (FAILED(autoCaller.rc())) return autoCaller.rc();
269
270 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
271
272 *aCancelable = mCancelable;
273
274 return S_OK;
275}
276
277/**
278 * Internal helper to compute the total percent value based on the member values and
279 * returns it as a "double". This is used both by GetPercent (which returns it as a
280 * rounded ULONG) and GetTimeRemaining().
281 *
282 * Requires locking by the caller!
283 *
284 * @return fractional percentage as a double value.
285 */
286double ProgressBase::calcTotalPercent()
287{
288 // avoid division by zero
289 if (m_ulTotalOperationsWeight == 0)
290 return 0;
291
292 double dPercent = ( (double)m_ulOperationsCompletedWeight // weight of operations that have been completed
293 + ((double)m_ulOperationPercent * (double)m_ulCurrentOperationWeight / (double)100) // plus partial weight of the current operation
294 ) * (double)100 / (double)m_ulTotalOperationsWeight;
295
296 return dPercent;
297}
298
299/**
300 * Internal helper for automatically timing out the operation.
301 *
302 * The caller should hold the object write lock.
303 */
304void ProgressBase::checkForAutomaticTimeout(void)
305{
306 if ( m_cMsTimeout
307 && mCancelable
308 && !mCanceled
309 && RTTimeMilliTS() - m_ullTimestamp > m_cMsTimeout
310 )
311 Cancel();
312}
313
314
315STDMETHODIMP ProgressBase::COMGETTER(TimeRemaining)(LONG *aTimeRemaining)
316{
317 CheckComArgOutPointerValid(aTimeRemaining);
318
319 AutoCaller autoCaller(this);
320 if (FAILED(autoCaller.rc())) return autoCaller.rc();
321
322 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
323
324 if (mCompleted)
325 *aTimeRemaining = 0;
326 else
327 {
328 double dPercentDone = calcTotalPercent();
329 if (dPercentDone < 1)
330 *aTimeRemaining = -1; // unreliable, or avoid division by 0 below
331 else
332 {
333 uint64_t ullTimeNow = RTTimeMilliTS();
334 uint64_t ullTimeElapsed = ullTimeNow - m_ullTimestamp;
335 uint64_t ullTimeTotal = (uint64_t)(ullTimeElapsed * 100 / dPercentDone);
336 uint64_t ullTimeRemaining = ullTimeTotal - ullTimeElapsed;
337
338// Log(("ProgressBase::GetTimeRemaining: dPercentDone %RI32, ullTimeNow = %RI64, ullTimeElapsed = %RI64, ullTimeTotal = %RI64, ullTimeRemaining = %RI64\n",
339// (uint32_t)dPercentDone, ullTimeNow, ullTimeElapsed, ullTimeTotal, ullTimeRemaining));
340
341 *aTimeRemaining = (LONG)(ullTimeRemaining / 1000);
342 }
343 }
344
345 return S_OK;
346}
347
348STDMETHODIMP ProgressBase::COMGETTER(Percent)(ULONG *aPercent)
349{
350 CheckComArgOutPointerValid(aPercent);
351
352 AutoCaller autoCaller(this);
353 if (FAILED(autoCaller.rc())) return autoCaller.rc();
354
355 checkForAutomaticTimeout();
356
357 /* checkForAutomaticTimeout requires a write lock. */
358 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
359
360 if (mCompleted && SUCCEEDED(mResultCode))
361 *aPercent = 100;
362 else
363 {
364 ULONG ulPercent = (ULONG)calcTotalPercent();
365 // do not report 100% until we're really really done with everything as the Qt GUI dismisses progress dialogs in that case
366 if ( ulPercent == 100
367 && ( m_ulOperationPercent < 100
368 || (m_ulCurrentOperation < m_cOperations -1)
369 )
370 )
371 *aPercent = 99;
372 else
373 *aPercent = ulPercent;
374 }
375
376 checkForAutomaticTimeout();
377
378 return S_OK;
379}
380
381STDMETHODIMP ProgressBase::COMGETTER(Completed)(BOOL *aCompleted)
382{
383 CheckComArgOutPointerValid(aCompleted);
384
385 AutoCaller autoCaller(this);
386 if (FAILED(autoCaller.rc())) return autoCaller.rc();
387
388 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
389
390 *aCompleted = mCompleted;
391
392 return S_OK;
393}
394
395STDMETHODIMP ProgressBase::COMGETTER(Canceled)(BOOL *aCanceled)
396{
397 CheckComArgOutPointerValid(aCanceled);
398
399 AutoCaller autoCaller(this);
400 if (FAILED(autoCaller.rc())) return autoCaller.rc();
401
402 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
403
404 *aCanceled = mCanceled;
405
406 return S_OK;
407}
408
409STDMETHODIMP ProgressBase::COMGETTER(ResultCode)(LONG *aResultCode)
410{
411 CheckComArgOutPointerValid(aResultCode);
412
413 AutoCaller autoCaller(this);
414 if (FAILED(autoCaller.rc())) return autoCaller.rc();
415
416 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
417
418 if (!mCompleted)
419 return setError(E_FAIL,
420 tr("Result code is not available, operation is still in progress"));
421
422 *aResultCode = mResultCode;
423
424 return S_OK;
425}
426
427STDMETHODIMP ProgressBase::COMGETTER(ErrorInfo)(IVirtualBoxErrorInfo **aErrorInfo)
428{
429 CheckComArgOutPointerValid(aErrorInfo);
430
431 AutoCaller autoCaller(this);
432 if (FAILED(autoCaller.rc())) return autoCaller.rc();
433
434 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
435
436 if (!mCompleted)
437 return setError(E_FAIL,
438 tr("Error info is not available, operation is still in progress"));
439
440 mErrorInfo.queryInterfaceTo(aErrorInfo);
441
442 return S_OK;
443}
444
445STDMETHODIMP ProgressBase::COMGETTER(OperationCount)(ULONG *aOperationCount)
446{
447 CheckComArgOutPointerValid(aOperationCount);
448
449 AutoCaller autoCaller(this);
450 if (FAILED(autoCaller.rc())) return autoCaller.rc();
451
452 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
453
454 *aOperationCount = m_cOperations;
455
456 return S_OK;
457}
458
459STDMETHODIMP ProgressBase::COMGETTER(Operation)(ULONG *aOperation)
460{
461 CheckComArgOutPointerValid(aOperation);
462
463 AutoCaller autoCaller(this);
464 if (FAILED(autoCaller.rc())) return autoCaller.rc();
465
466 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
467
468 *aOperation = m_ulCurrentOperation;
469
470 return S_OK;
471}
472
473STDMETHODIMP ProgressBase::COMGETTER(OperationDescription)(BSTR *aOperationDescription)
474{
475 CheckComArgOutPointerValid(aOperationDescription);
476
477 AutoCaller autoCaller(this);
478 if (FAILED(autoCaller.rc())) return autoCaller.rc();
479
480 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
481
482 m_bstrOperationDescription.cloneTo(aOperationDescription);
483
484 return S_OK;
485}
486
487STDMETHODIMP ProgressBase::COMGETTER(OperationPercent)(ULONG *aOperationPercent)
488{
489 CheckComArgOutPointerValid(aOperationPercent);
490
491 AutoCaller autoCaller(this);
492 if (FAILED(autoCaller.rc())) return autoCaller.rc();
493
494 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
495
496 if (mCompleted && SUCCEEDED(mResultCode))
497 *aOperationPercent = 100;
498 else
499 *aOperationPercent = m_ulOperationPercent;
500
501 return S_OK;
502}
503
504STDMETHODIMP ProgressBase::COMGETTER(OperationWeight)(ULONG *aOperationWeight)
505{
506 CheckComArgOutPointerValid(aOperationWeight);
507
508 AutoCaller autoCaller(this);
509 if (FAILED(autoCaller.rc())) return autoCaller.rc();
510
511 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
512
513 *aOperationWeight = m_ulCurrentOperationWeight;
514
515 return S_OK;
516}
517
518STDMETHODIMP ProgressBase::COMSETTER(Timeout)(ULONG aTimeout)
519{
520 AutoCaller autoCaller(this);
521 if (FAILED(autoCaller.rc())) return autoCaller.rc();
522
523 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
524
525 if (!mCancelable)
526 return setError(VBOX_E_INVALID_OBJECT_STATE,
527 tr("Operation cannot be canceled"));
528
529 LogThisFunc(("%#x => %#x\n", m_cMsTimeout, aTimeout));
530 m_cMsTimeout = aTimeout;
531 return S_OK;
532}
533
534STDMETHODIMP ProgressBase::COMGETTER(Timeout)(ULONG *aTimeout)
535{
536 CheckComArgOutPointerValid(aTimeout);
537
538 AutoCaller autoCaller(this);
539 if (FAILED(autoCaller.rc())) return autoCaller.rc();
540
541 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
542
543 *aTimeout = m_cMsTimeout;
544 return S_OK;
545}
546
547// public methods only for internal purposes
548////////////////////////////////////////////////////////////////////////////////
549
550/**
551 * Sets the cancelation callback, checking for cancelation first.
552 *
553 * @returns Success indicator.
554 * @retval true on success.
555 * @retval false if the progress object has already been canceled or is in an
556 * invalid state
557 *
558 * @param pfnCallback The function to be called upon cancelation.
559 * @param pvUser The callback argument.
560 */
561bool ProgressBase::setCancelCallback(void (*pfnCallback)(void *), void *pvUser)
562{
563 AutoCaller autoCaller(this);
564 AssertComRCReturn(autoCaller.rc(), false);
565
566 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
567
568 checkForAutomaticTimeout();
569 if (mCanceled)
570 return false;
571
572 m_pvCancelUserArg = pvUser;
573 m_pfnCancelCallback = pfnCallback;
574 return true;
575}
576
577////////////////////////////////////////////////////////////////////////////////
578// Progress class
579////////////////////////////////////////////////////////////////////////////////
580
581HRESULT Progress::FinalConstruct()
582{
583 HRESULT rc = ProgressBase::FinalConstruct();
584 if (FAILED(rc)) return rc;
585
586 mCompletedSem = NIL_RTSEMEVENTMULTI;
587 mWaitersCount = 0;
588
589 return S_OK;
590}
591
592void Progress::FinalRelease()
593{
594 uninit();
595 BaseFinalRelease();
596}
597
598// public initializer/uninitializer for internal purposes only
599////////////////////////////////////////////////////////////////////////////////
600
601/**
602 * Initializes the normal progress object. With this variant, one can have
603 * an arbitrary number of sub-operation which IProgress can analyze to
604 * have a weighted progress computed.
605 *
606 * For example, say that one IProgress is supposed to track the cloning
607 * of two hard disk images, which are 100 MB and 1000 MB in size, respectively,
608 * and each of these hard disks should be one sub-operation of the IProgress.
609 *
610 * Obviously the progress would be misleading if the progress displayed 50%
611 * after the smaller image was cloned and would then take much longer for
612 * the second half.
613 *
614 * With weighted progress, one can invoke the following calls:
615 *
616 * 1) create progress object with cOperations = 2 and ulTotalOperationsWeight =
617 * 1100 (100 MB plus 1100, but really the weights can be any ULONG); pass
618 * in ulFirstOperationWeight = 100 for the first sub-operation
619 *
620 * 2) Then keep calling setCurrentOperationProgress() with a percentage
621 * for the first image; the total progress will increase up to a value
622 * of 9% (100MB / 1100MB * 100%).
623 *
624 * 3) Then call setNextOperation with the second weight (1000 for the megabytes
625 * of the second disk).
626 *
627 * 4) Then keep calling setCurrentOperationProgress() with a percentage for
628 * the second image, where 100% of the operation will then yield a 100%
629 * progress of the entire task.
630 *
631 * Weighting is optional; you can simply assign a weight of 1 to each operation
632 * and pass ulTotalOperationsWeight == cOperations to this constructor (but
633 * for that variant and for backwards-compatibility a simpler constructor exists
634 * in ProgressImpl.h as well).
635 *
636 * Even simpler, if you need no sub-operations at all, pass in cOperations =
637 * ulTotalOperationsWeight = ulFirstOperationWeight = 1.
638 *
639 * @param aParent See ProgressBase::init().
640 * @param aInitiator See ProgressBase::init().
641 * @param aDescription See ProgressBase::init().
642 * @param aCancelable Flag whether the task maybe canceled.
643 * @param cOperations Number of operations within this task (at least 1).
644 * @param ulTotalOperationsWeight Total weight of operations; must be the sum of ulFirstOperationWeight and
645 * what is later passed with each subsequent setNextOperation() call.
646 * @param bstrFirstOperationDescription Description of the first operation.
647 * @param ulFirstOperationWeight Weight of first sub-operation.
648 * @param aId See ProgressBase::init().
649 */
650HRESULT Progress::init(
651#if !defined(VBOX_COM_INPROC)
652 VirtualBox *aParent,
653#endif
654 IUnknown *aInitiator,
655 CBSTR aDescription,
656 BOOL aCancelable,
657 ULONG cOperations,
658 ULONG ulTotalOperationsWeight,
659 CBSTR bstrFirstOperationDescription,
660 ULONG ulFirstOperationWeight,
661 OUT_GUID aId /* = NULL */)
662{
663 LogFlowThisFunc(("aDescription=\"%ls\", cOperations=%d, ulTotalOperationsWeight=%d, bstrFirstOperationDescription=\"%ls\", ulFirstOperationWeight=%d\n",
664 aDescription,
665 cOperations,
666 ulTotalOperationsWeight,
667 bstrFirstOperationDescription,
668 ulFirstOperationWeight));
669
670 AssertReturn(bstrFirstOperationDescription, E_INVALIDARG);
671 AssertReturn(ulTotalOperationsWeight >= 1, E_INVALIDARG);
672
673 /* Enclose the state transition NotReady->InInit->Ready */
674 AutoInitSpan autoInitSpan(this);
675 AssertReturn(autoInitSpan.isOk(), E_FAIL);
676
677 HRESULT rc = S_OK;
678
679 rc = ProgressBase::protectedInit(autoInitSpan,
680#if !defined(VBOX_COM_INPROC)
681 aParent,
682#endif
683 aInitiator, aDescription, aId);
684 if (FAILED(rc)) return rc;
685
686 mCancelable = aCancelable;
687
688 m_cOperations = cOperations;
689 m_ulTotalOperationsWeight = ulTotalOperationsWeight;
690 m_ulOperationsCompletedWeight = 0;
691 m_ulCurrentOperation = 0;
692 m_bstrOperationDescription = bstrFirstOperationDescription;
693 m_ulCurrentOperationWeight = ulFirstOperationWeight;
694 m_ulOperationPercent = 0;
695
696 int vrc = RTSemEventMultiCreate(&mCompletedSem);
697 ComAssertRCRet(vrc, E_FAIL);
698
699 RTSemEventMultiReset(mCompletedSem);
700
701 /* Confirm a successful initialization when it's the case */
702 if (SUCCEEDED(rc))
703 autoInitSpan.setSucceeded();
704
705 return rc;
706}
707
708/**
709 * Initializes the sub-progress object that represents a specific operation of
710 * the whole task.
711 *
712 * Objects initialized with this method are then combined together into the
713 * single task using a CombinedProgress instance, so it doesn't require the
714 * parent, initiator, description and doesn't create an ID. Note that calling
715 * respective getter methods on an object initialized with this method is
716 * useless. Such objects are used only to provide a separate wait semaphore and
717 * store individual operation descriptions.
718 *
719 * @param aCancelable Flag whether the task maybe canceled.
720 * @param aOperationCount Number of sub-operations within this task (at least 1).
721 * @param aOperationDescription Description of the individual operation.
722 */
723HRESULT Progress::init(BOOL aCancelable,
724 ULONG aOperationCount,
725 CBSTR aOperationDescription)
726{
727 LogFlowThisFunc(("aOperationDescription=\"%ls\"\n", aOperationDescription));
728
729 /* Enclose the state transition NotReady->InInit->Ready */
730 AutoInitSpan autoInitSpan(this);
731 AssertReturn(autoInitSpan.isOk(), E_FAIL);
732
733 HRESULT rc = S_OK;
734
735 rc = ProgressBase::protectedInit(autoInitSpan);
736 if (FAILED(rc)) return rc;
737
738 mCancelable = aCancelable;
739
740 // for this variant we assume for now that all operations are weighed "1"
741 // and equal total weight = operation count
742 m_cOperations = aOperationCount;
743 m_ulTotalOperationsWeight = aOperationCount;
744 m_ulOperationsCompletedWeight = 0;
745 m_ulCurrentOperation = 0;
746 m_bstrOperationDescription = aOperationDescription;
747 m_ulCurrentOperationWeight = 1;
748 m_ulOperationPercent = 0;
749
750 int vrc = RTSemEventMultiCreate(&mCompletedSem);
751 ComAssertRCRet(vrc, E_FAIL);
752
753 RTSemEventMultiReset(mCompletedSem);
754
755 /* Confirm a successful initialization when it's the case */
756 if (SUCCEEDED(rc))
757 autoInitSpan.setSucceeded();
758
759 return rc;
760}
761
762/**
763 * Uninitializes the instance and sets the ready flag to FALSE.
764 *
765 * Called either from FinalRelease() or by the parent when it gets destroyed.
766 */
767void Progress::uninit()
768{
769 LogFlowThisFunc(("\n"));
770
771 /* Enclose the state transition Ready->InUninit->NotReady */
772 AutoUninitSpan autoUninitSpan(this);
773 if (autoUninitSpan.uninitDone())
774 return;
775
776 /* wake up all threads still waiting on occasion */
777 if (mWaitersCount > 0)
778 {
779 LogFlow(("WARNING: There are still %d threads waiting for '%ls' completion!\n",
780 mWaitersCount, mDescription.raw()));
781 RTSemEventMultiSignal(mCompletedSem);
782 }
783
784 RTSemEventMultiDestroy(mCompletedSem);
785
786 ProgressBase::protectedUninit(autoUninitSpan);
787}
788
789// IProgress properties
790/////////////////////////////////////////////////////////////////////////////
791
792// IProgress methods
793/////////////////////////////////////////////////////////////////////////////
794
795/**
796 * @note XPCOM: when this method is not called on the main XPCOM thread, it
797 * simply blocks the thread until mCompletedSem is signalled. If the
798 * thread has its own event queue (hmm, what for?) that it must run, then
799 * calling this method will definitely freeze event processing.
800 */
801STDMETHODIMP Progress::WaitForCompletion(LONG aTimeout)
802{
803 LogFlowThisFuncEnter();
804 LogFlowThisFunc(("aTimeout=%d\n", aTimeout));
805
806 AutoCaller autoCaller(this);
807 if (FAILED(autoCaller.rc())) return autoCaller.rc();
808
809 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
810
811 /* if we're already completed, take a shortcut */
812 if (!mCompleted)
813 {
814 int vrc = VINF_SUCCESS;
815 bool fForever = aTimeout < 0;
816 int64_t timeLeft = aTimeout;
817 int64_t lastTime = RTTimeMilliTS();
818
819 while (!mCompleted && (fForever || timeLeft > 0))
820 {
821 mWaitersCount++;
822 alock.leave();
823 vrc = RTSemEventMultiWait(mCompletedSem,
824 fForever ? RT_INDEFINITE_WAIT : (RTMSINTERVAL)timeLeft);
825 alock.enter();
826 mWaitersCount--;
827
828 /* the last waiter resets the semaphore */
829 if (mWaitersCount == 0)
830 RTSemEventMultiReset(mCompletedSem);
831
832 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
833 break;
834
835 if (!fForever)
836 {
837 int64_t now = RTTimeMilliTS();
838 timeLeft -= now - lastTime;
839 lastTime = now;
840 }
841 }
842
843 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
844 return setError(VBOX_E_IPRT_ERROR,
845 tr("Failed to wait for the task completion (%Rrc)"),
846 vrc);
847 }
848
849 LogFlowThisFuncLeave();
850
851 return S_OK;
852}
853
854/**
855 * @note XPCOM: when this method is not called on the main XPCOM thread, it
856 * simply blocks the thread until mCompletedSem is signalled. If the
857 * thread has its own event queue (hmm, what for?) that it must run, then
858 * calling this method will definitely freeze event processing.
859 */
860STDMETHODIMP Progress::WaitForOperationCompletion(ULONG aOperation, LONG aTimeout)
861{
862 LogFlowThisFuncEnter();
863 LogFlowThisFunc(("aOperation=%d, aTimeout=%d\n", aOperation, aTimeout));
864
865 AutoCaller autoCaller(this);
866 if (FAILED(autoCaller.rc())) return autoCaller.rc();
867
868 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
869
870 CheckComArgExpr(aOperation, aOperation < m_cOperations);
871
872 /* if we're already completed or if the given operation is already done,
873 * then take a shortcut */
874 if ( !mCompleted
875 && aOperation >= m_ulCurrentOperation)
876 {
877 int vrc = VINF_SUCCESS;
878 bool fForever = aTimeout < 0;
879 int64_t timeLeft = aTimeout;
880 int64_t lastTime = RTTimeMilliTS();
881
882 while ( !mCompleted && aOperation >= m_ulCurrentOperation
883 && (fForever || timeLeft > 0))
884 {
885 mWaitersCount ++;
886 alock.leave();
887 vrc = RTSemEventMultiWait(mCompletedSem,
888 fForever ? RT_INDEFINITE_WAIT : (unsigned) timeLeft);
889 alock.enter();
890 mWaitersCount--;
891
892 /* the last waiter resets the semaphore */
893 if (mWaitersCount == 0)
894 RTSemEventMultiReset(mCompletedSem);
895
896 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
897 break;
898
899 if (!fForever)
900 {
901 int64_t now = RTTimeMilliTS();
902 timeLeft -= now - lastTime;
903 lastTime = now;
904 }
905 }
906
907 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
908 return setError(E_FAIL,
909 tr("Failed to wait for the operation completion (%Rrc)"),
910 vrc);
911 }
912
913 LogFlowThisFuncLeave();
914
915 return S_OK;
916}
917
918STDMETHODIMP Progress::WaitForAsyncProgressCompletion(IProgress *pProgressAsync)
919{
920 LogFlowThisFuncEnter();
921
922 CheckComArgNotNull(pProgressAsync);
923
924 AutoCaller autoCaller(this);
925 if (FAILED(autoCaller.rc())) return autoCaller.rc();
926
927 /* Note: we don't lock here, cause we just using public methods. */
928
929 HRESULT rc = S_OK;
930 BOOL fCancelable = FALSE;
931 BOOL fCompleted = FALSE;
932 BOOL fCanceled = FALSE;
933 ULONG currentPercent = 0;
934 ULONG cOp = 0;
935 /* Is the async process cancelable? */
936 rc = pProgressAsync->COMGETTER(Cancelable)(&fCancelable);
937 if (FAILED(rc)) return rc;
938 /* Loop as long as the sync process isn't completed. */
939 while (SUCCEEDED(pProgressAsync->COMGETTER(Completed(&fCompleted))))
940 {
941 /* We can forward any cancel request to the async process only when
942 * it is cancelable. */
943 if (fCancelable)
944 {
945 rc = COMGETTER(Canceled)(&fCanceled);
946 if (FAILED(rc)) return rc;
947 if (fCanceled)
948 {
949 rc = pProgressAsync->Cancel();
950 if (FAILED(rc)) return rc;
951 }
952 }
953 /* Even if the user canceled the process, we have to wait until the
954 async task has finished his work (cleanup and such). Otherwise there
955 will be sync trouble (still wrong state, dead locks, ...) on the
956 used objects. So just do nothing, but wait for the complete
957 notification. */
958 if (!fCanceled)
959 {
960 /* Check if the current operation has changed. It is also possible that
961 * in the meantime more than one async operation was finished. So we
962 * have to loop as long as we reached the same operation count. */
963 ULONG curOp;
964 for(;;)
965 {
966 rc = pProgressAsync->COMGETTER(Operation(&curOp));
967 if (FAILED(rc)) return rc;
968 if (cOp != curOp)
969 {
970 Bstr bstr;
971 ULONG currentWeight;
972 rc = pProgressAsync->COMGETTER(OperationDescription(bstr.asOutParam()));
973 if (FAILED(rc)) return rc;
974 rc = pProgressAsync->COMGETTER(OperationWeight(&currentWeight));
975 if (FAILED(rc)) return rc;
976 rc = SetNextOperation(bstr.raw(), currentWeight);
977 if (FAILED(rc)) return rc;
978 ++cOp;
979 }else
980 break;
981 }
982
983 rc = pProgressAsync->COMGETTER(OperationPercent(&currentPercent));
984 if (FAILED(rc)) return rc;
985 rc = SetCurrentOperationProgress(currentPercent);
986 if (FAILED(rc)) return rc;
987 }
988 if (fCompleted)
989 break;
990
991 /* Make sure the loop is not too tight */
992 rc = pProgressAsync->WaitForCompletion(100);
993 if (FAILED(rc)) return rc;
994 }
995
996 LogFlowThisFuncLeave();
997
998 return rc;
999}
1000
1001STDMETHODIMP Progress::Cancel()
1002{
1003 AutoCaller autoCaller(this);
1004 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1005
1006 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1007
1008 if (!mCancelable)
1009 return setError(VBOX_E_INVALID_OBJECT_STATE,
1010 tr("Operation cannot be canceled"));
1011
1012 if (!mCanceled)
1013 {
1014 LogThisFunc(("Canceling\n"));
1015 mCanceled = TRUE;
1016 if (m_pfnCancelCallback)
1017 m_pfnCancelCallback(m_pvCancelUserArg);
1018
1019 }
1020 else
1021 LogThisFunc(("Already canceled\n"));
1022
1023 return S_OK;
1024}
1025
1026/**
1027 * Updates the percentage value of the current operation.
1028 *
1029 * @param aPercent New percentage value of the operation in progress
1030 * (in range [0, 100]).
1031 */
1032STDMETHODIMP Progress::SetCurrentOperationProgress(ULONG aPercent)
1033{
1034 AutoCaller autoCaller(this);
1035 AssertComRCReturnRC(autoCaller.rc());
1036
1037 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1038
1039 AssertMsgReturn(aPercent <= 100, ("%u\n", aPercent), E_INVALIDARG);
1040
1041 checkForAutomaticTimeout();
1042 if (mCancelable && mCanceled)
1043 {
1044 Assert(!mCompleted);
1045 return E_FAIL;
1046 }
1047 AssertReturn(!mCompleted && !mCanceled, E_FAIL);
1048
1049 m_ulOperationPercent = aPercent;
1050
1051 return S_OK;
1052}
1053
1054/**
1055 * Signals that the current operation is successfully completed and advances to
1056 * the next operation. The operation percentage is reset to 0.
1057 *
1058 * @param aOperationDescription Description of the next operation.
1059 *
1060 * @note The current operation must not be the last one.
1061 */
1062STDMETHODIMP Progress::SetNextOperation(IN_BSTR bstrNextOperationDescription, ULONG ulNextOperationsWeight)
1063{
1064 AssertReturn(bstrNextOperationDescription, E_INVALIDARG);
1065
1066 AutoCaller autoCaller(this);
1067 AssertComRCReturnRC(autoCaller.rc());
1068
1069 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1070
1071 if (mCanceled)
1072 return E_FAIL;
1073 AssertReturn(!mCompleted, E_FAIL);
1074 AssertReturn(m_ulCurrentOperation + 1 < m_cOperations, E_FAIL);
1075
1076 ++m_ulCurrentOperation;
1077 m_ulOperationsCompletedWeight += m_ulCurrentOperationWeight;
1078
1079 m_bstrOperationDescription = bstrNextOperationDescription;
1080 m_ulCurrentOperationWeight = ulNextOperationsWeight;
1081 m_ulOperationPercent = 0;
1082
1083 Log(("Progress::setNextOperation(%ls): ulNextOperationsWeight = %d; m_ulCurrentOperation is now %d, m_ulOperationsCompletedWeight is now %d\n",
1084 m_bstrOperationDescription.raw(), ulNextOperationsWeight, m_ulCurrentOperation, m_ulOperationsCompletedWeight));
1085
1086 /* wake up all waiting threads */
1087 if (mWaitersCount > 0)
1088 RTSemEventMultiSignal(mCompletedSem);
1089
1090 return S_OK;
1091}
1092
1093// public methods only for internal purposes
1094/////////////////////////////////////////////////////////////////////////////
1095
1096/**
1097 * Sets the internal result code and attempts to retrieve additional error
1098 * info from the current thread. Gets called from Progress::notifyComplete(),
1099 * but can be called again to override a previous result set with
1100 * notifyComplete().
1101 *
1102 * @param aResultCode
1103 */
1104HRESULT Progress::setResultCode(HRESULT aResultCode)
1105{
1106 AutoCaller autoCaller(this);
1107 AssertComRCReturnRC(autoCaller.rc());
1108
1109 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1110
1111 mResultCode = aResultCode;
1112
1113 HRESULT rc = S_OK;
1114
1115 if (FAILED(aResultCode))
1116 {
1117 /* try to import error info from the current thread */
1118
1119#if !defined(VBOX_WITH_XPCOM)
1120
1121 ComPtr<IErrorInfo> err;
1122 rc = ::GetErrorInfo(0, err.asOutParam());
1123 if (rc == S_OK && err)
1124 {
1125 rc = err.queryInterfaceTo(mErrorInfo.asOutParam());
1126 if (SUCCEEDED(rc) && !mErrorInfo)
1127 rc = E_FAIL;
1128 }
1129
1130#else /* !defined(VBOX_WITH_XPCOM) */
1131
1132 nsCOMPtr<nsIExceptionService> es;
1133 es = do_GetService(NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
1134 if (NS_SUCCEEDED(rc))
1135 {
1136 nsCOMPtr <nsIExceptionManager> em;
1137 rc = es->GetCurrentExceptionManager(getter_AddRefs(em));
1138 if (NS_SUCCEEDED(rc))
1139 {
1140 ComPtr<nsIException> ex;
1141 rc = em->GetCurrentException(ex.asOutParam());
1142 if (NS_SUCCEEDED(rc) && ex)
1143 {
1144 rc = ex.queryInterfaceTo(mErrorInfo.asOutParam());
1145 if (NS_SUCCEEDED(rc) && !mErrorInfo)
1146 rc = E_FAIL;
1147 }
1148 }
1149 }
1150#endif /* !defined(VBOX_WITH_XPCOM) */
1151
1152 AssertMsg(rc == S_OK, ("Couldn't get error info (rc=%08X) while trying to set a failed result (%08X)!\n",
1153 rc, aResultCode));
1154 }
1155
1156 return rc;
1157}
1158
1159/**
1160 * Marks the whole task as complete and sets the result code.
1161 *
1162 * If the result code indicates a failure (|FAILED(@a aResultCode)|) then this
1163 * method will import the error info from the current thread and assign it to
1164 * the errorInfo attribute (it will return an error if no info is available in
1165 * such case).
1166 *
1167 * If the result code indicates a success (|SUCCEEDED(@a aResultCode)|) then
1168 * the current operation is set to the last.
1169 *
1170 * Note that this method may be called only once for the given Progress object.
1171 * Subsequent calls will assert.
1172 *
1173 * @param aResultCode Operation result code.
1174 */
1175HRESULT Progress::notifyComplete(HRESULT aResultCode)
1176{
1177 AutoCaller autoCaller(this);
1178 AssertComRCReturnRC(autoCaller.rc());
1179
1180 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1181
1182 AssertReturn(mCompleted == FALSE, E_FAIL);
1183
1184 LogFunc(("aResultCode=%d\n", aResultCode));
1185
1186 if (mCanceled && SUCCEEDED(aResultCode))
1187 aResultCode = E_FAIL;
1188
1189 HRESULT rc = setResultCode(aResultCode);
1190
1191 mCompleted = TRUE;
1192
1193 if (!FAILED(aResultCode))
1194 {
1195 m_ulCurrentOperation = m_cOperations - 1; /* last operation */
1196 m_ulOperationPercent = 100;
1197 }
1198
1199#if !defined VBOX_COM_INPROC
1200 /* remove from the global collection of pending progress operations */
1201 if (mParent)
1202 mParent->removeProgress(mId.ref());
1203#endif
1204
1205 /* wake up all waiting threads */
1206 if (mWaitersCount > 0)
1207 RTSemEventMultiSignal(mCompletedSem);
1208
1209 return rc;
1210}
1211
1212/**
1213 * Wrapper around Progress:notifyCompleteV.
1214 */
1215HRESULT Progress::notifyComplete(HRESULT aResultCode,
1216 const GUID &aIID,
1217 const char *pcszComponent,
1218 const char *aText,
1219 ...)
1220{
1221 va_list va;
1222 va_start(va, aText);
1223 HRESULT hrc = notifyCompleteV(aResultCode, aIID, pcszComponent, aText, va);
1224 va_end(va);
1225 return hrc;
1226}
1227
1228/**
1229 * Marks the operation as complete and attaches full error info.
1230 *
1231 * See com::SupportErrorInfoImpl::setError(HRESULT, const GUID &, const wchar_t
1232 * *, const char *, ...) for more info.
1233 *
1234 * @param aResultCode Operation result (error) code, must not be S_OK.
1235 * @param aIID IID of the interface that defines the error.
1236 * @param aComponent Name of the component that generates the error.
1237 * @param aText Error message (must not be null), an RTStrPrintf-like
1238 * format string in UTF-8 encoding.
1239 * @param va List of arguments for the format string.
1240 */
1241HRESULT Progress::notifyCompleteV(HRESULT aResultCode,
1242 const GUID &aIID,
1243 const char *pcszComponent,
1244 const char *aText,
1245 va_list va)
1246{
1247 Utf8Str text(aText, va);
1248
1249 AutoCaller autoCaller(this);
1250 AssertComRCReturnRC(autoCaller.rc());
1251
1252 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1253
1254 AssertReturn(mCompleted == FALSE, E_FAIL);
1255
1256 if (mCanceled && SUCCEEDED(aResultCode))
1257 aResultCode = E_FAIL;
1258
1259 mCompleted = TRUE;
1260 mResultCode = aResultCode;
1261
1262 AssertReturn(FAILED(aResultCode), E_FAIL);
1263
1264 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
1265 HRESULT rc = errorInfo.createObject();
1266 AssertComRC(rc);
1267 if (SUCCEEDED(rc))
1268 {
1269 errorInfo->init(aResultCode, aIID, pcszComponent, text);
1270 errorInfo.queryInterfaceTo(mErrorInfo.asOutParam());
1271 }
1272
1273#if !defined VBOX_COM_INPROC
1274 /* remove from the global collection of pending progress operations */
1275 if (mParent)
1276 mParent->removeProgress(mId.ref());
1277#endif
1278
1279 /* wake up all waiting threads */
1280 if (mWaitersCount > 0)
1281 RTSemEventMultiSignal(mCompletedSem);
1282
1283 return rc;
1284}
1285
1286/**
1287 * Notify the progress object that we're almost at the point of no return.
1288 *
1289 * This atomically checks for and disables cancelation. Calls to
1290 * IProgress::Cancel() made after a successful call to this method will fail
1291 * and the user can be told. While this isn't entirely clean behavior, it
1292 * prevents issues with an irreversible actually operation succeeding while the
1293 * user believe it was rolled back.
1294 *
1295 * @returns Success indicator.
1296 * @retval true on success.
1297 * @retval false if the progress object has already been canceled or is in an
1298 * invalid state
1299 */
1300bool Progress::notifyPointOfNoReturn(void)
1301{
1302 AutoCaller autoCaller(this);
1303 AssertComRCReturn(autoCaller.rc(), false);
1304
1305 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1306
1307 if (mCanceled)
1308 {
1309 LogThisFunc(("returns false\n"));
1310 return false;
1311 }
1312
1313 mCancelable = FALSE;
1314 LogThisFunc(("returns true\n"));
1315 return true;
1316}
1317
1318////////////////////////////////////////////////////////////////////////////////
1319// CombinedProgress class
1320////////////////////////////////////////////////////////////////////////////////
1321
1322HRESULT CombinedProgress::FinalConstruct()
1323{
1324 HRESULT rc = ProgressBase::FinalConstruct();
1325 if (FAILED(rc)) return rc;
1326
1327 mProgress = 0;
1328 mCompletedOperations = 0;
1329
1330 return BaseFinalConstruct();
1331}
1332
1333void CombinedProgress::FinalRelease()
1334{
1335 uninit();
1336 BaseFinalRelease();
1337}
1338
1339// public initializer/uninitializer for internal purposes only
1340////////////////////////////////////////////////////////////////////////////////
1341
1342/**
1343 * Initializes this object based on individual combined progresses.
1344 * Must be called only from #init()!
1345 *
1346 * @param aAutoInitSpan AutoInitSpan object instantiated by a subclass.
1347 * @param aParent See ProgressBase::init().
1348 * @param aInitiator See ProgressBase::init().
1349 * @param aDescription See ProgressBase::init().
1350 * @param aId See ProgressBase::init().
1351 */
1352HRESULT CombinedProgress::protectedInit(AutoInitSpan &aAutoInitSpan,
1353#if !defined(VBOX_COM_INPROC)
1354 VirtualBox *aParent,
1355#endif
1356 IUnknown *aInitiator,
1357 CBSTR aDescription, OUT_GUID aId)
1358{
1359 LogFlowThisFunc(("aDescription={%ls} mProgresses.size()=%d\n",
1360 aDescription, mProgresses.size()));
1361
1362 HRESULT rc = S_OK;
1363
1364 rc = ProgressBase::protectedInit(aAutoInitSpan,
1365#if !defined(VBOX_COM_INPROC)
1366 aParent,
1367#endif
1368 aInitiator, aDescription, aId);
1369 if (FAILED(rc)) return rc;
1370
1371 mProgress = 0; /* the first object */
1372 mCompletedOperations = 0;
1373
1374 mCompleted = FALSE;
1375 mCancelable = TRUE; /* until any progress returns FALSE */
1376 mCanceled = FALSE;
1377
1378 m_cOperations = 0; /* will be calculated later */
1379
1380 m_ulCurrentOperation = 0;
1381 rc = mProgresses[0]->COMGETTER(OperationDescription)(m_bstrOperationDescription.asOutParam());
1382 if (FAILED(rc)) return rc;
1383
1384 for (size_t i = 0; i < mProgresses.size(); i ++)
1385 {
1386 if (mCancelable)
1387 {
1388 BOOL cancelable = FALSE;
1389 rc = mProgresses[i]->COMGETTER(Cancelable)(&cancelable);
1390 if (FAILED(rc)) return rc;
1391
1392 if (!cancelable)
1393 mCancelable = FALSE;
1394 }
1395
1396 {
1397 ULONG opCount = 0;
1398 rc = mProgresses[i]->COMGETTER(OperationCount)(&opCount);
1399 if (FAILED(rc)) return rc;
1400
1401 m_cOperations += opCount;
1402 }
1403 }
1404
1405 rc = checkProgress();
1406 if (FAILED(rc)) return rc;
1407
1408 return rc;
1409}
1410
1411/**
1412 * Initializes the combined progress object given two normal progress
1413 * objects.
1414 *
1415 * @param aParent See ProgressBase::init().
1416 * @param aInitiator See ProgressBase::init().
1417 * @param aDescription See ProgressBase::init().
1418 * @param aProgress1 First normal progress object.
1419 * @param aProgress2 Second normal progress object.
1420 * @param aId See ProgressBase::init().
1421 */
1422HRESULT CombinedProgress::init(
1423#if !defined(VBOX_COM_INPROC)
1424 VirtualBox *aParent,
1425#endif
1426 IUnknown *aInitiator,
1427 CBSTR aDescription,
1428 IProgress *aProgress1,
1429 IProgress *aProgress2,
1430 OUT_GUID aId /* = NULL */)
1431{
1432 /* Enclose the state transition NotReady->InInit->Ready */
1433 AutoInitSpan autoInitSpan(this);
1434 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1435
1436 mProgresses.resize(2);
1437 mProgresses[0] = aProgress1;
1438 mProgresses[1] = aProgress2;
1439
1440 HRESULT rc = protectedInit(autoInitSpan,
1441#if !defined(VBOX_COM_INPROC)
1442 aParent,
1443#endif
1444 aInitiator,
1445 aDescription,
1446 aId);
1447
1448 /* Confirm a successful initialization when it's the case */
1449 if (SUCCEEDED(rc))
1450 autoInitSpan.setSucceeded();
1451
1452 return rc;
1453}
1454
1455/**
1456 * Uninitializes the instance and sets the ready flag to FALSE.
1457 *
1458 * Called either from FinalRelease() or by the parent when it gets destroyed.
1459 */
1460void CombinedProgress::uninit()
1461{
1462 LogFlowThisFunc(("\n"));
1463
1464 /* Enclose the state transition Ready->InUninit->NotReady */
1465 AutoUninitSpan autoUninitSpan(this);
1466 if (autoUninitSpan.uninitDone())
1467 return;
1468
1469 mProgress = 0;
1470 mProgresses.clear();
1471
1472 ProgressBase::protectedUninit(autoUninitSpan);
1473}
1474
1475// IProgress properties
1476////////////////////////////////////////////////////////////////////////////////
1477
1478STDMETHODIMP CombinedProgress::COMGETTER(Percent)(ULONG *aPercent)
1479{
1480 CheckComArgOutPointerValid(aPercent);
1481
1482 AutoCaller autoCaller(this);
1483 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1484
1485 /* checkProgress needs a write lock */
1486 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1487
1488 if (mCompleted && SUCCEEDED(mResultCode))
1489 *aPercent = 100;
1490 else
1491 {
1492 HRESULT rc = checkProgress();
1493 if (FAILED(rc)) return rc;
1494
1495 /* global percent =
1496 * (100 / m_cOperations) * mOperation +
1497 * ((100 / m_cOperations) / 100) * m_ulOperationPercent */
1498 *aPercent = (100 * m_ulCurrentOperation + m_ulOperationPercent) / m_cOperations;
1499 }
1500
1501 return S_OK;
1502}
1503
1504STDMETHODIMP CombinedProgress::COMGETTER(Completed)(BOOL *aCompleted)
1505{
1506 CheckComArgOutPointerValid(aCompleted);
1507
1508 AutoCaller autoCaller(this);
1509 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1510
1511 /* checkProgress needs a write lock */
1512 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1513
1514 HRESULT rc = checkProgress();
1515 if (FAILED(rc)) return rc;
1516
1517 return ProgressBase::COMGETTER(Completed)(aCompleted);
1518}
1519
1520STDMETHODIMP CombinedProgress::COMGETTER(Canceled)(BOOL *aCanceled)
1521{
1522 CheckComArgOutPointerValid(aCanceled);
1523
1524 AutoCaller autoCaller(this);
1525 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1526
1527 /* checkProgress needs a write lock */
1528 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1529
1530 HRESULT rc = checkProgress();
1531 if (FAILED(rc)) return rc;
1532
1533 return ProgressBase::COMGETTER(Canceled)(aCanceled);
1534}
1535
1536STDMETHODIMP CombinedProgress::COMGETTER(ResultCode)(LONG *aResultCode)
1537{
1538 CheckComArgOutPointerValid(aResultCode);
1539
1540 AutoCaller autoCaller(this);
1541 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1542
1543 /* checkProgress needs a write lock */
1544 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1545
1546 HRESULT rc = checkProgress();
1547 if (FAILED(rc)) return rc;
1548
1549 return ProgressBase::COMGETTER(ResultCode)(aResultCode);
1550}
1551
1552STDMETHODIMP CombinedProgress::COMGETTER(ErrorInfo)(IVirtualBoxErrorInfo **aErrorInfo)
1553{
1554 CheckComArgOutPointerValid(aErrorInfo);
1555
1556 AutoCaller autoCaller(this);
1557 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1558
1559 /* checkProgress needs a write lock */
1560 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1561
1562 HRESULT rc = checkProgress();
1563 if (FAILED(rc)) return rc;
1564
1565 return ProgressBase::COMGETTER(ErrorInfo)(aErrorInfo);
1566}
1567
1568STDMETHODIMP CombinedProgress::COMGETTER(Operation)(ULONG *aOperation)
1569{
1570 CheckComArgOutPointerValid(aOperation);
1571
1572 AutoCaller autoCaller(this);
1573 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1574
1575 /* checkProgress needs a write lock */
1576 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1577
1578 HRESULT rc = checkProgress();
1579 if (FAILED(rc)) return rc;
1580
1581 return ProgressBase::COMGETTER(Operation)(aOperation);
1582}
1583
1584STDMETHODIMP CombinedProgress::COMGETTER(OperationDescription)(BSTR *aOperationDescription)
1585{
1586 CheckComArgOutPointerValid(aOperationDescription);
1587
1588 AutoCaller autoCaller(this);
1589 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1590
1591 /* checkProgress needs a write lock */
1592 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1593
1594 HRESULT rc = checkProgress();
1595 if (FAILED(rc)) return rc;
1596
1597 return ProgressBase::COMGETTER(OperationDescription)(aOperationDescription);
1598}
1599
1600STDMETHODIMP CombinedProgress::COMGETTER(OperationPercent)(ULONG *aOperationPercent)
1601{
1602 CheckComArgOutPointerValid(aOperationPercent);
1603
1604 AutoCaller autoCaller(this);
1605 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1606
1607 /* checkProgress needs a write lock */
1608 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1609
1610 HRESULT rc = checkProgress();
1611 if (FAILED(rc)) return rc;
1612
1613 return ProgressBase::COMGETTER(OperationPercent)(aOperationPercent);
1614}
1615
1616STDMETHODIMP CombinedProgress::COMSETTER(Timeout)(ULONG aTimeout)
1617{
1618 NOREF(aTimeout);
1619 AssertFailed();
1620 return E_NOTIMPL;
1621}
1622
1623STDMETHODIMP CombinedProgress::COMGETTER(Timeout)(ULONG *aTimeout)
1624{
1625 CheckComArgOutPointerValid(aTimeout);
1626
1627 AssertFailed();
1628 return E_NOTIMPL;
1629}
1630
1631// IProgress methods
1632/////////////////////////////////////////////////////////////////////////////
1633
1634/**
1635 * @note XPCOM: when this method is called not on the main XPCOM thread, it
1636 * simply blocks the thread until mCompletedSem is signalled. If the
1637 * thread has its own event queue (hmm, what for?) that it must run, then
1638 * calling this method will definitely freeze event processing.
1639 */
1640STDMETHODIMP CombinedProgress::WaitForCompletion(LONG aTimeout)
1641{
1642 LogFlowThisFuncEnter();
1643 LogFlowThisFunc(("aTtimeout=%d\n", aTimeout));
1644
1645 AutoCaller autoCaller(this);
1646 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1647
1648 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1649
1650 /* if we're already completed, take a shortcut */
1651 if (!mCompleted)
1652 {
1653 HRESULT rc = S_OK;
1654 bool forever = aTimeout < 0;
1655 int64_t timeLeft = aTimeout;
1656 int64_t lastTime = RTTimeMilliTS();
1657
1658 while (!mCompleted && (forever || timeLeft > 0))
1659 {
1660 alock.leave();
1661 rc = mProgresses.back()->WaitForCompletion(forever ? -1 : (LONG) timeLeft);
1662 alock.enter();
1663
1664 if (SUCCEEDED(rc))
1665 rc = checkProgress();
1666
1667 if (FAILED(rc)) break;
1668
1669 if (!forever)
1670 {
1671 int64_t now = RTTimeMilliTS();
1672 timeLeft -= now - lastTime;
1673 lastTime = now;
1674 }
1675 }
1676
1677 if (FAILED(rc)) return rc;
1678 }
1679
1680 LogFlowThisFuncLeave();
1681
1682 return S_OK;
1683}
1684
1685/**
1686 * @note XPCOM: when this method is called not on the main XPCOM thread, it
1687 * simply blocks the thread until mCompletedSem is signalled. If the
1688 * thread has its own event queue (hmm, what for?) that it must run, then
1689 * calling this method will definitely freeze event processing.
1690 */
1691STDMETHODIMP CombinedProgress::WaitForOperationCompletion(ULONG aOperation, LONG aTimeout)
1692{
1693 LogFlowThisFuncEnter();
1694 LogFlowThisFunc(("aOperation=%d, aTimeout=%d\n", aOperation, aTimeout));
1695
1696 AutoCaller autoCaller(this);
1697 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1698
1699 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1700
1701 if (aOperation >= m_cOperations)
1702 return setError(E_FAIL,
1703 tr("Operation number must be in range [0, %d]"), m_ulCurrentOperation - 1);
1704
1705 /* if we're already completed or if the given operation is already done,
1706 * then take a shortcut */
1707 if (!mCompleted && aOperation >= m_ulCurrentOperation)
1708 {
1709 HRESULT rc = S_OK;
1710
1711 /* find the right progress object to wait for */
1712 size_t progress = mProgress;
1713 ULONG operation = 0, completedOps = mCompletedOperations;
1714 do
1715 {
1716 ULONG opCount = 0;
1717 rc = mProgresses[progress]->COMGETTER(OperationCount)(&opCount);
1718 if (FAILED(rc))
1719 return rc;
1720
1721 if (completedOps + opCount > aOperation)
1722 {
1723 /* found the right progress object */
1724 operation = aOperation - completedOps;
1725 break;
1726 }
1727
1728 completedOps += opCount;
1729 progress ++;
1730 ComAssertRet(progress < mProgresses.size(), E_FAIL);
1731 }
1732 while (1);
1733
1734 LogFlowThisFunc(("will wait for mProgresses [%d] (%d)\n",
1735 progress, operation));
1736
1737 bool forever = aTimeout < 0;
1738 int64_t timeLeft = aTimeout;
1739 int64_t lastTime = RTTimeMilliTS();
1740
1741 while (!mCompleted && aOperation >= m_ulCurrentOperation &&
1742 (forever || timeLeft > 0))
1743 {
1744 alock.leave();
1745 /* wait for the appropriate progress operation completion */
1746 rc = mProgresses[progress]-> WaitForOperationCompletion(operation,
1747 forever ? -1 : (LONG) timeLeft);
1748 alock.enter();
1749
1750 if (SUCCEEDED(rc))
1751 rc = checkProgress();
1752
1753 if (FAILED(rc)) break;
1754
1755 if (!forever)
1756 {
1757 int64_t now = RTTimeMilliTS();
1758 timeLeft -= now - lastTime;
1759 lastTime = now;
1760 }
1761 }
1762
1763 if (FAILED(rc)) return rc;
1764 }
1765
1766 LogFlowThisFuncLeave();
1767
1768 return S_OK;
1769}
1770
1771STDMETHODIMP CombinedProgress::Cancel()
1772{
1773 AutoCaller autoCaller(this);
1774 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1775
1776 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1777
1778 if (!mCancelable)
1779 return setError(E_FAIL, tr("Operation cannot be canceled"));
1780
1781 if (!mCanceled)
1782 {
1783 LogThisFunc(("Canceling\n"));
1784 mCanceled = TRUE;
1785/** @todo Teleportation: Shouldn't this be propagated to mProgresses? If
1786 * powerUp creates passes a combined progress object to the client, I
1787 * won't get called back since I'm only getting the powerupProgress ...
1788 * Or what? */
1789 if (m_pfnCancelCallback)
1790 m_pfnCancelCallback(m_pvCancelUserArg);
1791
1792 }
1793 else
1794 LogThisFunc(("Already canceled\n"));
1795
1796 return S_OK;
1797}
1798
1799// private methods
1800////////////////////////////////////////////////////////////////////////////////
1801
1802/**
1803 * Fetches the properties of the current progress object and, if it is
1804 * successfully completed, advances to the next uncompleted or unsuccessfully
1805 * completed object in the vector of combined progress objects.
1806 *
1807 * @note Must be called from under this object's write lock!
1808 */
1809HRESULT CombinedProgress::checkProgress()
1810{
1811 /* do nothing if we're already marked ourselves as completed */
1812 if (mCompleted)
1813 return S_OK;
1814
1815 AssertReturn(mProgress < mProgresses.size(), E_FAIL);
1816
1817 ComPtr<IProgress> progress = mProgresses[mProgress];
1818 ComAssertRet(!progress.isNull(), E_FAIL);
1819
1820 HRESULT rc = S_OK;
1821 BOOL fCompleted = FALSE;
1822
1823 do
1824 {
1825 rc = progress->COMGETTER(Completed)(&fCompleted);
1826 if (FAILED(rc))
1827 return rc;
1828
1829 if (fCompleted)
1830 {
1831 rc = progress->COMGETTER(Canceled)(&mCanceled);
1832 if (FAILED(rc))
1833 return rc;
1834
1835 LONG iRc;
1836 rc = progress->COMGETTER(ResultCode)(&iRc);
1837 if (FAILED(rc))
1838 return rc;
1839 mResultCode = iRc;
1840
1841 if (FAILED(mResultCode))
1842 {
1843 rc = progress->COMGETTER(ErrorInfo)(mErrorInfo.asOutParam());
1844 if (FAILED(rc))
1845 return rc;
1846 }
1847
1848 if (FAILED(mResultCode) || mCanceled)
1849 {
1850 mCompleted = TRUE;
1851 }
1852 else
1853 {
1854 ULONG opCount = 0;
1855 rc = progress->COMGETTER(OperationCount)(&opCount);
1856 if (FAILED(rc))
1857 return rc;
1858
1859 mCompletedOperations += opCount;
1860 mProgress ++;
1861
1862 if (mProgress < mProgresses.size())
1863 progress = mProgresses[mProgress];
1864 else
1865 mCompleted = TRUE;
1866 }
1867 }
1868 }
1869 while (fCompleted && !mCompleted);
1870
1871 rc = progress->COMGETTER(OperationPercent)(&m_ulOperationPercent);
1872 if (SUCCEEDED(rc))
1873 {
1874 ULONG operation = 0;
1875 rc = progress->COMGETTER(Operation)(&operation);
1876 if (SUCCEEDED(rc) && mCompletedOperations + operation > m_ulCurrentOperation)
1877 {
1878 m_ulCurrentOperation = mCompletedOperations + operation;
1879 rc = progress->COMGETTER(OperationDescription)(m_bstrOperationDescription.asOutParam());
1880 }
1881 }
1882
1883 return rc;
1884}
1885/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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