VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/MediumImpl.cpp@ 41842

Last change on this file since 41842 was 41243, checked in by vboxsync, 13 years ago

Main/Medium: small optimization, as it is only necessary to look at the original location if it isn't the same as the (potentially modified) one.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 269.3 KB
Line 
1/* $Id: MediumImpl.cpp 41243 2012-05-10 16:34:36Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2008-2012 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include "MediumImpl.h"
19#include "ProgressImpl.h"
20#include "SystemPropertiesImpl.h"
21#include "VirtualBoxImpl.h"
22
23#include "AutoCaller.h"
24#include "Logging.h"
25
26#include <VBox/com/array.h>
27#include "VBox/com/MultiResult.h"
28#include "VBox/com/ErrorInfo.h"
29
30#include <VBox/err.h>
31#include <VBox/settings.h>
32
33#include <iprt/param.h>
34#include <iprt/path.h>
35#include <iprt/file.h>
36#include <iprt/tcp.h>
37#include <iprt/cpp/utils.h>
38
39#include <VBox/vd.h>
40
41#include <algorithm>
42#include <list>
43
44typedef std::list<Guid> GuidList;
45
46////////////////////////////////////////////////////////////////////////////////
47//
48// Medium data definition
49//
50////////////////////////////////////////////////////////////////////////////////
51
52/** Describes how a machine refers to this medium. */
53struct BackRef
54{
55 /** Equality predicate for stdc++. */
56 struct EqualsTo : public std::unary_function <BackRef, bool>
57 {
58 explicit EqualsTo(const Guid &aMachineId) : machineId(aMachineId) {}
59
60 bool operator()(const argument_type &aThat) const
61 {
62 return aThat.machineId == machineId;
63 }
64
65 const Guid machineId;
66 };
67
68 BackRef(const Guid &aMachineId,
69 const Guid &aSnapshotId = Guid::Empty)
70 : machineId(aMachineId),
71 fInCurState(aSnapshotId.isEmpty())
72 {
73 if (!aSnapshotId.isEmpty())
74 llSnapshotIds.push_back(aSnapshotId);
75 }
76
77 Guid machineId;
78 bool fInCurState : 1;
79 GuidList llSnapshotIds;
80};
81
82typedef std::list<BackRef> BackRefList;
83
84struct Medium::Data
85{
86 Data()
87 : pVirtualBox(NULL),
88 state(MediumState_NotCreated),
89 variant(MediumVariant_Standard),
90 size(0),
91 readers(0),
92 preLockState(MediumState_NotCreated),
93 queryInfoSem(LOCKCLASS_MEDIUMQUERY),
94 queryInfoRunning(false),
95 type(MediumType_Normal),
96 devType(DeviceType_HardDisk),
97 logicalSize(0),
98 hddOpenMode(OpenReadWrite),
99 autoReset(false),
100 hostDrive(false),
101 implicit(false),
102 uOpenFlagsDef(VD_OPEN_FLAGS_IGNORE_FLUSH),
103 numCreateDiffTasks(0),
104 vdDiskIfaces(NULL),
105 vdImageIfaces(NULL)
106 { }
107
108 /** weak VirtualBox parent */
109 VirtualBox * const pVirtualBox;
110
111 // pParent and llChildren are protected by VirtualBox::getMediaTreeLockHandle()
112 ComObjPtr<Medium> pParent;
113 MediaList llChildren; // to add a child, just call push_back; to remove a child, call child->deparent() which does a lookup
114
115 GuidList llRegistryIDs; // media registries in which this medium is listed
116
117 const Guid id;
118 Utf8Str strDescription;
119 MediumState_T state;
120 MediumVariant_T variant;
121 Utf8Str strLocationFull;
122 uint64_t size;
123 Utf8Str strLastAccessError;
124
125 BackRefList backRefs;
126
127 size_t readers;
128 MediumState_T preLockState;
129
130 /** Special synchronization for operations which must wait for
131 * Medium::queryInfo in another thread to complete. Using a SemRW is
132 * not quite ideal, but at least it is subject to the lock validator,
133 * unlike the SemEventMulti which we had here for many years. Catching
134 * possible deadlocks is more important than a tiny bit of efficiency. */
135 RWLockHandle queryInfoSem;
136 bool queryInfoRunning : 1;
137
138 const Utf8Str strFormat;
139 ComObjPtr<MediumFormat> formatObj;
140
141 MediumType_T type;
142 DeviceType_T devType;
143 uint64_t logicalSize;
144
145 HDDOpenMode hddOpenMode;
146
147 bool autoReset : 1;
148
149 /** New UUID to be set on the next Medium::queryInfo call. */
150 const Guid uuidImage;
151 /** New parent UUID to be set on the next Medium::queryInfo call. */
152 const Guid uuidParentImage;
153
154 bool hostDrive : 1;
155
156 settings::StringsMap mapProperties;
157
158 bool implicit : 1;
159
160 /** Default flags passed to VDOpen(). */
161 unsigned uOpenFlagsDef;
162
163 uint32_t numCreateDiffTasks;
164
165 Utf8Str vdError; /*< Error remembered by the VD error callback. */
166
167 VDINTERFACEERROR vdIfError;
168
169 VDINTERFACECONFIG vdIfConfig;
170
171 VDINTERFACETCPNET vdIfTcpNet;
172
173 PVDINTERFACE vdDiskIfaces;
174 PVDINTERFACE vdImageIfaces;
175};
176
177typedef struct VDSOCKETINT
178{
179 /** Socket handle. */
180 RTSOCKET hSocket;
181} VDSOCKETINT, *PVDSOCKETINT;
182
183////////////////////////////////////////////////////////////////////////////////
184//
185// Globals
186//
187////////////////////////////////////////////////////////////////////////////////
188
189/**
190 * Medium::Task class for asynchronous operations.
191 *
192 * @note Instances of this class must be created using new() because the
193 * task thread function will delete them when the task is complete.
194 *
195 * @note The constructor of this class adds a caller on the managed Medium
196 * object which is automatically released upon destruction.
197 */
198class Medium::Task
199{
200public:
201 Task(Medium *aMedium, Progress *aProgress)
202 : mVDOperationIfaces(NULL),
203 mMedium(aMedium),
204 mMediumCaller(aMedium),
205 mThread(NIL_RTTHREAD),
206 mProgress(aProgress),
207 mVirtualBoxCaller(NULL)
208 {
209 AssertReturnVoidStmt(aMedium, mRC = E_FAIL);
210 mRC = mMediumCaller.rc();
211 if (FAILED(mRC))
212 return;
213
214 /* Get strong VirtualBox reference, see below. */
215 VirtualBox *pVirtualBox = aMedium->m->pVirtualBox;
216 mVirtualBox = pVirtualBox;
217 mVirtualBoxCaller.attach(pVirtualBox);
218 mRC = mVirtualBoxCaller.rc();
219 if (FAILED(mRC))
220 return;
221
222 /* Set up a per-operation progress interface, can be used freely (for
223 * binary operations you can use it either on the source or target). */
224 mVDIfProgress.pfnProgress = vdProgressCall;
225 int vrc = VDInterfaceAdd(&mVDIfProgress.Core,
226 "Medium::Task::vdInterfaceProgress",
227 VDINTERFACETYPE_PROGRESS,
228 mProgress,
229 sizeof(VDINTERFACEPROGRESS),
230 &mVDOperationIfaces);
231 AssertRC(vrc);
232 if (RT_FAILURE(vrc))
233 mRC = E_FAIL;
234 }
235
236 // Make all destructors virtual. Just in case.
237 virtual ~Task()
238 {}
239
240 HRESULT rc() const { return mRC; }
241 bool isOk() const { return SUCCEEDED(rc()); }
242
243 static int fntMediumTask(RTTHREAD aThread, void *pvUser);
244
245 bool isAsync() { return mThread != NIL_RTTHREAD; }
246
247 PVDINTERFACE mVDOperationIfaces;
248
249 const ComObjPtr<Medium> mMedium;
250 AutoCaller mMediumCaller;
251
252 friend HRESULT Medium::runNow(Medium::Task*);
253
254protected:
255 HRESULT mRC;
256 RTTHREAD mThread;
257
258private:
259 virtual HRESULT handler() = 0;
260
261 const ComObjPtr<Progress> mProgress;
262
263 static DECLCALLBACK(int) vdProgressCall(void *pvUser, unsigned uPercent);
264
265 VDINTERFACEPROGRESS mVDIfProgress;
266
267 /* Must have a strong VirtualBox reference during a task otherwise the
268 * reference count might drop to 0 while a task is still running. This
269 * would result in weird behavior, including deadlocks due to uninit and
270 * locking order issues. The deadlock often is not detectable because the
271 * uninit uses event semaphores which sabotages deadlock detection. */
272 ComObjPtr<VirtualBox> mVirtualBox;
273 AutoCaller mVirtualBoxCaller;
274};
275
276class Medium::CreateBaseTask : public Medium::Task
277{
278public:
279 CreateBaseTask(Medium *aMedium,
280 Progress *aProgress,
281 uint64_t aSize,
282 MediumVariant_T aVariant)
283 : Medium::Task(aMedium, aProgress),
284 mSize(aSize),
285 mVariant(aVariant)
286 {}
287
288 uint64_t mSize;
289 MediumVariant_T mVariant;
290
291private:
292 virtual HRESULT handler();
293};
294
295class Medium::CreateDiffTask : public Medium::Task
296{
297public:
298 CreateDiffTask(Medium *aMedium,
299 Progress *aProgress,
300 Medium *aTarget,
301 MediumVariant_T aVariant,
302 MediumLockList *aMediumLockList,
303 bool fKeepMediumLockList = false)
304 : Medium::Task(aMedium, aProgress),
305 mpMediumLockList(aMediumLockList),
306 mTarget(aTarget),
307 mVariant(aVariant),
308 mTargetCaller(aTarget),
309 mfKeepMediumLockList(fKeepMediumLockList)
310 {
311 AssertReturnVoidStmt(aTarget != NULL, mRC = E_FAIL);
312 mRC = mTargetCaller.rc();
313 if (FAILED(mRC))
314 return;
315 }
316
317 ~CreateDiffTask()
318 {
319 if (!mfKeepMediumLockList && mpMediumLockList)
320 delete mpMediumLockList;
321 }
322
323 MediumLockList *mpMediumLockList;
324
325 const ComObjPtr<Medium> mTarget;
326 MediumVariant_T mVariant;
327
328private:
329 virtual HRESULT handler();
330
331 AutoCaller mTargetCaller;
332 bool mfKeepMediumLockList;
333};
334
335class Medium::CloneTask : public Medium::Task
336{
337public:
338 CloneTask(Medium *aMedium,
339 Progress *aProgress,
340 Medium *aTarget,
341 MediumVariant_T aVariant,
342 Medium *aParent,
343 uint32_t idxSrcImageSame,
344 uint32_t idxDstImageSame,
345 MediumLockList *aSourceMediumLockList,
346 MediumLockList *aTargetMediumLockList,
347 bool fKeepSourceMediumLockList = false,
348 bool fKeepTargetMediumLockList = false)
349 : Medium::Task(aMedium, aProgress),
350 mTarget(aTarget),
351 mParent(aParent),
352 mpSourceMediumLockList(aSourceMediumLockList),
353 mpTargetMediumLockList(aTargetMediumLockList),
354 mVariant(aVariant),
355 midxSrcImageSame(idxSrcImageSame),
356 midxDstImageSame(idxDstImageSame),
357 mTargetCaller(aTarget),
358 mParentCaller(aParent),
359 mfKeepSourceMediumLockList(fKeepSourceMediumLockList),
360 mfKeepTargetMediumLockList(fKeepTargetMediumLockList)
361 {
362 AssertReturnVoidStmt(aTarget != NULL, mRC = E_FAIL);
363 mRC = mTargetCaller.rc();
364 if (FAILED(mRC))
365 return;
366 /* aParent may be NULL */
367 mRC = mParentCaller.rc();
368 if (FAILED(mRC))
369 return;
370 AssertReturnVoidStmt(aSourceMediumLockList != NULL, mRC = E_FAIL);
371 AssertReturnVoidStmt(aTargetMediumLockList != NULL, mRC = E_FAIL);
372 }
373
374 ~CloneTask()
375 {
376 if (!mfKeepSourceMediumLockList && mpSourceMediumLockList)
377 delete mpSourceMediumLockList;
378 if (!mfKeepTargetMediumLockList && mpTargetMediumLockList)
379 delete mpTargetMediumLockList;
380 }
381
382 const ComObjPtr<Medium> mTarget;
383 const ComObjPtr<Medium> mParent;
384 MediumLockList *mpSourceMediumLockList;
385 MediumLockList *mpTargetMediumLockList;
386 MediumVariant_T mVariant;
387 uint32_t midxSrcImageSame;
388 uint32_t midxDstImageSame;
389
390private:
391 virtual HRESULT handler();
392
393 AutoCaller mTargetCaller;
394 AutoCaller mParentCaller;
395 bool mfKeepSourceMediumLockList;
396 bool mfKeepTargetMediumLockList;
397};
398
399class Medium::CompactTask : public Medium::Task
400{
401public:
402 CompactTask(Medium *aMedium,
403 Progress *aProgress,
404 MediumLockList *aMediumLockList,
405 bool fKeepMediumLockList = false)
406 : Medium::Task(aMedium, aProgress),
407 mpMediumLockList(aMediumLockList),
408 mfKeepMediumLockList(fKeepMediumLockList)
409 {
410 AssertReturnVoidStmt(aMediumLockList != NULL, mRC = E_FAIL);
411 }
412
413 ~CompactTask()
414 {
415 if (!mfKeepMediumLockList && mpMediumLockList)
416 delete mpMediumLockList;
417 }
418
419 MediumLockList *mpMediumLockList;
420
421private:
422 virtual HRESULT handler();
423
424 bool mfKeepMediumLockList;
425};
426
427class Medium::ResizeTask : public Medium::Task
428{
429public:
430 ResizeTask(Medium *aMedium,
431 uint64_t aSize,
432 Progress *aProgress,
433 MediumLockList *aMediumLockList,
434 bool fKeepMediumLockList = false)
435 : Medium::Task(aMedium, aProgress),
436 mSize(aSize),
437 mpMediumLockList(aMediumLockList),
438 mfKeepMediumLockList(fKeepMediumLockList)
439 {
440 AssertReturnVoidStmt(aMediumLockList != NULL, mRC = E_FAIL);
441 }
442
443 ~ResizeTask()
444 {
445 if (!mfKeepMediumLockList && mpMediumLockList)
446 delete mpMediumLockList;
447 }
448
449 uint64_t mSize;
450 MediumLockList *mpMediumLockList;
451
452private:
453 virtual HRESULT handler();
454
455 bool mfKeepMediumLockList;
456};
457
458class Medium::ResetTask : public Medium::Task
459{
460public:
461 ResetTask(Medium *aMedium,
462 Progress *aProgress,
463 MediumLockList *aMediumLockList,
464 bool fKeepMediumLockList = false)
465 : Medium::Task(aMedium, aProgress),
466 mpMediumLockList(aMediumLockList),
467 mfKeepMediumLockList(fKeepMediumLockList)
468 {}
469
470 ~ResetTask()
471 {
472 if (!mfKeepMediumLockList && mpMediumLockList)
473 delete mpMediumLockList;
474 }
475
476 MediumLockList *mpMediumLockList;
477
478private:
479 virtual HRESULT handler();
480
481 bool mfKeepMediumLockList;
482};
483
484class Medium::DeleteTask : public Medium::Task
485{
486public:
487 DeleteTask(Medium *aMedium,
488 Progress *aProgress,
489 MediumLockList *aMediumLockList,
490 bool fKeepMediumLockList = false)
491 : Medium::Task(aMedium, aProgress),
492 mpMediumLockList(aMediumLockList),
493 mfKeepMediumLockList(fKeepMediumLockList)
494 {}
495
496 ~DeleteTask()
497 {
498 if (!mfKeepMediumLockList && mpMediumLockList)
499 delete mpMediumLockList;
500 }
501
502 MediumLockList *mpMediumLockList;
503
504private:
505 virtual HRESULT handler();
506
507 bool mfKeepMediumLockList;
508};
509
510class Medium::MergeTask : public Medium::Task
511{
512public:
513 MergeTask(Medium *aMedium,
514 Medium *aTarget,
515 bool fMergeForward,
516 Medium *aParentForTarget,
517 const MediaList &aChildrenToReparent,
518 Progress *aProgress,
519 MediumLockList *aMediumLockList,
520 bool fKeepMediumLockList = false)
521 : Medium::Task(aMedium, aProgress),
522 mTarget(aTarget),
523 mfMergeForward(fMergeForward),
524 mParentForTarget(aParentForTarget),
525 mChildrenToReparent(aChildrenToReparent),
526 mpMediumLockList(aMediumLockList),
527 mTargetCaller(aTarget),
528 mParentForTargetCaller(aParentForTarget),
529 mfChildrenCaller(false),
530 mfKeepMediumLockList(fKeepMediumLockList)
531 {
532 AssertReturnVoidStmt(aMediumLockList != NULL, mRC = E_FAIL);
533 for (MediaList::const_iterator it = mChildrenToReparent.begin();
534 it != mChildrenToReparent.end();
535 ++it)
536 {
537 HRESULT rc2 = (*it)->addCaller();
538 if (FAILED(rc2))
539 {
540 mRC = E_FAIL;
541 for (MediaList::const_iterator it2 = mChildrenToReparent.begin();
542 it2 != it;
543 --it2)
544 {
545 (*it2)->releaseCaller();
546 }
547 return;
548 }
549 }
550 mfChildrenCaller = true;
551 }
552
553 ~MergeTask()
554 {
555 if (!mfKeepMediumLockList && mpMediumLockList)
556 delete mpMediumLockList;
557 if (mfChildrenCaller)
558 {
559 for (MediaList::const_iterator it = mChildrenToReparent.begin();
560 it != mChildrenToReparent.end();
561 ++it)
562 {
563 (*it)->releaseCaller();
564 }
565 }
566 }
567
568 const ComObjPtr<Medium> mTarget;
569 bool mfMergeForward;
570 /* When mChildrenToReparent is empty then mParentForTarget is non-null.
571 * In other words: they are used in different cases. */
572 const ComObjPtr<Medium> mParentForTarget;
573 MediaList mChildrenToReparent;
574 MediumLockList *mpMediumLockList;
575
576private:
577 virtual HRESULT handler();
578
579 AutoCaller mTargetCaller;
580 AutoCaller mParentForTargetCaller;
581 bool mfChildrenCaller;
582 bool mfKeepMediumLockList;
583};
584
585class Medium::ExportTask : public Medium::Task
586{
587public:
588 ExportTask(Medium *aMedium,
589 Progress *aProgress,
590 const char *aFilename,
591 MediumFormat *aFormat,
592 MediumVariant_T aVariant,
593 VDINTERFACEIO *aVDImageIOIf,
594 void *aVDImageIOUser,
595 MediumLockList *aSourceMediumLockList,
596 bool fKeepSourceMediumLockList = false)
597 : Medium::Task(aMedium, aProgress),
598 mpSourceMediumLockList(aSourceMediumLockList),
599 mFilename(aFilename),
600 mFormat(aFormat),
601 mVariant(aVariant),
602 mfKeepSourceMediumLockList(fKeepSourceMediumLockList)
603 {
604 AssertReturnVoidStmt(aSourceMediumLockList != NULL, mRC = E_FAIL);
605
606 mVDImageIfaces = aMedium->m->vdImageIfaces;
607 if (aVDImageIOIf)
608 {
609 int vrc = VDInterfaceAdd(&aVDImageIOIf->Core, "Medium::vdInterfaceIO",
610 VDINTERFACETYPE_IO, aVDImageIOUser,
611 sizeof(VDINTERFACEIO), &mVDImageIfaces);
612 AssertRCReturnVoidStmt(vrc, mRC = E_FAIL);
613 }
614 }
615
616 ~ExportTask()
617 {
618 if (!mfKeepSourceMediumLockList && mpSourceMediumLockList)
619 delete mpSourceMediumLockList;
620 }
621
622 MediumLockList *mpSourceMediumLockList;
623 Utf8Str mFilename;
624 ComObjPtr<MediumFormat> mFormat;
625 MediumVariant_T mVariant;
626 PVDINTERFACE mVDImageIfaces;
627
628private:
629 virtual HRESULT handler();
630
631 bool mfKeepSourceMediumLockList;
632};
633
634class Medium::ImportTask : public Medium::Task
635{
636public:
637 ImportTask(Medium *aMedium,
638 Progress *aProgress,
639 const char *aFilename,
640 MediumFormat *aFormat,
641 MediumVariant_T aVariant,
642 VDINTERFACEIO *aVDImageIOIf,
643 void *aVDImageIOUser,
644 Medium *aParent,
645 MediumLockList *aTargetMediumLockList,
646 bool fKeepTargetMediumLockList = false)
647 : Medium::Task(aMedium, aProgress),
648 mFilename(aFilename),
649 mFormat(aFormat),
650 mVariant(aVariant),
651 mParent(aParent),
652 mpTargetMediumLockList(aTargetMediumLockList),
653 mParentCaller(aParent),
654 mfKeepTargetMediumLockList(fKeepTargetMediumLockList)
655 {
656 AssertReturnVoidStmt(aTargetMediumLockList != NULL, mRC = E_FAIL);
657 /* aParent may be NULL */
658 mRC = mParentCaller.rc();
659 if (FAILED(mRC))
660 return;
661
662 mVDImageIfaces = aMedium->m->vdImageIfaces;
663 if (aVDImageIOIf)
664 {
665 int vrc = VDInterfaceAdd(&aVDImageIOIf->Core, "Medium::vdInterfaceIO",
666 VDINTERFACETYPE_IO, aVDImageIOUser,
667 sizeof(VDINTERFACEIO), &mVDImageIfaces);
668 AssertRCReturnVoidStmt(vrc, mRC = E_FAIL);
669 }
670 }
671
672 ~ImportTask()
673 {
674 if (!mfKeepTargetMediumLockList && mpTargetMediumLockList)
675 delete mpTargetMediumLockList;
676 }
677
678 Utf8Str mFilename;
679 ComObjPtr<MediumFormat> mFormat;
680 MediumVariant_T mVariant;
681 const ComObjPtr<Medium> mParent;
682 MediumLockList *mpTargetMediumLockList;
683 PVDINTERFACE mVDImageIfaces;
684
685private:
686 virtual HRESULT handler();
687
688 AutoCaller mParentCaller;
689 bool mfKeepTargetMediumLockList;
690};
691
692/**
693 * Thread function for time-consuming medium tasks.
694 *
695 * @param pvUser Pointer to the Medium::Task instance.
696 */
697/* static */
698DECLCALLBACK(int) Medium::Task::fntMediumTask(RTTHREAD aThread, void *pvUser)
699{
700 LogFlowFuncEnter();
701 AssertReturn(pvUser, (int)E_INVALIDARG);
702 Medium::Task *pTask = static_cast<Medium::Task *>(pvUser);
703
704 pTask->mThread = aThread;
705
706 HRESULT rc = pTask->handler();
707
708 /* complete the progress if run asynchronously */
709 if (pTask->isAsync())
710 {
711 if (!pTask->mProgress.isNull())
712 pTask->mProgress->notifyComplete(rc);
713 }
714
715 /* pTask is no longer needed, delete it. */
716 delete pTask;
717
718 LogFlowFunc(("rc=%Rhrc\n", rc));
719 LogFlowFuncLeave();
720
721 return (int)rc;
722}
723
724/**
725 * PFNVDPROGRESS callback handler for Task operations.
726 *
727 * @param pvUser Pointer to the Progress instance.
728 * @param uPercent Completion percentage (0-100).
729 */
730/*static*/
731DECLCALLBACK(int) Medium::Task::vdProgressCall(void *pvUser, unsigned uPercent)
732{
733 Progress *that = static_cast<Progress *>(pvUser);
734
735 if (that != NULL)
736 {
737 /* update the progress object, capping it at 99% as the final percent
738 * is used for additional operations like setting the UUIDs and similar. */
739 HRESULT rc = that->SetCurrentOperationProgress(uPercent * 99 / 100);
740 if (FAILED(rc))
741 {
742 if (rc == E_FAIL)
743 return VERR_CANCELLED;
744 else
745 return VERR_INVALID_STATE;
746 }
747 }
748
749 return VINF_SUCCESS;
750}
751
752/**
753 * Implementation code for the "create base" task.
754 */
755HRESULT Medium::CreateBaseTask::handler()
756{
757 return mMedium->taskCreateBaseHandler(*this);
758}
759
760/**
761 * Implementation code for the "create diff" task.
762 */
763HRESULT Medium::CreateDiffTask::handler()
764{
765 return mMedium->taskCreateDiffHandler(*this);
766}
767
768/**
769 * Implementation code for the "clone" task.
770 */
771HRESULT Medium::CloneTask::handler()
772{
773 return mMedium->taskCloneHandler(*this);
774}
775
776/**
777 * Implementation code for the "compact" task.
778 */
779HRESULT Medium::CompactTask::handler()
780{
781 return mMedium->taskCompactHandler(*this);
782}
783
784/**
785 * Implementation code for the "resize" task.
786 */
787HRESULT Medium::ResizeTask::handler()
788{
789 return mMedium->taskResizeHandler(*this);
790}
791
792
793/**
794 * Implementation code for the "reset" task.
795 */
796HRESULT Medium::ResetTask::handler()
797{
798 return mMedium->taskResetHandler(*this);
799}
800
801/**
802 * Implementation code for the "delete" task.
803 */
804HRESULT Medium::DeleteTask::handler()
805{
806 return mMedium->taskDeleteHandler(*this);
807}
808
809/**
810 * Implementation code for the "merge" task.
811 */
812HRESULT Medium::MergeTask::handler()
813{
814 return mMedium->taskMergeHandler(*this);
815}
816
817/**
818 * Implementation code for the "export" task.
819 */
820HRESULT Medium::ExportTask::handler()
821{
822 return mMedium->taskExportHandler(*this);
823}
824
825/**
826 * Implementation code for the "import" task.
827 */
828HRESULT Medium::ImportTask::handler()
829{
830 return mMedium->taskImportHandler(*this);
831}
832
833////////////////////////////////////////////////////////////////////////////////
834//
835// Medium constructor / destructor
836//
837////////////////////////////////////////////////////////////////////////////////
838
839DEFINE_EMPTY_CTOR_DTOR(Medium)
840
841HRESULT Medium::FinalConstruct()
842{
843 m = new Data;
844
845 /* Initialize the callbacks of the VD error interface */
846 m->vdIfError.pfnError = vdErrorCall;
847 m->vdIfError.pfnMessage = NULL;
848
849 /* Initialize the callbacks of the VD config interface */
850 m->vdIfConfig.pfnAreKeysValid = vdConfigAreKeysValid;
851 m->vdIfConfig.pfnQuerySize = vdConfigQuerySize;
852 m->vdIfConfig.pfnQuery = vdConfigQuery;
853
854 /* Initialize the callbacks of the VD TCP interface (we always use the host
855 * IP stack for now) */
856 m->vdIfTcpNet.pfnSocketCreate = vdTcpSocketCreate;
857 m->vdIfTcpNet.pfnSocketDestroy = vdTcpSocketDestroy;
858 m->vdIfTcpNet.pfnClientConnect = vdTcpClientConnect;
859 m->vdIfTcpNet.pfnClientClose = vdTcpClientClose;
860 m->vdIfTcpNet.pfnIsClientConnected = vdTcpIsClientConnected;
861 m->vdIfTcpNet.pfnSelectOne = vdTcpSelectOne;
862 m->vdIfTcpNet.pfnRead = vdTcpRead;
863 m->vdIfTcpNet.pfnWrite = vdTcpWrite;
864 m->vdIfTcpNet.pfnSgWrite = vdTcpSgWrite;
865 m->vdIfTcpNet.pfnFlush = vdTcpFlush;
866 m->vdIfTcpNet.pfnSetSendCoalescing = vdTcpSetSendCoalescing;
867 m->vdIfTcpNet.pfnGetLocalAddress = vdTcpGetLocalAddress;
868 m->vdIfTcpNet.pfnGetPeerAddress = vdTcpGetPeerAddress;
869 m->vdIfTcpNet.pfnSelectOneEx = NULL;
870 m->vdIfTcpNet.pfnPoke = NULL;
871
872 /* Initialize the per-disk interface chain (could be done more globally,
873 * but it's not wasting much time or space so it's not worth it). */
874 int vrc;
875 vrc = VDInterfaceAdd(&m->vdIfError.Core,
876 "Medium::vdInterfaceError",
877 VDINTERFACETYPE_ERROR, this,
878 sizeof(VDINTERFACEERROR), &m->vdDiskIfaces);
879 AssertRCReturn(vrc, E_FAIL);
880
881 /* Initialize the per-image interface chain */
882 vrc = VDInterfaceAdd(&m->vdIfConfig.Core,
883 "Medium::vdInterfaceConfig",
884 VDINTERFACETYPE_CONFIG, this,
885 sizeof(VDINTERFACECONFIG), &m->vdImageIfaces);
886 AssertRCReturn(vrc, E_FAIL);
887
888 vrc = VDInterfaceAdd(&m->vdIfTcpNet.Core,
889 "Medium::vdInterfaceTcpNet",
890 VDINTERFACETYPE_TCPNET, this,
891 sizeof(VDINTERFACETCPNET), &m->vdImageIfaces);
892 AssertRCReturn(vrc, E_FAIL);
893
894 return BaseFinalConstruct();
895}
896
897void Medium::FinalRelease()
898{
899 uninit();
900
901 delete m;
902
903 BaseFinalRelease();
904}
905
906/**
907 * Initializes an empty hard disk object without creating or opening an associated
908 * storage unit.
909 *
910 * This gets called by VirtualBox::CreateHardDisk() in which case uuidMachineRegistry
911 * is empty since starting with VirtualBox 4.0, we no longer add opened media to a
912 * registry automatically (this is deferred until the medium is attached to a machine).
913 *
914 * This also gets called when VirtualBox creates diff images; in this case uuidMachineRegistry
915 * is set to the registry of the parent image to make sure they all end up in the same
916 * file.
917 *
918 * For hard disks that don't have the MediumFormatCapabilities_CreateFixed or
919 * MediumFormatCapabilities_CreateDynamic capability (and therefore cannot be created or deleted
920 * with the means of VirtualBox) the associated storage unit is assumed to be
921 * ready for use so the state of the hard disk object will be set to Created.
922 *
923 * @param aVirtualBox VirtualBox object.
924 * @param aFormat
925 * @param aLocation Storage unit location.
926 * @param uuidMachineRegistry The registry to which this medium should be added (global registry UUID or machine UUID or empty if none).
927 */
928HRESULT Medium::init(VirtualBox *aVirtualBox,
929 const Utf8Str &aFormat,
930 const Utf8Str &aLocation,
931 const Guid &uuidMachineRegistry)
932{
933 AssertReturn(aVirtualBox != NULL, E_FAIL);
934 AssertReturn(!aFormat.isEmpty(), E_FAIL);
935
936 /* Enclose the state transition NotReady->InInit->Ready */
937 AutoInitSpan autoInitSpan(this);
938 AssertReturn(autoInitSpan.isOk(), E_FAIL);
939
940 HRESULT rc = S_OK;
941
942 unconst(m->pVirtualBox) = aVirtualBox;
943
944 if (!uuidMachineRegistry.isEmpty())
945 m->llRegistryIDs.push_back(uuidMachineRegistry);
946
947 /* no storage yet */
948 m->state = MediumState_NotCreated;
949
950 /* cannot be a host drive */
951 m->hostDrive = false;
952
953 /* No storage unit is created yet, no need to call Medium::queryInfo */
954
955 rc = setFormat(aFormat);
956 if (FAILED(rc)) return rc;
957
958 rc = setLocation(aLocation);
959 if (FAILED(rc)) return rc;
960
961 if (!(m->formatObj->getCapabilities() & ( MediumFormatCapabilities_CreateFixed
962 | MediumFormatCapabilities_CreateDynamic))
963 )
964 {
965 /* Storage for hard disks of this format can neither be explicitly
966 * created by VirtualBox nor deleted, so we place the hard disk to
967 * Inaccessible state here and also add it to the registry. The
968 * state means that one has to use RefreshState() to update the
969 * medium format specific fields. */
970 m->state = MediumState_Inaccessible;
971 // create new UUID
972 unconst(m->id).create();
973
974 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
975 ComObjPtr<Medium> pMedium;
976 rc = m->pVirtualBox->registerMedium(this, &pMedium, DeviceType_HardDisk);
977 Assert(this == pMedium);
978 }
979
980 /* Confirm a successful initialization when it's the case */
981 if (SUCCEEDED(rc))
982 autoInitSpan.setSucceeded();
983
984 return rc;
985}
986
987/**
988 * Initializes the medium object by opening the storage unit at the specified
989 * location. The enOpenMode parameter defines whether the medium will be opened
990 * read/write or read-only.
991 *
992 * This gets called by VirtualBox::OpenMedium() and also by
993 * Machine::AttachDevice() and createImplicitDiffs() when new diff
994 * images are created.
995 *
996 * There is no registry for this case since starting with VirtualBox 4.0, we
997 * no longer add opened media to a registry automatically (this is deferred
998 * until the medium is attached to a machine).
999 *
1000 * For hard disks, the UUID, format and the parent of this medium will be
1001 * determined when reading the medium storage unit. For DVD and floppy images,
1002 * which have no UUIDs in their storage units, new UUIDs are created.
1003 * If the detected or set parent is not known to VirtualBox, then this method
1004 * will fail.
1005 *
1006 * @param aVirtualBox VirtualBox object.
1007 * @param aLocation Storage unit location.
1008 * @param enOpenMode Whether to open the medium read/write or read-only.
1009 * @param fForceNewUuid Whether a new UUID should be set to avoid duplicates.
1010 * @param aDeviceType Device type of medium.
1011 */
1012HRESULT Medium::init(VirtualBox *aVirtualBox,
1013 const Utf8Str &aLocation,
1014 HDDOpenMode enOpenMode,
1015 bool fForceNewUuid,
1016 DeviceType_T aDeviceType)
1017{
1018 AssertReturn(aVirtualBox, E_INVALIDARG);
1019 AssertReturn(!aLocation.isEmpty(), E_INVALIDARG);
1020
1021 /* Enclose the state transition NotReady->InInit->Ready */
1022 AutoInitSpan autoInitSpan(this);
1023 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1024
1025 HRESULT rc = S_OK;
1026
1027 unconst(m->pVirtualBox) = aVirtualBox;
1028
1029 /* there must be a storage unit */
1030 m->state = MediumState_Created;
1031
1032 /* remember device type for correct unregistering later */
1033 m->devType = aDeviceType;
1034
1035 /* cannot be a host drive */
1036 m->hostDrive = false;
1037
1038 /* remember the open mode (defaults to ReadWrite) */
1039 m->hddOpenMode = enOpenMode;
1040
1041 if (aDeviceType == DeviceType_DVD)
1042 m->type = MediumType_Readonly;
1043 else if (aDeviceType == DeviceType_Floppy)
1044 m->type = MediumType_Writethrough;
1045
1046 rc = setLocation(aLocation);
1047 if (FAILED(rc)) return rc;
1048
1049 /* get all the information about the medium from the storage unit */
1050 if (fForceNewUuid)
1051 unconst(m->uuidImage).create();
1052
1053 rc = queryInfo(fForceNewUuid /* fSetImageId */, false /* fSetParentId */);
1054
1055 if (SUCCEEDED(rc))
1056 {
1057 /* if the storage unit is not accessible, it's not acceptable for the
1058 * newly opened media so convert this into an error */
1059 if (m->state == MediumState_Inaccessible)
1060 {
1061 Assert(!m->strLastAccessError.isEmpty());
1062 rc = setError(E_FAIL, "%s", m->strLastAccessError.c_str());
1063 }
1064 else
1065 {
1066 AssertReturn(!m->id.isEmpty(), E_FAIL);
1067
1068 /* storage format must be detected by Medium::queryInfo if the
1069 * medium is accessible */
1070 AssertReturn(!m->strFormat.isEmpty(), E_FAIL);
1071 }
1072 }
1073
1074 /* Confirm a successful initialization when it's the case */
1075 if (SUCCEEDED(rc))
1076 autoInitSpan.setSucceeded();
1077
1078 return rc;
1079}
1080
1081/**
1082 * Initializes the medium object by loading its data from the given settings
1083 * node. In this mode, the medium will always be opened read/write.
1084 *
1085 * In this case, since we're loading from a registry, uuidMachineRegistry is
1086 * always set: it's either the global registry UUID or a machine UUID when
1087 * loading from a per-machine registry.
1088 *
1089 * @param aVirtualBox VirtualBox object.
1090 * @param aParent Parent medium disk or NULL for a root (base) medium.
1091 * @param aDeviceType Device type of the medium.
1092 * @param uuidMachineRegistry The registry to which this medium should be added (global registry UUID or machine UUID).
1093 * @param aNode Configuration settings.
1094 * @param strMachineFolder The machine folder with which to resolve relative paths; if empty, then we use the VirtualBox home directory
1095 *
1096 * @note Locks the medium tree for writing.
1097 */
1098HRESULT Medium::init(VirtualBox *aVirtualBox,
1099 Medium *aParent,
1100 DeviceType_T aDeviceType,
1101 const Guid &uuidMachineRegistry,
1102 const settings::Medium &data,
1103 const Utf8Str &strMachineFolder)
1104{
1105 using namespace settings;
1106
1107 AssertReturn(aVirtualBox, E_INVALIDARG);
1108
1109 /* Enclose the state transition NotReady->InInit->Ready */
1110 AutoInitSpan autoInitSpan(this);
1111 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1112
1113 HRESULT rc = S_OK;
1114
1115 unconst(m->pVirtualBox) = aVirtualBox;
1116
1117 if (!uuidMachineRegistry.isEmpty())
1118 m->llRegistryIDs.push_back(uuidMachineRegistry);
1119
1120 /* register with VirtualBox/parent early, since uninit() will
1121 * unconditionally unregister on failure */
1122 if (aParent)
1123 {
1124 // differencing medium: add to parent
1125 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1126 m->pParent = aParent;
1127 aParent->m->llChildren.push_back(this);
1128 }
1129
1130 /* see below why we don't call Medium::queryInfo (and therefore treat
1131 * the medium as inaccessible for now */
1132 m->state = MediumState_Inaccessible;
1133 m->strLastAccessError = tr("Accessibility check was not yet performed");
1134
1135 /* required */
1136 unconst(m->id) = data.uuid;
1137
1138 /* assume not a host drive */
1139 m->hostDrive = false;
1140
1141 /* optional */
1142 m->strDescription = data.strDescription;
1143
1144 /* required */
1145 if (aDeviceType == DeviceType_HardDisk)
1146 {
1147 AssertReturn(!data.strFormat.isEmpty(), E_FAIL);
1148 rc = setFormat(data.strFormat);
1149 if (FAILED(rc)) return rc;
1150 }
1151 else
1152 {
1153 /// @todo handle host drive settings here as well?
1154 if (!data.strFormat.isEmpty())
1155 rc = setFormat(data.strFormat);
1156 else
1157 rc = setFormat("RAW");
1158 if (FAILED(rc)) return rc;
1159 }
1160
1161 /* optional, only for diffs, default is false; we can only auto-reset
1162 * diff media so they must have a parent */
1163 if (aParent != NULL)
1164 m->autoReset = data.fAutoReset;
1165 else
1166 m->autoReset = false;
1167
1168 /* properties (after setting the format as it populates the map). Note that
1169 * if some properties are not supported but present in the settings file,
1170 * they will still be read and accessible (for possible backward
1171 * compatibility; we can also clean them up from the XML upon next
1172 * XML format version change if we wish) */
1173 for (settings::StringsMap::const_iterator it = data.properties.begin();
1174 it != data.properties.end();
1175 ++it)
1176 {
1177 const Utf8Str &name = it->first;
1178 const Utf8Str &value = it->second;
1179 m->mapProperties[name] = value;
1180 }
1181
1182 Utf8Str strFull;
1183 if (m->formatObj->getCapabilities() & MediumFormatCapabilities_File)
1184 {
1185 // compose full path of the medium, if it's not fully qualified...
1186 // slightly convoluted logic here. If the caller has given us a
1187 // machine folder, then a relative path will be relative to that:
1188 if ( !strMachineFolder.isEmpty()
1189 && !RTPathStartsWithRoot(data.strLocation.c_str())
1190 )
1191 {
1192 strFull = strMachineFolder;
1193 strFull += RTPATH_SLASH;
1194 strFull += data.strLocation;
1195 }
1196 else
1197 {
1198 // Otherwise use the old VirtualBox "make absolute path" logic:
1199 rc = m->pVirtualBox->calculateFullPath(data.strLocation, strFull);
1200 if (FAILED(rc)) return rc;
1201 }
1202 }
1203 else
1204 strFull = data.strLocation;
1205
1206 rc = setLocation(strFull);
1207 if (FAILED(rc)) return rc;
1208
1209 if (aDeviceType == DeviceType_HardDisk)
1210 {
1211 /* type is only for base hard disks */
1212 if (m->pParent.isNull())
1213 m->type = data.hdType;
1214 }
1215 else if (aDeviceType == DeviceType_DVD)
1216 m->type = MediumType_Readonly;
1217 else
1218 m->type = MediumType_Writethrough;
1219
1220 /* remember device type for correct unregistering later */
1221 m->devType = aDeviceType;
1222
1223 LogFlowThisFunc(("m->strLocationFull='%s', m->strFormat=%s, m->id={%RTuuid}\n",
1224 m->strLocationFull.c_str(), m->strFormat.c_str(), m->id.raw()));
1225
1226 /* Don't call Medium::queryInfo for registered media to prevent the calling
1227 * thread (i.e. the VirtualBox server startup thread) from an unexpected
1228 * freeze but mark it as initially inaccessible instead. The vital UUID,
1229 * location and format properties are read from the registry file above; to
1230 * get the actual state and the rest of the data, the user will have to call
1231 * COMGETTER(State). */
1232
1233 AutoWriteLock treeLock(aVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1234
1235 /* load all children */
1236 for (settings::MediaList::const_iterator it = data.llChildren.begin();
1237 it != data.llChildren.end();
1238 ++it)
1239 {
1240 const settings::Medium &med = *it;
1241
1242 ComObjPtr<Medium> pHD;
1243 pHD.createObject();
1244 rc = pHD->init(aVirtualBox,
1245 this, // parent
1246 aDeviceType,
1247 uuidMachineRegistry,
1248 med, // child data
1249 strMachineFolder);
1250 if (FAILED(rc)) break;
1251
1252 rc = m->pVirtualBox->registerMedium(pHD, &pHD, DeviceType_HardDisk);
1253 if (FAILED(rc)) break;
1254 }
1255
1256 /* Confirm a successful initialization when it's the case */
1257 if (SUCCEEDED(rc))
1258 autoInitSpan.setSucceeded();
1259
1260 return rc;
1261}
1262
1263/**
1264 * Initializes the medium object by providing the host drive information.
1265 * Not used for anything but the host floppy/host DVD case.
1266 *
1267 * There is no registry for this case.
1268 *
1269 * @param aVirtualBox VirtualBox object.
1270 * @param aDeviceType Device type of the medium.
1271 * @param aLocation Location of the host drive.
1272 * @param aDescription Comment for this host drive.
1273 *
1274 * @note Locks VirtualBox lock for writing.
1275 */
1276HRESULT Medium::init(VirtualBox *aVirtualBox,
1277 DeviceType_T aDeviceType,
1278 const Utf8Str &aLocation,
1279 const Utf8Str &aDescription /* = Utf8Str::Empty */)
1280{
1281 ComAssertRet(aDeviceType == DeviceType_DVD || aDeviceType == DeviceType_Floppy, E_INVALIDARG);
1282 ComAssertRet(!aLocation.isEmpty(), E_INVALIDARG);
1283
1284 /* Enclose the state transition NotReady->InInit->Ready */
1285 AutoInitSpan autoInitSpan(this);
1286 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1287
1288 unconst(m->pVirtualBox) = aVirtualBox;
1289
1290 // We do not store host drives in VirtualBox.xml or anywhere else, so if we want
1291 // host drives to be identifiable by UUID and not give the drive a different UUID
1292 // every time VirtualBox starts, we need to fake a reproducible UUID here:
1293 RTUUID uuid;
1294 RTUuidClear(&uuid);
1295 if (aDeviceType == DeviceType_DVD)
1296 memcpy(&uuid.au8[0], "DVD", 3);
1297 else
1298 memcpy(&uuid.au8[0], "FD", 2);
1299 /* use device name, adjusted to the end of uuid, shortened if necessary */
1300 size_t lenLocation = aLocation.length();
1301 if (lenLocation > 12)
1302 memcpy(&uuid.au8[4], aLocation.c_str() + (lenLocation - 12), 12);
1303 else
1304 memcpy(&uuid.au8[4 + 12 - lenLocation], aLocation.c_str(), lenLocation);
1305 unconst(m->id) = uuid;
1306
1307 if (aDeviceType == DeviceType_DVD)
1308 m->type = MediumType_Readonly;
1309 else
1310 m->type = MediumType_Writethrough;
1311 m->devType = aDeviceType;
1312 m->state = MediumState_Created;
1313 m->hostDrive = true;
1314 HRESULT rc = setFormat("RAW");
1315 if (FAILED(rc)) return rc;
1316 rc = setLocation(aLocation);
1317 if (FAILED(rc)) return rc;
1318 m->strDescription = aDescription;
1319
1320 autoInitSpan.setSucceeded();
1321 return S_OK;
1322}
1323
1324/**
1325 * Uninitializes the instance.
1326 *
1327 * Called either from FinalRelease() or by the parent when it gets destroyed.
1328 *
1329 * @note All children of this medium get uninitialized by calling their
1330 * uninit() methods.
1331 */
1332void Medium::uninit()
1333{
1334 /* Enclose the state transition Ready->InUninit->NotReady */
1335 AutoUninitSpan autoUninitSpan(this);
1336 if (autoUninitSpan.uninitDone())
1337 return;
1338
1339 if (!m->formatObj.isNull())
1340 {
1341 /* remove the caller reference we added in setFormat() */
1342 m->formatObj->releaseCaller();
1343 m->formatObj.setNull();
1344 }
1345
1346 if (m->state == MediumState_Deleting)
1347 {
1348 /* This medium has been already deleted (directly or as part of a
1349 * merge). Reparenting has already been done. */
1350 Assert(m->pParent.isNull());
1351 }
1352 else
1353 {
1354 MediaList::iterator it;
1355 for (it = m->llChildren.begin();
1356 it != m->llChildren.end();
1357 ++it)
1358 {
1359 Medium *pChild = *it;
1360 pChild->m->pParent.setNull();
1361 pChild->uninit();
1362 }
1363 m->llChildren.clear(); // this unsets all the ComPtrs and probably calls delete
1364
1365 if (m->pParent)
1366 {
1367 // this is a differencing disk: then remove it from the parent's children list
1368 deparent();
1369 }
1370 }
1371
1372 unconst(m->pVirtualBox) = NULL;
1373}
1374
1375/**
1376 * Internal helper that removes "this" from the list of children of its
1377 * parent. Used in uninit() and other places when reparenting is necessary.
1378 *
1379 * The caller must hold the medium tree lock!
1380 */
1381void Medium::deparent()
1382{
1383 MediaList &llParent = m->pParent->m->llChildren;
1384 for (MediaList::iterator it = llParent.begin();
1385 it != llParent.end();
1386 ++it)
1387 {
1388 Medium *pParentsChild = *it;
1389 if (this == pParentsChild)
1390 {
1391 llParent.erase(it);
1392 break;
1393 }
1394 }
1395 m->pParent.setNull();
1396}
1397
1398/**
1399 * Internal helper that removes "this" from the list of children of its
1400 * parent. Used in uninit() and other places when reparenting is necessary.
1401 *
1402 * The caller must hold the medium tree lock!
1403 */
1404void Medium::setParent(const ComObjPtr<Medium> &pParent)
1405{
1406 m->pParent = pParent;
1407 if (pParent)
1408 pParent->m->llChildren.push_back(this);
1409}
1410
1411
1412////////////////////////////////////////////////////////////////////////////////
1413//
1414// IMedium public methods
1415//
1416////////////////////////////////////////////////////////////////////////////////
1417
1418STDMETHODIMP Medium::COMGETTER(Id)(BSTR *aId)
1419{
1420 CheckComArgOutPointerValid(aId);
1421
1422 AutoCaller autoCaller(this);
1423 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1424
1425 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1426
1427 m->id.toUtf16().cloneTo(aId);
1428
1429 return S_OK;
1430}
1431
1432STDMETHODIMP Medium::COMGETTER(Description)(BSTR *aDescription)
1433{
1434 CheckComArgOutPointerValid(aDescription);
1435
1436 AutoCaller autoCaller(this);
1437 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1438
1439 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1440
1441 m->strDescription.cloneTo(aDescription);
1442
1443 return S_OK;
1444}
1445
1446STDMETHODIMP Medium::COMSETTER(Description)(IN_BSTR aDescription)
1447{
1448 AutoCaller autoCaller(this);
1449 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1450
1451// AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1452
1453 /// @todo update m->description and save the global registry (and local
1454 /// registries of portable VMs referring to this medium), this will also
1455 /// require to add the mRegistered flag to data
1456
1457 NOREF(aDescription);
1458
1459 ReturnComNotImplemented();
1460}
1461
1462STDMETHODIMP Medium::COMGETTER(State)(MediumState_T *aState)
1463{
1464 CheckComArgOutPointerValid(aState);
1465
1466 AutoCaller autoCaller(this);
1467 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1468
1469 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1470 *aState = m->state;
1471
1472 return S_OK;
1473}
1474
1475STDMETHODIMP Medium::COMGETTER(Variant)(ULONG *aVariant)
1476{
1477 CheckComArgOutPointerValid(aVariant);
1478
1479 AutoCaller autoCaller(this);
1480 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1481
1482 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1483 *aVariant = m->variant;
1484
1485 return S_OK;
1486}
1487
1488
1489STDMETHODIMP Medium::COMGETTER(Location)(BSTR *aLocation)
1490{
1491 CheckComArgOutPointerValid(aLocation);
1492
1493 AutoCaller autoCaller(this);
1494 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1495
1496 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1497
1498 m->strLocationFull.cloneTo(aLocation);
1499
1500 return S_OK;
1501}
1502
1503STDMETHODIMP Medium::COMSETTER(Location)(IN_BSTR aLocation)
1504{
1505 CheckComArgStrNotEmptyOrNull(aLocation);
1506
1507 AutoCaller autoCaller(this);
1508 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1509
1510 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1511
1512 /// @todo NEWMEDIA for file names, add the default extension if no extension
1513 /// is present (using the information from the VD backend which also implies
1514 /// that one more parameter should be passed to setLocation() requesting
1515 /// that functionality since it is only allowed when called from this method
1516
1517 /// @todo NEWMEDIA rename the file and set m->location on success, then save
1518 /// the global registry (and local registries of portable VMs referring to
1519 /// this medium), this will also require to add the mRegistered flag to data
1520
1521 ReturnComNotImplemented();
1522}
1523
1524STDMETHODIMP Medium::COMGETTER(Name)(BSTR *aName)
1525{
1526 CheckComArgOutPointerValid(aName);
1527
1528 AutoCaller autoCaller(this);
1529 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1530
1531 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1532
1533 getName().cloneTo(aName);
1534
1535 return S_OK;
1536}
1537
1538STDMETHODIMP Medium::COMGETTER(DeviceType)(DeviceType_T *aDeviceType)
1539{
1540 CheckComArgOutPointerValid(aDeviceType);
1541
1542 AutoCaller autoCaller(this);
1543 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1544
1545 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1546
1547 *aDeviceType = m->devType;
1548
1549 return S_OK;
1550}
1551
1552STDMETHODIMP Medium::COMGETTER(HostDrive)(BOOL *aHostDrive)
1553{
1554 CheckComArgOutPointerValid(aHostDrive);
1555
1556 AutoCaller autoCaller(this);
1557 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1558
1559 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1560
1561 *aHostDrive = m->hostDrive;
1562
1563 return S_OK;
1564}
1565
1566STDMETHODIMP Medium::COMGETTER(Size)(LONG64 *aSize)
1567{
1568 CheckComArgOutPointerValid(aSize);
1569
1570 AutoCaller autoCaller(this);
1571 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1572
1573 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1574
1575 *aSize = m->size;
1576
1577 return S_OK;
1578}
1579
1580STDMETHODIMP Medium::COMGETTER(Format)(BSTR *aFormat)
1581{
1582 CheckComArgOutPointerValid(aFormat);
1583
1584 AutoCaller autoCaller(this);
1585 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1586
1587 /* no need to lock, m->strFormat is const */
1588 m->strFormat.cloneTo(aFormat);
1589
1590 return S_OK;
1591}
1592
1593STDMETHODIMP Medium::COMGETTER(MediumFormat)(IMediumFormat **aMediumFormat)
1594{
1595 CheckComArgOutPointerValid(aMediumFormat);
1596
1597 AutoCaller autoCaller(this);
1598 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1599
1600 /* no need to lock, m->formatObj is const */
1601 m->formatObj.queryInterfaceTo(aMediumFormat);
1602
1603 return S_OK;
1604}
1605
1606STDMETHODIMP Medium::COMGETTER(Type)(MediumType_T *aType)
1607{
1608 CheckComArgOutPointerValid(aType);
1609
1610 AutoCaller autoCaller(this);
1611 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1612
1613 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1614
1615 *aType = m->type;
1616
1617 return S_OK;
1618}
1619
1620STDMETHODIMP Medium::COMSETTER(Type)(MediumType_T aType)
1621{
1622 AutoCaller autoCaller(this);
1623 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1624
1625 // we access mParent and members
1626 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1627 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
1628
1629 switch (m->state)
1630 {
1631 case MediumState_Created:
1632 case MediumState_Inaccessible:
1633 break;
1634 default:
1635 return setStateError();
1636 }
1637
1638 if (m->type == aType)
1639 {
1640 /* Nothing to do */
1641 return S_OK;
1642 }
1643
1644 DeviceType_T devType = getDeviceType();
1645 // DVD media can only be readonly.
1646 if (devType == DeviceType_DVD && aType != MediumType_Readonly)
1647 return setError(VBOX_E_INVALID_OBJECT_STATE,
1648 tr("Cannot change the type of DVD medium '%s'"),
1649 m->strLocationFull.c_str());
1650 // Floppy media can only be writethrough or readonly.
1651 if ( devType == DeviceType_Floppy
1652 && aType != MediumType_Writethrough
1653 && aType != MediumType_Readonly)
1654 return setError(VBOX_E_INVALID_OBJECT_STATE,
1655 tr("Cannot change the type of floppy medium '%s'"),
1656 m->strLocationFull.c_str());
1657
1658 /* cannot change the type of a differencing medium */
1659 if (m->pParent)
1660 return setError(VBOX_E_INVALID_OBJECT_STATE,
1661 tr("Cannot change the type of medium '%s' because it is a differencing medium"),
1662 m->strLocationFull.c_str());
1663
1664 /* Cannot change the type of a medium being in use by more than one VM.
1665 * If the change is to Immutable or MultiAttach then it must not be
1666 * directly attached to any VM, otherwise the assumptions about indirect
1667 * attachment elsewhere are violated and the VM becomes inaccessible.
1668 * Attaching an immutable medium triggers the diff creation, and this is
1669 * vital for the correct operation. */
1670 if ( m->backRefs.size() > 1
1671 || ( ( aType == MediumType_Immutable
1672 || aType == MediumType_MultiAttach)
1673 && m->backRefs.size() > 0))
1674 return setError(VBOX_E_INVALID_OBJECT_STATE,
1675 tr("Cannot change the type of medium '%s' because it is attached to %d virtual machines"),
1676 m->strLocationFull.c_str(), m->backRefs.size());
1677
1678 switch (aType)
1679 {
1680 case MediumType_Normal:
1681 case MediumType_Immutable:
1682 case MediumType_MultiAttach:
1683 {
1684 /* normal can be easily converted to immutable and vice versa even
1685 * if they have children as long as they are not attached to any
1686 * machine themselves */
1687 break;
1688 }
1689 case MediumType_Writethrough:
1690 case MediumType_Shareable:
1691 case MediumType_Readonly:
1692 {
1693 /* cannot change to writethrough, shareable or readonly
1694 * if there are children */
1695 if (getChildren().size() != 0)
1696 return setError(VBOX_E_OBJECT_IN_USE,
1697 tr("Cannot change type for medium '%s' since it has %d child media"),
1698 m->strLocationFull.c_str(), getChildren().size());
1699 if (aType == MediumType_Shareable)
1700 {
1701 MediumVariant_T variant = getVariant();
1702 if (!(variant & MediumVariant_Fixed))
1703 return setError(VBOX_E_INVALID_OBJECT_STATE,
1704 tr("Cannot change type for medium '%s' to 'Shareable' since it is a dynamic medium storage unit"),
1705 m->strLocationFull.c_str());
1706 }
1707 else if (aType == MediumType_Readonly && devType == DeviceType_HardDisk)
1708 {
1709 // Readonly hard disks are not allowed, this medium type is reserved for
1710 // DVDs and floppy images at the moment. Later we might allow readonly hard
1711 // disks, but that's extremely unusual and many guest OSes will have trouble.
1712 return setError(VBOX_E_INVALID_OBJECT_STATE,
1713 tr("Cannot change type for medium '%s' to 'Readonly' since it is a hard disk"),
1714 m->strLocationFull.c_str());
1715 }
1716 break;
1717 }
1718 default:
1719 AssertFailedReturn(E_FAIL);
1720 }
1721
1722 if (aType == MediumType_MultiAttach)
1723 {
1724 // This type is new with VirtualBox 4.0 and therefore requires settings
1725 // version 1.11 in the settings backend. Unfortunately it is not enough to do
1726 // the usual routine in MachineConfigFile::bumpSettingsVersionIfNeeded() for
1727 // two reasons: The medium type is a property of the media registry tree, which
1728 // can reside in the global config file (for pre-4.0 media); we would therefore
1729 // possibly need to bump the global config version. We don't want to do that though
1730 // because that might make downgrading to pre-4.0 impossible.
1731 // As a result, we can only use these two new types if the medium is NOT in the
1732 // global registry:
1733 const Guid &uuidGlobalRegistry = m->pVirtualBox->getGlobalRegistryId();
1734 if (isInRegistry(uuidGlobalRegistry))
1735 return setError(VBOX_E_INVALID_OBJECT_STATE,
1736 tr("Cannot change type for medium '%s': the media type 'MultiAttach' can only be used "
1737 "on media registered with a machine that was created with VirtualBox 4.0 or later"),
1738 m->strLocationFull.c_str());
1739 }
1740
1741 m->type = aType;
1742
1743 // save the settings
1744 mlock.release();
1745 treeLock.release();
1746 markRegistriesModified();
1747 m->pVirtualBox->saveModifiedRegistries();
1748
1749 return S_OK;
1750}
1751
1752STDMETHODIMP Medium::COMGETTER(AllowedTypes)(ComSafeArrayOut(MediumType_T, aAllowedTypes))
1753{
1754 CheckComArgOutSafeArrayPointerValid(aAllowedTypes);
1755 NOREF(aAllowedTypes);
1756#ifndef RT_OS_WINDOWS
1757 NOREF(aAllowedTypesSize);
1758#endif
1759
1760 AutoCaller autoCaller(this);
1761 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1762
1763 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1764
1765 ReturnComNotImplemented();
1766}
1767
1768STDMETHODIMP Medium::COMGETTER(Parent)(IMedium **aParent)
1769{
1770 CheckComArgOutPointerValid(aParent);
1771
1772 AutoCaller autoCaller(this);
1773 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1774
1775 /* we access mParent */
1776 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1777
1778 m->pParent.queryInterfaceTo(aParent);
1779
1780 return S_OK;
1781}
1782
1783STDMETHODIMP Medium::COMGETTER(Children)(ComSafeArrayOut(IMedium *, aChildren))
1784{
1785 CheckComArgOutSafeArrayPointerValid(aChildren);
1786
1787 AutoCaller autoCaller(this);
1788 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1789
1790 /* we access children */
1791 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1792
1793 SafeIfaceArray<IMedium> children(this->getChildren());
1794 children.detachTo(ComSafeArrayOutArg(aChildren));
1795
1796 return S_OK;
1797}
1798
1799STDMETHODIMP Medium::COMGETTER(Base)(IMedium **aBase)
1800{
1801 CheckComArgOutPointerValid(aBase);
1802
1803 /* base() will do callers/locking */
1804
1805 getBase().queryInterfaceTo(aBase);
1806
1807 return S_OK;
1808}
1809
1810STDMETHODIMP Medium::COMGETTER(ReadOnly)(BOOL *aReadOnly)
1811{
1812 CheckComArgOutPointerValid(aReadOnly);
1813
1814 AutoCaller autoCaller(this);
1815 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1816
1817 /* isReadOnly() will do locking */
1818
1819 *aReadOnly = isReadOnly();
1820
1821 return S_OK;
1822}
1823
1824STDMETHODIMP Medium::COMGETTER(LogicalSize)(LONG64 *aLogicalSize)
1825{
1826 CheckComArgOutPointerValid(aLogicalSize);
1827
1828 {
1829 AutoCaller autoCaller(this);
1830 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1831
1832 /* we access mParent */
1833 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1834
1835 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1836
1837 if (m->pParent.isNull())
1838 {
1839 *aLogicalSize = m->logicalSize;
1840
1841 return S_OK;
1842 }
1843 }
1844
1845 /* We assume that some backend may decide to return a meaningless value in
1846 * response to VDGetSize() for differencing media and therefore always
1847 * ask the base medium ourselves. */
1848
1849 /* base() will do callers/locking */
1850
1851 return getBase()->COMGETTER(LogicalSize)(aLogicalSize);
1852}
1853
1854STDMETHODIMP Medium::COMGETTER(AutoReset)(BOOL *aAutoReset)
1855{
1856 CheckComArgOutPointerValid(aAutoReset);
1857
1858 AutoCaller autoCaller(this);
1859 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1860
1861 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1862
1863 if (m->pParent.isNull())
1864 *aAutoReset = FALSE;
1865 else
1866 *aAutoReset = m->autoReset;
1867
1868 return S_OK;
1869}
1870
1871STDMETHODIMP Medium::COMSETTER(AutoReset)(BOOL aAutoReset)
1872{
1873 AutoCaller autoCaller(this);
1874 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1875
1876 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
1877
1878 if (m->pParent.isNull())
1879 return setError(VBOX_E_NOT_SUPPORTED,
1880 tr("Medium '%s' is not differencing"),
1881 m->strLocationFull.c_str());
1882
1883 if (m->autoReset != !!aAutoReset)
1884 {
1885 m->autoReset = !!aAutoReset;
1886
1887 // save the settings
1888 mlock.release();
1889 markRegistriesModified();
1890 m->pVirtualBox->saveModifiedRegistries();
1891 }
1892
1893 return S_OK;
1894}
1895
1896STDMETHODIMP Medium::COMGETTER(LastAccessError)(BSTR *aLastAccessError)
1897{
1898 CheckComArgOutPointerValid(aLastAccessError);
1899
1900 AutoCaller autoCaller(this);
1901 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1902
1903 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1904
1905 m->strLastAccessError.cloneTo(aLastAccessError);
1906
1907 return S_OK;
1908}
1909
1910STDMETHODIMP Medium::COMGETTER(MachineIds)(ComSafeArrayOut(BSTR,aMachineIds))
1911{
1912 CheckComArgOutSafeArrayPointerValid(aMachineIds);
1913
1914 AutoCaller autoCaller(this);
1915 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1916
1917 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1918
1919 com::SafeArray<BSTR> machineIds;
1920
1921 if (m->backRefs.size() != 0)
1922 {
1923 machineIds.reset(m->backRefs.size());
1924
1925 size_t i = 0;
1926 for (BackRefList::const_iterator it = m->backRefs.begin();
1927 it != m->backRefs.end(); ++it, ++i)
1928 {
1929 it->machineId.toUtf16().detachTo(&machineIds[i]);
1930 }
1931 }
1932
1933 machineIds.detachTo(ComSafeArrayOutArg(aMachineIds));
1934
1935 return S_OK;
1936}
1937
1938STDMETHODIMP Medium::SetIDs(BOOL aSetImageId,
1939 IN_BSTR aImageId,
1940 BOOL aSetParentId,
1941 IN_BSTR aParentId)
1942{
1943 AutoCaller autoCaller(this);
1944 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1945
1946 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1947
1948 switch (m->state)
1949 {
1950 case MediumState_Created:
1951 break;
1952 default:
1953 return setStateError();
1954 }
1955
1956 Guid imageId, parentId;
1957 if (aSetImageId)
1958 {
1959 if (Bstr(aImageId).isEmpty())
1960 imageId.create();
1961 else
1962 {
1963 imageId = Guid(aImageId);
1964 if (imageId.isEmpty())
1965 return setError(E_INVALIDARG, tr("Argument %s is empty"), "aImageId");
1966 }
1967 }
1968 if (aSetParentId)
1969 {
1970 if (Bstr(aParentId).isEmpty())
1971 parentId.create();
1972 else
1973 parentId = Guid(aParentId);
1974 }
1975
1976 unconst(m->uuidImage) = imageId;
1977 unconst(m->uuidParentImage) = parentId;
1978
1979 // must not hold any locks before calling Medium::queryInfo
1980 alock.release();
1981
1982 HRESULT rc = queryInfo(!!aSetImageId /* fSetImageId */,
1983 !!aSetParentId /* fSetParentId */);
1984
1985 return rc;
1986}
1987
1988STDMETHODIMP Medium::RefreshState(MediumState_T *aState)
1989{
1990 CheckComArgOutPointerValid(aState);
1991
1992 AutoCaller autoCaller(this);
1993 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1994
1995 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1996
1997 HRESULT rc = S_OK;
1998
1999 switch (m->state)
2000 {
2001 case MediumState_Created:
2002 case MediumState_Inaccessible:
2003 case MediumState_LockedRead:
2004 {
2005 // must not hold any locks before calling Medium::queryInfo
2006 alock.release();
2007
2008 rc = queryInfo(false /* fSetImageId */, false /* fSetParentId */);
2009
2010 alock.acquire();
2011 break;
2012 }
2013 default:
2014 break;
2015 }
2016
2017 *aState = m->state;
2018
2019 return rc;
2020}
2021
2022STDMETHODIMP Medium::GetSnapshotIds(IN_BSTR aMachineId,
2023 ComSafeArrayOut(BSTR, aSnapshotIds))
2024{
2025 CheckComArgExpr(aMachineId, Guid(aMachineId).isEmpty() == false);
2026 CheckComArgOutSafeArrayPointerValid(aSnapshotIds);
2027
2028 AutoCaller autoCaller(this);
2029 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2030
2031 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2032
2033 com::SafeArray<BSTR> snapshotIds;
2034
2035 Guid id(aMachineId);
2036 for (BackRefList::const_iterator it = m->backRefs.begin();
2037 it != m->backRefs.end(); ++it)
2038 {
2039 if (it->machineId == id)
2040 {
2041 size_t size = it->llSnapshotIds.size();
2042
2043 /* if the medium is attached to the machine in the current state, we
2044 * return its ID as the first element of the array */
2045 if (it->fInCurState)
2046 ++size;
2047
2048 if (size > 0)
2049 {
2050 snapshotIds.reset(size);
2051
2052 size_t j = 0;
2053 if (it->fInCurState)
2054 it->machineId.toUtf16().detachTo(&snapshotIds[j++]);
2055
2056 for (GuidList::const_iterator jt = it->llSnapshotIds.begin();
2057 jt != it->llSnapshotIds.end();
2058 ++jt, ++j)
2059 {
2060 (*jt).toUtf16().detachTo(&snapshotIds[j]);
2061 }
2062 }
2063
2064 break;
2065 }
2066 }
2067
2068 snapshotIds.detachTo(ComSafeArrayOutArg(aSnapshotIds));
2069
2070 return S_OK;
2071}
2072
2073/**
2074 * @note @a aState may be NULL if the state value is not needed (only for
2075 * in-process calls).
2076 */
2077STDMETHODIMP Medium::LockRead(MediumState_T *aState)
2078{
2079 AutoCaller autoCaller(this);
2080 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2081
2082 /* Must not hold the object lock, as we need control over it below. */
2083 Assert(!isWriteLockOnCurrentThread());
2084 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2085
2086 /* Wait for a concurrently running Medium::queryInfo to complete. */
2087 if (m->queryInfoRunning)
2088 {
2089 /* Must not hold the media tree lock, as Medium::queryInfo needs this
2090 * lock and thus we would run into a deadlock here. */
2091 Assert(!m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2092 while (m->queryInfoRunning)
2093 {
2094 alock.release();
2095 {
2096 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2097 }
2098 alock.acquire();
2099 }
2100 }
2101
2102 /* return the current state before */
2103 if (aState)
2104 *aState = m->state;
2105
2106 HRESULT rc = S_OK;
2107
2108 switch (m->state)
2109 {
2110 case MediumState_Created:
2111 case MediumState_Inaccessible:
2112 case MediumState_LockedRead:
2113 {
2114 ++m->readers;
2115
2116 ComAssertMsgBreak(m->readers != 0, ("Counter overflow"), rc = E_FAIL);
2117
2118 /* Remember pre-lock state */
2119 if (m->state != MediumState_LockedRead)
2120 m->preLockState = m->state;
2121
2122 LogFlowThisFunc(("Okay - prev state=%d readers=%d\n", m->state, m->readers));
2123 m->state = MediumState_LockedRead;
2124
2125 break;
2126 }
2127 default:
2128 {
2129 LogFlowThisFunc(("Failing - state=%d\n", m->state));
2130 rc = setStateError();
2131 break;
2132 }
2133 }
2134
2135 return rc;
2136}
2137
2138/**
2139 * @note @a aState may be NULL if the state value is not needed (only for
2140 * in-process calls).
2141 */
2142STDMETHODIMP Medium::UnlockRead(MediumState_T *aState)
2143{
2144 AutoCaller autoCaller(this);
2145 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2146
2147 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2148
2149 HRESULT rc = S_OK;
2150
2151 switch (m->state)
2152 {
2153 case MediumState_LockedRead:
2154 {
2155 ComAssertMsgBreak(m->readers != 0, ("Counter underflow"), rc = E_FAIL);
2156 --m->readers;
2157
2158 /* Reset the state after the last reader */
2159 if (m->readers == 0)
2160 {
2161 m->state = m->preLockState;
2162 /* There are cases where we inject the deleting state into
2163 * a medium locked for reading. Make sure #unmarkForDeletion()
2164 * gets the right state afterwards. */
2165 if (m->preLockState == MediumState_Deleting)
2166 m->preLockState = MediumState_Created;
2167 }
2168
2169 LogFlowThisFunc(("new state=%d\n", m->state));
2170 break;
2171 }
2172 default:
2173 {
2174 LogFlowThisFunc(("Failing - state=%d\n", m->state));
2175 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
2176 tr("Medium '%s' is not locked for reading"),
2177 m->strLocationFull.c_str());
2178 break;
2179 }
2180 }
2181
2182 /* return the current state after */
2183 if (aState)
2184 *aState = m->state;
2185
2186 return rc;
2187}
2188
2189/**
2190 * @note @a aState may be NULL if the state value is not needed (only for
2191 * in-process calls).
2192 */
2193STDMETHODIMP Medium::LockWrite(MediumState_T *aState)
2194{
2195 AutoCaller autoCaller(this);
2196 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2197
2198 /* Must not hold the object lock, as we need control over it below. */
2199 Assert(!isWriteLockOnCurrentThread());
2200 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2201
2202 /* Wait for a concurrently running Medium::queryInfo to complete. */
2203 if (m->queryInfoRunning)
2204 {
2205 /* Must not hold the media tree lock, as Medium::queryInfo needs this
2206 * lock and thus we would run into a deadlock here. */
2207 Assert(!m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2208 while (m->queryInfoRunning)
2209 {
2210 alock.release();
2211 {
2212 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2213 }
2214 alock.acquire();
2215 }
2216 }
2217
2218 /* return the current state before */
2219 if (aState)
2220 *aState = m->state;
2221
2222 HRESULT rc = S_OK;
2223
2224 switch (m->state)
2225 {
2226 case MediumState_Created:
2227 case MediumState_Inaccessible:
2228 {
2229 m->preLockState = m->state;
2230
2231 LogFlowThisFunc(("Okay - prev state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
2232 m->state = MediumState_LockedWrite;
2233 break;
2234 }
2235 default:
2236 {
2237 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
2238 rc = setStateError();
2239 break;
2240 }
2241 }
2242
2243 return rc;
2244}
2245
2246/**
2247 * @note @a aState may be NULL if the state value is not needed (only for
2248 * in-process calls).
2249 */
2250STDMETHODIMP Medium::UnlockWrite(MediumState_T *aState)
2251{
2252 AutoCaller autoCaller(this);
2253 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2254
2255 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2256
2257 HRESULT rc = S_OK;
2258
2259 switch (m->state)
2260 {
2261 case MediumState_LockedWrite:
2262 {
2263 m->state = m->preLockState;
2264 /* There are cases where we inject the deleting state into
2265 * a medium locked for writing. Make sure #unmarkForDeletion()
2266 * gets the right state afterwards. */
2267 if (m->preLockState == MediumState_Deleting)
2268 m->preLockState = MediumState_Created;
2269 LogFlowThisFunc(("new state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
2270 break;
2271 }
2272 default:
2273 {
2274 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
2275 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
2276 tr("Medium '%s' is not locked for writing"),
2277 m->strLocationFull.c_str());
2278 break;
2279 }
2280 }
2281
2282 /* return the current state after */
2283 if (aState)
2284 *aState = m->state;
2285
2286 return rc;
2287}
2288
2289STDMETHODIMP Medium::Close()
2290{
2291 AutoCaller autoCaller(this);
2292 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2293
2294 // make a copy of VirtualBox pointer which gets nulled by uninit()
2295 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
2296
2297 MultiResult mrc = close(autoCaller);
2298
2299 pVirtualBox->saveModifiedRegistries();
2300
2301 return mrc;
2302}
2303
2304STDMETHODIMP Medium::GetProperty(IN_BSTR aName, BSTR *aValue)
2305{
2306 CheckComArgStrNotEmptyOrNull(aName);
2307 CheckComArgOutPointerValid(aValue);
2308
2309 AutoCaller autoCaller(this);
2310 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2311
2312 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2313
2314 settings::StringsMap::const_iterator it = m->mapProperties.find(Utf8Str(aName));
2315 if (it == m->mapProperties.end())
2316 return setError(VBOX_E_OBJECT_NOT_FOUND,
2317 tr("Property '%ls' does not exist"), aName);
2318
2319 it->second.cloneTo(aValue);
2320
2321 return S_OK;
2322}
2323
2324STDMETHODIMP Medium::SetProperty(IN_BSTR aName, IN_BSTR aValue)
2325{
2326 CheckComArgStrNotEmptyOrNull(aName);
2327
2328 AutoCaller autoCaller(this);
2329 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2330
2331 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
2332
2333 switch (m->state)
2334 {
2335 case MediumState_Created:
2336 case MediumState_Inaccessible:
2337 break;
2338 default:
2339 return setStateError();
2340 }
2341
2342 settings::StringsMap::iterator it = m->mapProperties.find(Utf8Str(aName));
2343 if (it == m->mapProperties.end())
2344 return setError(VBOX_E_OBJECT_NOT_FOUND,
2345 tr("Property '%ls' does not exist"),
2346 aName);
2347
2348 it->second = aValue;
2349
2350 // save the settings
2351 mlock.release();
2352 markRegistriesModified();
2353 m->pVirtualBox->saveModifiedRegistries();
2354
2355 return S_OK;
2356}
2357
2358STDMETHODIMP Medium::GetProperties(IN_BSTR aNames,
2359 ComSafeArrayOut(BSTR, aReturnNames),
2360 ComSafeArrayOut(BSTR, aReturnValues))
2361{
2362 CheckComArgOutSafeArrayPointerValid(aReturnNames);
2363 CheckComArgOutSafeArrayPointerValid(aReturnValues);
2364
2365 AutoCaller autoCaller(this);
2366 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2367
2368 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2369
2370 /// @todo make use of aNames according to the documentation
2371 NOREF(aNames);
2372
2373 com::SafeArray<BSTR> names(m->mapProperties.size());
2374 com::SafeArray<BSTR> values(m->mapProperties.size());
2375 size_t i = 0;
2376
2377 for (settings::StringsMap::const_iterator it = m->mapProperties.begin();
2378 it != m->mapProperties.end();
2379 ++it)
2380 {
2381 it->first.cloneTo(&names[i]);
2382 it->second.cloneTo(&values[i]);
2383 ++i;
2384 }
2385
2386 names.detachTo(ComSafeArrayOutArg(aReturnNames));
2387 values.detachTo(ComSafeArrayOutArg(aReturnValues));
2388
2389 return S_OK;
2390}
2391
2392STDMETHODIMP Medium::SetProperties(ComSafeArrayIn(IN_BSTR, aNames),
2393 ComSafeArrayIn(IN_BSTR, aValues))
2394{
2395 CheckComArgSafeArrayNotNull(aNames);
2396 CheckComArgSafeArrayNotNull(aValues);
2397
2398 AutoCaller autoCaller(this);
2399 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2400
2401 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
2402
2403 com::SafeArray<IN_BSTR> names(ComSafeArrayInArg(aNames));
2404 com::SafeArray<IN_BSTR> values(ComSafeArrayInArg(aValues));
2405
2406 /* first pass: validate names */
2407 for (size_t i = 0;
2408 i < names.size();
2409 ++i)
2410 {
2411 if (m->mapProperties.find(Utf8Str(names[i])) == m->mapProperties.end())
2412 return setError(VBOX_E_OBJECT_NOT_FOUND,
2413 tr("Property '%ls' does not exist"), names[i]);
2414 }
2415
2416 /* second pass: assign */
2417 for (size_t i = 0;
2418 i < names.size();
2419 ++i)
2420 {
2421 settings::StringsMap::iterator it = m->mapProperties.find(Utf8Str(names[i]));
2422 AssertReturn(it != m->mapProperties.end(), E_FAIL);
2423
2424 it->second = Utf8Str(values[i]);
2425 }
2426
2427 // save the settings
2428 mlock.release();
2429 markRegistriesModified();
2430 m->pVirtualBox->saveModifiedRegistries();
2431
2432 return S_OK;
2433}
2434
2435STDMETHODIMP Medium::CreateBaseStorage(LONG64 aLogicalSize,
2436 ULONG aVariant,
2437 IProgress **aProgress)
2438{
2439 CheckComArgOutPointerValid(aProgress);
2440 if (aLogicalSize < 0)
2441 return setError(E_INVALIDARG, tr("The medium size argument (%lld) is negative"), aLogicalSize);
2442
2443 AutoCaller autoCaller(this);
2444 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2445
2446 HRESULT rc = S_OK;
2447 ComObjPtr <Progress> pProgress;
2448 Medium::Task *pTask = NULL;
2449
2450 try
2451 {
2452 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2453
2454 aVariant = (MediumVariant_T)((unsigned)aVariant & (unsigned)~MediumVariant_Diff);
2455 if ( !(aVariant & MediumVariant_Fixed)
2456 && !(m->formatObj->getCapabilities() & MediumFormatCapabilities_CreateDynamic))
2457 throw setError(VBOX_E_NOT_SUPPORTED,
2458 tr("Medium format '%s' does not support dynamic storage creation"),
2459 m->strFormat.c_str());
2460 if ( (aVariant & MediumVariant_Fixed)
2461 && !(m->formatObj->getCapabilities() & MediumFormatCapabilities_CreateDynamic))
2462 throw setError(VBOX_E_NOT_SUPPORTED,
2463 tr("Medium format '%s' does not support fixed storage creation"),
2464 m->strFormat.c_str());
2465
2466 if (m->state != MediumState_NotCreated)
2467 throw setStateError();
2468
2469 pProgress.createObject();
2470 rc = pProgress->init(m->pVirtualBox,
2471 static_cast<IMedium*>(this),
2472 (aVariant & MediumVariant_Fixed)
2473 ? BstrFmt(tr("Creating fixed medium storage unit '%s'"), m->strLocationFull.c_str()).raw()
2474 : BstrFmt(tr("Creating dynamic medium storage unit '%s'"), m->strLocationFull.c_str()).raw(),
2475 TRUE /* aCancelable */);
2476 if (FAILED(rc))
2477 throw rc;
2478
2479 /* setup task object to carry out the operation asynchronously */
2480 pTask = new Medium::CreateBaseTask(this, pProgress, aLogicalSize,
2481 (MediumVariant_T)aVariant);
2482 rc = pTask->rc();
2483 AssertComRC(rc);
2484 if (FAILED(rc))
2485 throw rc;
2486
2487 m->state = MediumState_Creating;
2488 }
2489 catch (HRESULT aRC) { rc = aRC; }
2490
2491 if (SUCCEEDED(rc))
2492 {
2493 rc = startThread(pTask);
2494
2495 if (SUCCEEDED(rc))
2496 pProgress.queryInterfaceTo(aProgress);
2497 }
2498 else if (pTask != NULL)
2499 delete pTask;
2500
2501 return rc;
2502}
2503
2504STDMETHODIMP Medium::DeleteStorage(IProgress **aProgress)
2505{
2506 CheckComArgOutPointerValid(aProgress);
2507
2508 AutoCaller autoCaller(this);
2509 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2510
2511 ComObjPtr<Progress> pProgress;
2512
2513 MultiResult mrc = deleteStorage(&pProgress,
2514 false /* aWait */);
2515 /* Must save the registries in any case, since an entry was removed. */
2516 m->pVirtualBox->saveModifiedRegistries();
2517
2518 if (SUCCEEDED(mrc))
2519 pProgress.queryInterfaceTo(aProgress);
2520
2521 return mrc;
2522}
2523
2524STDMETHODIMP Medium::CreateDiffStorage(IMedium *aTarget,
2525 ULONG aVariant,
2526 IProgress **aProgress)
2527{
2528 CheckComArgNotNull(aTarget);
2529 CheckComArgOutPointerValid(aProgress);
2530
2531 AutoCaller autoCaller(this);
2532 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2533
2534 ComObjPtr<Medium> diff = static_cast<Medium*>(aTarget);
2535
2536 // locking: we need the tree lock first because we access parent pointers
2537 AutoMultiWriteLock3 alock(&m->pVirtualBox->getMediaTreeLockHandle(),
2538 this->lockHandle(), diff->lockHandle() COMMA_LOCKVAL_SRC_POS);
2539
2540 if (m->type == MediumType_Writethrough)
2541 return setError(VBOX_E_INVALID_OBJECT_STATE,
2542 tr("Medium type of '%s' is Writethrough"),
2543 m->strLocationFull.c_str());
2544 else if (m->type == MediumType_Shareable)
2545 return setError(VBOX_E_INVALID_OBJECT_STATE,
2546 tr("Medium type of '%s' is Shareable"),
2547 m->strLocationFull.c_str());
2548 else if (m->type == MediumType_Readonly)
2549 return setError(VBOX_E_INVALID_OBJECT_STATE,
2550 tr("Medium type of '%s' is Readonly"),
2551 m->strLocationFull.c_str());
2552
2553 /* Apply the normal locking logic to the entire chain. */
2554 MediumLockList *pMediumLockList(new MediumLockList());
2555 alock.release();
2556 HRESULT rc = diff->createMediumLockList(true /* fFailIfInaccessible */,
2557 true /* fMediumLockWrite */,
2558 this,
2559 *pMediumLockList);
2560 alock.acquire();
2561 if (FAILED(rc))
2562 {
2563 delete pMediumLockList;
2564 return rc;
2565 }
2566
2567 alock.release();
2568 rc = pMediumLockList->Lock();
2569 alock.acquire();
2570 if (FAILED(rc))
2571 {
2572 delete pMediumLockList;
2573
2574 return setError(rc, tr("Could not lock medium when creating diff '%s'"),
2575 diff->getLocationFull().c_str());
2576 }
2577
2578 Guid parentMachineRegistry;
2579 if (getFirstRegistryMachineId(parentMachineRegistry))
2580 {
2581 /* since this medium has been just created it isn't associated yet */
2582 diff->m->llRegistryIDs.push_back(parentMachineRegistry);
2583 alock.release();
2584 diff->markRegistriesModified();
2585 alock.acquire();
2586 }
2587
2588 alock.release();
2589
2590 ComObjPtr <Progress> pProgress;
2591
2592 rc = createDiffStorage(diff, (MediumVariant_T)aVariant, pMediumLockList,
2593 &pProgress, false /* aWait */);
2594 if (FAILED(rc))
2595 delete pMediumLockList;
2596 else
2597 pProgress.queryInterfaceTo(aProgress);
2598
2599 return rc;
2600}
2601
2602STDMETHODIMP Medium::MergeTo(IMedium *aTarget, IProgress **aProgress)
2603{
2604 CheckComArgNotNull(aTarget);
2605 CheckComArgOutPointerValid(aProgress);
2606 ComAssertRet(aTarget != this, E_INVALIDARG);
2607
2608 AutoCaller autoCaller(this);
2609 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2610
2611 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aTarget);
2612
2613 bool fMergeForward = false;
2614 ComObjPtr<Medium> pParentForTarget;
2615 MediaList childrenToReparent;
2616 MediumLockList *pMediumLockList = NULL;
2617
2618 HRESULT rc = S_OK;
2619
2620 rc = prepareMergeTo(pTarget, NULL, NULL, true, fMergeForward,
2621 pParentForTarget, childrenToReparent, pMediumLockList);
2622 if (FAILED(rc)) return rc;
2623
2624 ComObjPtr <Progress> pProgress;
2625
2626 rc = mergeTo(pTarget, fMergeForward, pParentForTarget, childrenToReparent,
2627 pMediumLockList, &pProgress, false /* aWait */);
2628 if (FAILED(rc))
2629 cancelMergeTo(childrenToReparent, pMediumLockList);
2630 else
2631 pProgress.queryInterfaceTo(aProgress);
2632
2633 return rc;
2634}
2635
2636STDMETHODIMP Medium::CloneTo(IMedium *aTarget,
2637 ULONG aVariant,
2638 IMedium *aParent,
2639 IProgress **aProgress)
2640{
2641 CheckComArgNotNull(aTarget);
2642 CheckComArgOutPointerValid(aProgress);
2643 ComAssertRet(aTarget != this, E_INVALIDARG);
2644
2645 AutoCaller autoCaller(this);
2646 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2647
2648 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aTarget);
2649 ComObjPtr<Medium> pParent;
2650 if (aParent)
2651 pParent = static_cast<Medium*>(aParent);
2652
2653 HRESULT rc = S_OK;
2654 ComObjPtr<Progress> pProgress;
2655 Medium::Task *pTask = NULL;
2656
2657 try
2658 {
2659 // locking: we need the tree lock first because we access parent pointers
2660 // and we need to write-lock the media involved
2661 uint32_t cHandles = 3;
2662 LockHandle* pHandles[4] = { &m->pVirtualBox->getMediaTreeLockHandle(),
2663 this->lockHandle(),
2664 pTarget->lockHandle() };
2665 /* Only add parent to the lock if it is not null */
2666 if (!pParent.isNull())
2667 pHandles[cHandles++] = pParent->lockHandle();
2668 AutoWriteLock alock(cHandles,
2669 pHandles
2670 COMMA_LOCKVAL_SRC_POS);
2671
2672 if ( pTarget->m->state != MediumState_NotCreated
2673 && pTarget->m->state != MediumState_Created)
2674 throw pTarget->setStateError();
2675
2676 /* Build the source lock list. */
2677 MediumLockList *pSourceMediumLockList(new MediumLockList());
2678 alock.release();
2679 rc = createMediumLockList(true /* fFailIfInaccessible */,
2680 false /* fMediumLockWrite */,
2681 NULL,
2682 *pSourceMediumLockList);
2683 alock.acquire();
2684 if (FAILED(rc))
2685 {
2686 delete pSourceMediumLockList;
2687 throw rc;
2688 }
2689
2690 /* Build the target lock list (including the to-be parent chain). */
2691 MediumLockList *pTargetMediumLockList(new MediumLockList());
2692 alock.release();
2693 rc = pTarget->createMediumLockList(true /* fFailIfInaccessible */,
2694 true /* fMediumLockWrite */,
2695 pParent,
2696 *pTargetMediumLockList);
2697 alock.acquire();
2698 if (FAILED(rc))
2699 {
2700 delete pSourceMediumLockList;
2701 delete pTargetMediumLockList;
2702 throw rc;
2703 }
2704
2705 alock.release();
2706 rc = pSourceMediumLockList->Lock();
2707 alock.acquire();
2708 if (FAILED(rc))
2709 {
2710 delete pSourceMediumLockList;
2711 delete pTargetMediumLockList;
2712 throw setError(rc,
2713 tr("Failed to lock source media '%s'"),
2714 getLocationFull().c_str());
2715 }
2716 alock.release();
2717 rc = pTargetMediumLockList->Lock();
2718 alock.acquire();
2719 if (FAILED(rc))
2720 {
2721 delete pSourceMediumLockList;
2722 delete pTargetMediumLockList;
2723 throw setError(rc,
2724 tr("Failed to lock target media '%s'"),
2725 pTarget->getLocationFull().c_str());
2726 }
2727
2728 pProgress.createObject();
2729 rc = pProgress->init(m->pVirtualBox,
2730 static_cast <IMedium *>(this),
2731 BstrFmt(tr("Creating clone medium '%s'"), pTarget->m->strLocationFull.c_str()).raw(),
2732 TRUE /* aCancelable */);
2733 if (FAILED(rc))
2734 {
2735 delete pSourceMediumLockList;
2736 delete pTargetMediumLockList;
2737 throw rc;
2738 }
2739
2740 /* setup task object to carry out the operation asynchronously */
2741 pTask = new Medium::CloneTask(this, pProgress, pTarget,
2742 (MediumVariant_T)aVariant,
2743 pParent, UINT32_MAX, UINT32_MAX,
2744 pSourceMediumLockList, pTargetMediumLockList);
2745 rc = pTask->rc();
2746 AssertComRC(rc);
2747 if (FAILED(rc))
2748 throw rc;
2749
2750 if (pTarget->m->state == MediumState_NotCreated)
2751 pTarget->m->state = MediumState_Creating;
2752 }
2753 catch (HRESULT aRC) { rc = aRC; }
2754
2755 if (SUCCEEDED(rc))
2756 {
2757 rc = startThread(pTask);
2758
2759 if (SUCCEEDED(rc))
2760 pProgress.queryInterfaceTo(aProgress);
2761 }
2762 else if (pTask != NULL)
2763 delete pTask;
2764
2765 return rc;
2766}
2767
2768STDMETHODIMP Medium::Compact(IProgress **aProgress)
2769{
2770 CheckComArgOutPointerValid(aProgress);
2771
2772 AutoCaller autoCaller(this);
2773 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2774
2775 HRESULT rc = S_OK;
2776 ComObjPtr <Progress> pProgress;
2777 Medium::Task *pTask = NULL;
2778
2779 try
2780 {
2781 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2782
2783 /* Build the medium lock list. */
2784 MediumLockList *pMediumLockList(new MediumLockList());
2785 alock.release();
2786 rc = createMediumLockList(true /* fFailIfInaccessible */ ,
2787 true /* fMediumLockWrite */,
2788 NULL,
2789 *pMediumLockList);
2790 alock.acquire();
2791 if (FAILED(rc))
2792 {
2793 delete pMediumLockList;
2794 throw rc;
2795 }
2796
2797 alock.release();
2798 rc = pMediumLockList->Lock();
2799 alock.acquire();
2800 if (FAILED(rc))
2801 {
2802 delete pMediumLockList;
2803 throw setError(rc,
2804 tr("Failed to lock media when compacting '%s'"),
2805 getLocationFull().c_str());
2806 }
2807
2808 pProgress.createObject();
2809 rc = pProgress->init(m->pVirtualBox,
2810 static_cast <IMedium *>(this),
2811 BstrFmt(tr("Compacting medium '%s'"), m->strLocationFull.c_str()).raw(),
2812 TRUE /* aCancelable */);
2813 if (FAILED(rc))
2814 {
2815 delete pMediumLockList;
2816 throw rc;
2817 }
2818
2819 /* setup task object to carry out the operation asynchronously */
2820 pTask = new Medium::CompactTask(this, pProgress, pMediumLockList);
2821 rc = pTask->rc();
2822 AssertComRC(rc);
2823 if (FAILED(rc))
2824 throw rc;
2825 }
2826 catch (HRESULT aRC) { rc = aRC; }
2827
2828 if (SUCCEEDED(rc))
2829 {
2830 rc = startThread(pTask);
2831
2832 if (SUCCEEDED(rc))
2833 pProgress.queryInterfaceTo(aProgress);
2834 }
2835 else if (pTask != NULL)
2836 delete pTask;
2837
2838 return rc;
2839}
2840
2841STDMETHODIMP Medium::Resize(LONG64 aLogicalSize, IProgress **aProgress)
2842{
2843 CheckComArgOutPointerValid(aProgress);
2844
2845 AutoCaller autoCaller(this);
2846 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2847
2848 HRESULT rc = S_OK;
2849 ComObjPtr <Progress> pProgress;
2850 Medium::Task *pTask = NULL;
2851
2852 try
2853 {
2854 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2855
2856 /* Build the medium lock list. */
2857 MediumLockList *pMediumLockList(new MediumLockList());
2858 alock.release();
2859 rc = createMediumLockList(true /* fFailIfInaccessible */ ,
2860 true /* fMediumLockWrite */,
2861 NULL,
2862 *pMediumLockList);
2863 alock.acquire();
2864 if (FAILED(rc))
2865 {
2866 delete pMediumLockList;
2867 throw rc;
2868 }
2869
2870 alock.release();
2871 rc = pMediumLockList->Lock();
2872 alock.acquire();
2873 if (FAILED(rc))
2874 {
2875 delete pMediumLockList;
2876 throw setError(rc,
2877 tr("Failed to lock media when compacting '%s'"),
2878 getLocationFull().c_str());
2879 }
2880
2881 pProgress.createObject();
2882 rc = pProgress->init(m->pVirtualBox,
2883 static_cast <IMedium *>(this),
2884 BstrFmt(tr("Compacting medium '%s'"), m->strLocationFull.c_str()).raw(),
2885 TRUE /* aCancelable */);
2886 if (FAILED(rc))
2887 {
2888 delete pMediumLockList;
2889 throw rc;
2890 }
2891
2892 /* setup task object to carry out the operation asynchronously */
2893 pTask = new Medium::ResizeTask(this, aLogicalSize, pProgress, pMediumLockList);
2894 rc = pTask->rc();
2895 AssertComRC(rc);
2896 if (FAILED(rc))
2897 throw rc;
2898 }
2899 catch (HRESULT aRC) { rc = aRC; }
2900
2901 if (SUCCEEDED(rc))
2902 {
2903 rc = startThread(pTask);
2904
2905 if (SUCCEEDED(rc))
2906 pProgress.queryInterfaceTo(aProgress);
2907 }
2908 else if (pTask != NULL)
2909 delete pTask;
2910
2911 return rc;
2912}
2913
2914STDMETHODIMP Medium::Reset(IProgress **aProgress)
2915{
2916 CheckComArgOutPointerValid(aProgress);
2917
2918 AutoCaller autoCaller(this);
2919 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2920
2921 HRESULT rc = S_OK;
2922 ComObjPtr <Progress> pProgress;
2923 Medium::Task *pTask = NULL;
2924
2925 try
2926 {
2927 /* canClose() needs the tree lock */
2928 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
2929 this->lockHandle()
2930 COMMA_LOCKVAL_SRC_POS);
2931
2932 LogFlowThisFunc(("ENTER for medium %s\n", m->strLocationFull.c_str()));
2933
2934 if (m->pParent.isNull())
2935 throw setError(VBOX_E_NOT_SUPPORTED,
2936 tr("Medium type of '%s' is not differencing"),
2937 m->strLocationFull.c_str());
2938
2939 rc = canClose();
2940 if (FAILED(rc))
2941 throw rc;
2942
2943 /* Build the medium lock list. */
2944 MediumLockList *pMediumLockList(new MediumLockList());
2945 multilock.release();
2946 rc = createMediumLockList(true /* fFailIfInaccessible */,
2947 true /* fMediumLockWrite */,
2948 NULL,
2949 *pMediumLockList);
2950 multilock.acquire();
2951 if (FAILED(rc))
2952 {
2953 delete pMediumLockList;
2954 throw rc;
2955 }
2956
2957 multilock.release();
2958 rc = pMediumLockList->Lock();
2959 multilock.acquire();
2960 if (FAILED(rc))
2961 {
2962 delete pMediumLockList;
2963 throw setError(rc,
2964 tr("Failed to lock media when resetting '%s'"),
2965 getLocationFull().c_str());
2966 }
2967
2968 pProgress.createObject();
2969 rc = pProgress->init(m->pVirtualBox,
2970 static_cast<IMedium*>(this),
2971 BstrFmt(tr("Resetting differencing medium '%s'"), m->strLocationFull.c_str()).raw(),
2972 FALSE /* aCancelable */);
2973 if (FAILED(rc))
2974 throw rc;
2975
2976 /* setup task object to carry out the operation asynchronously */
2977 pTask = new Medium::ResetTask(this, pProgress, pMediumLockList);
2978 rc = pTask->rc();
2979 AssertComRC(rc);
2980 if (FAILED(rc))
2981 throw rc;
2982 }
2983 catch (HRESULT aRC) { rc = aRC; }
2984
2985 if (SUCCEEDED(rc))
2986 {
2987 rc = startThread(pTask);
2988
2989 if (SUCCEEDED(rc))
2990 pProgress.queryInterfaceTo(aProgress);
2991 }
2992 else
2993 {
2994 /* Note: on success, the task will unlock this */
2995 {
2996 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2997 HRESULT rc2 = UnlockWrite(NULL);
2998 AssertComRC(rc2);
2999 }
3000 if (pTask != NULL)
3001 delete pTask;
3002 }
3003
3004 LogFlowThisFunc(("LEAVE, rc=%Rhrc\n", rc));
3005
3006 return rc;
3007}
3008
3009////////////////////////////////////////////////////////////////////////////////
3010//
3011// Medium public internal methods
3012//
3013////////////////////////////////////////////////////////////////////////////////
3014
3015/**
3016 * Internal method to return the medium's parent medium. Must have caller + locking!
3017 * @return
3018 */
3019const ComObjPtr<Medium>& Medium::getParent() const
3020{
3021 return m->pParent;
3022}
3023
3024/**
3025 * Internal method to return the medium's list of child media. Must have caller + locking!
3026 * @return
3027 */
3028const MediaList& Medium::getChildren() const
3029{
3030 return m->llChildren;
3031}
3032
3033/**
3034 * Internal method to return the medium's GUID. Must have caller + locking!
3035 * @return
3036 */
3037const Guid& Medium::getId() const
3038{
3039 return m->id;
3040}
3041
3042/**
3043 * Internal method to return the medium's state. Must have caller + locking!
3044 * @return
3045 */
3046MediumState_T Medium::getState() const
3047{
3048 return m->state;
3049}
3050
3051/**
3052 * Internal method to return the medium's variant. Must have caller + locking!
3053 * @return
3054 */
3055MediumVariant_T Medium::getVariant() const
3056{
3057 return m->variant;
3058}
3059
3060/**
3061 * Internal method which returns true if this medium represents a host drive.
3062 * @return
3063 */
3064bool Medium::isHostDrive() const
3065{
3066 return m->hostDrive;
3067}
3068
3069/**
3070 * Internal method to return the medium's full location. Must have caller + locking!
3071 * @return
3072 */
3073const Utf8Str& Medium::getLocationFull() const
3074{
3075 return m->strLocationFull;
3076}
3077
3078/**
3079 * Internal method to return the medium's format string. Must have caller + locking!
3080 * @return
3081 */
3082const Utf8Str& Medium::getFormat() const
3083{
3084 return m->strFormat;
3085}
3086
3087/**
3088 * Internal method to return the medium's format object. Must have caller + locking!
3089 * @return
3090 */
3091const ComObjPtr<MediumFormat>& Medium::getMediumFormat() const
3092{
3093 return m->formatObj;
3094}
3095
3096/**
3097 * Internal method that returns true if the medium is represented by a file on the host disk
3098 * (and not iSCSI or something).
3099 * @return
3100 */
3101bool Medium::isMediumFormatFile() const
3102{
3103 if ( m->formatObj
3104 && (m->formatObj->getCapabilities() & MediumFormatCapabilities_File)
3105 )
3106 return true;
3107 return false;
3108}
3109
3110/**
3111 * Internal method to return the medium's size. Must have caller + locking!
3112 * @return
3113 */
3114uint64_t Medium::getSize() const
3115{
3116 return m->size;
3117}
3118
3119/**
3120 * Returns the medium device type. Must have caller + locking!
3121 * @return
3122 */
3123DeviceType_T Medium::getDeviceType() const
3124{
3125 return m->devType;
3126}
3127
3128/**
3129 * Returns the medium type. Must have caller + locking!
3130 * @return
3131 */
3132MediumType_T Medium::getType() const
3133{
3134 return m->type;
3135}
3136
3137/**
3138 * Returns a short version of the location attribute.
3139 *
3140 * @note Must be called from under this object's read or write lock.
3141 */
3142Utf8Str Medium::getName()
3143{
3144 Utf8Str name = RTPathFilename(m->strLocationFull.c_str());
3145 return name;
3146}
3147
3148/**
3149 * This adds the given UUID to the list of media registries in which this
3150 * medium should be registered. The UUID can either be a machine UUID,
3151 * to add a machine registry, or the global registry UUID as returned by
3152 * VirtualBox::getGlobalRegistryId().
3153 *
3154 * Note that for hard disks, this method does nothing if the medium is
3155 * already in another registry to avoid having hard disks in more than
3156 * one registry, which causes trouble with keeping diff images in sync.
3157 * See getFirstRegistryMachineId() for details.
3158 *
3159 * If fRecurse == true, then the media tree lock must be held for reading.
3160 *
3161 * @param id
3162 * @param fRecurse If true, recurses into child media to make sure the whole tree has registries in sync.
3163 * @return true if the registry was added; false if the given id was already on the list.
3164 */
3165bool Medium::addRegistry(const Guid& id, bool fRecurse)
3166{
3167 AutoCaller autoCaller(this);
3168 if (FAILED(autoCaller.rc()))
3169 return false;
3170 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3171
3172 bool fAdd = true;
3173
3174 // hard disks cannot be in more than one registry
3175 if ( m->devType == DeviceType_HardDisk
3176 && m->llRegistryIDs.size() > 0)
3177 fAdd = false;
3178
3179 // no need to add the UUID twice
3180 if (fAdd)
3181 {
3182 for (GuidList::const_iterator it = m->llRegistryIDs.begin();
3183 it != m->llRegistryIDs.end();
3184 ++it)
3185 {
3186 if ((*it) == id)
3187 {
3188 fAdd = false;
3189 break;
3190 }
3191 }
3192 }
3193
3194 if (fAdd)
3195 m->llRegistryIDs.push_back(id);
3196
3197 if (fRecurse)
3198 {
3199 // Get private list of children and release medium lock straight away.
3200 MediaList llChildren(m->llChildren);
3201 alock.release();
3202
3203 for (MediaList::iterator it = llChildren.begin();
3204 it != llChildren.end();
3205 ++it)
3206 {
3207 Medium *pChild = *it;
3208 fAdd |= pChild->addRegistry(id, true);
3209 }
3210 }
3211
3212 return fAdd;
3213}
3214
3215/**
3216 * Removes the given UUID from the list of media registry UUIDs. Returns true
3217 * if found or false if not.
3218 *
3219 * If fRecurse == true, then the media tree lock must be held for reading.
3220 *
3221 * @param id
3222 * @param fRecurse If true, recurses into child media to make sure the whole tree has registries in sync.
3223 * @return
3224 */
3225bool Medium::removeRegistry(const Guid& id, bool fRecurse)
3226{
3227 AutoCaller autoCaller(this);
3228 if (FAILED(autoCaller.rc()))
3229 return false;
3230 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3231
3232 bool fRemove = false;
3233
3234 for (GuidList::iterator it = m->llRegistryIDs.begin();
3235 it != m->llRegistryIDs.end();
3236 ++it)
3237 {
3238 if ((*it) == id)
3239 {
3240 m->llRegistryIDs.erase(it);
3241 fRemove = true;
3242 break;
3243 }
3244 }
3245
3246 if (fRecurse)
3247 {
3248 // Get private list of children and release medium lock straight away.
3249 MediaList llChildren(m->llChildren);
3250 alock.release();
3251
3252 for (MediaList::iterator it = llChildren.begin();
3253 it != llChildren.end();
3254 ++it)
3255 {
3256 Medium *pChild = *it;
3257 fRemove |= pChild->removeRegistry(id, true);
3258 }
3259 }
3260
3261 return fRemove;
3262}
3263
3264/**
3265 * Returns true if id is in the list of media registries for this medium.
3266 *
3267 * Must have caller + read locking!
3268 *
3269 * @param id
3270 * @return
3271 */
3272bool Medium::isInRegistry(const Guid& id)
3273{
3274 for (GuidList::const_iterator it = m->llRegistryIDs.begin();
3275 it != m->llRegistryIDs.end();
3276 ++it)
3277 {
3278 if (*it == id)
3279 return true;
3280 }
3281
3282 return false;
3283}
3284
3285/**
3286 * Internal method to return the medium's first registry machine (i.e. the machine in whose
3287 * machine XML this medium is listed).
3288 *
3289 * Every attached medium must now (4.0) reside in at least one media registry, which is identified
3290 * by a UUID. This is either a machine UUID if the machine is from 4.0 or newer, in which case
3291 * machines have their own media registries, or it is the pseudo-UUID of the VirtualBox
3292 * object if the machine is old and still needs the global registry in VirtualBox.xml.
3293 *
3294 * By definition, hard disks may only be in one media registry, in which all its children
3295 * will be stored as well. Otherwise we run into problems with having keep multiple registries
3296 * in sync. (This is the "cloned VM" case in which VM1 may link to the disks of VM2; in this
3297 * case, only VM2's registry is used for the disk in question.)
3298 *
3299 * If there is no medium registry, particularly if the medium has not been attached yet, this
3300 * does not modify uuid and returns false.
3301 *
3302 * ISOs and RAWs, by contrast, can be in more than one repository to make things easier for
3303 * the user.
3304 *
3305 * Must have caller + locking!
3306 *
3307 * @param uuid Receives first registry machine UUID, if available.
3308 * @return true if uuid was set.
3309 */
3310bool Medium::getFirstRegistryMachineId(Guid &uuid) const
3311{
3312 if (m->llRegistryIDs.size())
3313 {
3314 uuid = m->llRegistryIDs.front();
3315 return true;
3316 }
3317 return false;
3318}
3319
3320/**
3321 * Marks all the registries in which this medium is registered as modified.
3322 */
3323void Medium::markRegistriesModified()
3324{
3325 AutoCaller autoCaller(this);
3326 if (FAILED(autoCaller.rc())) return;
3327
3328 // Get local copy, as keeping the lock over VirtualBox::markRegistryModified
3329 // causes trouble with the lock order
3330 GuidList llRegistryIDs;
3331 {
3332 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3333 llRegistryIDs = m->llRegistryIDs;
3334 }
3335
3336 for (GuidList::const_iterator it = llRegistryIDs.begin();
3337 it != llRegistryIDs.end();
3338 ++it)
3339 {
3340 m->pVirtualBox->markRegistryModified(*it);
3341 }
3342}
3343
3344/**
3345 * Adds the given machine and optionally the snapshot to the list of the objects
3346 * this medium is attached to.
3347 *
3348 * @param aMachineId Machine ID.
3349 * @param aSnapshotId Snapshot ID; when non-empty, adds a snapshot attachment.
3350 */
3351HRESULT Medium::addBackReference(const Guid &aMachineId,
3352 const Guid &aSnapshotId /*= Guid::Empty*/)
3353{
3354 AssertReturn(!aMachineId.isEmpty(), E_FAIL);
3355
3356 LogFlowThisFunc(("ENTER, aMachineId: {%RTuuid}, aSnapshotId: {%RTuuid}\n", aMachineId.raw(), aSnapshotId.raw()));
3357
3358 AutoCaller autoCaller(this);
3359 AssertComRCReturnRC(autoCaller.rc());
3360
3361 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3362
3363 switch (m->state)
3364 {
3365 case MediumState_Created:
3366 case MediumState_Inaccessible:
3367 case MediumState_LockedRead:
3368 case MediumState_LockedWrite:
3369 break;
3370
3371 default:
3372 return setStateError();
3373 }
3374
3375 if (m->numCreateDiffTasks > 0)
3376 return setError(VBOX_E_OBJECT_IN_USE,
3377 tr("Cannot attach medium '%s' {%RTuuid}: %u differencing child media are being created"),
3378 m->strLocationFull.c_str(),
3379 m->id.raw(),
3380 m->numCreateDiffTasks);
3381
3382 BackRefList::iterator it = std::find_if(m->backRefs.begin(),
3383 m->backRefs.end(),
3384 BackRef::EqualsTo(aMachineId));
3385 if (it == m->backRefs.end())
3386 {
3387 BackRef ref(aMachineId, aSnapshotId);
3388 m->backRefs.push_back(ref);
3389
3390 return S_OK;
3391 }
3392
3393 // if the caller has not supplied a snapshot ID, then we're attaching
3394 // to a machine a medium which represents the machine's current state,
3395 // so set the flag
3396 if (aSnapshotId.isEmpty())
3397 {
3398 /* sanity: no duplicate attachments */
3399 if (it->fInCurState)
3400 return setError(VBOX_E_OBJECT_IN_USE,
3401 tr("Cannot attach medium '%s' {%RTuuid}: medium is already associated with the current state of machine uuid {%RTuuid}!"),
3402 m->strLocationFull.c_str(),
3403 m->id.raw(),
3404 aMachineId.raw());
3405 it->fInCurState = true;
3406
3407 return S_OK;
3408 }
3409
3410 // otherwise: a snapshot medium is being attached
3411
3412 /* sanity: no duplicate attachments */
3413 for (GuidList::const_iterator jt = it->llSnapshotIds.begin();
3414 jt != it->llSnapshotIds.end();
3415 ++jt)
3416 {
3417 const Guid &idOldSnapshot = *jt;
3418
3419 if (idOldSnapshot == aSnapshotId)
3420 {
3421#ifdef DEBUG
3422 dumpBackRefs();
3423#endif
3424 return setError(VBOX_E_OBJECT_IN_USE,
3425 tr("Cannot attach medium '%s' {%RTuuid} from snapshot '%RTuuid': medium is already in use by this snapshot!"),
3426 m->strLocationFull.c_str(),
3427 m->id.raw(),
3428 aSnapshotId.raw());
3429 }
3430 }
3431
3432 it->llSnapshotIds.push_back(aSnapshotId);
3433 it->fInCurState = false;
3434
3435 LogFlowThisFuncLeave();
3436
3437 return S_OK;
3438}
3439
3440/**
3441 * Removes the given machine and optionally the snapshot from the list of the
3442 * objects this medium is attached to.
3443 *
3444 * @param aMachineId Machine ID.
3445 * @param aSnapshotId Snapshot ID; when non-empty, removes the snapshot
3446 * attachment.
3447 */
3448HRESULT Medium::removeBackReference(const Guid &aMachineId,
3449 const Guid &aSnapshotId /*= Guid::Empty*/)
3450{
3451 AssertReturn(!aMachineId.isEmpty(), E_FAIL);
3452
3453 AutoCaller autoCaller(this);
3454 AssertComRCReturnRC(autoCaller.rc());
3455
3456 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3457
3458 BackRefList::iterator it =
3459 std::find_if(m->backRefs.begin(), m->backRefs.end(),
3460 BackRef::EqualsTo(aMachineId));
3461 AssertReturn(it != m->backRefs.end(), E_FAIL);
3462
3463 if (aSnapshotId.isEmpty())
3464 {
3465 /* remove the current state attachment */
3466 it->fInCurState = false;
3467 }
3468 else
3469 {
3470 /* remove the snapshot attachment */
3471 GuidList::iterator jt = std::find(it->llSnapshotIds.begin(),
3472 it->llSnapshotIds.end(),
3473 aSnapshotId);
3474
3475 AssertReturn(jt != it->llSnapshotIds.end(), E_FAIL);
3476 it->llSnapshotIds.erase(jt);
3477 }
3478
3479 /* if the backref becomes empty, remove it */
3480 if (it->fInCurState == false && it->llSnapshotIds.size() == 0)
3481 m->backRefs.erase(it);
3482
3483 return S_OK;
3484}
3485
3486/**
3487 * Internal method to return the medium's list of backrefs. Must have caller + locking!
3488 * @return
3489 */
3490const Guid* Medium::getFirstMachineBackrefId() const
3491{
3492 if (!m->backRefs.size())
3493 return NULL;
3494
3495 return &m->backRefs.front().machineId;
3496}
3497
3498/**
3499 * Internal method which returns a machine that either this medium or one of its children
3500 * is attached to. This is used for finding a replacement media registry when an existing
3501 * media registry is about to be deleted in VirtualBox::unregisterMachine().
3502 *
3503 * Must have caller + locking, *and* caller must hold the media tree lock!
3504 * @return
3505 */
3506const Guid* Medium::getAnyMachineBackref() const
3507{
3508 if (m->backRefs.size())
3509 return &m->backRefs.front().machineId;
3510
3511 for (MediaList::iterator it = m->llChildren.begin();
3512 it != m->llChildren.end();
3513 ++it)
3514 {
3515 Medium *pChild = *it;
3516 // recurse for this child
3517 const Guid* puuid;
3518 if ((puuid = pChild->getAnyMachineBackref()))
3519 return puuid;
3520 }
3521
3522 return NULL;
3523}
3524
3525const Guid* Medium::getFirstMachineBackrefSnapshotId() const
3526{
3527 if (!m->backRefs.size())
3528 return NULL;
3529
3530 const BackRef &ref = m->backRefs.front();
3531 if (!ref.llSnapshotIds.size())
3532 return NULL;
3533
3534 return &ref.llSnapshotIds.front();
3535}
3536
3537size_t Medium::getMachineBackRefCount() const
3538{
3539 return m->backRefs.size();
3540}
3541
3542#ifdef DEBUG
3543/**
3544 * Debugging helper that gets called after VirtualBox initialization that writes all
3545 * machine backreferences to the debug log.
3546 */
3547void Medium::dumpBackRefs()
3548{
3549 AutoCaller autoCaller(this);
3550 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3551
3552 LogFlowThisFunc(("Dumping backrefs for medium '%s':\n", m->strLocationFull.c_str()));
3553
3554 for (BackRefList::iterator it2 = m->backRefs.begin();
3555 it2 != m->backRefs.end();
3556 ++it2)
3557 {
3558 const BackRef &ref = *it2;
3559 LogFlowThisFunc((" Backref from machine {%RTuuid} (fInCurState: %d)\n", ref.machineId.raw(), ref.fInCurState));
3560
3561 for (GuidList::const_iterator jt2 = it2->llSnapshotIds.begin();
3562 jt2 != it2->llSnapshotIds.end();
3563 ++jt2)
3564 {
3565 const Guid &id = *jt2;
3566 LogFlowThisFunc((" Backref from snapshot {%RTuuid}\n", id.raw()));
3567 }
3568 }
3569}
3570#endif
3571
3572/**
3573 * Checks if the given change of \a aOldPath to \a aNewPath affects the location
3574 * of this media and updates it if necessary to reflect the new location.
3575 *
3576 * @param aOldPath Old path (full).
3577 * @param aNewPath New path (full).
3578 *
3579 * @note Locks this object for writing.
3580 */
3581HRESULT Medium::updatePath(const Utf8Str &strOldPath, const Utf8Str &strNewPath)
3582{
3583 AssertReturn(!strOldPath.isEmpty(), E_FAIL);
3584 AssertReturn(!strNewPath.isEmpty(), E_FAIL);
3585
3586 AutoCaller autoCaller(this);
3587 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3588
3589 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3590
3591 LogFlowThisFunc(("locationFull.before='%s'\n", m->strLocationFull.c_str()));
3592
3593 const char *pcszMediumPath = m->strLocationFull.c_str();
3594
3595 if (RTPathStartsWith(pcszMediumPath, strOldPath.c_str()))
3596 {
3597 Utf8Str newPath(strNewPath);
3598 newPath.append(pcszMediumPath + strOldPath.length());
3599 unconst(m->strLocationFull) = newPath;
3600
3601 LogFlowThisFunc(("locationFull.after='%s'\n", m->strLocationFull.c_str()));
3602 // we changed something
3603 return S_OK;
3604 }
3605
3606 // no change was necessary, signal error which the caller needs to interpret
3607 return VBOX_E_FILE_ERROR;
3608}
3609
3610/**
3611 * Returns the base medium of the media chain this medium is part of.
3612 *
3613 * The base medium is found by walking up the parent-child relationship axis.
3614 * If the medium doesn't have a parent (i.e. it's a base medium), it
3615 * returns itself in response to this method.
3616 *
3617 * @param aLevel Where to store the number of ancestors of this medium
3618 * (zero for the base), may be @c NULL.
3619 *
3620 * @note Locks medium tree for reading.
3621 */
3622ComObjPtr<Medium> Medium::getBase(uint32_t *aLevel /*= NULL*/)
3623{
3624 ComObjPtr<Medium> pBase;
3625 uint32_t level;
3626
3627 AutoCaller autoCaller(this);
3628 AssertReturn(autoCaller.isOk(), pBase);
3629
3630 /* we access mParent */
3631 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3632
3633 pBase = this;
3634 level = 0;
3635
3636 if (m->pParent)
3637 {
3638 for (;;)
3639 {
3640 AutoCaller baseCaller(pBase);
3641 AssertReturn(baseCaller.isOk(), pBase);
3642
3643 if (pBase->m->pParent.isNull())
3644 break;
3645
3646 pBase = pBase->m->pParent;
3647 ++level;
3648 }
3649 }
3650
3651 if (aLevel != NULL)
3652 *aLevel = level;
3653
3654 return pBase;
3655}
3656
3657/**
3658 * Returns @c true if this medium cannot be modified because it has
3659 * dependents (children) or is part of the snapshot. Related to the medium
3660 * type and posterity, not to the current media state.
3661 *
3662 * @note Locks this object and medium tree for reading.
3663 */
3664bool Medium::isReadOnly()
3665{
3666 AutoCaller autoCaller(this);
3667 AssertComRCReturn(autoCaller.rc(), false);
3668
3669 /* we access children */
3670 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3671
3672 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3673
3674 switch (m->type)
3675 {
3676 case MediumType_Normal:
3677 {
3678 if (getChildren().size() != 0)
3679 return true;
3680
3681 for (BackRefList::const_iterator it = m->backRefs.begin();
3682 it != m->backRefs.end(); ++it)
3683 if (it->llSnapshotIds.size() != 0)
3684 return true;
3685
3686 if (m->variant & MediumVariant_VmdkStreamOptimized)
3687 return true;
3688
3689 return false;
3690 }
3691 case MediumType_Immutable:
3692 case MediumType_MultiAttach:
3693 return true;
3694 case MediumType_Writethrough:
3695 case MediumType_Shareable:
3696 case MediumType_Readonly: /* explicit readonly media has no diffs */
3697 return false;
3698 default:
3699 break;
3700 }
3701
3702 AssertFailedReturn(false);
3703}
3704
3705/**
3706 * Internal method to return the medium's size. Must have caller + locking!
3707 * @return
3708 */
3709void Medium::updateId(const Guid &id)
3710{
3711 unconst(m->id) = id;
3712}
3713
3714/**
3715 * Saves medium data by appending a new child node to the given
3716 * parent XML settings node.
3717 *
3718 * @param data Settings struct to be updated.
3719 * @param strHardDiskFolder Folder for which paths should be relative.
3720 *
3721 * @note Locks this object, medium tree and children for reading.
3722 */
3723HRESULT Medium::saveSettings(settings::Medium &data,
3724 const Utf8Str &strHardDiskFolder)
3725{
3726 AutoCaller autoCaller(this);
3727 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3728
3729 /* we access mParent */
3730 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3731
3732 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3733
3734 data.uuid = m->id;
3735
3736 // make path relative if needed
3737 if ( !strHardDiskFolder.isEmpty()
3738 && RTPathStartsWith(m->strLocationFull.c_str(), strHardDiskFolder.c_str())
3739 )
3740 data.strLocation = m->strLocationFull.substr(strHardDiskFolder.length() + 1);
3741 else
3742 data.strLocation = m->strLocationFull;
3743 data.strFormat = m->strFormat;
3744
3745 /* optional, only for diffs, default is false */
3746 if (m->pParent)
3747 data.fAutoReset = m->autoReset;
3748 else
3749 data.fAutoReset = false;
3750
3751 /* optional */
3752 data.strDescription = m->strDescription;
3753
3754 /* optional properties */
3755 data.properties.clear();
3756 for (settings::StringsMap::const_iterator it = m->mapProperties.begin();
3757 it != m->mapProperties.end();
3758 ++it)
3759 {
3760 /* only save properties that have non-default values */
3761 if (!it->second.isEmpty())
3762 {
3763 const Utf8Str &name = it->first;
3764 const Utf8Str &value = it->second;
3765 data.properties[name] = value;
3766 }
3767 }
3768
3769 /* only for base media */
3770 if (m->pParent.isNull())
3771 data.hdType = m->type;
3772
3773 /* save all children */
3774 for (MediaList::const_iterator it = getChildren().begin();
3775 it != getChildren().end();
3776 ++it)
3777 {
3778 settings::Medium med;
3779 HRESULT rc = (*it)->saveSettings(med, strHardDiskFolder);
3780 AssertComRCReturnRC(rc);
3781 data.llChildren.push_back(med);
3782 }
3783
3784 return S_OK;
3785}
3786
3787/**
3788 * Constructs a medium lock list for this medium. The lock is not taken.
3789 *
3790 * @note Caller MUST NOT hold the media tree or medium lock.
3791 *
3792 * @param fFailIfInaccessible If true, this fails with an error if a medium is inaccessible. If false,
3793 * inaccessible media are silently skipped and not locked (i.e. their state remains "Inaccessible");
3794 * this is necessary for a VM's removable media VM startup for which we do not want to fail.
3795 * @param fMediumLockWrite Whether to associate a write lock with this medium.
3796 * @param pToBeParent Medium which will become the parent of this medium.
3797 * @param mediumLockList Where to store the resulting list.
3798 */
3799HRESULT Medium::createMediumLockList(bool fFailIfInaccessible,
3800 bool fMediumLockWrite,
3801 Medium *pToBeParent,
3802 MediumLockList &mediumLockList)
3803{
3804 Assert(!m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
3805 Assert(!isWriteLockOnCurrentThread());
3806
3807 AutoCaller autoCaller(this);
3808 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3809
3810 HRESULT rc = S_OK;
3811
3812 /* paranoid sanity checking if the medium has a to-be parent medium */
3813 if (pToBeParent)
3814 {
3815 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3816 ComAssertRet(getParent().isNull(), E_FAIL);
3817 ComAssertRet(getChildren().size() == 0, E_FAIL);
3818 }
3819
3820 ErrorInfoKeeper eik;
3821 MultiResult mrc(S_OK);
3822
3823 ComObjPtr<Medium> pMedium = this;
3824 while (!pMedium.isNull())
3825 {
3826 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
3827
3828 /* Accessibility check must be first, otherwise locking interferes
3829 * with getting the medium state. Lock lists are not created for
3830 * fun, and thus getting the medium status is no luxury. */
3831 MediumState_T mediumState = pMedium->getState();
3832 if (mediumState == MediumState_Inaccessible)
3833 {
3834 alock.release();
3835 rc = pMedium->queryInfo(false /* fSetImageId */, false /* fSetParentId */);
3836 alock.acquire();
3837 if (FAILED(rc)) return rc;
3838
3839 mediumState = pMedium->getState();
3840 if (mediumState == MediumState_Inaccessible)
3841 {
3842 // ignore inaccessible ISO media and silently return S_OK,
3843 // otherwise VM startup (esp. restore) may fail without good reason
3844 if (!fFailIfInaccessible)
3845 return S_OK;
3846
3847 // otherwise report an error
3848 Bstr error;
3849 rc = pMedium->COMGETTER(LastAccessError)(error.asOutParam());
3850 if (FAILED(rc)) return rc;
3851
3852 /* collect multiple errors */
3853 eik.restore();
3854 Assert(!error.isEmpty());
3855 mrc = setError(E_FAIL,
3856 "%ls",
3857 error.raw());
3858 // error message will be something like
3859 // "Could not open the medium ... VD: error VERR_FILE_NOT_FOUND opening image file ... (VERR_FILE_NOT_FOUND).
3860 eik.fetch();
3861 }
3862 }
3863
3864 if (pMedium == this)
3865 mediumLockList.Prepend(pMedium, fMediumLockWrite);
3866 else
3867 mediumLockList.Prepend(pMedium, false);
3868
3869 pMedium = pMedium->getParent();
3870 if (pMedium.isNull() && pToBeParent)
3871 {
3872 pMedium = pToBeParent;
3873 pToBeParent = NULL;
3874 }
3875 }
3876
3877 return mrc;
3878}
3879
3880/**
3881 * Creates a new differencing storage unit using the format of the given target
3882 * medium and the location. Note that @c aTarget must be NotCreated.
3883 *
3884 * The @a aMediumLockList parameter contains the associated medium lock list,
3885 * which must be in locked state. If @a aWait is @c true then the caller is
3886 * responsible for unlocking.
3887 *
3888 * If @a aProgress is not NULL but the object it points to is @c null then a
3889 * new progress object will be created and assigned to @a *aProgress on
3890 * success, otherwise the existing progress object is used. If @a aProgress is
3891 * NULL, then no progress object is created/used at all.
3892 *
3893 * When @a aWait is @c false, this method will create a thread to perform the
3894 * create operation asynchronously and will return immediately. Otherwise, it
3895 * will perform the operation on the calling thread and will not return to the
3896 * caller until the operation is completed. Note that @a aProgress cannot be
3897 * NULL when @a aWait is @c false (this method will assert in this case).
3898 *
3899 * @param aTarget Target medium.
3900 * @param aVariant Precise medium variant to create.
3901 * @param aMediumLockList List of media which should be locked.
3902 * @param aProgress Where to find/store a Progress object to track
3903 * operation completion.
3904 * @param aWait @c true if this method should block instead of
3905 * creating an asynchronous thread.
3906 *
3907 * @note Locks this object and @a aTarget for writing.
3908 */
3909HRESULT Medium::createDiffStorage(ComObjPtr<Medium> &aTarget,
3910 MediumVariant_T aVariant,
3911 MediumLockList *aMediumLockList,
3912 ComObjPtr<Progress> *aProgress,
3913 bool aWait)
3914{
3915 AssertReturn(!aTarget.isNull(), E_FAIL);
3916 AssertReturn(aMediumLockList, E_FAIL);
3917 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
3918
3919 AutoCaller autoCaller(this);
3920 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3921
3922 AutoCaller targetCaller(aTarget);
3923 if (FAILED(targetCaller.rc())) return targetCaller.rc();
3924
3925 HRESULT rc = S_OK;
3926 ComObjPtr<Progress> pProgress;
3927 Medium::Task *pTask = NULL;
3928
3929 try
3930 {
3931 AutoMultiWriteLock2 alock(this, aTarget COMMA_LOCKVAL_SRC_POS);
3932
3933 ComAssertThrow( m->type != MediumType_Writethrough
3934 && m->type != MediumType_Shareable
3935 && m->type != MediumType_Readonly, E_FAIL);
3936 ComAssertThrow(m->state == MediumState_LockedRead, E_FAIL);
3937
3938 if (aTarget->m->state != MediumState_NotCreated)
3939 throw aTarget->setStateError();
3940
3941 /* Check that the medium is not attached to the current state of
3942 * any VM referring to it. */
3943 for (BackRefList::const_iterator it = m->backRefs.begin();
3944 it != m->backRefs.end();
3945 ++it)
3946 {
3947 if (it->fInCurState)
3948 {
3949 /* Note: when a VM snapshot is being taken, all normal media
3950 * attached to the VM in the current state will be, as an
3951 * exception, also associated with the snapshot which is about
3952 * to create (see SnapshotMachine::init()) before deassociating
3953 * them from the current state (which takes place only on
3954 * success in Machine::fixupHardDisks()), so that the size of
3955 * snapshotIds will be 1 in this case. The extra condition is
3956 * used to filter out this legal situation. */
3957 if (it->llSnapshotIds.size() == 0)
3958 throw setError(VBOX_E_INVALID_OBJECT_STATE,
3959 tr("Medium '%s' is attached to a virtual machine with UUID {%RTuuid}. No differencing media based on it may be created until it is detached"),
3960 m->strLocationFull.c_str(), it->machineId.raw());
3961
3962 Assert(it->llSnapshotIds.size() == 1);
3963 }
3964 }
3965
3966 if (aProgress != NULL)
3967 {
3968 /* use the existing progress object... */
3969 pProgress = *aProgress;
3970
3971 /* ...but create a new one if it is null */
3972 if (pProgress.isNull())
3973 {
3974 pProgress.createObject();
3975 rc = pProgress->init(m->pVirtualBox,
3976 static_cast<IMedium*>(this),
3977 BstrFmt(tr("Creating differencing medium storage unit '%s'"), aTarget->m->strLocationFull.c_str()).raw(),
3978 TRUE /* aCancelable */);
3979 if (FAILED(rc))
3980 throw rc;
3981 }
3982 }
3983
3984 /* setup task object to carry out the operation sync/async */
3985 pTask = new Medium::CreateDiffTask(this, pProgress, aTarget, aVariant,
3986 aMediumLockList,
3987 aWait /* fKeepMediumLockList */);
3988 rc = pTask->rc();
3989 AssertComRC(rc);
3990 if (FAILED(rc))
3991 throw rc;
3992
3993 /* register a task (it will deregister itself when done) */
3994 ++m->numCreateDiffTasks;
3995 Assert(m->numCreateDiffTasks != 0); /* overflow? */
3996
3997 aTarget->m->state = MediumState_Creating;
3998 }
3999 catch (HRESULT aRC) { rc = aRC; }
4000
4001 if (SUCCEEDED(rc))
4002 {
4003 if (aWait)
4004 rc = runNow(pTask);
4005 else
4006 rc = startThread(pTask);
4007
4008 if (SUCCEEDED(rc) && aProgress != NULL)
4009 *aProgress = pProgress;
4010 }
4011 else if (pTask != NULL)
4012 delete pTask;
4013
4014 return rc;
4015}
4016
4017/**
4018 * Returns a preferred format for differencing media.
4019 */
4020Utf8Str Medium::getPreferredDiffFormat()
4021{
4022 AutoCaller autoCaller(this);
4023 AssertComRCReturn(autoCaller.rc(), Utf8Str::Empty);
4024
4025 /* check that our own format supports diffs */
4026 if (!(m->formatObj->getCapabilities() & MediumFormatCapabilities_Differencing))
4027 {
4028 /* use the default format if not */
4029 Utf8Str tmp;
4030 m->pVirtualBox->getDefaultHardDiskFormat(tmp);
4031 return tmp;
4032 }
4033
4034 /* m->strFormat is const, no need to lock */
4035 return m->strFormat;
4036}
4037
4038/**
4039 * Implementation for the public Medium::Close() with the exception of calling
4040 * VirtualBox::saveRegistries(), in case someone wants to call this for several
4041 * media.
4042 *
4043 * After this returns with success, uninit() has been called on the medium, and
4044 * the object is no longer usable ("not ready" state).
4045 *
4046 * @param autoCaller AutoCaller instance which must have been created on the caller's stack for this medium. This gets released here
4047 * upon which the Medium instance gets uninitialized.
4048 * @return
4049 */
4050HRESULT Medium::close(AutoCaller &autoCaller)
4051{
4052 // we're accessing parent/child and backrefs, so lock the tree first, then ourselves
4053 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
4054 this->lockHandle()
4055 COMMA_LOCKVAL_SRC_POS);
4056
4057 LogFlowFunc(("ENTER for %s\n", getLocationFull().c_str()));
4058
4059 bool wasCreated = true;
4060
4061 switch (m->state)
4062 {
4063 case MediumState_NotCreated:
4064 wasCreated = false;
4065 break;
4066 case MediumState_Created:
4067 case MediumState_Inaccessible:
4068 break;
4069 default:
4070 return setStateError();
4071 }
4072
4073 if (m->backRefs.size() != 0)
4074 return setError(VBOX_E_OBJECT_IN_USE,
4075 tr("Medium '%s' cannot be closed because it is still attached to %d virtual machines"),
4076 m->strLocationFull.c_str(), m->backRefs.size());
4077
4078 // perform extra media-dependent close checks
4079 HRESULT rc = canClose();
4080 if (FAILED(rc)) return rc;
4081
4082 if (wasCreated)
4083 {
4084 // remove from the list of known media before performing actual
4085 // uninitialization (to keep the media registry consistent on
4086 // failure to do so)
4087 rc = unregisterWithVirtualBox();
4088 if (FAILED(rc)) return rc;
4089
4090 multilock.release();
4091 markRegistriesModified();
4092 m->pVirtualBox->saveModifiedRegistries();
4093 multilock.acquire();
4094 }
4095
4096 // release the AutoCaller, as otherwise uninit() will simply hang
4097 autoCaller.release();
4098
4099 // Keep the locks held until after uninit, as otherwise the consistency
4100 // of the medium tree cannot be guaranteed.
4101 uninit();
4102
4103 LogFlowFuncLeave();
4104
4105 return rc;
4106}
4107
4108/**
4109 * Deletes the medium storage unit.
4110 *
4111 * If @a aProgress is not NULL but the object it points to is @c null then a new
4112 * progress object will be created and assigned to @a *aProgress on success,
4113 * otherwise the existing progress object is used. If Progress is NULL, then no
4114 * progress object is created/used at all.
4115 *
4116 * When @a aWait is @c false, this method will create a thread to perform the
4117 * delete operation asynchronously and will return immediately. Otherwise, it
4118 * will perform the operation on the calling thread and will not return to the
4119 * caller until the operation is completed. Note that @a aProgress cannot be
4120 * NULL when @a aWait is @c false (this method will assert in this case).
4121 *
4122 * @param aProgress Where to find/store a Progress object to track operation
4123 * completion.
4124 * @param aWait @c true if this method should block instead of creating
4125 * an asynchronous thread.
4126 *
4127 * @note Locks mVirtualBox and this object for writing. Locks medium tree for
4128 * writing.
4129 */
4130HRESULT Medium::deleteStorage(ComObjPtr<Progress> *aProgress,
4131 bool aWait)
4132{
4133 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
4134
4135 AutoCaller autoCaller(this);
4136 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4137
4138 HRESULT rc = S_OK;
4139 ComObjPtr<Progress> pProgress;
4140 Medium::Task *pTask = NULL;
4141
4142 try
4143 {
4144 /* we're accessing the media tree, and canClose() needs it too */
4145 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
4146 this->lockHandle()
4147 COMMA_LOCKVAL_SRC_POS);
4148 LogFlowThisFunc(("aWait=%RTbool locationFull=%s\n", aWait, getLocationFull().c_str() ));
4149
4150 if ( !(m->formatObj->getCapabilities() & ( MediumFormatCapabilities_CreateDynamic
4151 | MediumFormatCapabilities_CreateFixed)))
4152 throw setError(VBOX_E_NOT_SUPPORTED,
4153 tr("Medium format '%s' does not support storage deletion"),
4154 m->strFormat.c_str());
4155
4156 /* Note that we are fine with Inaccessible state too: a) for symmetry
4157 * with create calls and b) because it doesn't really harm to try, if
4158 * it is really inaccessible, the delete operation will fail anyway.
4159 * Accepting Inaccessible state is especially important because all
4160 * registered media are initially Inaccessible upon VBoxSVC startup
4161 * until COMGETTER(RefreshState) is called. Accept Deleting state
4162 * because some callers need to put the medium in this state early
4163 * to prevent races. */
4164 switch (m->state)
4165 {
4166 case MediumState_Created:
4167 case MediumState_Deleting:
4168 case MediumState_Inaccessible:
4169 break;
4170 default:
4171 throw setStateError();
4172 }
4173
4174 if (m->backRefs.size() != 0)
4175 {
4176 Utf8Str strMachines;
4177 for (BackRefList::const_iterator it = m->backRefs.begin();
4178 it != m->backRefs.end();
4179 ++it)
4180 {
4181 const BackRef &b = *it;
4182 if (strMachines.length())
4183 strMachines.append(", ");
4184 strMachines.append(b.machineId.toString().c_str());
4185 }
4186#ifdef DEBUG
4187 dumpBackRefs();
4188#endif
4189 throw setError(VBOX_E_OBJECT_IN_USE,
4190 tr("Cannot delete storage: medium '%s' is still attached to the following %d virtual machine(s): %s"),
4191 m->strLocationFull.c_str(),
4192 m->backRefs.size(),
4193 strMachines.c_str());
4194 }
4195
4196 rc = canClose();
4197 if (FAILED(rc))
4198 throw rc;
4199
4200 /* go to Deleting state, so that the medium is not actually locked */
4201 if (m->state != MediumState_Deleting)
4202 {
4203 rc = markForDeletion();
4204 if (FAILED(rc))
4205 throw rc;
4206 }
4207
4208 /* Build the medium lock list. */
4209 MediumLockList *pMediumLockList(new MediumLockList());
4210 multilock.release();
4211 rc = createMediumLockList(true /* fFailIfInaccessible */,
4212 true /* fMediumLockWrite */,
4213 NULL,
4214 *pMediumLockList);
4215 multilock.acquire();
4216 if (FAILED(rc))
4217 {
4218 delete pMediumLockList;
4219 throw rc;
4220 }
4221
4222 multilock.release();
4223 rc = pMediumLockList->Lock();
4224 multilock.acquire();
4225 if (FAILED(rc))
4226 {
4227 delete pMediumLockList;
4228 throw setError(rc,
4229 tr("Failed to lock media when deleting '%s'"),
4230 getLocationFull().c_str());
4231 }
4232
4233 /* try to remove from the list of known media before performing
4234 * actual deletion (we favor the consistency of the media registry
4235 * which would have been broken if unregisterWithVirtualBox() failed
4236 * after we successfully deleted the storage) */
4237 rc = unregisterWithVirtualBox();
4238 if (FAILED(rc))
4239 throw rc;
4240 // no longer need lock
4241 multilock.release();
4242 markRegistriesModified();
4243 m->pVirtualBox->saveModifiedRegistries();
4244
4245 if (aProgress != NULL)
4246 {
4247 /* use the existing progress object... */
4248 pProgress = *aProgress;
4249
4250 /* ...but create a new one if it is null */
4251 if (pProgress.isNull())
4252 {
4253 pProgress.createObject();
4254 rc = pProgress->init(m->pVirtualBox,
4255 static_cast<IMedium*>(this),
4256 BstrFmt(tr("Deleting medium storage unit '%s'"), m->strLocationFull.c_str()).raw(),
4257 FALSE /* aCancelable */);
4258 if (FAILED(rc))
4259 throw rc;
4260 }
4261 }
4262
4263 /* setup task object to carry out the operation sync/async */
4264 pTask = new Medium::DeleteTask(this, pProgress, pMediumLockList);
4265 rc = pTask->rc();
4266 AssertComRC(rc);
4267 if (FAILED(rc))
4268 throw rc;
4269 }
4270 catch (HRESULT aRC) { rc = aRC; }
4271
4272 if (SUCCEEDED(rc))
4273 {
4274 if (aWait)
4275 rc = runNow(pTask);
4276 else
4277 rc = startThread(pTask);
4278
4279 if (SUCCEEDED(rc) && aProgress != NULL)
4280 *aProgress = pProgress;
4281
4282 }
4283 else
4284 {
4285 if (pTask)
4286 delete pTask;
4287
4288 /* Undo deleting state if necessary. */
4289 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4290 /* Make sure that any error signalled by unmarkForDeletion() is not
4291 * ending up in the error list (if the caller uses MultiResult). It
4292 * usually is spurious, as in most cases the medium hasn't been marked
4293 * for deletion when the error was thrown above. */
4294 ErrorInfoKeeper eik;
4295 unmarkForDeletion();
4296 }
4297
4298 return rc;
4299}
4300
4301/**
4302 * Mark a medium for deletion.
4303 *
4304 * @note Caller must hold the write lock on this medium!
4305 */
4306HRESULT Medium::markForDeletion()
4307{
4308 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
4309 switch (m->state)
4310 {
4311 case MediumState_Created:
4312 case MediumState_Inaccessible:
4313 m->preLockState = m->state;
4314 m->state = MediumState_Deleting;
4315 return S_OK;
4316 default:
4317 return setStateError();
4318 }
4319}
4320
4321/**
4322 * Removes the "mark for deletion".
4323 *
4324 * @note Caller must hold the write lock on this medium!
4325 */
4326HRESULT Medium::unmarkForDeletion()
4327{
4328 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
4329 switch (m->state)
4330 {
4331 case MediumState_Deleting:
4332 m->state = m->preLockState;
4333 return S_OK;
4334 default:
4335 return setStateError();
4336 }
4337}
4338
4339/**
4340 * Mark a medium for deletion which is in locked state.
4341 *
4342 * @note Caller must hold the write lock on this medium!
4343 */
4344HRESULT Medium::markLockedForDeletion()
4345{
4346 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
4347 if ( ( m->state == MediumState_LockedRead
4348 || m->state == MediumState_LockedWrite)
4349 && m->preLockState == MediumState_Created)
4350 {
4351 m->preLockState = MediumState_Deleting;
4352 return S_OK;
4353 }
4354 else
4355 return setStateError();
4356}
4357
4358/**
4359 * Removes the "mark for deletion" for a medium in locked state.
4360 *
4361 * @note Caller must hold the write lock on this medium!
4362 */
4363HRESULT Medium::unmarkLockedForDeletion()
4364{
4365 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
4366 if ( ( m->state == MediumState_LockedRead
4367 || m->state == MediumState_LockedWrite)
4368 && m->preLockState == MediumState_Deleting)
4369 {
4370 m->preLockState = MediumState_Created;
4371 return S_OK;
4372 }
4373 else
4374 return setStateError();
4375}
4376
4377/**
4378 * Prepares this (source) medium, target medium and all intermediate media
4379 * for the merge operation.
4380 *
4381 * This method is to be called prior to calling the #mergeTo() to perform
4382 * necessary consistency checks and place involved media to appropriate
4383 * states. If #mergeTo() is not called or fails, the state modifications
4384 * performed by this method must be undone by #cancelMergeTo().
4385 *
4386 * See #mergeTo() for more information about merging.
4387 *
4388 * @param pTarget Target medium.
4389 * @param aMachineId Allowed machine attachment. NULL means do not check.
4390 * @param aSnapshotId Allowed snapshot attachment. NULL or empty UUID means
4391 * do not check.
4392 * @param fLockMedia Flag whether to lock the medium lock list or not.
4393 * If set to false and the medium lock list locking fails
4394 * later you must call #cancelMergeTo().
4395 * @param fMergeForward Resulting merge direction (out).
4396 * @param pParentForTarget New parent for target medium after merge (out).
4397 * @param aChildrenToReparent List of children of the source which will have
4398 * to be reparented to the target after merge (out).
4399 * @param aMediumLockList Medium locking information (out).
4400 *
4401 * @note Locks medium tree for reading. Locks this object, aTarget and all
4402 * intermediate media for writing.
4403 */
4404HRESULT Medium::prepareMergeTo(const ComObjPtr<Medium> &pTarget,
4405 const Guid *aMachineId,
4406 const Guid *aSnapshotId,
4407 bool fLockMedia,
4408 bool &fMergeForward,
4409 ComObjPtr<Medium> &pParentForTarget,
4410 MediaList &aChildrenToReparent,
4411 MediumLockList * &aMediumLockList)
4412{
4413 AssertReturn(pTarget != NULL, E_FAIL);
4414 AssertReturn(pTarget != this, E_FAIL);
4415
4416 AutoCaller autoCaller(this);
4417 AssertComRCReturnRC(autoCaller.rc());
4418
4419 AutoCaller targetCaller(pTarget);
4420 AssertComRCReturnRC(targetCaller.rc());
4421
4422 HRESULT rc = S_OK;
4423 fMergeForward = false;
4424 pParentForTarget.setNull();
4425 aChildrenToReparent.clear();
4426 Assert(aMediumLockList == NULL);
4427 aMediumLockList = NULL;
4428
4429 try
4430 {
4431 // locking: we need the tree lock first because we access parent pointers
4432 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4433
4434 /* more sanity checking and figuring out the merge direction */
4435 ComObjPtr<Medium> pMedium = getParent();
4436 while (!pMedium.isNull() && pMedium != pTarget)
4437 pMedium = pMedium->getParent();
4438 if (pMedium == pTarget)
4439 fMergeForward = false;
4440 else
4441 {
4442 pMedium = pTarget->getParent();
4443 while (!pMedium.isNull() && pMedium != this)
4444 pMedium = pMedium->getParent();
4445 if (pMedium == this)
4446 fMergeForward = true;
4447 else
4448 {
4449 Utf8Str tgtLoc;
4450 {
4451 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4452 tgtLoc = pTarget->getLocationFull();
4453 }
4454
4455 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4456 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4457 tr("Media '%s' and '%s' are unrelated"),
4458 m->strLocationFull.c_str(), tgtLoc.c_str());
4459 }
4460 }
4461
4462 /* Build the lock list. */
4463 aMediumLockList = new MediumLockList();
4464 treeLock.release();
4465 if (fMergeForward)
4466 rc = pTarget->createMediumLockList(true /* fFailIfInaccessible */,
4467 true /* fMediumLockWrite */,
4468 NULL,
4469 *aMediumLockList);
4470 else
4471 rc = createMediumLockList(true /* fFailIfInaccessible */,
4472 false /* fMediumLockWrite */,
4473 NULL,
4474 *aMediumLockList);
4475 treeLock.acquire();
4476 if (FAILED(rc))
4477 throw rc;
4478
4479 /* Sanity checking, must be after lock list creation as it depends on
4480 * valid medium states. The medium objects must be accessible. Only
4481 * do this if immediate locking is requested, otherwise it fails when
4482 * we construct a medium lock list for an already running VM. Snapshot
4483 * deletion uses this to simplify its life. */
4484 if (fLockMedia)
4485 {
4486 {
4487 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4488 if (m->state != MediumState_Created)
4489 throw setStateError();
4490 }
4491 {
4492 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4493 if (pTarget->m->state != MediumState_Created)
4494 throw pTarget->setStateError();
4495 }
4496 }
4497
4498 /* check medium attachment and other sanity conditions */
4499 if (fMergeForward)
4500 {
4501 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4502 if (getChildren().size() > 1)
4503 {
4504 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4505 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4506 m->strLocationFull.c_str(), getChildren().size());
4507 }
4508 /* One backreference is only allowed if the machine ID is not empty
4509 * and it matches the machine the medium is attached to (including
4510 * the snapshot ID if not empty). */
4511 if ( m->backRefs.size() != 0
4512 && ( !aMachineId
4513 || m->backRefs.size() != 1
4514 || aMachineId->isEmpty()
4515 || *getFirstMachineBackrefId() != *aMachineId
4516 || ( (!aSnapshotId || !aSnapshotId->isEmpty())
4517 && *getFirstMachineBackrefSnapshotId() != *aSnapshotId)))
4518 throw setError(VBOX_E_OBJECT_IN_USE,
4519 tr("Medium '%s' is attached to %d virtual machines"),
4520 m->strLocationFull.c_str(), m->backRefs.size());
4521 if (m->type == MediumType_Immutable)
4522 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4523 tr("Medium '%s' is immutable"),
4524 m->strLocationFull.c_str());
4525 if (m->type == MediumType_MultiAttach)
4526 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4527 tr("Medium '%s' is multi-attach"),
4528 m->strLocationFull.c_str());
4529 }
4530 else
4531 {
4532 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4533 if (pTarget->getChildren().size() > 1)
4534 {
4535 throw setError(VBOX_E_OBJECT_IN_USE,
4536 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4537 pTarget->m->strLocationFull.c_str(),
4538 pTarget->getChildren().size());
4539 }
4540 if (pTarget->m->type == MediumType_Immutable)
4541 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4542 tr("Medium '%s' is immutable"),
4543 pTarget->m->strLocationFull.c_str());
4544 if (pTarget->m->type == MediumType_MultiAttach)
4545 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4546 tr("Medium '%s' is multi-attach"),
4547 pTarget->m->strLocationFull.c_str());
4548 }
4549 ComObjPtr<Medium> pLast(fMergeForward ? (Medium *)pTarget : this);
4550 ComObjPtr<Medium> pLastIntermediate = pLast->getParent();
4551 for (pLast = pLastIntermediate;
4552 !pLast.isNull() && pLast != pTarget && pLast != this;
4553 pLast = pLast->getParent())
4554 {
4555 AutoReadLock alock(pLast COMMA_LOCKVAL_SRC_POS);
4556 if (pLast->getChildren().size() > 1)
4557 {
4558 throw setError(VBOX_E_OBJECT_IN_USE,
4559 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4560 pLast->m->strLocationFull.c_str(),
4561 pLast->getChildren().size());
4562 }
4563 if (pLast->m->backRefs.size() != 0)
4564 throw setError(VBOX_E_OBJECT_IN_USE,
4565 tr("Medium '%s' is attached to %d virtual machines"),
4566 pLast->m->strLocationFull.c_str(),
4567 pLast->m->backRefs.size());
4568
4569 }
4570
4571 /* Update medium states appropriately */
4572 {
4573 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4574
4575 if (m->state == MediumState_Created)
4576 {
4577 rc = markForDeletion();
4578 if (FAILED(rc))
4579 throw rc;
4580 }
4581 else
4582 {
4583 if (fLockMedia)
4584 throw setStateError();
4585 else if ( m->state == MediumState_LockedWrite
4586 || m->state == MediumState_LockedRead)
4587 {
4588 /* Either mark it for deletion in locked state or allow
4589 * others to have done so. */
4590 if (m->preLockState == MediumState_Created)
4591 markLockedForDeletion();
4592 else if (m->preLockState != MediumState_Deleting)
4593 throw setStateError();
4594 }
4595 else
4596 throw setStateError();
4597 }
4598 }
4599
4600 if (fMergeForward)
4601 {
4602 /* we will need parent to reparent target */
4603 pParentForTarget = getParent();
4604 }
4605 else
4606 {
4607 /* we will need to reparent children of the source */
4608 for (MediaList::const_iterator it = getChildren().begin();
4609 it != getChildren().end();
4610 ++it)
4611 {
4612 pMedium = *it;
4613 if (fLockMedia)
4614 {
4615 rc = pMedium->LockWrite(NULL);
4616 if (FAILED(rc))
4617 throw rc;
4618 }
4619
4620 aChildrenToReparent.push_back(pMedium);
4621 }
4622 }
4623 for (pLast = pLastIntermediate;
4624 !pLast.isNull() && pLast != pTarget && pLast != this;
4625 pLast = pLast->getParent())
4626 {
4627 AutoWriteLock alock(pLast COMMA_LOCKVAL_SRC_POS);
4628 if (pLast->m->state == MediumState_Created)
4629 {
4630 rc = pLast->markForDeletion();
4631 if (FAILED(rc))
4632 throw rc;
4633 }
4634 else
4635 throw pLast->setStateError();
4636 }
4637
4638 /* Tweak the lock list in the backward merge case, as the target
4639 * isn't marked to be locked for writing yet. */
4640 if (!fMergeForward)
4641 {
4642 MediumLockList::Base::iterator lockListBegin =
4643 aMediumLockList->GetBegin();
4644 MediumLockList::Base::iterator lockListEnd =
4645 aMediumLockList->GetEnd();
4646 lockListEnd--;
4647 for (MediumLockList::Base::iterator it = lockListBegin;
4648 it != lockListEnd;
4649 ++it)
4650 {
4651 MediumLock &mediumLock = *it;
4652 if (mediumLock.GetMedium() == pTarget)
4653 {
4654 HRESULT rc2 = mediumLock.UpdateLock(true);
4655 AssertComRC(rc2);
4656 break;
4657 }
4658 }
4659 }
4660
4661 if (fLockMedia)
4662 {
4663 treeLock.release();
4664 rc = aMediumLockList->Lock();
4665 treeLock.acquire();
4666 if (FAILED(rc))
4667 {
4668 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4669 throw setError(rc,
4670 tr("Failed to lock media when merging to '%s'"),
4671 pTarget->getLocationFull().c_str());
4672 }
4673 }
4674 }
4675 catch (HRESULT aRC) { rc = aRC; }
4676
4677 if (FAILED(rc))
4678 {
4679 delete aMediumLockList;
4680 aMediumLockList = NULL;
4681 }
4682
4683 return rc;
4684}
4685
4686/**
4687 * Merges this medium to the specified medium which must be either its
4688 * direct ancestor or descendant.
4689 *
4690 * Given this medium is SOURCE and the specified medium is TARGET, we will
4691 * get two variants of the merge operation:
4692 *
4693 * forward merge
4694 * ------------------------->
4695 * [Extra] <- SOURCE <- Intermediate <- TARGET
4696 * Any Del Del LockWr
4697 *
4698 *
4699 * backward merge
4700 * <-------------------------
4701 * TARGET <- Intermediate <- SOURCE <- [Extra]
4702 * LockWr Del Del LockWr
4703 *
4704 * Each diagram shows the involved media on the media chain where
4705 * SOURCE and TARGET belong. Under each medium there is a state value which
4706 * the medium must have at a time of the mergeTo() call.
4707 *
4708 * The media in the square braces may be absent (e.g. when the forward
4709 * operation takes place and SOURCE is the base medium, or when the backward
4710 * merge operation takes place and TARGET is the last child in the chain) but if
4711 * they present they are involved too as shown.
4712 *
4713 * Neither the source medium nor intermediate media may be attached to
4714 * any VM directly or in the snapshot, otherwise this method will assert.
4715 *
4716 * The #prepareMergeTo() method must be called prior to this method to place all
4717 * involved to necessary states and perform other consistency checks.
4718 *
4719 * If @a aWait is @c true then this method will perform the operation on the
4720 * calling thread and will not return to the caller until the operation is
4721 * completed. When this method succeeds, all intermediate medium objects in
4722 * the chain will be uninitialized, the state of the target medium (and all
4723 * involved extra media) will be restored. @a aMediumLockList will not be
4724 * deleted, whether the operation is successful or not. The caller has to do
4725 * this if appropriate. Note that this (source) medium is not uninitialized
4726 * because of possible AutoCaller instances held by the caller of this method
4727 * on the current thread. It's therefore the responsibility of the caller to
4728 * call Medium::uninit() after releasing all callers.
4729 *
4730 * If @a aWait is @c false then this method will create a thread to perform the
4731 * operation asynchronously and will return immediately. If the operation
4732 * succeeds, the thread will uninitialize the source medium object and all
4733 * intermediate medium objects in the chain, reset the state of the target
4734 * medium (and all involved extra media) and delete @a aMediumLockList.
4735 * If the operation fails, the thread will only reset the states of all
4736 * involved media and delete @a aMediumLockList.
4737 *
4738 * When this method fails (regardless of the @a aWait mode), it is a caller's
4739 * responsibility to undo state changes and delete @a aMediumLockList using
4740 * #cancelMergeTo().
4741 *
4742 * If @a aProgress is not NULL but the object it points to is @c null then a new
4743 * progress object will be created and assigned to @a *aProgress on success,
4744 * otherwise the existing progress object is used. If Progress is NULL, then no
4745 * progress object is created/used at all. Note that @a aProgress cannot be
4746 * NULL when @a aWait is @c false (this method will assert in this case).
4747 *
4748 * @param pTarget Target medium.
4749 * @param fMergeForward Merge direction.
4750 * @param pParentForTarget New parent for target medium after merge.
4751 * @param aChildrenToReparent List of children of the source which will have
4752 * to be reparented to the target after merge.
4753 * @param aMediumLockList Medium locking information.
4754 * @param aProgress Where to find/store a Progress object to track operation
4755 * completion.
4756 * @param aWait @c true if this method should block instead of creating
4757 * an asynchronous thread.
4758 *
4759 * @note Locks the tree lock for writing. Locks the media from the chain
4760 * for writing.
4761 */
4762HRESULT Medium::mergeTo(const ComObjPtr<Medium> &pTarget,
4763 bool fMergeForward,
4764 const ComObjPtr<Medium> &pParentForTarget,
4765 const MediaList &aChildrenToReparent,
4766 MediumLockList *aMediumLockList,
4767 ComObjPtr <Progress> *aProgress,
4768 bool aWait)
4769{
4770 AssertReturn(pTarget != NULL, E_FAIL);
4771 AssertReturn(pTarget != this, E_FAIL);
4772 AssertReturn(aMediumLockList != NULL, E_FAIL);
4773 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
4774
4775 AutoCaller autoCaller(this);
4776 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4777
4778 AutoCaller targetCaller(pTarget);
4779 AssertComRCReturnRC(targetCaller.rc());
4780
4781 HRESULT rc = S_OK;
4782 ComObjPtr <Progress> pProgress;
4783 Medium::Task *pTask = NULL;
4784
4785 try
4786 {
4787 if (aProgress != NULL)
4788 {
4789 /* use the existing progress object... */
4790 pProgress = *aProgress;
4791
4792 /* ...but create a new one if it is null */
4793 if (pProgress.isNull())
4794 {
4795 Utf8Str tgtName;
4796 {
4797 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4798 tgtName = pTarget->getName();
4799 }
4800
4801 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4802
4803 pProgress.createObject();
4804 rc = pProgress->init(m->pVirtualBox,
4805 static_cast<IMedium*>(this),
4806 BstrFmt(tr("Merging medium '%s' to '%s'"),
4807 getName().c_str(),
4808 tgtName.c_str()).raw(),
4809 TRUE /* aCancelable */);
4810 if (FAILED(rc))
4811 throw rc;
4812 }
4813 }
4814
4815 /* setup task object to carry out the operation sync/async */
4816 pTask = new Medium::MergeTask(this, pTarget, fMergeForward,
4817 pParentForTarget, aChildrenToReparent,
4818 pProgress, aMediumLockList,
4819 aWait /* fKeepMediumLockList */);
4820 rc = pTask->rc();
4821 AssertComRC(rc);
4822 if (FAILED(rc))
4823 throw rc;
4824 }
4825 catch (HRESULT aRC) { rc = aRC; }
4826
4827 if (SUCCEEDED(rc))
4828 {
4829 if (aWait)
4830 rc = runNow(pTask);
4831 else
4832 rc = startThread(pTask);
4833
4834 if (SUCCEEDED(rc) && aProgress != NULL)
4835 *aProgress = pProgress;
4836 }
4837 else if (pTask != NULL)
4838 delete pTask;
4839
4840 return rc;
4841}
4842
4843/**
4844 * Undoes what #prepareMergeTo() did. Must be called if #mergeTo() is not
4845 * called or fails. Frees memory occupied by @a aMediumLockList and unlocks
4846 * the medium objects in @a aChildrenToReparent.
4847 *
4848 * @param aChildrenToReparent List of children of the source which will have
4849 * to be reparented to the target after merge.
4850 * @param aMediumLockList Medium locking information.
4851 *
4852 * @note Locks the media from the chain for writing.
4853 */
4854void Medium::cancelMergeTo(const MediaList &aChildrenToReparent,
4855 MediumLockList *aMediumLockList)
4856{
4857 AutoCaller autoCaller(this);
4858 AssertComRCReturnVoid(autoCaller.rc());
4859
4860 AssertReturnVoid(aMediumLockList != NULL);
4861
4862 /* Revert media marked for deletion to previous state. */
4863 HRESULT rc;
4864 MediumLockList::Base::const_iterator mediumListBegin =
4865 aMediumLockList->GetBegin();
4866 MediumLockList::Base::const_iterator mediumListEnd =
4867 aMediumLockList->GetEnd();
4868 for (MediumLockList::Base::const_iterator it = mediumListBegin;
4869 it != mediumListEnd;
4870 ++it)
4871 {
4872 const MediumLock &mediumLock = *it;
4873 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
4874 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
4875
4876 if (pMedium->m->state == MediumState_Deleting)
4877 {
4878 rc = pMedium->unmarkForDeletion();
4879 AssertComRC(rc);
4880 }
4881 }
4882
4883 /* the destructor will do the work */
4884 delete aMediumLockList;
4885
4886 /* unlock the children which had to be reparented */
4887 for (MediaList::const_iterator it = aChildrenToReparent.begin();
4888 it != aChildrenToReparent.end();
4889 ++it)
4890 {
4891 const ComObjPtr<Medium> &pMedium = *it;
4892
4893 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
4894 pMedium->UnlockWrite(NULL);
4895 }
4896}
4897
4898/**
4899 * Fix the parent UUID of all children to point to this medium as their
4900 * parent.
4901 */
4902HRESULT Medium::fixParentUuidOfChildren(const MediaList &childrenToReparent)
4903{
4904 Assert(!isWriteLockOnCurrentThread());
4905 Assert(!m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
4906 MediumLockList mediumLockList;
4907 HRESULT rc = createMediumLockList(true /* fFailIfInaccessible */,
4908 false /* fMediumLockWrite */,
4909 this,
4910 mediumLockList);
4911 AssertComRCReturnRC(rc);
4912
4913 try
4914 {
4915 PVBOXHDD hdd;
4916 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
4917 ComAssertRCThrow(vrc, E_FAIL);
4918
4919 try
4920 {
4921 MediumLockList::Base::iterator lockListBegin =
4922 mediumLockList.GetBegin();
4923 MediumLockList::Base::iterator lockListEnd =
4924 mediumLockList.GetEnd();
4925 for (MediumLockList::Base::iterator it = lockListBegin;
4926 it != lockListEnd;
4927 ++it)
4928 {
4929 MediumLock &mediumLock = *it;
4930 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
4931 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
4932
4933 // open the medium
4934 vrc = VDOpen(hdd,
4935 pMedium->m->strFormat.c_str(),
4936 pMedium->m->strLocationFull.c_str(),
4937 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
4938 pMedium->m->vdImageIfaces);
4939 if (RT_FAILURE(vrc))
4940 throw vrc;
4941 }
4942
4943 for (MediaList::const_iterator it = childrenToReparent.begin();
4944 it != childrenToReparent.end();
4945 ++it)
4946 {
4947 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
4948 vrc = VDOpen(hdd,
4949 (*it)->m->strFormat.c_str(),
4950 (*it)->m->strLocationFull.c_str(),
4951 VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
4952 (*it)->m->vdImageIfaces);
4953 if (RT_FAILURE(vrc))
4954 throw vrc;
4955
4956 vrc = VDSetParentUuid(hdd, VD_LAST_IMAGE, m->id.raw());
4957 if (RT_FAILURE(vrc))
4958 throw vrc;
4959
4960 vrc = VDClose(hdd, false /* fDelete */);
4961 if (RT_FAILURE(vrc))
4962 throw vrc;
4963
4964 (*it)->UnlockWrite(NULL);
4965 }
4966 }
4967 catch (HRESULT aRC) { rc = aRC; }
4968 catch (int aVRC)
4969 {
4970 rc = setError(E_FAIL,
4971 tr("Could not update medium UUID references to parent '%s' (%s)"),
4972 m->strLocationFull.c_str(),
4973 vdError(aVRC).c_str());
4974 }
4975
4976 VDDestroy(hdd);
4977 }
4978 catch (HRESULT aRC) { rc = aRC; }
4979
4980 return rc;
4981}
4982
4983/**
4984 * Used by IAppliance to export disk images.
4985 *
4986 * @param aFilename Filename to create (UTF8).
4987 * @param aFormat Medium format for creating @a aFilename.
4988 * @param aVariant Which exact image format variant to use
4989 * for the destination image.
4990 * @param aVDImageIOCallbacks Pointer to the callback table for a
4991 * VDINTERFACEIO interface. May be NULL.
4992 * @param aVDImageIOUser Opaque data for the callbacks.
4993 * @param aProgress Progress object to use.
4994 * @return
4995 * @note The source format is defined by the Medium instance.
4996 */
4997HRESULT Medium::exportFile(const char *aFilename,
4998 const ComObjPtr<MediumFormat> &aFormat,
4999 MediumVariant_T aVariant,
5000 PVDINTERFACEIO aVDImageIOIf, void *aVDImageIOUser,
5001 const ComObjPtr<Progress> &aProgress)
5002{
5003 AssertPtrReturn(aFilename, E_INVALIDARG);
5004 AssertReturn(!aFormat.isNull(), E_INVALIDARG);
5005 AssertReturn(!aProgress.isNull(), E_INVALIDARG);
5006
5007 AutoCaller autoCaller(this);
5008 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5009
5010 HRESULT rc = S_OK;
5011 Medium::Task *pTask = NULL;
5012
5013 try
5014 {
5015 // This needs no extra locks besides what is done in the called methods.
5016
5017 /* Build the source lock list. */
5018 MediumLockList *pSourceMediumLockList(new MediumLockList());
5019 rc = createMediumLockList(true /* fFailIfInaccessible */,
5020 false /* fMediumLockWrite */,
5021 NULL,
5022 *pSourceMediumLockList);
5023 if (FAILED(rc))
5024 {
5025 delete pSourceMediumLockList;
5026 throw rc;
5027 }
5028
5029 rc = pSourceMediumLockList->Lock();
5030 if (FAILED(rc))
5031 {
5032 delete pSourceMediumLockList;
5033 throw setError(rc,
5034 tr("Failed to lock source media '%s'"),
5035 getLocationFull().c_str());
5036 }
5037
5038 /* setup task object to carry out the operation asynchronously */
5039 pTask = new Medium::ExportTask(this, aProgress, aFilename, aFormat,
5040 aVariant, aVDImageIOIf,
5041 aVDImageIOUser, pSourceMediumLockList);
5042 rc = pTask->rc();
5043 AssertComRC(rc);
5044 if (FAILED(rc))
5045 throw rc;
5046 }
5047 catch (HRESULT aRC) { rc = aRC; }
5048
5049 if (SUCCEEDED(rc))
5050 rc = startThread(pTask);
5051 else if (pTask != NULL)
5052 delete pTask;
5053
5054 return rc;
5055}
5056
5057/**
5058 * Used by IAppliance to import disk images.
5059 *
5060 * @param aFilename Filename to read (UTF8).
5061 * @param aFormat Medium format for reading @a aFilename.
5062 * @param aVariant Which exact image format variant to use
5063 * for the destination image.
5064 * @param aVDImageIOCallbacks Pointer to the callback table for a
5065 * VDINTERFACEIO interface. May be NULL.
5066 * @param aVDImageIOUser Opaque data for the callbacks.
5067 * @param aParent Parent medium. May be NULL.
5068 * @param aProgress Progress object to use.
5069 * @return
5070 * @note The destination format is defined by the Medium instance.
5071 */
5072HRESULT Medium::importFile(const char *aFilename,
5073 const ComObjPtr<MediumFormat> &aFormat,
5074 MediumVariant_T aVariant,
5075 PVDINTERFACEIO aVDImageIOIf, void *aVDImageIOUser,
5076 const ComObjPtr<Medium> &aParent,
5077 const ComObjPtr<Progress> &aProgress)
5078{
5079 AssertPtrReturn(aFilename, E_INVALIDARG);
5080 AssertReturn(!aFormat.isNull(), E_INVALIDARG);
5081 AssertReturn(!aProgress.isNull(), E_INVALIDARG);
5082
5083 AutoCaller autoCaller(this);
5084 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5085
5086 HRESULT rc = S_OK;
5087 Medium::Task *pTask = NULL;
5088
5089 try
5090 {
5091 // locking: we need the tree lock first because we access parent pointers
5092 // and we need to write-lock the media involved
5093 uint32_t cHandles = 2;
5094 LockHandle* pHandles[3] = { &m->pVirtualBox->getMediaTreeLockHandle(),
5095 this->lockHandle() };
5096 /* Only add parent to the lock if it is not null */
5097 if (!aParent.isNull())
5098 pHandles[cHandles++] = aParent->lockHandle();
5099 AutoWriteLock alock(cHandles,
5100 pHandles
5101 COMMA_LOCKVAL_SRC_POS);
5102
5103 if ( m->state != MediumState_NotCreated
5104 && m->state != MediumState_Created)
5105 throw setStateError();
5106
5107 /* Build the target lock list. */
5108 MediumLockList *pTargetMediumLockList(new MediumLockList());
5109 alock.release();
5110 rc = createMediumLockList(true /* fFailIfInaccessible */,
5111 true /* fMediumLockWrite */,
5112 aParent,
5113 *pTargetMediumLockList);
5114 alock.acquire();
5115 if (FAILED(rc))
5116 {
5117 delete pTargetMediumLockList;
5118 throw rc;
5119 }
5120
5121 alock.release();
5122 rc = pTargetMediumLockList->Lock();
5123 alock.acquire();
5124 if (FAILED(rc))
5125 {
5126 delete pTargetMediumLockList;
5127 throw setError(rc,
5128 tr("Failed to lock target media '%s'"),
5129 getLocationFull().c_str());
5130 }
5131
5132 /* setup task object to carry out the operation asynchronously */
5133 pTask = new Medium::ImportTask(this, aProgress, aFilename, aFormat,
5134 aVariant, aVDImageIOIf,
5135 aVDImageIOUser, aParent,
5136 pTargetMediumLockList);
5137 rc = pTask->rc();
5138 AssertComRC(rc);
5139 if (FAILED(rc))
5140 throw rc;
5141
5142 if (m->state == MediumState_NotCreated)
5143 m->state = MediumState_Creating;
5144 }
5145 catch (HRESULT aRC) { rc = aRC; }
5146
5147 if (SUCCEEDED(rc))
5148 rc = startThread(pTask);
5149 else if (pTask != NULL)
5150 delete pTask;
5151
5152 return rc;
5153}
5154
5155/**
5156 * Internal version of the public CloneTo API which allows to enable certain
5157 * optimizations to improve speed during VM cloning.
5158 *
5159 * @param aTarget Target medium
5160 * @param aVariant Which exact image format variant to use
5161 * for the destination image.
5162 * @param aParent Parent medium. May be NULL.
5163 * @param aProgress Progress object to use.
5164 * @param idxSrcImageSame The last image in the source chain which has the
5165 * same content as the given image in the destination
5166 * chain. Use UINT32_MAX to disable this optimization.
5167 * @param idxDstImageSame The last image in the destination chain which has the
5168 * same content as the given image in the source chain.
5169 * Use UINT32_MAX to disable this optimization.
5170 * @return
5171 */
5172HRESULT Medium::cloneToEx(const ComObjPtr<Medium> &aTarget, ULONG aVariant,
5173 const ComObjPtr<Medium> &aParent, IProgress **aProgress,
5174 uint32_t idxSrcImageSame, uint32_t idxDstImageSame)
5175{
5176 CheckComArgNotNull(aTarget);
5177 CheckComArgOutPointerValid(aProgress);
5178 ComAssertRet(aTarget != this, E_INVALIDARG);
5179
5180 AutoCaller autoCaller(this);
5181 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5182
5183 HRESULT rc = S_OK;
5184 ComObjPtr<Progress> pProgress;
5185 Medium::Task *pTask = NULL;
5186
5187 try
5188 {
5189 // locking: we need the tree lock first because we access parent pointers
5190 // and we need to write-lock the media involved
5191 uint32_t cHandles = 3;
5192 LockHandle* pHandles[4] = { &m->pVirtualBox->getMediaTreeLockHandle(),
5193 this->lockHandle(),
5194 aTarget->lockHandle() };
5195 /* Only add parent to the lock if it is not null */
5196 if (!aParent.isNull())
5197 pHandles[cHandles++] = aParent->lockHandle();
5198 AutoWriteLock alock(cHandles,
5199 pHandles
5200 COMMA_LOCKVAL_SRC_POS);
5201
5202 if ( aTarget->m->state != MediumState_NotCreated
5203 && aTarget->m->state != MediumState_Created)
5204 throw aTarget->setStateError();
5205
5206 /* Build the source lock list. */
5207 MediumLockList *pSourceMediumLockList(new MediumLockList());
5208 alock.release();
5209 rc = createMediumLockList(true /* fFailIfInaccessible */,
5210 false /* fMediumLockWrite */,
5211 NULL,
5212 *pSourceMediumLockList);
5213 alock.acquire();
5214 if (FAILED(rc))
5215 {
5216 delete pSourceMediumLockList;
5217 throw rc;
5218 }
5219
5220 /* Build the target lock list (including the to-be parent chain). */
5221 MediumLockList *pTargetMediumLockList(new MediumLockList());
5222 alock.release();
5223 rc = aTarget->createMediumLockList(true /* fFailIfInaccessible */,
5224 true /* fMediumLockWrite */,
5225 aParent,
5226 *pTargetMediumLockList);
5227 alock.acquire();
5228 if (FAILED(rc))
5229 {
5230 delete pSourceMediumLockList;
5231 delete pTargetMediumLockList;
5232 throw rc;
5233 }
5234
5235 alock.release();
5236 rc = pSourceMediumLockList->Lock();
5237 alock.acquire();
5238 if (FAILED(rc))
5239 {
5240 delete pSourceMediumLockList;
5241 delete pTargetMediumLockList;
5242 throw setError(rc,
5243 tr("Failed to lock source media '%s'"),
5244 getLocationFull().c_str());
5245 }
5246 alock.release();
5247 rc = pTargetMediumLockList->Lock();
5248 alock.acquire();
5249 if (FAILED(rc))
5250 {
5251 delete pSourceMediumLockList;
5252 delete pTargetMediumLockList;
5253 throw setError(rc,
5254 tr("Failed to lock target media '%s'"),
5255 aTarget->getLocationFull().c_str());
5256 }
5257
5258 pProgress.createObject();
5259 rc = pProgress->init(m->pVirtualBox,
5260 static_cast <IMedium *>(this),
5261 BstrFmt(tr("Creating clone medium '%s'"), aTarget->m->strLocationFull.c_str()).raw(),
5262 TRUE /* aCancelable */);
5263 if (FAILED(rc))
5264 {
5265 delete pSourceMediumLockList;
5266 delete pTargetMediumLockList;
5267 throw rc;
5268 }
5269
5270 /* setup task object to carry out the operation asynchronously */
5271 pTask = new Medium::CloneTask(this, pProgress, aTarget,
5272 (MediumVariant_T)aVariant,
5273 aParent, idxSrcImageSame,
5274 idxDstImageSame, pSourceMediumLockList,
5275 pTargetMediumLockList);
5276 rc = pTask->rc();
5277 AssertComRC(rc);
5278 if (FAILED(rc))
5279 throw rc;
5280
5281 if (aTarget->m->state == MediumState_NotCreated)
5282 aTarget->m->state = MediumState_Creating;
5283 }
5284 catch (HRESULT aRC) { rc = aRC; }
5285
5286 if (SUCCEEDED(rc))
5287 {
5288 rc = startThread(pTask);
5289
5290 if (SUCCEEDED(rc))
5291 pProgress.queryInterfaceTo(aProgress);
5292 }
5293 else if (pTask != NULL)
5294 delete pTask;
5295
5296 return rc;
5297}
5298
5299////////////////////////////////////////////////////////////////////////////////
5300//
5301// Private methods
5302//
5303////////////////////////////////////////////////////////////////////////////////
5304
5305/**
5306 * Queries information from the medium.
5307 *
5308 * As a result of this call, the accessibility state and data members such as
5309 * size and description will be updated with the current information.
5310 *
5311 * @note This method may block during a system I/O call that checks storage
5312 * accessibility.
5313 *
5314 * @note Caller MUST NOT hold the media tree or medium lock.
5315 *
5316 * @note Locks mParent for reading. Locks this object for writing.
5317 *
5318 * @param fSetImageId Whether to reset the UUID contained in the image file to the UUID in the medium instance data (see SetIDs())
5319 * @param fSetParentId Whether to reset the parent UUID contained in the image file to the parent UUID in the medium instance data (see SetIDs())
5320 * @return
5321 */
5322HRESULT Medium::queryInfo(bool fSetImageId, bool fSetParentId)
5323{
5324 Assert(!isWriteLockOnCurrentThread());
5325 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5326
5327 if ( m->state != MediumState_Created
5328 && m->state != MediumState_Inaccessible
5329 && m->state != MediumState_LockedRead)
5330 return E_FAIL;
5331
5332 HRESULT rc = S_OK;
5333
5334 int vrc = VINF_SUCCESS;
5335
5336 /* check if a blocking queryInfo() call is in progress on some other thread,
5337 * and wait for it to finish if so instead of querying data ourselves */
5338 if (m->queryInfoRunning)
5339 {
5340 Assert( m->state == MediumState_LockedRead
5341 || m->state == MediumState_LockedWrite);
5342
5343 while (m->queryInfoRunning)
5344 {
5345 alock.release();
5346 {
5347 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
5348 }
5349 alock.acquire();
5350 }
5351
5352 return S_OK;
5353 }
5354
5355 bool success = false;
5356 Utf8Str lastAccessError;
5357
5358 /* are we dealing with a new medium constructed using the existing
5359 * location? */
5360 bool isImport = m->id.isEmpty();
5361 unsigned uOpenFlags = VD_OPEN_FLAGS_INFO;
5362
5363 /* Note that we don't use VD_OPEN_FLAGS_READONLY when opening new
5364 * media because that would prevent necessary modifications
5365 * when opening media of some third-party formats for the first
5366 * time in VirtualBox (such as VMDK for which VDOpen() needs to
5367 * generate an UUID if it is missing) */
5368 if ( m->hddOpenMode == OpenReadOnly
5369 || m->type == MediumType_Readonly
5370 || (!isImport && !fSetImageId && !fSetParentId)
5371 )
5372 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
5373
5374 /* Open shareable medium with the appropriate flags */
5375 if (m->type == MediumType_Shareable)
5376 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
5377
5378 /* Lock the medium, which makes the behavior much more consistent */
5379 alock.release();
5380 if (uOpenFlags & (VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_SHAREABLE))
5381 rc = LockRead(NULL);
5382 else
5383 rc = LockWrite(NULL);
5384 if (FAILED(rc)) return rc;
5385 alock.acquire();
5386
5387 /* Copies of the input state fields which are not read-only,
5388 * as we're dropping the lock. CAUTION: be extremely careful what
5389 * you do with the contents of this medium object, as you will
5390 * create races if there are concurrent changes. */
5391 Utf8Str format(m->strFormat);
5392 Utf8Str location(m->strLocationFull);
5393 ComObjPtr<MediumFormat> formatObj = m->formatObj;
5394
5395 /* "Output" values which can't be set because the lock isn't held
5396 * at the time the values are determined. */
5397 Guid mediumId = m->id;
5398 uint64_t mediumSize = 0;
5399 uint64_t mediumLogicalSize = 0;
5400
5401 /* Flag whether a base image has a non-zero parent UUID and thus
5402 * need repairing after it was closed again. */
5403 bool fRepairImageZeroParentUuid = false;
5404
5405 /* release the object lock before a lengthy operation, and take the
5406 * opportunity to have a media tree lock, too, which isn't held initially */
5407 m->queryInfoRunning = true;
5408 alock.release();
5409 Assert(!isWriteLockOnCurrentThread());
5410 Assert(!m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5411 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5412 treeLock.release();
5413
5414 /* Note that taking the queryInfoSem after leaving the object lock above
5415 * can lead to short spinning of the loops waiting for queryInfo() to
5416 * complete. This is unavoidable since the other order causes a lock order
5417 * violation: here it would be requesting the object lock (at the beginning
5418 * of the method), then queryInfoSem, and below the other way round. */
5419 AutoWriteLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
5420
5421 try
5422 {
5423 /* skip accessibility checks for host drives */
5424 if (m->hostDrive)
5425 {
5426 success = true;
5427 throw S_OK;
5428 }
5429
5430 PVBOXHDD hdd;
5431 vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
5432 ComAssertRCThrow(vrc, E_FAIL);
5433
5434 try
5435 {
5436 /** @todo This kind of opening of media is assuming that diff
5437 * media can be opened as base media. Should be documented that
5438 * it must work for all medium format backends. */
5439 vrc = VDOpen(hdd,
5440 format.c_str(),
5441 location.c_str(),
5442 uOpenFlags | m->uOpenFlagsDef,
5443 m->vdImageIfaces);
5444 if (RT_FAILURE(vrc))
5445 {
5446 lastAccessError = Utf8StrFmt(tr("Could not open the medium '%s'%s"),
5447 location.c_str(), vdError(vrc).c_str());
5448 throw S_OK;
5449 }
5450
5451 if (formatObj->getCapabilities() & MediumFormatCapabilities_Uuid)
5452 {
5453 /* Modify the UUIDs if necessary. The associated fields are
5454 * not modified by other code, so no need to copy. */
5455 if (fSetImageId)
5456 {
5457 alock.acquire();
5458 vrc = VDSetUuid(hdd, 0, m->uuidImage.raw());
5459 alock.release();
5460 ComAssertRCThrow(vrc, E_FAIL);
5461 mediumId = m->uuidImage;
5462 }
5463 if (fSetParentId)
5464 {
5465 alock.acquire();
5466 vrc = VDSetParentUuid(hdd, 0, m->uuidParentImage.raw());
5467 alock.release();
5468 ComAssertRCThrow(vrc, E_FAIL);
5469 }
5470 /* zap the information, these are no long-term members */
5471 alock.acquire();
5472 unconst(m->uuidImage).clear();
5473 unconst(m->uuidParentImage).clear();
5474 alock.release();
5475
5476 /* check the UUID */
5477 RTUUID uuid;
5478 vrc = VDGetUuid(hdd, 0, &uuid);
5479 ComAssertRCThrow(vrc, E_FAIL);
5480
5481 if (isImport)
5482 {
5483 mediumId = uuid;
5484
5485 if (mediumId.isEmpty() && (m->hddOpenMode == OpenReadOnly))
5486 // only when importing a VDMK that has no UUID, create one in memory
5487 mediumId.create();
5488 }
5489 else
5490 {
5491 Assert(!mediumId.isEmpty());
5492
5493 if (mediumId != uuid)
5494 {
5495 /** @todo r=klaus this always refers to VirtualBox.xml as the medium registry, even for new VMs */
5496 lastAccessError = Utf8StrFmt(
5497 tr("UUID {%RTuuid} of the medium '%s' does not match the value {%RTuuid} stored in the media registry ('%s')"),
5498 &uuid,
5499 location.c_str(),
5500 mediumId.raw(),
5501 m->pVirtualBox->settingsFilePath().c_str());
5502 throw S_OK;
5503 }
5504 }
5505 }
5506 else
5507 {
5508 /* the backend does not support storing UUIDs within the
5509 * underlying storage so use what we store in XML */
5510
5511 if (fSetImageId)
5512 {
5513 /* set the UUID if an API client wants to change it */
5514 alock.acquire();
5515 mediumId = m->uuidImage;
5516 alock.release();
5517 }
5518 else if (isImport)
5519 {
5520 /* generate an UUID for an imported UUID-less medium */
5521 mediumId.create();
5522 }
5523 }
5524
5525 /* get the medium variant */
5526 unsigned uImageFlags;
5527 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
5528 ComAssertRCThrow(vrc, E_FAIL);
5529 alock.acquire();
5530 m->variant = (MediumVariant_T)uImageFlags;
5531 alock.release();
5532
5533 /* check/get the parent uuid and update corresponding state */
5534 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
5535 {
5536 RTUUID parentId;
5537 vrc = VDGetParentUuid(hdd, 0, &parentId);
5538 ComAssertRCThrow(vrc, E_FAIL);
5539
5540 /* streamOptimized VMDK images are only accepted as base
5541 * images, as this allows automatic repair of OVF appliances.
5542 * Since such images don't support random writes they will not
5543 * be created for diff images. Only an overly smart user might
5544 * manually create this case. Too bad for him. */
5545 if ( isImport
5546 && !(uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED))
5547 {
5548 /* the parent must be known to us. Note that we freely
5549 * call locking methods of mVirtualBox and parent, as all
5550 * relevant locks must be already held. There may be no
5551 * concurrent access to the just opened medium on other
5552 * threads yet (and init() will fail if this method reports
5553 * MediumState_Inaccessible) */
5554
5555 Guid id = parentId;
5556 ComObjPtr<Medium> pParent;
5557 rc = m->pVirtualBox->findHardDiskById(id, false /* aSetError */, &pParent);
5558 if (FAILED(rc))
5559 {
5560 lastAccessError = Utf8StrFmt(
5561 tr("Parent medium with UUID {%RTuuid} of the medium '%s' is not found in the media registry ('%s')"),
5562 &parentId, location.c_str(),
5563 m->pVirtualBox->settingsFilePath().c_str());
5564 throw S_OK;
5565 }
5566
5567 /* we set mParent & children() */
5568 treeLock.acquire();
5569
5570 Assert(m->pParent.isNull());
5571 m->pParent = pParent;
5572 m->pParent->m->llChildren.push_back(this);
5573
5574 treeLock.release();
5575 }
5576 else
5577 {
5578 /* we access mParent */
5579 treeLock.acquire();
5580
5581 /* check that parent UUIDs match. Note that there's no need
5582 * for the parent's AutoCaller (our lifetime is bound to
5583 * it) */
5584
5585 if (m->pParent.isNull())
5586 {
5587 /* Due to a bug in VDCopy() in VirtualBox 3.0.0-3.0.14
5588 * and 3.1.0-3.1.8 there are base images out there
5589 * which have a non-zero parent UUID. No point in
5590 * complaining about them, instead automatically
5591 * repair the problem. Later we can bring back the
5592 * error message, but we should wait until really
5593 * most users have repaired their images, either with
5594 * VBoxFixHdd or this way. */
5595#if 1
5596 fRepairImageZeroParentUuid = true;
5597#else /* 0 */
5598 lastAccessError = Utf8StrFmt(
5599 tr("Medium type of '%s' is differencing but it is not associated with any parent medium in the media registry ('%s')"),
5600 location.c_str(),
5601 m->pVirtualBox->settingsFilePath().c_str());
5602 treeLock.release();
5603 throw S_OK;
5604#endif /* 0 */
5605 }
5606
5607 {
5608 AutoReadLock parentLock(m->pParent COMMA_LOCKVAL_SRC_POS);
5609 if ( !fRepairImageZeroParentUuid
5610 && m->pParent->getState() != MediumState_Inaccessible
5611 && m->pParent->getId() != parentId)
5612 {
5613 /** @todo r=klaus this always refers to VirtualBox.xml as the medium registry, even for new VMs */
5614 lastAccessError = Utf8StrFmt(
5615 tr("Parent UUID {%RTuuid} of the medium '%s' does not match UUID {%RTuuid} of its parent medium stored in the media registry ('%s')"),
5616 &parentId, location.c_str(),
5617 m->pParent->getId().raw(),
5618 m->pVirtualBox->settingsFilePath().c_str());
5619 parentLock.release();
5620 treeLock.release();
5621 throw S_OK;
5622 }
5623 }
5624
5625 /// @todo NEWMEDIA what to do if the parent is not
5626 /// accessible while the diff is? Probably nothing. The
5627 /// real code will detect the mismatch anyway.
5628
5629 treeLock.release();
5630 }
5631 }
5632
5633 mediumSize = VDGetFileSize(hdd, 0);
5634 mediumLogicalSize = VDGetSize(hdd, 0);
5635
5636 success = true;
5637 }
5638 catch (HRESULT aRC)
5639 {
5640 rc = aRC;
5641 }
5642
5643 vrc = VDDestroy(hdd);
5644 if (RT_FAILURE(vrc))
5645 {
5646 lastAccessError = Utf8StrFmt(tr("Could not update and close the medium '%s'%s"),
5647 location.c_str(), vdError(vrc).c_str());
5648 success = false;
5649 throw S_OK;
5650 }
5651 }
5652 catch (HRESULT aRC)
5653 {
5654 rc = aRC;
5655 }
5656
5657 treeLock.acquire();
5658 alock.acquire();
5659
5660 if (isImport || fSetImageId)
5661 unconst(m->id) = mediumId;
5662
5663 if (success)
5664 {
5665 m->size = mediumSize;
5666 m->logicalSize = mediumLogicalSize;
5667 m->strLastAccessError.setNull();
5668 }
5669 else
5670 {
5671 m->strLastAccessError = lastAccessError;
5672 LogWarningFunc(("'%s' is not accessible (error='%s', rc=%Rhrc, vrc=%Rrc)\n",
5673 location.c_str(), m->strLastAccessError.c_str(),
5674 rc, vrc));
5675 }
5676
5677 /* unblock anyone waiting for the queryInfo results */
5678 qlock.release();
5679 m->queryInfoRunning = false;
5680
5681 /* Set the proper state according to the result of the check */
5682 if (success)
5683 m->preLockState = MediumState_Created;
5684 else
5685 m->preLockState = MediumState_Inaccessible;
5686
5687 HRESULT rc2;
5688 if (uOpenFlags & (VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_SHAREABLE))
5689 rc2 = UnlockRead(NULL);
5690 else
5691 rc2 = UnlockWrite(NULL);
5692 if (SUCCEEDED(rc) && FAILED(rc2))
5693 rc = rc2;
5694 if (FAILED(rc)) return rc;
5695
5696 /* If this is a base image which incorrectly has a parent UUID set,
5697 * repair the image now by zeroing the parent UUID. This is only done
5698 * when we have structural information from a config file, on import
5699 * this is not possible. If someone would accidentally call openMedium
5700 * with a diff image before the base is registered this would destroy
5701 * the diff. Not acceptable. */
5702 if (fRepairImageZeroParentUuid)
5703 {
5704 rc = LockWrite(NULL);
5705 if (FAILED(rc)) return rc;
5706
5707 alock.release();
5708
5709 try
5710 {
5711 PVBOXHDD hdd;
5712 vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
5713 ComAssertRCThrow(vrc, E_FAIL);
5714
5715 try
5716 {
5717 vrc = VDOpen(hdd,
5718 format.c_str(),
5719 location.c_str(),
5720 (uOpenFlags & ~VD_OPEN_FLAGS_READONLY) | m->uOpenFlagsDef,
5721 m->vdImageIfaces);
5722 if (RT_FAILURE(vrc))
5723 throw S_OK;
5724
5725 RTUUID zeroParentUuid;
5726 RTUuidClear(&zeroParentUuid);
5727 vrc = VDSetParentUuid(hdd, 0, &zeroParentUuid);
5728 ComAssertRCThrow(vrc, E_FAIL);
5729 }
5730 catch (HRESULT aRC)
5731 {
5732 rc = aRC;
5733 }
5734
5735 VDDestroy(hdd);
5736 }
5737 catch (HRESULT aRC)
5738 {
5739 rc = aRC;
5740 }
5741
5742 rc = UnlockWrite(NULL);
5743 if (SUCCEEDED(rc) && FAILED(rc2))
5744 rc = rc2;
5745 if (FAILED(rc)) return rc;
5746 }
5747
5748 return rc;
5749}
5750
5751/**
5752 * Performs extra checks if the medium can be closed and returns S_OK in
5753 * this case. Otherwise, returns a respective error message. Called by
5754 * Close() under the medium tree lock and the medium lock.
5755 *
5756 * @note Also reused by Medium::Reset().
5757 *
5758 * @note Caller must hold the media tree write lock!
5759 */
5760HRESULT Medium::canClose()
5761{
5762 Assert(m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5763
5764 if (getChildren().size() != 0)
5765 return setError(VBOX_E_OBJECT_IN_USE,
5766 tr("Cannot close medium '%s' because it has %d child media"),
5767 m->strLocationFull.c_str(), getChildren().size());
5768
5769 return S_OK;
5770}
5771
5772/**
5773 * Unregisters this medium with mVirtualBox. Called by close() under the medium tree lock.
5774 *
5775 * @note Caller must have locked the media tree lock for writing!
5776 */
5777HRESULT Medium::unregisterWithVirtualBox()
5778{
5779 /* Note that we need to de-associate ourselves from the parent to let
5780 * unregisterMedium() properly save the registry */
5781
5782 /* we modify mParent and access children */
5783 Assert(m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5784
5785 Medium *pParentBackup = m->pParent;
5786 AssertReturn(getChildren().size() == 0, E_FAIL);
5787 if (m->pParent)
5788 deparent();
5789
5790 HRESULT rc = m->pVirtualBox->unregisterMedium(this);
5791 if (FAILED(rc))
5792 {
5793 if (pParentBackup)
5794 {
5795 // re-associate with the parent as we are still relatives in the registry
5796 m->pParent = pParentBackup;
5797 m->pParent->m->llChildren.push_back(this);
5798 }
5799 }
5800
5801 return rc;
5802}
5803
5804/**
5805 * Sets the extended error info according to the current media state.
5806 *
5807 * @note Must be called from under this object's write or read lock.
5808 */
5809HRESULT Medium::setStateError()
5810{
5811 HRESULT rc = E_FAIL;
5812
5813 switch (m->state)
5814 {
5815 case MediumState_NotCreated:
5816 {
5817 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5818 tr("Storage for the medium '%s' is not created"),
5819 m->strLocationFull.c_str());
5820 break;
5821 }
5822 case MediumState_Created:
5823 {
5824 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5825 tr("Storage for the medium '%s' is already created"),
5826 m->strLocationFull.c_str());
5827 break;
5828 }
5829 case MediumState_LockedRead:
5830 {
5831 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5832 tr("Medium '%s' is locked for reading by another task"),
5833 m->strLocationFull.c_str());
5834 break;
5835 }
5836 case MediumState_LockedWrite:
5837 {
5838 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5839 tr("Medium '%s' is locked for writing by another task"),
5840 m->strLocationFull.c_str());
5841 break;
5842 }
5843 case MediumState_Inaccessible:
5844 {
5845 /* be in sync with Console::powerUpThread() */
5846 if (!m->strLastAccessError.isEmpty())
5847 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5848 tr("Medium '%s' is not accessible. %s"),
5849 m->strLocationFull.c_str(), m->strLastAccessError.c_str());
5850 else
5851 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5852 tr("Medium '%s' is not accessible"),
5853 m->strLocationFull.c_str());
5854 break;
5855 }
5856 case MediumState_Creating:
5857 {
5858 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5859 tr("Storage for the medium '%s' is being created"),
5860 m->strLocationFull.c_str());
5861 break;
5862 }
5863 case MediumState_Deleting:
5864 {
5865 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
5866 tr("Storage for the medium '%s' is being deleted"),
5867 m->strLocationFull.c_str());
5868 break;
5869 }
5870 default:
5871 {
5872 AssertFailed();
5873 break;
5874 }
5875 }
5876
5877 return rc;
5878}
5879
5880/**
5881 * Sets the value of m->strLocationFull. The given location must be a fully
5882 * qualified path; relative paths are not supported here.
5883 *
5884 * As a special exception, if the specified location is a file path that ends with '/'
5885 * then the file name part will be generated by this method automatically in the format
5886 * '{<uuid>}.<ext>' where <uuid> is a fresh UUID that this method will generate
5887 * and assign to this medium, and <ext> is the default extension for this
5888 * medium's storage format. Note that this procedure requires the media state to
5889 * be NotCreated and will return a failure otherwise.
5890 *
5891 * @param aLocation Location of the storage unit. If the location is a FS-path,
5892 * then it can be relative to the VirtualBox home directory.
5893 * @param aFormat Optional fallback format if it is an import and the format
5894 * cannot be determined.
5895 *
5896 * @note Must be called from under this object's write lock.
5897 */
5898HRESULT Medium::setLocation(const Utf8Str &aLocation,
5899 const Utf8Str &aFormat /* = Utf8Str::Empty */)
5900{
5901 AssertReturn(!aLocation.isEmpty(), E_FAIL);
5902
5903 AutoCaller autoCaller(this);
5904 AssertComRCReturnRC(autoCaller.rc());
5905
5906 /* formatObj may be null only when initializing from an existing path and
5907 * no format is known yet */
5908 AssertReturn( (!m->strFormat.isEmpty() && !m->formatObj.isNull())
5909 || ( autoCaller.state() == InInit
5910 && m->state != MediumState_NotCreated
5911 && m->id.isEmpty()
5912 && m->strFormat.isEmpty()
5913 && m->formatObj.isNull()),
5914 E_FAIL);
5915
5916 /* are we dealing with a new medium constructed using the existing
5917 * location? */
5918 bool isImport = m->strFormat.isEmpty();
5919
5920 if ( isImport
5921 || ( (m->formatObj->getCapabilities() & MediumFormatCapabilities_File)
5922 && !m->hostDrive))
5923 {
5924 Guid id;
5925
5926 Utf8Str locationFull(aLocation);
5927
5928 if (m->state == MediumState_NotCreated)
5929 {
5930 /* must be a file (formatObj must be already known) */
5931 Assert(m->formatObj->getCapabilities() & MediumFormatCapabilities_File);
5932
5933 if (RTPathFilename(aLocation.c_str()) == NULL)
5934 {
5935 /* no file name is given (either an empty string or ends with a
5936 * slash), generate a new UUID + file name if the state allows
5937 * this */
5938
5939 ComAssertMsgRet(!m->formatObj->getFileExtensions().empty(),
5940 ("Must be at least one extension if it is MediumFormatCapabilities_File\n"),
5941 E_FAIL);
5942
5943 Utf8Str strExt = m->formatObj->getFileExtensions().front();
5944 ComAssertMsgRet(!strExt.isEmpty(),
5945 ("Default extension must not be empty\n"),
5946 E_FAIL);
5947
5948 id.create();
5949
5950 locationFull = Utf8StrFmt("%s{%RTuuid}.%s",
5951 aLocation.c_str(), id.raw(), strExt.c_str());
5952 }
5953 }
5954
5955 // we must always have full paths now (if it refers to a file)
5956 if ( ( m->formatObj.isNull()
5957 || m->formatObj->getCapabilities() & MediumFormatCapabilities_File)
5958 && !RTPathStartsWithRoot(locationFull.c_str()))
5959 return setError(VBOX_E_FILE_ERROR,
5960 tr("The given path '%s' is not fully qualified"),
5961 locationFull.c_str());
5962
5963 /* detect the backend from the storage unit if importing */
5964 if (isImport)
5965 {
5966 VDTYPE enmType = VDTYPE_INVALID;
5967 char *backendName = NULL;
5968
5969 int vrc = VINF_SUCCESS;
5970
5971 /* is it a file? */
5972 {
5973 RTFILE file;
5974 vrc = RTFileOpen(&file, locationFull.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
5975 if (RT_SUCCESS(vrc))
5976 RTFileClose(file);
5977 }
5978 if (RT_SUCCESS(vrc))
5979 {
5980 vrc = VDGetFormat(NULL /* pVDIfsDisk */, NULL /* pVDIfsImage */,
5981 locationFull.c_str(), &backendName, &enmType);
5982 }
5983 else if ( vrc != VERR_FILE_NOT_FOUND
5984 && vrc != VERR_PATH_NOT_FOUND
5985 && vrc != VERR_ACCESS_DENIED
5986 && locationFull != aLocation)
5987 {
5988 /* assume it's not a file, restore the original location */
5989 locationFull = aLocation;
5990 vrc = VDGetFormat(NULL /* pVDIfsDisk */, NULL /* pVDIfsImage */,
5991 locationFull.c_str(), &backendName, &enmType);
5992 }
5993
5994 if (RT_FAILURE(vrc))
5995 {
5996 if (vrc == VERR_ACCESS_DENIED)
5997 return setError(VBOX_E_FILE_ERROR,
5998 tr("Permission problem accessing the file for the medium '%s' (%Rrc)"),
5999 locationFull.c_str(), vrc);
6000 else if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
6001 return setError(VBOX_E_FILE_ERROR,
6002 tr("Could not find file for the medium '%s' (%Rrc)"),
6003 locationFull.c_str(), vrc);
6004 else if (aFormat.isEmpty())
6005 return setError(VBOX_E_IPRT_ERROR,
6006 tr("Could not get the storage format of the medium '%s' (%Rrc)"),
6007 locationFull.c_str(), vrc);
6008 else
6009 {
6010 HRESULT rc = setFormat(aFormat);
6011 /* setFormat() must not fail since we've just used the backend so
6012 * the format object must be there */
6013 AssertComRCReturnRC(rc);
6014 }
6015 }
6016 else if ( enmType == VDTYPE_INVALID
6017 || m->devType != convertToDeviceType(enmType))
6018 {
6019 /*
6020 * The user tried to use a image as a device which is not supported
6021 * by the backend.
6022 */
6023 return setError(E_FAIL,
6024 tr("The medium '%s' can't be used as the requested device type"),
6025 locationFull.c_str());
6026 }
6027 else
6028 {
6029 ComAssertRet(backendName != NULL && *backendName != '\0', E_FAIL);
6030
6031 HRESULT rc = setFormat(backendName);
6032 RTStrFree(backendName);
6033
6034 /* setFormat() must not fail since we've just used the backend so
6035 * the format object must be there */
6036 AssertComRCReturnRC(rc);
6037 }
6038 }
6039
6040 m->strLocationFull = locationFull;
6041
6042 /* is it still a file? */
6043 if ( (m->formatObj->getCapabilities() & MediumFormatCapabilities_File)
6044 && (m->state == MediumState_NotCreated)
6045 )
6046 /* assign a new UUID (this UUID will be used when calling
6047 * VDCreateBase/VDCreateDiff as a wanted UUID). Note that we
6048 * also do that if we didn't generate it to make sure it is
6049 * either generated by us or reset to null */
6050 unconst(m->id) = id;
6051 }
6052 else
6053 m->strLocationFull = aLocation;
6054
6055 return S_OK;
6056}
6057
6058/**
6059 * Checks that the format ID is valid and sets it on success.
6060 *
6061 * Note that this method will caller-reference the format object on success!
6062 * This reference must be released somewhere to let the MediumFormat object be
6063 * uninitialized.
6064 *
6065 * @note Must be called from under this object's write lock.
6066 */
6067HRESULT Medium::setFormat(const Utf8Str &aFormat)
6068{
6069 /* get the format object first */
6070 {
6071 SystemProperties *pSysProps = m->pVirtualBox->getSystemProperties();
6072 AutoReadLock propsLock(pSysProps COMMA_LOCKVAL_SRC_POS);
6073
6074 unconst(m->formatObj) = pSysProps->mediumFormat(aFormat);
6075 if (m->formatObj.isNull())
6076 return setError(E_INVALIDARG,
6077 tr("Invalid medium storage format '%s'"),
6078 aFormat.c_str());
6079
6080 /* reference the format permanently to prevent its unexpected
6081 * uninitialization */
6082 HRESULT rc = m->formatObj->addCaller();
6083 AssertComRCReturnRC(rc);
6084
6085 /* get properties (preinsert them as keys in the map). Note that the
6086 * map doesn't grow over the object life time since the set of
6087 * properties is meant to be constant. */
6088
6089 Assert(m->mapProperties.empty());
6090
6091 for (MediumFormat::PropertyList::const_iterator it = m->formatObj->getProperties().begin();
6092 it != m->formatObj->getProperties().end();
6093 ++it)
6094 {
6095 m->mapProperties.insert(std::make_pair(it->strName, Utf8Str::Empty));
6096 }
6097 }
6098
6099 unconst(m->strFormat) = aFormat;
6100
6101 return S_OK;
6102}
6103
6104/**
6105 * Converts the Medium device type to the VD type.
6106 */
6107VDTYPE Medium::convertDeviceType()
6108{
6109 VDTYPE enmType;
6110
6111 switch (m->devType)
6112 {
6113 case DeviceType_HardDisk:
6114 enmType = VDTYPE_HDD;
6115 break;
6116 case DeviceType_DVD:
6117 enmType = VDTYPE_DVD;
6118 break;
6119 case DeviceType_Floppy:
6120 enmType = VDTYPE_FLOPPY;
6121 break;
6122 default:
6123 ComAssertFailedRet(VDTYPE_INVALID);
6124 }
6125
6126 return enmType;
6127}
6128
6129/**
6130 * Converts from the VD type to the medium type.
6131 */
6132DeviceType_T Medium::convertToDeviceType(VDTYPE enmType)
6133{
6134 DeviceType_T devType;
6135
6136 switch (enmType)
6137 {
6138 case VDTYPE_HDD:
6139 devType = DeviceType_HardDisk;
6140 break;
6141 case VDTYPE_DVD:
6142 devType = DeviceType_DVD;
6143 break;
6144 case VDTYPE_FLOPPY:
6145 devType = DeviceType_Floppy;
6146 break;
6147 default:
6148 ComAssertFailedRet(DeviceType_Null);
6149 }
6150
6151 return devType;
6152}
6153
6154/**
6155 * Returns the last error message collected by the vdErrorCall callback and
6156 * resets it.
6157 *
6158 * The error message is returned prepended with a dot and a space, like this:
6159 * <code>
6160 * ". <error_text> (%Rrc)"
6161 * </code>
6162 * to make it easily appendable to a more general error message. The @c %Rrc
6163 * format string is given @a aVRC as an argument.
6164 *
6165 * If there is no last error message collected by vdErrorCall or if it is a
6166 * null or empty string, then this function returns the following text:
6167 * <code>
6168 * " (%Rrc)"
6169 * </code>
6170 *
6171 * @note Doesn't do any object locking; it is assumed that the caller makes sure
6172 * the callback isn't called by more than one thread at a time.
6173 *
6174 * @param aVRC VBox error code to use when no error message is provided.
6175 */
6176Utf8Str Medium::vdError(int aVRC)
6177{
6178 Utf8Str error;
6179
6180 if (m->vdError.isEmpty())
6181 error = Utf8StrFmt(" (%Rrc)", aVRC);
6182 else
6183 error = Utf8StrFmt(".\n%s", m->vdError.c_str());
6184
6185 m->vdError.setNull();
6186
6187 return error;
6188}
6189
6190/**
6191 * Error message callback.
6192 *
6193 * Puts the reported error message to the m->vdError field.
6194 *
6195 * @note Doesn't do any object locking; it is assumed that the caller makes sure
6196 * the callback isn't called by more than one thread at a time.
6197 *
6198 * @param pvUser The opaque data passed on container creation.
6199 * @param rc The VBox error code.
6200 * @param RT_SRC_POS_DECL Use RT_SRC_POS.
6201 * @param pszFormat Error message format string.
6202 * @param va Error message arguments.
6203 */
6204/*static*/
6205DECLCALLBACK(void) Medium::vdErrorCall(void *pvUser, int rc, RT_SRC_POS_DECL,
6206 const char *pszFormat, va_list va)
6207{
6208 NOREF(pszFile); NOREF(iLine); NOREF(pszFunction); /* RT_SRC_POS_DECL */
6209
6210 Medium *that = static_cast<Medium*>(pvUser);
6211 AssertReturnVoid(that != NULL);
6212
6213 if (that->m->vdError.isEmpty())
6214 that->m->vdError =
6215 Utf8StrFmt("%s (%Rrc)", Utf8Str(pszFormat, va).c_str(), rc);
6216 else
6217 that->m->vdError =
6218 Utf8StrFmt("%s.\n%s (%Rrc)", that->m->vdError.c_str(),
6219 Utf8Str(pszFormat, va).c_str(), rc);
6220}
6221
6222/* static */
6223DECLCALLBACK(bool) Medium::vdConfigAreKeysValid(void *pvUser,
6224 const char * /* pszzValid */)
6225{
6226 Medium *that = static_cast<Medium*>(pvUser);
6227 AssertReturn(that != NULL, false);
6228
6229 /* we always return true since the only keys we have are those found in
6230 * VDBACKENDINFO */
6231 return true;
6232}
6233
6234/* static */
6235DECLCALLBACK(int) Medium::vdConfigQuerySize(void *pvUser,
6236 const char *pszName,
6237 size_t *pcbValue)
6238{
6239 AssertReturn(VALID_PTR(pcbValue), VERR_INVALID_POINTER);
6240
6241 Medium *that = static_cast<Medium*>(pvUser);
6242 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
6243
6244 settings::StringsMap::const_iterator it = that->m->mapProperties.find(Utf8Str(pszName));
6245 if (it == that->m->mapProperties.end())
6246 return VERR_CFGM_VALUE_NOT_FOUND;
6247
6248 /* we interpret null values as "no value" in Medium */
6249 if (it->second.isEmpty())
6250 return VERR_CFGM_VALUE_NOT_FOUND;
6251
6252 *pcbValue = it->second.length() + 1 /* include terminator */;
6253
6254 return VINF_SUCCESS;
6255}
6256
6257/* static */
6258DECLCALLBACK(int) Medium::vdConfigQuery(void *pvUser,
6259 const char *pszName,
6260 char *pszValue,
6261 size_t cchValue)
6262{
6263 AssertReturn(VALID_PTR(pszValue), VERR_INVALID_POINTER);
6264
6265 Medium *that = static_cast<Medium*>(pvUser);
6266 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
6267
6268 settings::StringsMap::const_iterator it = that->m->mapProperties.find(Utf8Str(pszName));
6269 if (it == that->m->mapProperties.end())
6270 return VERR_CFGM_VALUE_NOT_FOUND;
6271
6272 /* we interpret null values as "no value" in Medium */
6273 if (it->second.isEmpty())
6274 return VERR_CFGM_VALUE_NOT_FOUND;
6275
6276 const Utf8Str &value = it->second;
6277 if (value.length() >= cchValue)
6278 return VERR_CFGM_NOT_ENOUGH_SPACE;
6279
6280 memcpy(pszValue, value.c_str(), value.length() + 1);
6281
6282 return VINF_SUCCESS;
6283}
6284
6285DECLCALLBACK(int) Medium::vdTcpSocketCreate(uint32_t fFlags, PVDSOCKET pSock)
6286{
6287 PVDSOCKETINT pSocketInt = NULL;
6288
6289 if ((fFlags & VD_INTERFACETCPNET_CONNECT_EXTENDED_SELECT) != 0)
6290 return VERR_NOT_SUPPORTED;
6291
6292 pSocketInt = (PVDSOCKETINT)RTMemAllocZ(sizeof(VDSOCKETINT));
6293 if (!pSocketInt)
6294 return VERR_NO_MEMORY;
6295
6296 pSocketInt->hSocket = NIL_RTSOCKET;
6297 *pSock = pSocketInt;
6298 return VINF_SUCCESS;
6299}
6300
6301DECLCALLBACK(int) Medium::vdTcpSocketDestroy(VDSOCKET Sock)
6302{
6303 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6304
6305 if (pSocketInt->hSocket != NIL_RTSOCKET)
6306 RTTcpClientCloseEx(pSocketInt->hSocket, false /*fGracefulShutdown*/);
6307
6308 RTMemFree(pSocketInt);
6309
6310 return VINF_SUCCESS;
6311}
6312
6313DECLCALLBACK(int) Medium::vdTcpClientConnect(VDSOCKET Sock, const char *pszAddress, uint32_t uPort)
6314{
6315 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6316
6317 return RTTcpClientConnect(pszAddress, uPort, &pSocketInt->hSocket);
6318}
6319
6320DECLCALLBACK(int) Medium::vdTcpClientClose(VDSOCKET Sock)
6321{
6322 int rc = VINF_SUCCESS;
6323 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6324
6325 rc = RTTcpClientCloseEx(pSocketInt->hSocket, false /*fGracefulShutdown*/);
6326 pSocketInt->hSocket = NIL_RTSOCKET;
6327 return rc;
6328}
6329
6330DECLCALLBACK(bool) Medium::vdTcpIsClientConnected(VDSOCKET Sock)
6331{
6332 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6333 return pSocketInt->hSocket != NIL_RTSOCKET;
6334}
6335
6336DECLCALLBACK(int) Medium::vdTcpSelectOne(VDSOCKET Sock, RTMSINTERVAL cMillies)
6337{
6338 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6339 return RTTcpSelectOne(pSocketInt->hSocket, cMillies);
6340}
6341
6342DECLCALLBACK(int) Medium::vdTcpRead(VDSOCKET Sock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
6343{
6344 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6345 return RTTcpRead(pSocketInt->hSocket, pvBuffer, cbBuffer, pcbRead);
6346}
6347
6348DECLCALLBACK(int) Medium::vdTcpWrite(VDSOCKET Sock, const void *pvBuffer, size_t cbBuffer)
6349{
6350 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6351 return RTTcpWrite(pSocketInt->hSocket, pvBuffer, cbBuffer);
6352}
6353
6354DECLCALLBACK(int) Medium::vdTcpSgWrite(VDSOCKET Sock, PCRTSGBUF pSgBuf)
6355{
6356 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6357 return RTTcpSgWrite(pSocketInt->hSocket, pSgBuf);
6358}
6359
6360DECLCALLBACK(int) Medium::vdTcpFlush(VDSOCKET Sock)
6361{
6362 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6363 return RTTcpFlush(pSocketInt->hSocket);
6364}
6365
6366DECLCALLBACK(int) Medium::vdTcpSetSendCoalescing(VDSOCKET Sock, bool fEnable)
6367{
6368 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6369 return RTTcpSetSendCoalescing(pSocketInt->hSocket, fEnable);
6370}
6371
6372DECLCALLBACK(int) Medium::vdTcpGetLocalAddress(VDSOCKET Sock, PRTNETADDR pAddr)
6373{
6374 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6375 return RTTcpGetLocalAddress(pSocketInt->hSocket, pAddr);
6376}
6377
6378DECLCALLBACK(int) Medium::vdTcpGetPeerAddress(VDSOCKET Sock, PRTNETADDR pAddr)
6379{
6380 PVDSOCKETINT pSocketInt = (PVDSOCKETINT)Sock;
6381 return RTTcpGetPeerAddress(pSocketInt->hSocket, pAddr);
6382}
6383
6384/**
6385 * Starts a new thread driven by the appropriate Medium::Task::handler() method.
6386 *
6387 * @note When the task is executed by this method, IProgress::notifyComplete()
6388 * is automatically called for the progress object associated with this
6389 * task when the task is finished to signal the operation completion for
6390 * other threads asynchronously waiting for it.
6391 */
6392HRESULT Medium::startThread(Medium::Task *pTask)
6393{
6394#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
6395 /* Extreme paranoia: The calling thread should not hold the medium
6396 * tree lock or any medium lock. Since there is no separate lock class
6397 * for medium objects be even more strict: no other object locks. */
6398 Assert(!AutoLockHoldsLocksInClass(LOCKCLASS_LISTOFMEDIA));
6399 Assert(!AutoLockHoldsLocksInClass(getLockingClass()));
6400#endif
6401
6402 /// @todo use a more descriptive task name
6403 int vrc = RTThreadCreate(NULL, Medium::Task::fntMediumTask, pTask,
6404 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
6405 "Medium::Task");
6406 if (RT_FAILURE(vrc))
6407 {
6408 delete pTask;
6409 return setError(E_FAIL, "Could not create Medium::Task thread (%Rrc)\n", vrc);
6410 }
6411
6412 return S_OK;
6413}
6414
6415/**
6416 * Runs Medium::Task::handler() on the current thread instead of creating
6417 * a new one.
6418 *
6419 * This call implies that it is made on another temporary thread created for
6420 * some asynchronous task. Avoid calling it from a normal thread since the task
6421 * operations are potentially lengthy and will block the calling thread in this
6422 * case.
6423 *
6424 * @note When the task is executed by this method, IProgress::notifyComplete()
6425 * is not called for the progress object associated with this task when
6426 * the task is finished. Instead, the result of the operation is returned
6427 * by this method directly and it's the caller's responsibility to
6428 * complete the progress object in this case.
6429 */
6430HRESULT Medium::runNow(Medium::Task *pTask)
6431{
6432#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
6433 /* Extreme paranoia: The calling thread should not hold the medium
6434 * tree lock or any medium lock. Since there is no separate lock class
6435 * for medium objects be even more strict: no other object locks. */
6436 Assert(!AutoLockHoldsLocksInClass(LOCKCLASS_LISTOFMEDIA));
6437 Assert(!AutoLockHoldsLocksInClass(getLockingClass()));
6438#endif
6439
6440 /* NIL_RTTHREAD indicates synchronous call. */
6441 return (HRESULT)Medium::Task::fntMediumTask(NIL_RTTHREAD, pTask);
6442}
6443
6444/**
6445 * Implementation code for the "create base" task.
6446 *
6447 * This only gets started from Medium::CreateBaseStorage() and always runs
6448 * asynchronously. As a result, we always save the VirtualBox.xml file when
6449 * we're done here.
6450 *
6451 * @param task
6452 * @return
6453 */
6454HRESULT Medium::taskCreateBaseHandler(Medium::CreateBaseTask &task)
6455{
6456 HRESULT rc = S_OK;
6457
6458 /* these parameters we need after creation */
6459 uint64_t size = 0, logicalSize = 0;
6460 MediumVariant_T variant = MediumVariant_Standard;
6461 bool fGenerateUuid = false;
6462
6463 try
6464 {
6465 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
6466
6467 /* The object may request a specific UUID (through a special form of
6468 * the setLocation() argument). Otherwise we have to generate it */
6469 Guid id = m->id;
6470 fGenerateUuid = id.isEmpty();
6471 if (fGenerateUuid)
6472 {
6473 id.create();
6474 /* VirtualBox::registerMedium() will need UUID */
6475 unconst(m->id) = id;
6476 }
6477
6478 Utf8Str format(m->strFormat);
6479 Utf8Str location(m->strLocationFull);
6480 uint64_t capabilities = m->formatObj->getCapabilities();
6481 ComAssertThrow(capabilities & ( MediumFormatCapabilities_CreateFixed
6482 | MediumFormatCapabilities_CreateDynamic), E_FAIL);
6483 Assert(m->state == MediumState_Creating);
6484
6485 PVBOXHDD hdd;
6486 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
6487 ComAssertRCThrow(vrc, E_FAIL);
6488
6489 /* unlock before the potentially lengthy operation */
6490 thisLock.release();
6491
6492 try
6493 {
6494 /* ensure the directory exists */
6495 if (capabilities & MediumFormatCapabilities_File)
6496 {
6497 rc = VirtualBox::ensureFilePathExists(location, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
6498 if (FAILED(rc))
6499 throw rc;
6500 }
6501
6502 VDGEOMETRY geo = { 0, 0, 0 }; /* auto-detect */
6503
6504 vrc = VDCreateBase(hdd,
6505 format.c_str(),
6506 location.c_str(),
6507 task.mSize,
6508 task.mVariant & ~MediumVariant_NoCreateDir,
6509 NULL,
6510 &geo,
6511 &geo,
6512 id.raw(),
6513 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
6514 m->vdImageIfaces,
6515 task.mVDOperationIfaces);
6516 if (RT_FAILURE(vrc))
6517 throw setError(VBOX_E_FILE_ERROR,
6518 tr("Could not create the medium storage unit '%s'%s"),
6519 location.c_str(), vdError(vrc).c_str());
6520
6521 size = VDGetFileSize(hdd, 0);
6522 logicalSize = VDGetSize(hdd, 0);
6523 unsigned uImageFlags;
6524 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
6525 if (RT_SUCCESS(vrc))
6526 variant = (MediumVariant_T)uImageFlags;
6527 }
6528 catch (HRESULT aRC) { rc = aRC; }
6529
6530 VDDestroy(hdd);
6531 }
6532 catch (HRESULT aRC) { rc = aRC; }
6533
6534 if (SUCCEEDED(rc))
6535 {
6536 /* register with mVirtualBox as the last step and move to
6537 * Created state only on success (leaving an orphan file is
6538 * better than breaking media registry consistency) */
6539 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
6540 ComObjPtr<Medium> pMedium;
6541 rc = m->pVirtualBox->registerMedium(this, &pMedium, DeviceType_HardDisk);
6542 Assert(this == pMedium);
6543 }
6544
6545 // re-acquire the lock before changing state
6546 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
6547
6548 if (SUCCEEDED(rc))
6549 {
6550 m->state = MediumState_Created;
6551
6552 m->size = size;
6553 m->logicalSize = logicalSize;
6554 m->variant = variant;
6555
6556 thisLock.release();
6557 markRegistriesModified();
6558 if (task.isAsync())
6559 {
6560 // in asynchronous mode, save settings now
6561 m->pVirtualBox->saveModifiedRegistries();
6562 }
6563 }
6564 else
6565 {
6566 /* back to NotCreated on failure */
6567 m->state = MediumState_NotCreated;
6568
6569 /* reset UUID to prevent it from being reused next time */
6570 if (fGenerateUuid)
6571 unconst(m->id).clear();
6572 }
6573
6574 return rc;
6575}
6576
6577/**
6578 * Implementation code for the "create diff" task.
6579 *
6580 * This task always gets started from Medium::createDiffStorage() and can run
6581 * synchronously or asynchronously depending on the "wait" parameter passed to
6582 * that function. If we run synchronously, the caller expects the medium
6583 * registry modification to be set before returning; otherwise (in asynchronous
6584 * mode), we save the settings ourselves.
6585 *
6586 * @param task
6587 * @return
6588 */
6589HRESULT Medium::taskCreateDiffHandler(Medium::CreateDiffTask &task)
6590{
6591 HRESULT rcTmp = S_OK;
6592
6593 const ComObjPtr<Medium> &pTarget = task.mTarget;
6594
6595 uint64_t size = 0, logicalSize = 0;
6596 MediumVariant_T variant = MediumVariant_Standard;
6597 bool fGenerateUuid = false;
6598
6599 try
6600 {
6601 /* Lock both in {parent,child} order. */
6602 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
6603
6604 /* The object may request a specific UUID (through a special form of
6605 * the setLocation() argument). Otherwise we have to generate it */
6606 Guid targetId = pTarget->m->id;
6607 fGenerateUuid = targetId.isEmpty();
6608 if (fGenerateUuid)
6609 {
6610 targetId.create();
6611 /* VirtualBox::registerMedium() will need UUID */
6612 unconst(pTarget->m->id) = targetId;
6613 }
6614
6615 Guid id = m->id;
6616
6617 Utf8Str targetFormat(pTarget->m->strFormat);
6618 Utf8Str targetLocation(pTarget->m->strLocationFull);
6619 uint64_t capabilities = pTarget->m->formatObj->getCapabilities();
6620 ComAssertThrow(capabilities & MediumFormatCapabilities_CreateDynamic, E_FAIL);
6621
6622 Assert(pTarget->m->state == MediumState_Creating);
6623 Assert(m->state == MediumState_LockedRead);
6624
6625 PVBOXHDD hdd;
6626 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
6627 ComAssertRCThrow(vrc, E_FAIL);
6628
6629 /* the two media are now protected by their non-default states;
6630 * unlock the media before the potentially lengthy operation */
6631 mediaLock.release();
6632
6633 try
6634 {
6635 /* Open all media in the target chain but the last. */
6636 MediumLockList::Base::const_iterator targetListBegin =
6637 task.mpMediumLockList->GetBegin();
6638 MediumLockList::Base::const_iterator targetListEnd =
6639 task.mpMediumLockList->GetEnd();
6640 for (MediumLockList::Base::const_iterator it = targetListBegin;
6641 it != targetListEnd;
6642 ++it)
6643 {
6644 const MediumLock &mediumLock = *it;
6645 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
6646
6647 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
6648
6649 /* Skip over the target diff medium */
6650 if (pMedium->m->state == MediumState_Creating)
6651 continue;
6652
6653 /* sanity check */
6654 Assert(pMedium->m->state == MediumState_LockedRead);
6655
6656 /* Open all media in appropriate mode. */
6657 vrc = VDOpen(hdd,
6658 pMedium->m->strFormat.c_str(),
6659 pMedium->m->strLocationFull.c_str(),
6660 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
6661 pMedium->m->vdImageIfaces);
6662 if (RT_FAILURE(vrc))
6663 throw setError(VBOX_E_FILE_ERROR,
6664 tr("Could not open the medium storage unit '%s'%s"),
6665 pMedium->m->strLocationFull.c_str(),
6666 vdError(vrc).c_str());
6667 }
6668
6669 /* ensure the target directory exists */
6670 if (capabilities & MediumFormatCapabilities_File)
6671 {
6672 HRESULT rc = VirtualBox::ensureFilePathExists(targetLocation, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
6673 if (FAILED(rc))
6674 throw rc;
6675 }
6676
6677 vrc = VDCreateDiff(hdd,
6678 targetFormat.c_str(),
6679 targetLocation.c_str(),
6680 (task.mVariant & ~MediumVariant_NoCreateDir) | VD_IMAGE_FLAGS_DIFF,
6681 NULL,
6682 targetId.raw(),
6683 id.raw(),
6684 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
6685 pTarget->m->vdImageIfaces,
6686 task.mVDOperationIfaces);
6687 if (RT_FAILURE(vrc))
6688 throw setError(VBOX_E_FILE_ERROR,
6689 tr("Could not create the differencing medium storage unit '%s'%s"),
6690 targetLocation.c_str(), vdError(vrc).c_str());
6691
6692 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
6693 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
6694 unsigned uImageFlags;
6695 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
6696 if (RT_SUCCESS(vrc))
6697 variant = (MediumVariant_T)uImageFlags;
6698 }
6699 catch (HRESULT aRC) { rcTmp = aRC; }
6700
6701 VDDestroy(hdd);
6702 }
6703 catch (HRESULT aRC) { rcTmp = aRC; }
6704
6705 MultiResult mrc(rcTmp);
6706
6707 if (SUCCEEDED(mrc))
6708 {
6709 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
6710
6711 Assert(pTarget->m->pParent.isNull());
6712
6713 /* associate the child with the parent */
6714 pTarget->m->pParent = this;
6715 m->llChildren.push_back(pTarget);
6716
6717 /** @todo r=klaus neither target nor base() are locked,
6718 * potential race! */
6719 /* diffs for immutable media are auto-reset by default */
6720 pTarget->m->autoReset = (getBase()->m->type == MediumType_Immutable);
6721
6722 /* register with mVirtualBox as the last step and move to
6723 * Created state only on success (leaving an orphan file is
6724 * better than breaking media registry consistency) */
6725 ComObjPtr<Medium> pMedium;
6726 mrc = m->pVirtualBox->registerMedium(pTarget, &pMedium, DeviceType_HardDisk);
6727 Assert(pTarget == pMedium);
6728
6729 if (FAILED(mrc))
6730 /* break the parent association on failure to register */
6731 deparent();
6732 }
6733
6734 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
6735
6736 if (SUCCEEDED(mrc))
6737 {
6738 pTarget->m->state = MediumState_Created;
6739
6740 pTarget->m->size = size;
6741 pTarget->m->logicalSize = logicalSize;
6742 pTarget->m->variant = variant;
6743 }
6744 else
6745 {
6746 /* back to NotCreated on failure */
6747 pTarget->m->state = MediumState_NotCreated;
6748
6749 pTarget->m->autoReset = false;
6750
6751 /* reset UUID to prevent it from being reused next time */
6752 if (fGenerateUuid)
6753 unconst(pTarget->m->id).clear();
6754 }
6755
6756 // deregister the task registered in createDiffStorage()
6757 Assert(m->numCreateDiffTasks != 0);
6758 --m->numCreateDiffTasks;
6759
6760 mediaLock.release();
6761 markRegistriesModified();
6762 if (task.isAsync())
6763 {
6764 // in asynchronous mode, save settings now
6765 m->pVirtualBox->saveModifiedRegistries();
6766 }
6767
6768 /* Note that in sync mode, it's the caller's responsibility to
6769 * unlock the medium. */
6770
6771 return mrc;
6772}
6773
6774/**
6775 * Implementation code for the "merge" task.
6776 *
6777 * This task always gets started from Medium::mergeTo() and can run
6778 * synchronously or asynchronously depending on the "wait" parameter passed to
6779 * that function. If we run synchronously, the caller expects the medium
6780 * registry modification to be set before returning; otherwise (in asynchronous
6781 * mode), we save the settings ourselves.
6782 *
6783 * @param task
6784 * @return
6785 */
6786HRESULT Medium::taskMergeHandler(Medium::MergeTask &task)
6787{
6788 HRESULT rcTmp = S_OK;
6789
6790 const ComObjPtr<Medium> &pTarget = task.mTarget;
6791
6792 try
6793 {
6794 PVBOXHDD hdd;
6795 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
6796 ComAssertRCThrow(vrc, E_FAIL);
6797
6798 try
6799 {
6800 // Similar code appears in SessionMachine::onlineMergeMedium, so
6801 // if you make any changes below check whether they are applicable
6802 // in that context as well.
6803
6804 unsigned uTargetIdx = VD_LAST_IMAGE;
6805 unsigned uSourceIdx = VD_LAST_IMAGE;
6806 /* Open all media in the chain. */
6807 MediumLockList::Base::iterator lockListBegin =
6808 task.mpMediumLockList->GetBegin();
6809 MediumLockList::Base::iterator lockListEnd =
6810 task.mpMediumLockList->GetEnd();
6811 unsigned i = 0;
6812 for (MediumLockList::Base::iterator it = lockListBegin;
6813 it != lockListEnd;
6814 ++it)
6815 {
6816 MediumLock &mediumLock = *it;
6817 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
6818
6819 if (pMedium == this)
6820 uSourceIdx = i;
6821 else if (pMedium == pTarget)
6822 uTargetIdx = i;
6823
6824 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
6825
6826 /*
6827 * complex sanity (sane complexity)
6828 *
6829 * The current medium must be in the Deleting (medium is merged)
6830 * or LockedRead (parent medium) state if it is not the target.
6831 * If it is the target it must be in the LockedWrite state.
6832 */
6833 Assert( ( pMedium != pTarget
6834 && ( pMedium->m->state == MediumState_Deleting
6835 || pMedium->m->state == MediumState_LockedRead))
6836 || ( pMedium == pTarget
6837 && pMedium->m->state == MediumState_LockedWrite));
6838
6839 /*
6840 * Medium must be the target, in the LockedRead state
6841 * or Deleting state where it is not allowed to be attached
6842 * to a virtual machine.
6843 */
6844 Assert( pMedium == pTarget
6845 || pMedium->m->state == MediumState_LockedRead
6846 || ( pMedium->m->backRefs.size() == 0
6847 && pMedium->m->state == MediumState_Deleting));
6848 /* The source medium must be in Deleting state. */
6849 Assert( pMedium != this
6850 || pMedium->m->state == MediumState_Deleting);
6851
6852 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
6853
6854 if ( pMedium->m->state == MediumState_LockedRead
6855 || pMedium->m->state == MediumState_Deleting)
6856 uOpenFlags = VD_OPEN_FLAGS_READONLY;
6857 if (pMedium->m->type == MediumType_Shareable)
6858 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
6859
6860 /* Open the medium */
6861 vrc = VDOpen(hdd,
6862 pMedium->m->strFormat.c_str(),
6863 pMedium->m->strLocationFull.c_str(),
6864 uOpenFlags | m->uOpenFlagsDef,
6865 pMedium->m->vdImageIfaces);
6866 if (RT_FAILURE(vrc))
6867 throw vrc;
6868
6869 i++;
6870 }
6871
6872 ComAssertThrow( uSourceIdx != VD_LAST_IMAGE
6873 && uTargetIdx != VD_LAST_IMAGE, E_FAIL);
6874
6875 vrc = VDMerge(hdd, uSourceIdx, uTargetIdx,
6876 task.mVDOperationIfaces);
6877 if (RT_FAILURE(vrc))
6878 throw vrc;
6879
6880 /* update parent UUIDs */
6881 if (!task.mfMergeForward)
6882 {
6883 /* we need to update UUIDs of all source's children
6884 * which cannot be part of the container at once so
6885 * add each one in there individually */
6886 if (task.mChildrenToReparent.size() > 0)
6887 {
6888 for (MediaList::const_iterator it = task.mChildrenToReparent.begin();
6889 it != task.mChildrenToReparent.end();
6890 ++it)
6891 {
6892 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
6893 vrc = VDOpen(hdd,
6894 (*it)->m->strFormat.c_str(),
6895 (*it)->m->strLocationFull.c_str(),
6896 VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
6897 (*it)->m->vdImageIfaces);
6898 if (RT_FAILURE(vrc))
6899 throw vrc;
6900
6901 vrc = VDSetParentUuid(hdd, VD_LAST_IMAGE,
6902 pTarget->m->id.raw());
6903 if (RT_FAILURE(vrc))
6904 throw vrc;
6905
6906 vrc = VDClose(hdd, false /* fDelete */);
6907 if (RT_FAILURE(vrc))
6908 throw vrc;
6909
6910 (*it)->UnlockWrite(NULL);
6911 }
6912 }
6913 }
6914 }
6915 catch (HRESULT aRC) { rcTmp = aRC; }
6916 catch (int aVRC)
6917 {
6918 rcTmp = setError(VBOX_E_FILE_ERROR,
6919 tr("Could not merge the medium '%s' to '%s'%s"),
6920 m->strLocationFull.c_str(),
6921 pTarget->m->strLocationFull.c_str(),
6922 vdError(aVRC).c_str());
6923 }
6924
6925 VDDestroy(hdd);
6926 }
6927 catch (HRESULT aRC) { rcTmp = aRC; }
6928
6929 ErrorInfoKeeper eik;
6930 MultiResult mrc(rcTmp);
6931 HRESULT rc2;
6932
6933 if (SUCCEEDED(mrc))
6934 {
6935 /* all media but the target were successfully deleted by
6936 * VDMerge; reparent the last one and uninitialize deleted media. */
6937
6938 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
6939
6940 if (task.mfMergeForward)
6941 {
6942 /* first, unregister the target since it may become a base
6943 * medium which needs re-registration */
6944 rc2 = m->pVirtualBox->unregisterMedium(pTarget);
6945 AssertComRC(rc2);
6946
6947 /* then, reparent it and disconnect the deleted branch at
6948 * both ends (chain->parent() is source's parent) */
6949 pTarget->deparent();
6950 pTarget->m->pParent = task.mParentForTarget;
6951 if (pTarget->m->pParent)
6952 {
6953 pTarget->m->pParent->m->llChildren.push_back(pTarget);
6954 deparent();
6955 }
6956
6957 /* then, register again */
6958 ComObjPtr<Medium> pMedium;
6959 rc2 = m->pVirtualBox->registerMedium(pTarget, &pMedium,
6960 DeviceType_HardDisk);
6961 AssertComRC(rc2);
6962 }
6963 else
6964 {
6965 Assert(pTarget->getChildren().size() == 1);
6966 Medium *targetChild = pTarget->getChildren().front();
6967
6968 /* disconnect the deleted branch at the elder end */
6969 targetChild->deparent();
6970
6971 /* reparent source's children and disconnect the deleted
6972 * branch at the younger end */
6973 if (task.mChildrenToReparent.size() > 0)
6974 {
6975 /* obey {parent,child} lock order */
6976 AutoWriteLock sourceLock(this COMMA_LOCKVAL_SRC_POS);
6977
6978 for (MediaList::const_iterator it = task.mChildrenToReparent.begin();
6979 it != task.mChildrenToReparent.end();
6980 it++)
6981 {
6982 Medium *pMedium = *it;
6983 AutoWriteLock childLock(pMedium COMMA_LOCKVAL_SRC_POS);
6984
6985 pMedium->deparent(); // removes pMedium from source
6986 pMedium->setParent(pTarget);
6987 }
6988 }
6989 }
6990
6991 /* unregister and uninitialize all media removed by the merge */
6992 MediumLockList::Base::iterator lockListBegin =
6993 task.mpMediumLockList->GetBegin();
6994 MediumLockList::Base::iterator lockListEnd =
6995 task.mpMediumLockList->GetEnd();
6996 for (MediumLockList::Base::iterator it = lockListBegin;
6997 it != lockListEnd;
6998 )
6999 {
7000 MediumLock &mediumLock = *it;
7001 /* Create a real copy of the medium pointer, as the medium
7002 * lock deletion below would invalidate the referenced object. */
7003 const ComObjPtr<Medium> pMedium = mediumLock.GetMedium();
7004
7005 /* The target and all media not merged (readonly) are skipped */
7006 if ( pMedium == pTarget
7007 || pMedium->m->state == MediumState_LockedRead)
7008 {
7009 ++it;
7010 continue;
7011 }
7012
7013 rc2 = pMedium->m->pVirtualBox->unregisterMedium(pMedium);
7014 AssertComRC(rc2);
7015
7016 /* now, uninitialize the deleted medium (note that
7017 * due to the Deleting state, uninit() will not touch
7018 * the parent-child relationship so we need to
7019 * uninitialize each disk individually) */
7020
7021 /* note that the operation initiator medium (which is
7022 * normally also the source medium) is a special case
7023 * -- there is one more caller added by Task to it which
7024 * we must release. Also, if we are in sync mode, the
7025 * caller may still hold an AutoCaller instance for it
7026 * and therefore we cannot uninit() it (it's therefore
7027 * the caller's responsibility) */
7028 if (pMedium == this)
7029 {
7030 Assert(getChildren().size() == 0);
7031 Assert(m->backRefs.size() == 0);
7032 task.mMediumCaller.release();
7033 }
7034
7035 /* Delete the medium lock list entry, which also releases the
7036 * caller added by MergeChain before uninit() and updates the
7037 * iterator to point to the right place. */
7038 rc2 = task.mpMediumLockList->RemoveByIterator(it);
7039 AssertComRC(rc2);
7040
7041 if (task.isAsync() || pMedium != this)
7042 pMedium->uninit();
7043 }
7044 }
7045
7046 markRegistriesModified();
7047 if (task.isAsync())
7048 {
7049 // in asynchronous mode, save settings now
7050 eik.restore();
7051 m->pVirtualBox->saveModifiedRegistries();
7052 eik.fetch();
7053 }
7054
7055 if (FAILED(mrc))
7056 {
7057 /* Here we come if either VDMerge() failed (in which case we
7058 * assume that it tried to do everything to make a further
7059 * retry possible -- e.g. not deleted intermediate media
7060 * and so on) or VirtualBox::saveRegistries() failed (where we
7061 * should have the original tree but with intermediate storage
7062 * units deleted by VDMerge()). We have to only restore states
7063 * (through the MergeChain dtor) unless we are run synchronously
7064 * in which case it's the responsibility of the caller as stated
7065 * in the mergeTo() docs. The latter also implies that we
7066 * don't own the merge chain, so release it in this case. */
7067 if (task.isAsync())
7068 {
7069 Assert(task.mChildrenToReparent.size() == 0);
7070 cancelMergeTo(task.mChildrenToReparent, task.mpMediumLockList);
7071 }
7072 }
7073
7074 return mrc;
7075}
7076
7077/**
7078 * Implementation code for the "clone" task.
7079 *
7080 * This only gets started from Medium::CloneTo() and always runs asynchronously.
7081 * As a result, we always save the VirtualBox.xml file when we're done here.
7082 *
7083 * @param task
7084 * @return
7085 */
7086HRESULT Medium::taskCloneHandler(Medium::CloneTask &task)
7087{
7088 HRESULT rcTmp = S_OK;
7089
7090 const ComObjPtr<Medium> &pTarget = task.mTarget;
7091 const ComObjPtr<Medium> &pParent = task.mParent;
7092
7093 bool fCreatingTarget = false;
7094
7095 uint64_t size = 0, logicalSize = 0;
7096 MediumVariant_T variant = MediumVariant_Standard;
7097 bool fGenerateUuid = false;
7098
7099 try
7100 {
7101 /* Lock all in {parent,child} order. The lock is also used as a
7102 * signal from the task initiator (which releases it only after
7103 * RTThreadCreate()) that we can start the job. */
7104 AutoMultiWriteLock3 thisLock(this, pTarget, pParent COMMA_LOCKVAL_SRC_POS);
7105
7106 fCreatingTarget = pTarget->m->state == MediumState_Creating;
7107
7108 /* The object may request a specific UUID (through a special form of
7109 * the setLocation() argument). Otherwise we have to generate it */
7110 Guid targetId = pTarget->m->id;
7111 fGenerateUuid = targetId.isEmpty();
7112 if (fGenerateUuid)
7113 {
7114 targetId.create();
7115 /* VirtualBox::registerMedium() will need UUID */
7116 unconst(pTarget->m->id) = targetId;
7117 }
7118
7119 PVBOXHDD hdd;
7120 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7121 ComAssertRCThrow(vrc, E_FAIL);
7122
7123 try
7124 {
7125 /* Open all media in the source chain. */
7126 MediumLockList::Base::const_iterator sourceListBegin =
7127 task.mpSourceMediumLockList->GetBegin();
7128 MediumLockList::Base::const_iterator sourceListEnd =
7129 task.mpSourceMediumLockList->GetEnd();
7130 for (MediumLockList::Base::const_iterator it = sourceListBegin;
7131 it != sourceListEnd;
7132 ++it)
7133 {
7134 const MediumLock &mediumLock = *it;
7135 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7136 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7137
7138 /* sanity check */
7139 Assert(pMedium->m->state == MediumState_LockedRead);
7140
7141 /** Open all media in read-only mode. */
7142 vrc = VDOpen(hdd,
7143 pMedium->m->strFormat.c_str(),
7144 pMedium->m->strLocationFull.c_str(),
7145 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
7146 pMedium->m->vdImageIfaces);
7147 if (RT_FAILURE(vrc))
7148 throw setError(VBOX_E_FILE_ERROR,
7149 tr("Could not open the medium storage unit '%s'%s"),
7150 pMedium->m->strLocationFull.c_str(),
7151 vdError(vrc).c_str());
7152 }
7153
7154 Utf8Str targetFormat(pTarget->m->strFormat);
7155 Utf8Str targetLocation(pTarget->m->strLocationFull);
7156 uint64_t capabilities = pTarget->m->formatObj->getCapabilities();
7157
7158 Assert( pTarget->m->state == MediumState_Creating
7159 || pTarget->m->state == MediumState_LockedWrite);
7160 Assert(m->state == MediumState_LockedRead);
7161 Assert( pParent.isNull()
7162 || pParent->m->state == MediumState_LockedRead);
7163
7164 /* unlock before the potentially lengthy operation */
7165 thisLock.release();
7166
7167 /* ensure the target directory exists */
7168 if (capabilities & MediumFormatCapabilities_File)
7169 {
7170 HRESULT rc = VirtualBox::ensureFilePathExists(targetLocation, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
7171 if (FAILED(rc))
7172 throw rc;
7173 }
7174
7175 PVBOXHDD targetHdd;
7176 vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &targetHdd);
7177 ComAssertRCThrow(vrc, E_FAIL);
7178
7179 try
7180 {
7181 /* Open all media in the target chain. */
7182 MediumLockList::Base::const_iterator targetListBegin =
7183 task.mpTargetMediumLockList->GetBegin();
7184 MediumLockList::Base::const_iterator targetListEnd =
7185 task.mpTargetMediumLockList->GetEnd();
7186 for (MediumLockList::Base::const_iterator it = targetListBegin;
7187 it != targetListEnd;
7188 ++it)
7189 {
7190 const MediumLock &mediumLock = *it;
7191 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7192
7193 /* If the target medium is not created yet there's no
7194 * reason to open it. */
7195 if (pMedium == pTarget && fCreatingTarget)
7196 continue;
7197
7198 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7199
7200 /* sanity check */
7201 Assert( pMedium->m->state == MediumState_LockedRead
7202 || pMedium->m->state == MediumState_LockedWrite);
7203
7204 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
7205 if (pMedium->m->state != MediumState_LockedWrite)
7206 uOpenFlags = VD_OPEN_FLAGS_READONLY;
7207 if (pMedium->m->type == MediumType_Shareable)
7208 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
7209
7210 /* Open all media in appropriate mode. */
7211 vrc = VDOpen(targetHdd,
7212 pMedium->m->strFormat.c_str(),
7213 pMedium->m->strLocationFull.c_str(),
7214 uOpenFlags | m->uOpenFlagsDef,
7215 pMedium->m->vdImageIfaces);
7216 if (RT_FAILURE(vrc))
7217 throw setError(VBOX_E_FILE_ERROR,
7218 tr("Could not open the medium storage unit '%s'%s"),
7219 pMedium->m->strLocationFull.c_str(),
7220 vdError(vrc).c_str());
7221 }
7222
7223 /** @todo r=klaus target isn't locked, race getting the state */
7224 if (task.midxSrcImageSame == UINT32_MAX)
7225 {
7226 vrc = VDCopy(hdd,
7227 VD_LAST_IMAGE,
7228 targetHdd,
7229 targetFormat.c_str(),
7230 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
7231 false /* fMoveByRename */,
7232 0 /* cbSize */,
7233 task.mVariant & ~MediumVariant_NoCreateDir,
7234 targetId.raw(),
7235 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
7236 NULL /* pVDIfsOperation */,
7237 pTarget->m->vdImageIfaces,
7238 task.mVDOperationIfaces);
7239 }
7240 else
7241 {
7242 vrc = VDCopyEx(hdd,
7243 VD_LAST_IMAGE,
7244 targetHdd,
7245 targetFormat.c_str(),
7246 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
7247 false /* fMoveByRename */,
7248 0 /* cbSize */,
7249 task.midxSrcImageSame,
7250 task.midxDstImageSame,
7251 task.mVariant & ~MediumVariant_NoCreateDir,
7252 targetId.raw(),
7253 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
7254 NULL /* pVDIfsOperation */,
7255 pTarget->m->vdImageIfaces,
7256 task.mVDOperationIfaces);
7257 }
7258 if (RT_FAILURE(vrc))
7259 throw setError(VBOX_E_FILE_ERROR,
7260 tr("Could not create the clone medium '%s'%s"),
7261 targetLocation.c_str(), vdError(vrc).c_str());
7262
7263 size = VDGetFileSize(targetHdd, VD_LAST_IMAGE);
7264 logicalSize = VDGetSize(targetHdd, VD_LAST_IMAGE);
7265 unsigned uImageFlags;
7266 vrc = VDGetImageFlags(targetHdd, 0, &uImageFlags);
7267 if (RT_SUCCESS(vrc))
7268 variant = (MediumVariant_T)uImageFlags;
7269 }
7270 catch (HRESULT aRC) { rcTmp = aRC; }
7271
7272 VDDestroy(targetHdd);
7273 }
7274 catch (HRESULT aRC) { rcTmp = aRC; }
7275
7276 VDDestroy(hdd);
7277 }
7278 catch (HRESULT aRC) { rcTmp = aRC; }
7279
7280 ErrorInfoKeeper eik;
7281 MultiResult mrc(rcTmp);
7282
7283 /* Only do the parent changes for newly created media. */
7284 if (SUCCEEDED(mrc) && fCreatingTarget)
7285 {
7286 /* we set mParent & children() */
7287 AutoWriteLock alock2(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
7288
7289 Assert(pTarget->m->pParent.isNull());
7290
7291 if (pParent)
7292 {
7293 /* associate the clone with the parent and deassociate
7294 * from VirtualBox */
7295 pTarget->m->pParent = pParent;
7296 pParent->m->llChildren.push_back(pTarget);
7297
7298 /* register with mVirtualBox as the last step and move to
7299 * Created state only on success (leaving an orphan file is
7300 * better than breaking media registry consistency) */
7301 eik.restore();
7302 ComObjPtr<Medium> pMedium;
7303 mrc = pParent->m->pVirtualBox->registerMedium(pTarget, &pMedium,
7304 DeviceType_HardDisk);
7305 Assert(pTarget == pMedium);
7306 eik.fetch();
7307
7308 if (FAILED(mrc))
7309 /* break parent association on failure to register */
7310 pTarget->deparent(); // removes target from parent
7311 }
7312 else
7313 {
7314 /* just register */
7315 eik.restore();
7316 ComObjPtr<Medium> pMedium;
7317 mrc = m->pVirtualBox->registerMedium(pTarget, &pMedium,
7318 DeviceType_HardDisk);
7319 Assert(pTarget == pMedium);
7320 eik.fetch();
7321 }
7322 }
7323
7324 if (fCreatingTarget)
7325 {
7326 AutoWriteLock mLock(pTarget COMMA_LOCKVAL_SRC_POS);
7327
7328 if (SUCCEEDED(mrc))
7329 {
7330 pTarget->m->state = MediumState_Created;
7331
7332 pTarget->m->size = size;
7333 pTarget->m->logicalSize = logicalSize;
7334 pTarget->m->variant = variant;
7335 }
7336 else
7337 {
7338 /* back to NotCreated on failure */
7339 pTarget->m->state = MediumState_NotCreated;
7340
7341 /* reset UUID to prevent it from being reused next time */
7342 if (fGenerateUuid)
7343 unconst(pTarget->m->id).clear();
7344 }
7345 }
7346
7347 // now, at the end of this task (always asynchronous), save the settings
7348 if (SUCCEEDED(mrc))
7349 {
7350 // save the settings
7351 markRegistriesModified();
7352 /* collect multiple errors */
7353 eik.restore();
7354 m->pVirtualBox->saveModifiedRegistries();
7355 eik.fetch();
7356 }
7357
7358 /* Everything is explicitly unlocked when the task exits,
7359 * as the task destruction also destroys the source chain. */
7360
7361 /* Make sure the source chain is released early. It could happen
7362 * that we get a deadlock in Appliance::Import when Medium::Close
7363 * is called & the source chain is released at the same time. */
7364 task.mpSourceMediumLockList->Clear();
7365
7366 return mrc;
7367}
7368
7369/**
7370 * Implementation code for the "delete" task.
7371 *
7372 * This task always gets started from Medium::deleteStorage() and can run
7373 * synchronously or asynchronously depending on the "wait" parameter passed to
7374 * that function.
7375 *
7376 * @param task
7377 * @return
7378 */
7379HRESULT Medium::taskDeleteHandler(Medium::DeleteTask &task)
7380{
7381 NOREF(task);
7382 HRESULT rc = S_OK;
7383
7384 try
7385 {
7386 /* The lock is also used as a signal from the task initiator (which
7387 * releases it only after RTThreadCreate()) that we can start the job */
7388 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7389
7390 PVBOXHDD hdd;
7391 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7392 ComAssertRCThrow(vrc, E_FAIL);
7393
7394 Utf8Str format(m->strFormat);
7395 Utf8Str location(m->strLocationFull);
7396
7397 /* unlock before the potentially lengthy operation */
7398 Assert(m->state == MediumState_Deleting);
7399 thisLock.release();
7400
7401 try
7402 {
7403 vrc = VDOpen(hdd,
7404 format.c_str(),
7405 location.c_str(),
7406 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
7407 m->vdImageIfaces);
7408 if (RT_SUCCESS(vrc))
7409 vrc = VDClose(hdd, true /* fDelete */);
7410
7411 if (RT_FAILURE(vrc))
7412 throw setError(VBOX_E_FILE_ERROR,
7413 tr("Could not delete the medium storage unit '%s'%s"),
7414 location.c_str(), vdError(vrc).c_str());
7415
7416 }
7417 catch (HRESULT aRC) { rc = aRC; }
7418
7419 VDDestroy(hdd);
7420 }
7421 catch (HRESULT aRC) { rc = aRC; }
7422
7423 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7424
7425 /* go to the NotCreated state even on failure since the storage
7426 * may have been already partially deleted and cannot be used any
7427 * more. One will be able to manually re-open the storage if really
7428 * needed to re-register it. */
7429 m->state = MediumState_NotCreated;
7430
7431 /* Reset UUID to prevent Create* from reusing it again */
7432 unconst(m->id).clear();
7433
7434 return rc;
7435}
7436
7437/**
7438 * Implementation code for the "reset" task.
7439 *
7440 * This always gets started asynchronously from Medium::Reset().
7441 *
7442 * @param task
7443 * @return
7444 */
7445HRESULT Medium::taskResetHandler(Medium::ResetTask &task)
7446{
7447 HRESULT rc = S_OK;
7448
7449 uint64_t size = 0, logicalSize = 0;
7450 MediumVariant_T variant = MediumVariant_Standard;
7451
7452 try
7453 {
7454 /* The lock is also used as a signal from the task initiator (which
7455 * releases it only after RTThreadCreate()) that we can start the job */
7456 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7457
7458 /// @todo Below we use a pair of delete/create operations to reset
7459 /// the diff contents but the most efficient way will of course be
7460 /// to add a VDResetDiff() API call
7461
7462 PVBOXHDD hdd;
7463 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7464 ComAssertRCThrow(vrc, E_FAIL);
7465
7466 Guid id = m->id;
7467 Utf8Str format(m->strFormat);
7468 Utf8Str location(m->strLocationFull);
7469
7470 Medium *pParent = m->pParent;
7471 Guid parentId = pParent->m->id;
7472 Utf8Str parentFormat(pParent->m->strFormat);
7473 Utf8Str parentLocation(pParent->m->strLocationFull);
7474
7475 Assert(m->state == MediumState_LockedWrite);
7476
7477 /* unlock before the potentially lengthy operation */
7478 thisLock.release();
7479
7480 try
7481 {
7482 /* Open all media in the target chain but the last. */
7483 MediumLockList::Base::const_iterator targetListBegin =
7484 task.mpMediumLockList->GetBegin();
7485 MediumLockList::Base::const_iterator targetListEnd =
7486 task.mpMediumLockList->GetEnd();
7487 for (MediumLockList::Base::const_iterator it = targetListBegin;
7488 it != targetListEnd;
7489 ++it)
7490 {
7491 const MediumLock &mediumLock = *it;
7492 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7493
7494 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7495
7496 /* sanity check, "this" is checked above */
7497 Assert( pMedium == this
7498 || pMedium->m->state == MediumState_LockedRead);
7499
7500 /* Open all media in appropriate mode. */
7501 vrc = VDOpen(hdd,
7502 pMedium->m->strFormat.c_str(),
7503 pMedium->m->strLocationFull.c_str(),
7504 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
7505 pMedium->m->vdImageIfaces);
7506 if (RT_FAILURE(vrc))
7507 throw setError(VBOX_E_FILE_ERROR,
7508 tr("Could not open the medium storage unit '%s'%s"),
7509 pMedium->m->strLocationFull.c_str(),
7510 vdError(vrc).c_str());
7511
7512 /* Done when we hit the media which should be reset */
7513 if (pMedium == this)
7514 break;
7515 }
7516
7517 /* first, delete the storage unit */
7518 vrc = VDClose(hdd, true /* fDelete */);
7519 if (RT_FAILURE(vrc))
7520 throw setError(VBOX_E_FILE_ERROR,
7521 tr("Could not delete the medium storage unit '%s'%s"),
7522 location.c_str(), vdError(vrc).c_str());
7523
7524 /* next, create it again */
7525 vrc = VDOpen(hdd,
7526 parentFormat.c_str(),
7527 parentLocation.c_str(),
7528 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
7529 m->vdImageIfaces);
7530 if (RT_FAILURE(vrc))
7531 throw setError(VBOX_E_FILE_ERROR,
7532 tr("Could not open the medium storage unit '%s'%s"),
7533 parentLocation.c_str(), vdError(vrc).c_str());
7534
7535 vrc = VDCreateDiff(hdd,
7536 format.c_str(),
7537 location.c_str(),
7538 /// @todo use the same medium variant as before
7539 VD_IMAGE_FLAGS_NONE,
7540 NULL,
7541 id.raw(),
7542 parentId.raw(),
7543 VD_OPEN_FLAGS_NORMAL,
7544 m->vdImageIfaces,
7545 task.mVDOperationIfaces);
7546 if (RT_FAILURE(vrc))
7547 throw setError(VBOX_E_FILE_ERROR,
7548 tr("Could not create the differencing medium storage unit '%s'%s"),
7549 location.c_str(), vdError(vrc).c_str());
7550
7551 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
7552 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
7553 unsigned uImageFlags;
7554 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
7555 if (RT_SUCCESS(vrc))
7556 variant = (MediumVariant_T)uImageFlags;
7557 }
7558 catch (HRESULT aRC) { rc = aRC; }
7559
7560 VDDestroy(hdd);
7561 }
7562 catch (HRESULT aRC) { rc = aRC; }
7563
7564 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7565
7566 m->size = size;
7567 m->logicalSize = logicalSize;
7568 m->variant = variant;
7569
7570 if (task.isAsync())
7571 {
7572 /* unlock ourselves when done */
7573 HRESULT rc2 = UnlockWrite(NULL);
7574 AssertComRC(rc2);
7575 }
7576
7577 /* Note that in sync mode, it's the caller's responsibility to
7578 * unlock the medium. */
7579
7580 return rc;
7581}
7582
7583/**
7584 * Implementation code for the "compact" task.
7585 *
7586 * @param task
7587 * @return
7588 */
7589HRESULT Medium::taskCompactHandler(Medium::CompactTask &task)
7590{
7591 HRESULT rc = S_OK;
7592
7593 /* Lock all in {parent,child} order. The lock is also used as a
7594 * signal from the task initiator (which releases it only after
7595 * RTThreadCreate()) that we can start the job. */
7596 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7597
7598 try
7599 {
7600 PVBOXHDD hdd;
7601 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7602 ComAssertRCThrow(vrc, E_FAIL);
7603
7604 try
7605 {
7606 /* Open all media in the chain. */
7607 MediumLockList::Base::const_iterator mediumListBegin =
7608 task.mpMediumLockList->GetBegin();
7609 MediumLockList::Base::const_iterator mediumListEnd =
7610 task.mpMediumLockList->GetEnd();
7611 MediumLockList::Base::const_iterator mediumListLast =
7612 mediumListEnd;
7613 mediumListLast--;
7614 for (MediumLockList::Base::const_iterator it = mediumListBegin;
7615 it != mediumListEnd;
7616 ++it)
7617 {
7618 const MediumLock &mediumLock = *it;
7619 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7620 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7621
7622 /* sanity check */
7623 if (it == mediumListLast)
7624 Assert(pMedium->m->state == MediumState_LockedWrite);
7625 else
7626 Assert(pMedium->m->state == MediumState_LockedRead);
7627
7628 /* Open all media but last in read-only mode. Do not handle
7629 * shareable media, as compaction and sharing are mutually
7630 * exclusive. */
7631 vrc = VDOpen(hdd,
7632 pMedium->m->strFormat.c_str(),
7633 pMedium->m->strLocationFull.c_str(),
7634 m->uOpenFlagsDef | (it == mediumListLast) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
7635 pMedium->m->vdImageIfaces);
7636 if (RT_FAILURE(vrc))
7637 throw setError(VBOX_E_FILE_ERROR,
7638 tr("Could not open the medium storage unit '%s'%s"),
7639 pMedium->m->strLocationFull.c_str(),
7640 vdError(vrc).c_str());
7641 }
7642
7643 Assert(m->state == MediumState_LockedWrite);
7644
7645 Utf8Str location(m->strLocationFull);
7646
7647 /* unlock before the potentially lengthy operation */
7648 thisLock.release();
7649
7650 vrc = VDCompact(hdd, VD_LAST_IMAGE, task.mVDOperationIfaces);
7651 if (RT_FAILURE(vrc))
7652 {
7653 if (vrc == VERR_NOT_SUPPORTED)
7654 throw setError(VBOX_E_NOT_SUPPORTED,
7655 tr("Compacting is not yet supported for medium '%s'"),
7656 location.c_str());
7657 else if (vrc == VERR_NOT_IMPLEMENTED)
7658 throw setError(E_NOTIMPL,
7659 tr("Compacting is not implemented, medium '%s'"),
7660 location.c_str());
7661 else
7662 throw setError(VBOX_E_FILE_ERROR,
7663 tr("Could not compact medium '%s'%s"),
7664 location.c_str(),
7665 vdError(vrc).c_str());
7666 }
7667 }
7668 catch (HRESULT aRC) { rc = aRC; }
7669
7670 VDDestroy(hdd);
7671 }
7672 catch (HRESULT aRC) { rc = aRC; }
7673
7674 /* Everything is explicitly unlocked when the task exits,
7675 * as the task destruction also destroys the media chain. */
7676
7677 return rc;
7678}
7679
7680/**
7681 * Implementation code for the "resize" task.
7682 *
7683 * @param task
7684 * @return
7685 */
7686HRESULT Medium::taskResizeHandler(Medium::ResizeTask &task)
7687{
7688 HRESULT rc = S_OK;
7689
7690 /* Lock all in {parent,child} order. The lock is also used as a
7691 * signal from the task initiator (which releases it only after
7692 * RTThreadCreate()) that we can start the job. */
7693 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7694
7695 try
7696 {
7697 PVBOXHDD hdd;
7698 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7699 ComAssertRCThrow(vrc, E_FAIL);
7700
7701 try
7702 {
7703 /* Open all media in the chain. */
7704 MediumLockList::Base::const_iterator mediumListBegin =
7705 task.mpMediumLockList->GetBegin();
7706 MediumLockList::Base::const_iterator mediumListEnd =
7707 task.mpMediumLockList->GetEnd();
7708 MediumLockList::Base::const_iterator mediumListLast =
7709 mediumListEnd;
7710 mediumListLast--;
7711 for (MediumLockList::Base::const_iterator it = mediumListBegin;
7712 it != mediumListEnd;
7713 ++it)
7714 {
7715 const MediumLock &mediumLock = *it;
7716 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7717 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7718
7719 /* sanity check */
7720 if (it == mediumListLast)
7721 Assert(pMedium->m->state == MediumState_LockedWrite);
7722 else
7723 Assert(pMedium->m->state == MediumState_LockedRead);
7724
7725 /* Open all media but last in read-only mode. Do not handle
7726 * shareable media, as compaction and sharing are mutually
7727 * exclusive. */
7728 vrc = VDOpen(hdd,
7729 pMedium->m->strFormat.c_str(),
7730 pMedium->m->strLocationFull.c_str(),
7731 m->uOpenFlagsDef | (it == mediumListLast) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
7732 pMedium->m->vdImageIfaces);
7733 if (RT_FAILURE(vrc))
7734 throw setError(VBOX_E_FILE_ERROR,
7735 tr("Could not open the medium storage unit '%s'%s"),
7736 pMedium->m->strLocationFull.c_str(),
7737 vdError(vrc).c_str());
7738 }
7739
7740 Assert(m->state == MediumState_LockedWrite);
7741
7742 Utf8Str location(m->strLocationFull);
7743
7744 /* unlock before the potentially lengthy operation */
7745 thisLock.release();
7746
7747 VDGEOMETRY geo = {0, 0, 0}; /* auto */
7748 vrc = VDResize(hdd, task.mSize, &geo, &geo, task.mVDOperationIfaces);
7749 if (RT_FAILURE(vrc))
7750 {
7751 if (vrc == VERR_NOT_SUPPORTED)
7752 throw setError(VBOX_E_NOT_SUPPORTED,
7753 tr("Compacting is not yet supported for medium '%s'"),
7754 location.c_str());
7755 else if (vrc == VERR_NOT_IMPLEMENTED)
7756 throw setError(E_NOTIMPL,
7757 tr("Compacting is not implemented, medium '%s'"),
7758 location.c_str());
7759 else
7760 throw setError(VBOX_E_FILE_ERROR,
7761 tr("Could not compact medium '%s'%s"),
7762 location.c_str(),
7763 vdError(vrc).c_str());
7764 }
7765 }
7766 catch (HRESULT aRC) { rc = aRC; }
7767
7768 VDDestroy(hdd);
7769 }
7770 catch (HRESULT aRC) { rc = aRC; }
7771
7772 /* Everything is explicitly unlocked when the task exits,
7773 * as the task destruction also destroys the media chain. */
7774
7775 return rc;
7776}
7777
7778/**
7779 * Implementation code for the "export" task.
7780 *
7781 * This only gets started from Medium::exportFile() and always runs
7782 * asynchronously. It doesn't touch anything configuration related, so
7783 * we never save the VirtualBox.xml file here.
7784 *
7785 * @param task
7786 * @return
7787 */
7788HRESULT Medium::taskExportHandler(Medium::ExportTask &task)
7789{
7790 HRESULT rc = S_OK;
7791
7792 try
7793 {
7794 /* Lock all in {parent,child} order. The lock is also used as a
7795 * signal from the task initiator (which releases it only after
7796 * RTThreadCreate()) that we can start the job. */
7797 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
7798
7799 PVBOXHDD hdd;
7800 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7801 ComAssertRCThrow(vrc, E_FAIL);
7802
7803 try
7804 {
7805 /* Open all media in the source chain. */
7806 MediumLockList::Base::const_iterator sourceListBegin =
7807 task.mpSourceMediumLockList->GetBegin();
7808 MediumLockList::Base::const_iterator sourceListEnd =
7809 task.mpSourceMediumLockList->GetEnd();
7810 for (MediumLockList::Base::const_iterator it = sourceListBegin;
7811 it != sourceListEnd;
7812 ++it)
7813 {
7814 const MediumLock &mediumLock = *it;
7815 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7816 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
7817
7818 /* sanity check */
7819 Assert(pMedium->m->state == MediumState_LockedRead);
7820
7821 /* Open all media in read-only mode. */
7822 vrc = VDOpen(hdd,
7823 pMedium->m->strFormat.c_str(),
7824 pMedium->m->strLocationFull.c_str(),
7825 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
7826 pMedium->m->vdImageIfaces);
7827 if (RT_FAILURE(vrc))
7828 throw setError(VBOX_E_FILE_ERROR,
7829 tr("Could not open the medium storage unit '%s'%s"),
7830 pMedium->m->strLocationFull.c_str(),
7831 vdError(vrc).c_str());
7832 }
7833
7834 Utf8Str targetFormat(task.mFormat->getId());
7835 Utf8Str targetLocation(task.mFilename);
7836 uint64_t capabilities = task.mFormat->getCapabilities();
7837
7838 Assert(m->state == MediumState_LockedRead);
7839
7840 /* unlock before the potentially lengthy operation */
7841 thisLock.release();
7842
7843 /* ensure the target directory exists */
7844 if (capabilities & MediumFormatCapabilities_File)
7845 {
7846 rc = VirtualBox::ensureFilePathExists(targetLocation, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
7847 if (FAILED(rc))
7848 throw rc;
7849 }
7850
7851 PVBOXHDD targetHdd;
7852 vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &targetHdd);
7853 ComAssertRCThrow(vrc, E_FAIL);
7854
7855 try
7856 {
7857 vrc = VDCopy(hdd,
7858 VD_LAST_IMAGE,
7859 targetHdd,
7860 targetFormat.c_str(),
7861 targetLocation.c_str(),
7862 false /* fMoveByRename */,
7863 0 /* cbSize */,
7864 task.mVariant & ~MediumVariant_NoCreateDir,
7865 NULL /* pDstUuid */,
7866 VD_OPEN_FLAGS_NORMAL | VD_OPEN_FLAGS_SEQUENTIAL,
7867 NULL /* pVDIfsOperation */,
7868 task.mVDImageIfaces,
7869 task.mVDOperationIfaces);
7870 if (RT_FAILURE(vrc))
7871 throw setError(VBOX_E_FILE_ERROR,
7872 tr("Could not create the clone medium '%s'%s"),
7873 targetLocation.c_str(), vdError(vrc).c_str());
7874 }
7875 catch (HRESULT aRC) { rc = aRC; }
7876
7877 VDDestroy(targetHdd);
7878 }
7879 catch (HRESULT aRC) { rc = aRC; }
7880
7881 VDDestroy(hdd);
7882 }
7883 catch (HRESULT aRC) { rc = aRC; }
7884
7885 /* Everything is explicitly unlocked when the task exits,
7886 * as the task destruction also destroys the source chain. */
7887
7888 /* Make sure the source chain is released early, otherwise it can
7889 * lead to deadlocks with concurrent IAppliance activities. */
7890 task.mpSourceMediumLockList->Clear();
7891
7892 return rc;
7893}
7894
7895/**
7896 * Implementation code for the "import" task.
7897 *
7898 * This only gets started from Medium::importFile() and always runs
7899 * asynchronously. It potentially touches the media registry, so we
7900 * always save the VirtualBox.xml file when we're done here.
7901 *
7902 * @param task
7903 * @return
7904 */
7905HRESULT Medium::taskImportHandler(Medium::ImportTask &task)
7906{
7907 HRESULT rcTmp = S_OK;
7908
7909 const ComObjPtr<Medium> &pParent = task.mParent;
7910
7911 bool fCreatingTarget = false;
7912
7913 uint64_t size = 0, logicalSize = 0;
7914 MediumVariant_T variant = MediumVariant_Standard;
7915 bool fGenerateUuid = false;
7916
7917 try
7918 {
7919 /* Lock all in {parent,child} order. The lock is also used as a
7920 * signal from the task initiator (which releases it only after
7921 * RTThreadCreate()) that we can start the job. */
7922 AutoMultiWriteLock2 thisLock(this, pParent COMMA_LOCKVAL_SRC_POS);
7923
7924 fCreatingTarget = m->state == MediumState_Creating;
7925
7926 /* The object may request a specific UUID (through a special form of
7927 * the setLocation() argument). Otherwise we have to generate it */
7928 Guid targetId = m->id;
7929 fGenerateUuid = targetId.isEmpty();
7930 if (fGenerateUuid)
7931 {
7932 targetId.create();
7933 /* VirtualBox::registerMedium() will need UUID */
7934 unconst(m->id) = targetId;
7935 }
7936
7937
7938 PVBOXHDD hdd;
7939 int vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &hdd);
7940 ComAssertRCThrow(vrc, E_FAIL);
7941
7942 try
7943 {
7944 /* Open source medium. */
7945 vrc = VDOpen(hdd,
7946 task.mFormat->getId().c_str(),
7947 task.mFilename.c_str(),
7948 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_SEQUENTIAL | m->uOpenFlagsDef,
7949 task.mVDImageIfaces);
7950 if (RT_FAILURE(vrc))
7951 throw setError(VBOX_E_FILE_ERROR,
7952 tr("Could not open the medium storage unit '%s'%s"),
7953 task.mFilename.c_str(),
7954 vdError(vrc).c_str());
7955
7956 Utf8Str targetFormat(m->strFormat);
7957 Utf8Str targetLocation(m->strLocationFull);
7958 uint64_t capabilities = task.mFormat->getCapabilities();
7959
7960 Assert( m->state == MediumState_Creating
7961 || m->state == MediumState_LockedWrite);
7962 Assert( pParent.isNull()
7963 || pParent->m->state == MediumState_LockedRead);
7964
7965 /* unlock before the potentially lengthy operation */
7966 thisLock.release();
7967
7968 /* ensure the target directory exists */
7969 if (capabilities & MediumFormatCapabilities_File)
7970 {
7971 HRESULT rc = VirtualBox::ensureFilePathExists(targetLocation, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
7972 if (FAILED(rc))
7973 throw rc;
7974 }
7975
7976 PVBOXHDD targetHdd;
7977 vrc = VDCreate(m->vdDiskIfaces, convertDeviceType(), &targetHdd);
7978 ComAssertRCThrow(vrc, E_FAIL);
7979
7980 try
7981 {
7982 /* Open all media in the target chain. */
7983 MediumLockList::Base::const_iterator targetListBegin =
7984 task.mpTargetMediumLockList->GetBegin();
7985 MediumLockList::Base::const_iterator targetListEnd =
7986 task.mpTargetMediumLockList->GetEnd();
7987 for (MediumLockList::Base::const_iterator it = targetListBegin;
7988 it != targetListEnd;
7989 ++it)
7990 {
7991 const MediumLock &mediumLock = *it;
7992 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
7993
7994 /* If the target medium is not created yet there's no
7995 * reason to open it. */
7996 if (pMedium == this && fCreatingTarget)
7997 continue;
7998
7999 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
8000
8001 /* sanity check */
8002 Assert( pMedium->m->state == MediumState_LockedRead
8003 || pMedium->m->state == MediumState_LockedWrite);
8004
8005 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
8006 if (pMedium->m->state != MediumState_LockedWrite)
8007 uOpenFlags = VD_OPEN_FLAGS_READONLY;
8008 if (pMedium->m->type == MediumType_Shareable)
8009 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
8010
8011 /* Open all media in appropriate mode. */
8012 vrc = VDOpen(targetHdd,
8013 pMedium->m->strFormat.c_str(),
8014 pMedium->m->strLocationFull.c_str(),
8015 uOpenFlags | m->uOpenFlagsDef,
8016 pMedium->m->vdImageIfaces);
8017 if (RT_FAILURE(vrc))
8018 throw setError(VBOX_E_FILE_ERROR,
8019 tr("Could not open the medium storage unit '%s'%s"),
8020 pMedium->m->strLocationFull.c_str(),
8021 vdError(vrc).c_str());
8022 }
8023
8024 /** @todo r=klaus target isn't locked, race getting the state */
8025 vrc = VDCopy(hdd,
8026 VD_LAST_IMAGE,
8027 targetHdd,
8028 targetFormat.c_str(),
8029 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
8030 false /* fMoveByRename */,
8031 0 /* cbSize */,
8032 task.mVariant & ~MediumVariant_NoCreateDir,
8033 targetId.raw(),
8034 VD_OPEN_FLAGS_NORMAL,
8035 NULL /* pVDIfsOperation */,
8036 m->vdImageIfaces,
8037 task.mVDOperationIfaces);
8038 if (RT_FAILURE(vrc))
8039 throw setError(VBOX_E_FILE_ERROR,
8040 tr("Could not create the clone medium '%s'%s"),
8041 targetLocation.c_str(), vdError(vrc).c_str());
8042
8043 size = VDGetFileSize(targetHdd, VD_LAST_IMAGE);
8044 logicalSize = VDGetSize(targetHdd, VD_LAST_IMAGE);
8045 unsigned uImageFlags;
8046 vrc = VDGetImageFlags(targetHdd, 0, &uImageFlags);
8047 if (RT_SUCCESS(vrc))
8048 variant = (MediumVariant_T)uImageFlags;
8049 }
8050 catch (HRESULT aRC) { rcTmp = aRC; }
8051
8052 VDDestroy(targetHdd);
8053 }
8054 catch (HRESULT aRC) { rcTmp = aRC; }
8055
8056 VDDestroy(hdd);
8057 }
8058 catch (HRESULT aRC) { rcTmp = aRC; }
8059
8060 ErrorInfoKeeper eik;
8061 MultiResult mrc(rcTmp);
8062
8063 /* Only do the parent changes for newly created media. */
8064 if (SUCCEEDED(mrc) && fCreatingTarget)
8065 {
8066 /* we set mParent & children() */
8067 AutoWriteLock alock2(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
8068
8069 Assert(m->pParent.isNull());
8070
8071 if (pParent)
8072 {
8073 /* associate the clone with the parent and deassociate
8074 * from VirtualBox */
8075 m->pParent = pParent;
8076 pParent->m->llChildren.push_back(this);
8077
8078 /* register with mVirtualBox as the last step and move to
8079 * Created state only on success (leaving an orphan file is
8080 * better than breaking media registry consistency) */
8081 eik.restore();
8082 ComObjPtr<Medium> pMedium;
8083 mrc = pParent->m->pVirtualBox->registerMedium(this, &pMedium,
8084 DeviceType_HardDisk);
8085 Assert(this == pMedium);
8086 eik.fetch();
8087
8088 if (FAILED(mrc))
8089 /* break parent association on failure to register */
8090 this->deparent(); // removes target from parent
8091 }
8092 else
8093 {
8094 /* just register */
8095 eik.restore();
8096 ComObjPtr<Medium> pMedium;
8097 mrc = m->pVirtualBox->registerMedium(this, &pMedium, DeviceType_HardDisk);
8098 Assert(this == pMedium);
8099 eik.fetch();
8100 }
8101 }
8102
8103 if (fCreatingTarget)
8104 {
8105 AutoWriteLock mLock(this COMMA_LOCKVAL_SRC_POS);
8106
8107 if (SUCCEEDED(mrc))
8108 {
8109 m->state = MediumState_Created;
8110
8111 m->size = size;
8112 m->logicalSize = logicalSize;
8113 m->variant = variant;
8114 }
8115 else
8116 {
8117 /* back to NotCreated on failure */
8118 m->state = MediumState_NotCreated;
8119
8120 /* reset UUID to prevent it from being reused next time */
8121 if (fGenerateUuid)
8122 unconst(m->id).clear();
8123 }
8124 }
8125
8126 // now, at the end of this task (always asynchronous), save the settings
8127 {
8128 // save the settings
8129 markRegistriesModified();
8130 /* collect multiple errors */
8131 eik.restore();
8132 m->pVirtualBox->saveModifiedRegistries();
8133 eik.fetch();
8134 }
8135
8136 /* Everything is explicitly unlocked when the task exits,
8137 * as the task destruction also destroys the target chain. */
8138
8139 /* Make sure the target chain is released early, otherwise it can
8140 * lead to deadlocks with concurrent IAppliance activities. */
8141 task.mpTargetMediumLockList->Clear();
8142
8143 return mrc;
8144}
8145
8146/* 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