VirtualBox

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

Last change on this file since 51441 was 51414, checked in by vboxsync, 11 years ago

Main/Medium: fix some copy/paste issues, resulted in not so perfect error messages for OVF export and import, talking about cloning instead. Backport candidate, no regression risk.

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