VirtualBox

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

Last change on this file since 74789 was 74759, checked in by vboxsync, 6 years ago

Main/Progress: clean up error handling and eliminate useless throwing

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.9 KB
Line 
1/* $Id: ProgressImpl.cpp 74759 2018-10-11 11:24:47Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox Progress COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2017 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
22#if defined(VBOX_WITH_XPCOM)
23#include <nsIServiceManager.h>
24#include <nsIExceptionService.h>
25#include <nsCOMPtr.h>
26#endif /* defined(VBOX_WITH_XPCOM) */
27
28#include "ProgressImpl.h"
29
30#if !defined(VBOX_COM_INPROC)
31# include "VirtualBoxImpl.h"
32#endif
33#include "VirtualBoxErrorInfoImpl.h"
34
35#include "Logging.h"
36
37#include <iprt/time.h>
38#include <iprt/semaphore.h>
39#include <iprt/cpp/utils.h>
40
41#include <VBox/err.h>
42#include "AutoCaller.h"
43
44#include "VBoxEvents.h"
45
46Progress::Progress()
47#if !defined(VBOX_COM_INPROC)
48 : mParent(NULL)
49#endif
50{
51}
52
53Progress::~Progress()
54{
55}
56
57
58HRESULT Progress::FinalConstruct()
59{
60 mCancelable = FALSE;
61 mCompleted = FALSE;
62 mCanceled = FALSE;
63 mResultCode = S_OK;
64
65 m_cOperations
66 = m_ulTotalOperationsWeight
67 = m_ulOperationsCompletedWeight
68 = m_ulCurrentOperation
69 = m_ulCurrentOperationWeight
70 = m_ulOperationPercent
71 = m_cMsTimeout
72 = 0;
73
74 // get creation timestamp
75 m_ullTimestamp = RTTimeMilliTS();
76
77 m_pfnCancelCallback = NULL;
78 m_pvCancelUserArg = NULL;
79
80 mCompletedSem = NIL_RTSEMEVENTMULTI;
81 mWaitersCount = 0;
82
83 return Progress::BaseFinalConstruct();
84}
85
86void Progress::FinalRelease()
87{
88 uninit();
89 BaseFinalRelease();
90}
91
92// public initializer/uninitializer for internal purposes only
93////////////////////////////////////////////////////////////////////////////////
94
95/**
96 * Initializes the normal progress object. With this variant, one can have
97 * an arbitrary number of sub-operation which IProgress can analyze to
98 * have a weighted progress computed.
99 *
100 * For example, say that one IProgress is supposed to track the cloning
101 * of two hard disk images, which are 100 MB and 1000 MB in size, respectively,
102 * and each of these hard disks should be one sub-operation of the IProgress.
103 *
104 * Obviously the progress would be misleading if the progress displayed 50%
105 * after the smaller image was cloned and would then take much longer for
106 * the second half.
107 *
108 * With weighted progress, one can invoke the following calls:
109 *
110 * 1) create progress object with cOperations = 2 and ulTotalOperationsWeight =
111 * 1100 (100 MB plus 1100, but really the weights can be any ULONG); pass
112 * in ulFirstOperationWeight = 100 for the first sub-operation
113 *
114 * 2) Then keep calling setCurrentOperationProgress() with a percentage
115 * for the first image; the total progress will increase up to a value
116 * of 9% (100MB / 1100MB * 100%).
117 *
118 * 3) Then call setNextOperation with the second weight (1000 for the megabytes
119 * of the second disk).
120 *
121 * 4) Then keep calling setCurrentOperationProgress() with a percentage for
122 * the second image, where 100% of the operation will then yield a 100%
123 * progress of the entire task.
124 *
125 * Weighting is optional; you can simply assign a weight of 1 to each operation
126 * and pass ulTotalOperationsWeight == cOperations to this constructor (but
127 * for that variant and for backwards-compatibility a simpler constructor exists
128 * in ProgressImpl.h as well).
129 *
130 * Even simpler, if you need no sub-operations at all, pass in cOperations =
131 * ulTotalOperationsWeight = ulFirstOperationWeight = 1.
132 *
133 * @param aParent Parent object (only for server-side Progress objects).
134 * @param aInitiator Initiator of the task (for server-side objects. Can be
135 * NULL which means initiator = parent, otherwise must not
136 * be NULL).
137 * @param aDescription Overall task description.
138 * @param aCancelable Flag whether the task maybe canceled.
139 * @param cOperations Number of operations within this task (at least 1).
140 * @param ulTotalOperationsWeight Total weight of operations; must be the sum of ulFirstOperationWeight and
141 * what is later passed with each subsequent setNextOperation() call.
142 * @param aFirstOperationDescription Description of the first operation.
143 * @param ulFirstOperationWeight Weight of first sub-operation.
144 */
145HRESULT Progress::init(
146#if !defined(VBOX_COM_INPROC)
147 VirtualBox *aParent,
148#endif
149 IUnknown *aInitiator,
150 const Utf8Str &aDescription,
151 BOOL aCancelable,
152 ULONG cOperations,
153 ULONG ulTotalOperationsWeight,
154 const Utf8Str &aFirstOperationDescription,
155 ULONG ulFirstOperationWeight)
156{
157 LogFlowThisFunc(("aDescription=\"%s\", cOperations=%d, ulTotalOperationsWeight=%d, aFirstOperationDescription=\"%s\", ulFirstOperationWeight=%d\n",
158 aDescription.c_str(),
159 cOperations,
160 ulTotalOperationsWeight,
161 aFirstOperationDescription.c_str(),
162 ulFirstOperationWeight));
163
164 AssertReturn(ulTotalOperationsWeight >= 1, E_INVALIDARG);
165
166 /* Enclose the state transition NotReady->InInit->Ready */
167 AutoInitSpan autoInitSpan(this);
168 AssertReturn(autoInitSpan.isOk(), E_FAIL);
169
170 HRESULT rc = unconst(pEventSource).createObject();
171 if (FAILED(rc))
172 return rc;
173
174 rc = pEventSource->init();
175 if (FAILED(rc))
176 return rc;
177
178#if !defined(VBOX_COM_INPROC)
179 AssertReturn(aParent, E_INVALIDARG);
180#else
181 AssertReturn(aInitiator, E_INVALIDARG);
182#endif
183
184#if !defined(VBOX_COM_INPROC)
185 /* share parent weakly */
186 unconst(mParent) = aParent;
187#endif
188
189#if !defined(VBOX_COM_INPROC)
190 /* assign (and therefore addref) initiator only if it is not VirtualBox
191 * (to avoid cycling); otherwise mInitiator will remain null which means
192 * that it is the same as the parent */
193 if (aInitiator)
194 {
195 ComObjPtr<VirtualBox> pVirtualBox(mParent);
196 if (!(pVirtualBox == aInitiator))
197 unconst(mInitiator) = aInitiator;
198 }
199#else
200 unconst(mInitiator) = aInitiator;
201#endif
202
203 unconst(mId).create();
204
205#if !defined(VBOX_COM_INPROC)
206 /* add to the global collection of progress operations (note: after
207 * creating mId) */
208 mParent->i_addProgress(this);
209#endif
210
211 unconst(mDescription) = aDescription;
212
213 mCancelable = aCancelable;
214
215 m_cOperations = cOperations;
216 m_ulTotalOperationsWeight = ulTotalOperationsWeight;
217 m_ulOperationsCompletedWeight = 0;
218 m_ulCurrentOperation = 0;
219 m_operationDescription = aFirstOperationDescription;
220 m_ulCurrentOperationWeight = ulFirstOperationWeight;
221 m_ulOperationPercent = 0;
222
223 int vrc = RTSemEventMultiCreate(&mCompletedSem);
224 ComAssertRCRet(vrc, E_FAIL);
225
226 RTSemEventMultiReset(mCompletedSem);
227
228 /* Confirm a successful initialization. */
229 autoInitSpan.setSucceeded();
230
231 return S_OK;
232}
233
234/**
235 * Initializes the sub-progress object that represents a specific operation of
236 * the whole task.
237 *
238 * Objects initialized with this method are then combined together into the
239 * single task using a Progress instance, so it doesn't require the
240 * parent, initiator, description and doesn't create an ID. Note that calling
241 * respective getter methods on an object initialized with this method is
242 * useless. Such objects are used only to provide a separate wait semaphore and
243 * store individual operation descriptions.
244 *
245 * @param aCancelable Flag whether the task maybe canceled.
246 * @param aOperationCount Number of sub-operations within this task (at least 1).
247 * @param aOperationDescription Description of the individual operation.
248 */
249HRESULT Progress::init(BOOL aCancelable,
250 ULONG aOperationCount,
251 const Utf8Str &aOperationDescription)
252{
253 LogFlowThisFunc(("aOperationDescription=\"%s\"\n", aOperationDescription.c_str()));
254
255 /* Enclose the state transition NotReady->InInit->Ready */
256 AutoInitSpan autoInitSpan(this);
257 AssertReturn(autoInitSpan.isOk(), E_FAIL);
258
259 mCancelable = aCancelable;
260
261 // for this variant we assume for now that all operations are weighed "1"
262 // and equal total weight = operation count
263 m_cOperations = aOperationCount;
264 m_ulTotalOperationsWeight = aOperationCount;
265 m_ulOperationsCompletedWeight = 0;
266 m_ulCurrentOperation = 0;
267 m_operationDescription = aOperationDescription;
268 m_ulCurrentOperationWeight = 1;
269 m_ulOperationPercent = 0;
270
271 int vrc = RTSemEventMultiCreate(&mCompletedSem);
272 ComAssertRCRet(vrc, E_FAIL);
273
274 RTSemEventMultiReset(mCompletedSem);
275
276 /* Confirm a successful initialization. */
277 autoInitSpan.setSucceeded();
278
279 return S_OK;
280}
281
282
283/**
284 * Uninitializes the instance and sets the ready flag to FALSE.
285 *
286 * Called either from FinalRelease() or by the parent when it gets destroyed.
287 */
288void Progress::uninit()
289{
290 LogFlowThisFunc(("\n"));
291
292 /* Enclose the state transition Ready->InUninit->NotReady */
293 AutoUninitSpan autoUninitSpan(this);
294 if (autoUninitSpan.uninitDone())
295 return;
296
297 /* wake up all threads still waiting on occasion */
298 if (mWaitersCount > 0)
299 {
300 LogFlow(("WARNING: There are still %d threads waiting for '%s' completion!\n",
301 mWaitersCount, mDescription.c_str()));
302 RTSemEventMultiSignal(mCompletedSem);
303 }
304
305 RTSemEventMultiDestroy(mCompletedSem);
306
307 /* release initiator (effective only if mInitiator has been assigned in init()) */
308 unconst(mInitiator).setNull();
309
310#if !defined(VBOX_COM_INPROC)
311 if (mParent)
312 {
313 /* remove the added progress on failure to complete the initialization */
314 if (autoUninitSpan.initFailed() && mId.isValid() && !mId.isZero())
315 mParent->i_removeProgress(mId.ref());
316
317 unconst(mParent) = NULL;
318 }
319#endif
320}
321
322
323// public methods only for internal purposes
324////////////////////////////////////////////////////////////////////////////////
325
326/**
327 * Marks the whole task as complete and sets the result code.
328 *
329 * If the result code indicates a failure (|FAILED(@a aResultCode)|) then this
330 * method will import the error info from the current thread and assign it to
331 * the errorInfo attribute (it will return an error if no info is available in
332 * such case).
333 *
334 * If the result code indicates a success (|SUCCEEDED(@a aResultCode)|) then
335 * the current operation is set to the last.
336 *
337 * Note that this method may be called only once for the given Progress object.
338 * Subsequent calls will assert.
339 *
340 * @param aResultCode Operation result code.
341 */
342HRESULT Progress::i_notifyComplete(HRESULT aResultCode)
343{
344 HRESULT rc;
345 ComPtr<IVirtualBoxErrorInfo> errorInfo;
346 if (FAILED(aResultCode))
347 {
348 /* try to import error info from the current thread */
349#if !defined(VBOX_WITH_XPCOM)
350 ComPtr<IErrorInfo> err;
351 rc = ::GetErrorInfo(0, err.asOutParam());
352 if (rc == S_OK && err)
353 rc = err.queryInterfaceTo(errorInfo.asOutParam());
354#else /* !defined(VBOX_WITH_XPCOM) */
355 nsCOMPtr<nsIExceptionService> es;
356 es = do_GetService(NS_EXCEPTIONSERVICE_CONTRACTID, &rc);
357 if (NS_SUCCEEDED(rc))
358 {
359 nsCOMPtr <nsIExceptionManager> em;
360 rc = es->GetCurrentExceptionManager(getter_AddRefs(em));
361 if (NS_SUCCEEDED(rc))
362 {
363 ComPtr<nsIException> ex;
364 rc = em->GetCurrentException(ex.asOutParam());
365 if (NS_SUCCEEDED(rc) && ex)
366 rc = ex.queryInterfaceTo(errorInfo.asOutParam());
367 }
368 }
369#endif /* !defined(VBOX_WITH_XPCOM) */
370 }
371
372 return i_notifyCompleteEI(aResultCode, errorInfo);
373}
374
375/**
376 * Wrapper around Progress:notifyCompleteV.
377 */
378HRESULT Progress::i_notifyComplete(HRESULT aResultCode,
379 const GUID &aIID,
380 const char *pcszComponent,
381 const char *aText,
382 ...)
383{
384 va_list va;
385 va_start(va, aText);
386 HRESULT hrc = i_notifyCompleteV(aResultCode, aIID, pcszComponent, aText, va);
387 va_end(va);
388 return hrc;
389}
390
391/**
392 * Marks the operation as complete and attaches full error info.
393 *
394 * @param aResultCode Operation result (error) code, must not be S_OK.
395 * @param aIID IID of the interface that defines the error.
396 * @param pcszComponent Name of the component that generates the error.
397 * @param aText Error message (must not be null), an RTStrPrintf-like
398 * format string in UTF-8 encoding.
399 * @param va List of arguments for the format string.
400 */
401HRESULT Progress::i_notifyCompleteV(HRESULT aResultCode,
402 const GUID &aIID,
403 const char *pcszComponent,
404 const char *aText,
405 va_list va)
406{
407 /* expected to be used only in case of error */
408 Assert(FAILED(aResultCode));
409
410 Utf8Str text(aText, va);
411 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
412 HRESULT rc = errorInfo.createObject();
413 AssertComRCReturnRC(rc);
414 errorInfo->init(aResultCode, aIID, pcszComponent, text);
415
416 return i_notifyCompleteEI(aResultCode, errorInfo);
417}
418
419/**
420 * Marks the operation as complete and attaches full error info.
421 *
422 * This is where the actual work is done, the related methods all end up here.
423 *
424 * @param aResultCode Operation result (error) code, must not be S_OK.
425 * @param aErrorInfo List of arguments for the format string.
426 */
427HRESULT Progress::i_notifyCompleteEI(HRESULT aResultCode, const ComPtr<IVirtualBoxErrorInfo> &aErrorInfo)
428{
429 LogThisFunc(("aResultCode=%d\n", aResultCode));
430 /* on failure we expect error info, on success there must be none */
431 AssertMsg(FAILED(aResultCode) ^ aErrorInfo.isNull(),
432 ("No error info but trying to set a failed result (%08X)!\n",
433 aResultCode));
434
435 AutoCaller autoCaller(this);
436 AssertComRCReturnRC(autoCaller.rc());
437
438 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
439
440 AssertReturn(mCompleted == FALSE, E_FAIL);
441
442 if (mCanceled && SUCCEEDED(aResultCode))
443 aResultCode = E_FAIL;
444
445 mCompleted = TRUE;
446 mResultCode = aResultCode;
447 if (SUCCEEDED(aResultCode))
448 {
449 m_ulCurrentOperation = m_cOperations - 1; /* last operation */
450 m_ulOperationPercent = 100;
451 }
452 mErrorInfo = aErrorInfo;
453
454#if !defined(VBOX_COM_INPROC)
455 /* remove from the global collection of pending progress operations */
456 if (mParent)
457 mParent->i_removeProgress(mId.ref());
458#endif
459
460 /* wake up all waiting threads */
461 if (mWaitersCount > 0)
462 RTSemEventMultiSignal(mCompletedSem);
463
464 fireProgressTaskCompletedEvent(pEventSource, mId.toUtf16().raw());
465
466 return S_OK;
467}
468
469HRESULT Progress::i_waitForOtherProgressCompletion(const ComPtr<IProgress> &aProgressOther)
470{
471 LogFlowThisFuncEnter();
472
473 /* Note: no locking needed, because we just use public methods. */
474
475 HRESULT rc = S_OK;
476 BOOL fCancelable = FALSE;
477 BOOL fCompleted = FALSE;
478 BOOL fCanceled = FALSE;
479 ULONG prevPercent = UINT32_MAX;
480 ULONG currentPercent = 0;
481 ULONG cOp = 0;
482 /* Is the async process cancelable? */
483 rc = aProgressOther->COMGETTER(Cancelable)(&fCancelable);
484 if (FAILED(rc)) return rc;
485 /* Loop as long as the sync process isn't completed. */
486 while (SUCCEEDED(aProgressOther->COMGETTER(Completed(&fCompleted))))
487 {
488 /* We can forward any cancel request to the async process only when
489 * it is cancelable. */
490 if (fCancelable)
491 {
492 rc = COMGETTER(Canceled)(&fCanceled);
493 if (FAILED(rc)) return rc;
494 if (fCanceled)
495 {
496 rc = aProgressOther->Cancel();
497 if (FAILED(rc)) return rc;
498 }
499 }
500 /* Even if the user canceled the process, we have to wait until the
501 async task has finished his work (cleanup and such). Otherwise there
502 will be sync trouble (still wrong state, dead locks, ...) on the
503 used objects. So just do nothing, but wait for the complete
504 notification. */
505 if (!fCanceled)
506 {
507 /* Check if the current operation has changed. It is also possible that
508 * in the meantime more than one async operation was finished. So we
509 * have to loop as long as we reached the same operation count. */
510 ULONG curOp;
511 for (;;)
512 {
513 rc = aProgressOther->COMGETTER(Operation(&curOp));
514 if (FAILED(rc)) return rc;
515 if (cOp != curOp)
516 {
517 Bstr bstr;
518 ULONG currentWeight;
519 rc = aProgressOther->COMGETTER(OperationDescription(bstr.asOutParam()));
520 if (FAILED(rc)) return rc;
521 rc = aProgressOther->COMGETTER(OperationWeight(&currentWeight));
522 if (FAILED(rc)) return rc;
523 rc = SetNextOperation(bstr.raw(), currentWeight);
524 if (FAILED(rc)) return rc;
525 ++cOp;
526 }
527 else
528 break;
529 }
530
531 rc = aProgressOther->COMGETTER(OperationPercent(&currentPercent));
532 if (FAILED(rc)) return rc;
533 if (currentPercent != prevPercent)
534 {
535 prevPercent = currentPercent;
536 rc = SetCurrentOperationProgress(currentPercent);
537 if (FAILED(rc)) return rc;
538 }
539 }
540 if (fCompleted)
541 break;
542
543 /* Make sure the loop is not too tight */
544 rc = aProgressOther->WaitForCompletion(100);
545 if (FAILED(rc)) return rc;
546 }
547
548 /* Transfer error information if applicable and report the error status
549 * back to the caller to make this as easy as possible. */
550 LONG iRc;
551 rc = aProgressOther->COMGETTER(ResultCode)(&iRc);
552 if (FAILED(rc)) return rc;
553 if (FAILED(iRc))
554 {
555 setError(ProgressErrorInfo(aProgressOther));
556 rc = iRc;
557 }
558
559 LogFlowThisFuncLeave();
560 return rc;
561}
562
563/**
564 * Notify the progress object that we're almost at the point of no return.
565 *
566 * This atomically checks for and disables cancelation. Calls to
567 * IProgress::Cancel() made after a successful call to this method will fail
568 * and the user can be told. While this isn't entirely clean behavior, it
569 * prevents issues with an irreversible actually operation succeeding while the
570 * user believe it was rolled back.
571 *
572 * @returns Success indicator.
573 * @retval true on success.
574 * @retval false if the progress object has already been canceled or is in an
575 * invalid state
576 */
577bool Progress::i_notifyPointOfNoReturn(void)
578{
579 AutoCaller autoCaller(this);
580 AssertComRCReturn(autoCaller.rc(), false);
581
582 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
583
584 if (mCanceled)
585 {
586 LogThisFunc(("returns false\n"));
587 return false;
588 }
589
590 mCancelable = FALSE;
591 LogThisFunc(("returns true\n"));
592 return true;
593}
594
595/**
596 * Sets the cancelation callback, checking for cancelation first.
597 *
598 * @returns Success indicator.
599 * @retval true on success.
600 * @retval false if the progress object has already been canceled or is in an
601 * invalid state
602 *
603 * @param pfnCallback The function to be called upon cancelation.
604 * @param pvUser The callback argument.
605 */
606bool Progress::i_setCancelCallback(void (*pfnCallback)(void *), void *pvUser)
607{
608 AutoCaller autoCaller(this);
609 AssertComRCReturn(autoCaller.rc(), false);
610
611 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
612
613 i_checkForAutomaticTimeout();
614 if (mCanceled)
615 return false;
616
617 m_pvCancelUserArg = pvUser;
618 m_pfnCancelCallback = pfnCallback;
619 return true;
620}
621
622/**
623 * @callback_method_impl{FNRTPROGRESS,
624 * Works the progress of the current operation.}
625 */
626/*static*/ DECLCALLBACK(int) Progress::i_iprtProgressCallback(unsigned uPercentage, void *pvUser)
627{
628 Progress *pThis = (Progress *)pvUser;
629
630 /*
631 * Same as setCurrentOperationProgress, except we don't fail on mCompleted.
632 */
633 AutoWriteLock alock(pThis COMMA_LOCKVAL_SRC_POS);
634 int vrc = VINF_SUCCESS;
635 if (!pThis->mCompleted)
636 {
637 pThis->i_checkForAutomaticTimeout();
638 if (!pThis->mCanceled)
639 {
640 if (uPercentage > pThis->m_ulOperationPercent)
641 pThis->setCurrentOperationProgress(uPercentage);
642 }
643 else
644 {
645 Assert(pThis->mCancelable);
646 vrc = VERR_CANCELLED;
647 }
648 }
649 /* else ignored */
650 return vrc;
651}
652
653/**
654 * @callback_method_impl{FNVDPROGRESS,
655 * Progress::i_iprtProgressCallback with parameters switched around.}
656 */
657/*static*/ DECLCALLBACK(int) Progress::i_vdProgressCallback(void *pvUser, unsigned uPercentage)
658{
659 return i_iprtProgressCallback(uPercentage, pvUser);
660}
661
662// IProgress properties
663/////////////////////////////////////////////////////////////////////////////
664
665HRESULT Progress::getId(com::Guid &aId)
666{
667 /* mId is constant during life time, no need to lock */
668 aId = mId;
669
670 return S_OK;
671}
672
673HRESULT Progress::getDescription(com::Utf8Str &aDescription)
674{
675 /* mDescription is constant during life time, no need to lock */
676 aDescription = mDescription;
677
678 return S_OK;
679}
680HRESULT Progress::getInitiator(ComPtr<IUnknown> &aInitiator)
681{
682 /* mInitiator/mParent are constant during life time, no need to lock */
683#if !defined(VBOX_COM_INPROC)
684 if (mInitiator)
685 mInitiator.queryInterfaceTo(aInitiator.asOutParam());
686 else
687 {
688 ComObjPtr<VirtualBox> pVirtualBox(mParent);
689 pVirtualBox.queryInterfaceTo(aInitiator.asOutParam());
690 }
691#else
692 mInitiator.queryInterfaceTo(aInitiator.asOutParam());
693#endif
694
695 return S_OK;
696}
697
698HRESULT Progress::getCancelable(BOOL *aCancelable)
699{
700 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
701
702 *aCancelable = mCancelable;
703
704 return S_OK;
705}
706
707HRESULT Progress::getPercent(ULONG *aPercent)
708{
709 /* i_checkForAutomaticTimeout requires a write lock. */
710 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
711
712 if (mCompleted && SUCCEEDED(mResultCode))
713 *aPercent = 100;
714 else
715 {
716 ULONG ulPercent = (ULONG)i_calcTotalPercent();
717 // do not report 100% until we're really really done with everything
718 // as the Qt GUI dismisses progress dialogs in that case
719 if ( ulPercent == 100
720 && ( m_ulOperationPercent < 100
721 || (m_ulCurrentOperation < m_cOperations -1)
722 )
723 )
724 *aPercent = 99;
725 else
726 *aPercent = ulPercent;
727 }
728
729 i_checkForAutomaticTimeout();
730
731 return S_OK;
732}
733
734HRESULT Progress::getTimeRemaining(LONG *aTimeRemaining)
735{
736 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
737
738 if (mCompleted)
739 *aTimeRemaining = 0;
740 else
741 {
742 double dPercentDone = i_calcTotalPercent();
743 if (dPercentDone < 1)
744 *aTimeRemaining = -1; // unreliable, or avoid division by 0 below
745 else
746 {
747 uint64_t ullTimeNow = RTTimeMilliTS();
748 uint64_t ullTimeElapsed = ullTimeNow - m_ullTimestamp;
749 uint64_t ullTimeTotal = (uint64_t)((double)ullTimeElapsed * 100 / dPercentDone);
750 uint64_t ullTimeRemaining = ullTimeTotal - ullTimeElapsed;
751
752// LogFunc(("dPercentDone = %RI32, ullTimeNow = %RI64, ullTimeElapsed = %RI64, ullTimeTotal = %RI64, ullTimeRemaining = %RI64\n",
753// (uint32_t)dPercentDone, ullTimeNow, ullTimeElapsed, ullTimeTotal, ullTimeRemaining));
754
755 *aTimeRemaining = (LONG)(ullTimeRemaining / 1000);
756 }
757 }
758
759 return S_OK;
760}
761
762HRESULT Progress::getCompleted(BOOL *aCompleted)
763{
764 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
765
766 *aCompleted = mCompleted;
767
768 return S_OK;
769}
770
771HRESULT Progress::getCanceled(BOOL *aCanceled)
772{
773 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
774
775 *aCanceled = mCanceled;
776
777 return S_OK;
778}
779
780HRESULT Progress::getResultCode(LONG *aResultCode)
781{
782 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
783
784 if (!mCompleted)
785 return setError(E_FAIL, tr("Result code is not available, operation is still in progress"));
786
787 *aResultCode = mResultCode;
788
789 return S_OK;
790}
791
792HRESULT Progress::getErrorInfo(ComPtr<IVirtualBoxErrorInfo> &aErrorInfo)
793{
794 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
795
796 if (!mCompleted)
797 return setError(E_FAIL, tr("Error info is not available, operation is still in progress"));
798
799 mErrorInfo.queryInterfaceTo(aErrorInfo.asOutParam());
800
801 return S_OK;
802}
803
804HRESULT Progress::getOperationCount(ULONG *aOperationCount)
805{
806 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
807
808 *aOperationCount = m_cOperations;
809
810 return S_OK;
811}
812
813HRESULT Progress::getOperation(ULONG *aOperation)
814{
815 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
816
817 *aOperation = m_ulCurrentOperation;
818
819 return S_OK;
820}
821
822HRESULT Progress::getOperationDescription(com::Utf8Str &aOperationDescription)
823{
824 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
825
826 aOperationDescription = m_operationDescription;
827
828 return S_OK;
829}
830
831HRESULT Progress::getOperationPercent(ULONG *aOperationPercent)
832{
833 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
834
835 if (mCompleted && SUCCEEDED(mResultCode))
836 *aOperationPercent = 100;
837 else
838 *aOperationPercent = m_ulOperationPercent;
839
840 return S_OK;
841}
842
843HRESULT Progress::getOperationWeight(ULONG *aOperationWeight)
844{
845 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
846
847 *aOperationWeight = m_ulCurrentOperationWeight;
848
849 return S_OK;
850}
851
852HRESULT Progress::getTimeout(ULONG *aTimeout)
853{
854 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
855
856 *aTimeout = m_cMsTimeout;
857
858 return S_OK;
859}
860
861HRESULT Progress::setTimeout(ULONG aTimeout)
862{
863 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
864
865 if (!mCancelable)
866 return setError(VBOX_E_INVALID_OBJECT_STATE, tr("Operation cannot be canceled"));
867 m_cMsTimeout = aTimeout;
868
869 return S_OK;
870}
871
872
873// IProgress methods
874/////////////////////////////////////////////////////////////////////////////
875
876/**
877 * Updates the percentage value of the current operation.
878 *
879 * @param aPercent New percentage value of the operation in progress
880 * (in range [0, 100]).
881 */
882HRESULT Progress::setCurrentOperationProgress(ULONG aPercent)
883{
884 AssertMsgReturn(aPercent <= 100, ("%u\n", aPercent), E_INVALIDARG);
885
886 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
887
888 i_checkForAutomaticTimeout();
889 if (mCancelable && mCanceled)
890 AssertReturn(!mCompleted, E_FAIL);
891 AssertReturn(!mCompleted && !mCanceled, E_FAIL);
892
893 if (m_ulOperationPercent != aPercent)
894 {
895 m_ulOperationPercent = aPercent;
896 ULONG actualPercent = 0;
897 getPercent(&actualPercent);
898 fireProgressPercentageChangedEvent(pEventSource, mId.toUtf16().raw(), actualPercent);
899 }
900
901 return S_OK;
902}
903
904/**
905 * Signals that the current operation is successfully completed and advances to
906 * the next operation. The operation percentage is reset to 0.
907 *
908 * @param aNextOperationDescription Description of the next operation.
909 * @param aNextOperationsWeight Weight of the next operation.
910 *
911 * @note The current operation must not be the last one.
912 */
913HRESULT Progress::setNextOperation(const com::Utf8Str &aNextOperationDescription, ULONG aNextOperationsWeight)
914{
915 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
916
917 if (mCanceled)
918 return E_FAIL;
919 AssertReturn(!mCompleted, E_FAIL);
920 AssertReturn(m_ulCurrentOperation + 1 < m_cOperations, E_FAIL);
921
922 ++m_ulCurrentOperation;
923 m_ulOperationsCompletedWeight += m_ulCurrentOperationWeight;
924
925 m_operationDescription = aNextOperationDescription;
926 m_ulCurrentOperationWeight = aNextOperationsWeight;
927 m_ulOperationPercent = 0;
928
929 LogThisFunc(("%s: aNextOperationsWeight = %d; m_ulCurrentOperation is now %d, m_ulOperationsCompletedWeight is now %d\n",
930 m_operationDescription.c_str(), aNextOperationsWeight, m_ulCurrentOperation, m_ulOperationsCompletedWeight));
931
932 /* wake up all waiting threads */
933 if (mWaitersCount > 0)
934 RTSemEventMultiSignal(mCompletedSem);
935
936 ULONG actualPercent = 0;
937 getPercent(&actualPercent);
938 fireProgressPercentageChangedEvent(pEventSource, mId.toUtf16().raw(), actualPercent);
939
940 return S_OK;
941}
942
943/**
944 * @note XPCOM: when this method is not called on the main XPCOM thread, it
945 * simply blocks the thread until mCompletedSem is signalled. If the
946 * thread has its own event queue (hmm, what for?) that it must run, then
947 * calling this method will definitely freeze event processing.
948 */
949HRESULT Progress::waitForCompletion(LONG aTimeout)
950{
951 LogFlowThisFuncEnter();
952 LogFlowThisFunc(("aTimeout=%d\n", aTimeout));
953
954 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
955
956 /* if we're already completed, take a shortcut */
957 if (!mCompleted)
958 {
959 int vrc = VINF_SUCCESS;
960 bool fForever = aTimeout < 0;
961 int64_t timeLeft = aTimeout;
962 int64_t lastTime = RTTimeMilliTS();
963
964 while (!mCompleted && (fForever || timeLeft > 0))
965 {
966 mWaitersCount++;
967 alock.release();
968 vrc = RTSemEventMultiWait(mCompletedSem,
969 fForever ? RT_INDEFINITE_WAIT : (RTMSINTERVAL)timeLeft);
970 alock.acquire();
971 mWaitersCount--;
972
973 /* the last waiter resets the semaphore */
974 if (mWaitersCount == 0)
975 RTSemEventMultiReset(mCompletedSem);
976
977 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
978 break;
979
980 if (!fForever)
981 {
982 int64_t now = RTTimeMilliTS();
983 timeLeft -= now - lastTime;
984 lastTime = now;
985 }
986 }
987
988 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
989 return setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("Failed to wait for the task completion (%Rrc)"), vrc);
990 }
991
992 LogFlowThisFuncLeave();
993
994 return S_OK;
995}
996
997/**
998 * @note XPCOM: when this method is not called on the main XPCOM thread, it
999 * simply blocks the thread until mCompletedSem is signalled. If the
1000 * thread has its own event queue (hmm, what for?) that it must run, then
1001 * calling this method will definitely freeze event processing.
1002 */
1003HRESULT Progress::waitForOperationCompletion(ULONG aOperation, LONG aTimeout)
1004
1005{
1006 LogFlowThisFuncEnter();
1007 LogFlowThisFunc(("aOperation=%d, aTimeout=%d\n", aOperation, aTimeout));
1008
1009 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1010
1011 CheckComArgExpr(aOperation, aOperation < m_cOperations);
1012
1013 /* if we're already completed or if the given operation is already done,
1014 * then take a shortcut */
1015 if ( !mCompleted
1016 && aOperation >= m_ulCurrentOperation)
1017 {
1018 int vrc = VINF_SUCCESS;
1019 bool fForever = aTimeout < 0;
1020 int64_t timeLeft = aTimeout;
1021 int64_t lastTime = RTTimeMilliTS();
1022
1023 while ( !mCompleted && aOperation >= m_ulCurrentOperation
1024 && (fForever || timeLeft > 0))
1025 {
1026 mWaitersCount ++;
1027 alock.release();
1028 vrc = RTSemEventMultiWait(mCompletedSem,
1029 fForever ? RT_INDEFINITE_WAIT : (unsigned) timeLeft);
1030 alock.acquire();
1031 mWaitersCount--;
1032
1033 /* the last waiter resets the semaphore */
1034 if (mWaitersCount == 0)
1035 RTSemEventMultiReset(mCompletedSem);
1036
1037 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
1038 break;
1039
1040 if (!fForever)
1041 {
1042 int64_t now = RTTimeMilliTS();
1043 timeLeft -= now - lastTime;
1044 lastTime = now;
1045 }
1046 }
1047
1048 if (RT_FAILURE(vrc) && vrc != VERR_TIMEOUT)
1049 return setErrorBoth(E_FAIL, vrc, tr("Failed to wait for the operation completion (%Rrc)"), vrc);
1050 }
1051
1052 LogFlowThisFuncLeave();
1053
1054 return S_OK;
1055}
1056
1057HRESULT Progress::cancel()
1058{
1059 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1060
1061 if (!mCancelable)
1062 return setError(VBOX_E_INVALID_OBJECT_STATE, tr("Operation cannot be canceled"));
1063
1064 if (!mCanceled)
1065 {
1066 LogThisFunc(("Canceling\n"));
1067 mCanceled = TRUE;
1068 if (m_pfnCancelCallback)
1069 m_pfnCancelCallback(m_pvCancelUserArg);
1070
1071 }
1072 else
1073 LogThisFunc(("Already canceled\n"));
1074
1075 return S_OK;
1076}
1077
1078HRESULT Progress::getEventSource(ComPtr<IEventSource> &aEventSource)
1079{
1080 /* event source is const, no need to lock */
1081 pEventSource.queryInterfaceTo(aEventSource.asOutParam());
1082 return S_OK;
1083}
1084
1085// private internal helpers
1086/////////////////////////////////////////////////////////////////////////////
1087
1088/**
1089 * Internal helper to compute the total percent value based on the member values and
1090 * returns it as a "double". This is used both by GetPercent (which returns it as a
1091 * rounded ULONG) and GetTimeRemaining().
1092 *
1093 * Requires locking by the caller!
1094 *
1095 * @return fractional percentage as a double value.
1096 */
1097double Progress::i_calcTotalPercent()
1098{
1099 // avoid division by zero
1100 if (m_ulTotalOperationsWeight == 0)
1101 return 0.0;
1102
1103 double dPercent = ( (double)m_ulOperationsCompletedWeight // weight of operations that have been completed
1104 + ((double)m_ulOperationPercent *
1105 (double)m_ulCurrentOperationWeight / 100.0) // plus partial weight of the current operation
1106 ) * 100.0 / (double)m_ulTotalOperationsWeight;
1107
1108 return dPercent;
1109}
1110
1111/**
1112 * Internal helper for automatically timing out the operation.
1113 *
1114 * The caller must hold the object write lock.
1115 */
1116void Progress::i_checkForAutomaticTimeout(void)
1117{
1118 AssertReturnVoid(isWriteLockOnCurrentThread());
1119
1120 if ( m_cMsTimeout
1121 && mCancelable
1122 && !mCanceled
1123 && RTTimeMilliTS() - m_ullTimestamp > m_cMsTimeout)
1124 Cancel();
1125}
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