VirtualBox

source: vbox/trunk/src/VBox/Main/include/ProgressImpl.h@ 24933

Last change on this file since 24933 was 23827, checked in by vboxsync, 15 years ago

Main: Cancellation fixes (almost perfect).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.8 KB
Line 
1/* $Id: ProgressImpl.h 23827 2009-10-16 15:50:35Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#ifndef ____H_PROGRESSIMPL
24#define ____H_PROGRESSIMPL
25
26#include "VirtualBoxBase.h"
27
28#include <VBox/com/SupportErrorInfo.h>
29
30#include <iprt/semaphore.h>
31
32#include <vector>
33
34class VirtualBox;
35
36////////////////////////////////////////////////////////////////////////////////
37
38/**
39 * Base component class for progress objects.
40 */
41class ATL_NO_VTABLE ProgressBase :
42 public VirtualBoxBase,
43 public com::SupportErrorInfoBase,
44 public VirtualBoxSupportTranslation<ProgressBase>,
45 VBOX_SCRIPTABLE_IMPL(IProgress)
46{
47protected:
48
49 VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT (ProgressBase)
50
51 DECLARE_EMPTY_CTOR_DTOR (ProgressBase)
52
53 HRESULT FinalConstruct();
54
55 // protected initializer/uninitializer for internal purposes only
56 HRESULT protectedInit (AutoInitSpan &aAutoInitSpan,
57#if !defined (VBOX_COM_INPROC)
58 VirtualBox *aParent,
59#endif
60 IUnknown *aInitiator,
61 CBSTR aDescription, OUT_GUID aId = NULL);
62 HRESULT protectedInit (AutoInitSpan &aAutoInitSpan);
63 void protectedUninit (AutoUninitSpan &aAutoUninitSpan);
64
65public:
66
67 // IProgress properties
68 STDMETHOD(COMGETTER(Id)) (BSTR *aId);
69 STDMETHOD(COMGETTER(Description)) (BSTR *aDescription);
70 STDMETHOD(COMGETTER(Initiator)) (IUnknown **aInitiator);
71
72 // IProgress properties
73 STDMETHOD(COMGETTER(Cancelable)) (BOOL *aCancelable);
74 STDMETHOD(COMGETTER(Percent)) (ULONG *aPercent);
75 STDMETHOD(COMGETTER(TimeRemaining)) (LONG *aTimeRemaining);
76 STDMETHOD(COMGETTER(Completed)) (BOOL *aCompleted);
77 STDMETHOD(COMGETTER(Canceled)) (BOOL *aCanceled);
78 STDMETHOD(COMGETTER(ResultCode)) (LONG *aResultCode);
79 STDMETHOD(COMGETTER(ErrorInfo)) (IVirtualBoxErrorInfo **aErrorInfo);
80 STDMETHOD(COMGETTER(OperationCount)) (ULONG *aOperationCount);
81 STDMETHOD(COMGETTER(Operation)) (ULONG *aCount);
82 STDMETHOD(COMGETTER(OperationDescription)) (BSTR *aOperationDescription);
83 STDMETHOD(COMGETTER(OperationPercent)) (ULONG *aOperationPercent);
84
85 // public methods only for internal purposes
86
87 static HRESULT setErrorInfoOnThread (IProgress *aProgress);
88 bool setCancelCallback(void (*pfnCallback)(void *), void *pvUser);
89
90
91 // unsafe inline public methods for internal purposes only (ensure there is
92 // a caller and a read lock before calling them!)
93
94 BOOL completed() const { return mCompleted; }
95 HRESULT resultCode() const { return mResultCode; }
96 double calcTotalPercent();
97
98protected:
99
100#if !defined (VBOX_COM_INPROC)
101 /** Weak parent. */
102 const ComObjPtr<VirtualBox, ComWeakRef> mParent;
103#endif
104
105 const ComPtr<IUnknown> mInitiator;
106
107 const Guid mId;
108 const Bstr mDescription;
109
110 uint64_t m_ullTimestamp; // progress object creation timestamp, for ETA computation
111
112 void (*m_pfnCancelCallback)(void *);
113 void *m_pvCancelUserArg;
114
115 /* The fields below are to be properly initalized by subclasses */
116
117 BOOL mCompleted;
118 BOOL mCancelable;
119 BOOL mCanceled;
120 HRESULT mResultCode;
121 ComPtr<IVirtualBoxErrorInfo> mErrorInfo;
122
123 ULONG m_cOperations; // number of operations (so that progress dialog can display something like 1/3)
124 ULONG m_ulTotalOperationsWeight; // sum of weights of all operations, given to constructor
125
126 ULONG m_ulOperationsCompletedWeight; // summed-up weight of operations that have been completed; initially 0
127
128 ULONG m_ulCurrentOperation; // operations counter, incremented with each setNextOperation()
129 Bstr m_bstrOperationDescription; // name of current operation; initially from constructor, changed with setNextOperation()
130 ULONG m_ulCurrentOperationWeight; // weight of current operation, given to setNextOperation()
131 ULONG m_ulOperationPercent; // percentage of current operation, set with setCurrentOperationProgress()
132};
133
134////////////////////////////////////////////////////////////////////////////////
135
136/**
137 * Normal progress object.
138 */
139class ATL_NO_VTABLE Progress :
140 public com::SupportErrorInfoDerived<ProgressBase, Progress, IProgress>,
141 public VirtualBoxSupportTranslation<Progress>
142{
143
144public:
145
146 VIRTUALBOXSUPPORTTRANSLATION_OVERRIDE (Progress)
147
148 DECLARE_NOT_AGGREGATABLE (Progress)
149
150 DECLARE_PROTECT_FINAL_CONSTRUCT()
151
152 BEGIN_COM_MAP (Progress)
153 COM_INTERFACE_ENTRY (ISupportErrorInfo)
154 COM_INTERFACE_ENTRY (IProgress)
155 COM_INTERFACE_ENTRY2 (IDispatch, IProgress)
156 END_COM_MAP()
157
158 HRESULT FinalConstruct();
159 void FinalRelease();
160
161 // public initializer/uninitializer for internal purposes only
162
163 /**
164 * Simplified constructor for progress objects that have only one
165 * operation as a task.
166 * @param aParent
167 * @param aInitiator
168 * @param aDescription
169 * @param aCancelable
170 * @param aId
171 * @return
172 */
173 HRESULT init(
174#if !defined (VBOX_COM_INPROC)
175 VirtualBox *aParent,
176#endif
177 IUnknown *aInitiator,
178 CBSTR aDescription,
179 BOOL aCancelable,
180 OUT_GUID aId = NULL)
181 {
182 return init(
183#if !defined (VBOX_COM_INPROC)
184 aParent,
185#endif
186 aInitiator,
187 aDescription,
188 aCancelable,
189 1, // cOperations
190 1, // ulTotalOperationsWeight
191 aDescription, // bstrFirstOperationDescription
192 1, // ulFirstOperationWeight
193 aId);
194 }
195
196 /**
197 * Not quite so simplified constructor for progress objects that have
198 * more than one operation, but all sub-operations are weighed the same.
199 * @param aParent
200 * @param aInitiator
201 * @param aDescription
202 * @param aCancelable
203 * @param cOperations
204 * @param bstrFirstOperationDescription
205 * @param aId
206 * @return
207 */
208 HRESULT init(
209#if !defined (VBOX_COM_INPROC)
210 VirtualBox *aParent,
211#endif
212 IUnknown *aInitiator,
213 CBSTR aDescription, BOOL aCancelable,
214 ULONG cOperations,
215 CBSTR bstrFirstOperationDescription,
216 OUT_GUID aId = NULL)
217 {
218 return init(
219#if !defined (VBOX_COM_INPROC)
220 aParent,
221#endif
222 aInitiator,
223 aDescription,
224 aCancelable,
225 cOperations, // cOperations
226 cOperations, // ulTotalOperationsWeight = cOperations
227 bstrFirstOperationDescription, // bstrFirstOperationDescription
228 1, // ulFirstOperationWeight: weigh them all the same
229 aId);
230 }
231
232 HRESULT init(
233#if !defined (VBOX_COM_INPROC)
234 VirtualBox *aParent,
235#endif
236 IUnknown *aInitiator,
237 CBSTR aDescription,
238 BOOL aCancelable,
239 ULONG cOperations,
240 ULONG ulTotalOperationsWeight,
241 CBSTR bstrFirstOperationDescription,
242 ULONG ulFirstOperationWeight,
243 OUT_GUID aId = NULL);
244
245 HRESULT init(BOOL aCancelable,
246 ULONG aOperationCount,
247 CBSTR aOperationDescription);
248
249 void uninit();
250
251 // IProgress methods
252 STDMETHOD(WaitForCompletion)(LONG aTimeout);
253 STDMETHOD(WaitForOperationCompletion)(ULONG aOperation, LONG aTimeout);
254 STDMETHOD(Cancel)();
255
256 STDMETHOD(SetCurrentOperationProgress)(ULONG aPercent);
257 STDMETHOD(SetNextOperation)(IN_BSTR bstrNextOperationDescription, ULONG ulNextOperationsWeight);
258
259 // public methods only for internal purposes
260
261 HRESULT setResultCode(HRESULT aResultCode);
262
263 HRESULT notifyComplete(HRESULT aResultCode);
264 HRESULT notifyComplete(HRESULT aResultCode,
265 const GUID &aIID,
266 const Bstr &aComponent,
267 const char *aText, ...);
268 bool notifyPointOfNoReturn(void);
269
270 /** For com::SupportErrorInfoImpl. */
271 static const char *ComponentName() { return "Progress"; }
272
273private:
274
275 RTSEMEVENTMULTI mCompletedSem;
276 ULONG mWaitersCount;
277};
278
279////////////////////////////////////////////////////////////////////////////////
280
281/**
282 * The CombinedProgress class allows to combine several progress objects to a
283 * single progress component. This single progress component will treat all
284 * operations of individual progress objects as a single sequence of operations
285 * that follow each other in the same order as progress objects are passed to
286 * the #init() method.
287 *
288 * Individual progress objects are sequentially combined so that this progress
289 * object:
290 *
291 * - is cancelable only if all progresses are cancelable.
292 * - is canceled once a progress that follows next to successfully completed
293 * ones reports it was canceled.
294 * - is completed successfully only after all progresses are completed
295 * successfully.
296 * - is completed unsuccessfully once a progress that follows next to
297 * successfully completed ones reports it was completed unsuccessfully;
298 * the result code and error info of the unsuccessful progress
299 * will be reported as the result code and error info of this progress.
300 * - returns N as the operation number, where N equals to the number of
301 * operations in all successfully completed progresses starting from the
302 * first one plus the operation number of the next (not yet complete)
303 * progress; the operation description of the latter one is reported as
304 * the operation description of this progress object.
305 * - returns P as the percent value, where P equals to the sum of percents
306 * of all successfully completed progresses starting from the
307 * first one plus the percent value of the next (not yet complete)
308 * progress, normalized to 100%.
309 *
310 * @note It's the respoisibility of the combined progress object creator to
311 * complete individual progresses in the right order: if, let's say, the
312 * last progress is completed before all previous ones,
313 * #WaitForCompletion(-1) will most likely give 100% CPU load because it
314 * will be in a loop calling a method that returns immediately.
315 */
316class ATL_NO_VTABLE CombinedProgress :
317 public com::SupportErrorInfoDerived<ProgressBase, CombinedProgress, IProgress>,
318 public VirtualBoxSupportTranslation<CombinedProgress>
319{
320
321public:
322
323 VIRTUALBOXSUPPORTTRANSLATION_OVERRIDE (CombinedProgress)
324
325 DECLARE_NOT_AGGREGATABLE (CombinedProgress)
326
327 DECLARE_PROTECT_FINAL_CONSTRUCT()
328
329 BEGIN_COM_MAP (CombinedProgress)
330 COM_INTERFACE_ENTRY (ISupportErrorInfo)
331 COM_INTERFACE_ENTRY (IProgress)
332 COM_INTERFACE_ENTRY2 (IDispatch, IProgress)
333 END_COM_MAP()
334
335 HRESULT FinalConstruct();
336 void FinalRelease();
337
338 // public initializer/uninitializer for internal purposes only
339
340 HRESULT init (
341#if !defined (VBOX_COM_INPROC)
342 VirtualBox *aParent,
343#endif
344 IUnknown *aInitiator,
345 CBSTR aDescription,
346 IProgress *aProgress1, IProgress *aProgress2,
347 OUT_GUID aId = NULL);
348
349 /**
350 * Initializes the combined progress object given the first and the last
351 * normal progress object from the list.
352 *
353 * @param aParent See ProgressBase::init().
354 * @param aInitiator See ProgressBase::init().
355 * @param aDescription See ProgressBase::init().
356 * @param aFirstProgress Iterator of the first normal progress object.
357 * @param aSecondProgress Iterator of the last normal progress object.
358 * @param aId See ProgressBase::init().
359 */
360 template <typename InputIterator>
361 HRESULT init (
362#if !defined (VBOX_COM_INPROC)
363 VirtualBox *aParent,
364#endif
365 IUnknown *aInitiator,
366 CBSTR aDescription,
367 InputIterator aFirstProgress, InputIterator aLastProgress,
368 OUT_GUID aId = NULL)
369 {
370 /* Enclose the state transition NotReady->InInit->Ready */
371 AutoInitSpan autoInitSpan (this);
372 AssertReturn (autoInitSpan.isOk(), E_FAIL);
373
374 mProgresses = ProgressVector (aFirstProgress, aLastProgress);
375
376 HRESULT rc = protectedInit (autoInitSpan,
377#if !defined (VBOX_COM_INPROC)
378 aParent,
379#endif
380 aInitiator, aDescription, aId);
381
382 /* Confirm a successful initialization when it's the case */
383 if (SUCCEEDED (rc))
384 autoInitSpan.setSucceeded();
385
386 return rc;
387 }
388
389protected:
390
391 HRESULT protectedInit (AutoInitSpan &aAutoInitSpan,
392#if !defined (VBOX_COM_INPROC)
393 VirtualBox *aParent,
394#endif
395 IUnknown *aInitiator,
396 CBSTR aDescription, OUT_GUID aId);
397
398public:
399
400 void uninit();
401
402 // IProgress properties
403 STDMETHOD(COMGETTER(Percent)) (ULONG *aPercent);
404 STDMETHOD(COMGETTER(Completed)) (BOOL *aCompleted);
405 STDMETHOD(COMGETTER(Canceled)) (BOOL *aCanceled);
406 STDMETHOD(COMGETTER(ResultCode)) (LONG *aResultCode);
407 STDMETHOD(COMGETTER(ErrorInfo)) (IVirtualBoxErrorInfo **aErrorInfo);
408 STDMETHOD(COMGETTER(Operation)) (ULONG *aCount);
409 STDMETHOD(COMGETTER(OperationDescription)) (BSTR *aOperationDescription);
410 STDMETHOD(COMGETTER(OperationPercent)) (ULONG *aOperationPercent);
411
412 // IProgress methods
413 STDMETHOD(WaitForCompletion) (LONG aTimeout);
414 STDMETHOD(WaitForOperationCompletion) (ULONG aOperation, LONG aTimeout);
415 STDMETHOD(Cancel)();
416
417 STDMETHOD(SetCurrentOperationProgress)(ULONG aPercent)
418 {
419 NOREF(aPercent);
420 return E_NOTIMPL;
421 }
422
423 STDMETHOD(SetNextOperation)(IN_BSTR bstrNextOperationDescription, ULONG ulNextOperationsWeight)
424 {
425 NOREF(bstrNextOperationDescription); NOREF(ulNextOperationsWeight);
426 return E_NOTIMPL;
427 }
428
429 // public methods only for internal purposes
430
431 /** For com::SupportErrorInfoImpl. */
432 static const char *ComponentName() { return "CombinedProgress"; }
433
434private:
435
436 HRESULT checkProgress();
437
438 typedef std::vector <ComPtr<IProgress> > ProgressVector;
439 ProgressVector mProgresses;
440
441 size_t mProgress;
442 ULONG mCompletedOperations;
443};
444
445#endif /* ____H_PROGRESSIMPL */
446
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