VirtualBox

source: vbox/trunk/src/VBox/Main/MediumImpl.cpp@ 28669

Last change on this file since 28669 was 28587, checked in by vboxsync, 15 years ago

Main/Medium: fix parameter passing for mergeTo(), eliminating unnecessary indirection and copying.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 194.5 KB
Line 
1/* $Id: MediumImpl.cpp 28587 2010-04-22 11:38:59Z vboxsync $ */
2
3/** @file
4 *
5 * VirtualBox COM class implementation
6 */
7
8/*
9 * Copyright (C) 2008-2010 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24#include "MediumImpl.h"
25#include "ProgressImpl.h"
26#include "SystemPropertiesImpl.h"
27#include "VirtualBoxImpl.h"
28
29#include "AutoCaller.h"
30#include "Logging.h"
31
32#include <VBox/com/array.h>
33#include <VBox/com/SupportErrorInfo.h>
34
35#include <VBox/err.h>
36#include <VBox/settings.h>
37
38#include <iprt/param.h>
39#include <iprt/path.h>
40#include <iprt/file.h>
41#include <iprt/tcp.h>
42
43#include <VBox/VBoxHDD.h>
44
45#include <algorithm>
46
47////////////////////////////////////////////////////////////////////////////////
48//
49// Medium data definition
50//
51////////////////////////////////////////////////////////////////////////////////
52
53/** Describes how a machine refers to this image. */
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 typedef std::list<Guid> GuidList;
70
71 BackRef(const Guid &aMachineId,
72 const Guid &aSnapshotId = Guid::Empty)
73 : machineId(aMachineId),
74 fInCurState(aSnapshotId.isEmpty())
75 {
76 if (!aSnapshotId.isEmpty())
77 llSnapshotIds.push_back(aSnapshotId);
78 }
79
80 Guid machineId;
81 bool fInCurState : 1;
82 GuidList llSnapshotIds;
83};
84
85typedef std::list<BackRef> BackRefList;
86
87struct Medium::Data
88{
89 Data()
90 : pVirtualBox(NULL),
91 state(MediumState_NotCreated),
92 size(0),
93 readers(0),
94 preLockState(MediumState_NotCreated),
95 queryInfoSem(NIL_RTSEMEVENTMULTI),
96 queryInfoRunning(false),
97 type(MediumType_Normal),
98 devType(DeviceType_HardDisk),
99 logicalSize(0),
100 hddOpenMode(OpenReadWrite),
101 autoReset(false),
102 setImageId(false),
103 setParentId(false),
104 hostDrive(FALSE),
105 implicit(false),
106 numCreateDiffTasks(0),
107 vdDiskIfaces(NULL)
108 {}
109
110 /** weak VirtualBox parent */
111 VirtualBox * const pVirtualBox;
112
113 const Guid id;
114 Utf8Str strDescription;
115 MediumState_T state;
116 Utf8Str strLocation;
117 Utf8Str strLocationFull;
118 uint64_t size;
119 Utf8Str strLastAccessError;
120
121 // pParent and llChildren are protected by VirtualBox::getMediaTreeLockHandle()
122 ComObjPtr<Medium> pParent;
123 MediaList llChildren; // to add a child, just call push_back; to remove a child, call child->deparent() which does a lookup
124
125 BackRefList backRefs;
126
127 size_t readers;
128 MediumState_T preLockState;
129
130 RTSEMEVENTMULTI queryInfoSem;
131 bool queryInfoRunning : 1;
132
133 const Utf8Str strFormat;
134 ComObjPtr<MediumFormat> formatObj;
135
136 MediumType_T type;
137 DeviceType_T devType;
138 uint64_t logicalSize; /*< In MBytes. */
139
140 HDDOpenMode hddOpenMode;
141
142 BOOL autoReset : 1;
143
144 /** the following members are invalid after changing UUID on open */
145 BOOL setImageId : 1;
146 BOOL setParentId : 1;
147 const Guid imageId;
148 const Guid parentId;
149
150 BOOL hostDrive : 1;
151
152 typedef std::map <Bstr, Bstr> PropertyMap;
153 PropertyMap properties;
154
155 bool implicit : 1;
156
157 uint32_t numCreateDiffTasks;
158
159 Utf8Str vdError; /*< Error remembered by the VD error callback. */
160
161 VDINTERFACE vdIfError;
162 VDINTERFACEERROR vdIfCallsError;
163
164 VDINTERFACE vdIfConfig;
165 VDINTERFACECONFIG vdIfCallsConfig;
166
167 VDINTERFACE vdIfTcpNet;
168 VDINTERFACETCPNET vdIfCallsTcpNet;
169
170 PVDINTERFACE vdDiskIfaces;
171};
172
173////////////////////////////////////////////////////////////////////////////////
174//
175// Globals
176//
177////////////////////////////////////////////////////////////////////////////////
178
179/**
180 * Medium::Task class for asynchronous operations.
181 *
182 * @note Instances of this class must be created using new() because the
183 * task thread function will delete them when the task is complete.
184 *
185 * @note The constructor of this class adds a caller on the managed Medium
186 * object which is automatically released upon destruction.
187 */
188class Medium::Task
189{
190public:
191 Task(Medium *aMedium, Progress *aProgress)
192 : mVDOperationIfaces(NULL),
193 m_pfNeedsSaveSettings(NULL),
194 mMedium(aMedium),
195 mMediumCaller(aMedium),
196 mThread(NIL_RTTHREAD),
197 mProgress(aProgress)
198 {
199 AssertReturnVoidStmt(aMedium, mRC = E_FAIL);
200 mRC = mMediumCaller.rc();
201 if (FAILED(mRC))
202 return;
203
204 /* Set up a per-operation progress interface, can be used freely (for
205 * binary operations you can use it either on the source or target). */
206 mVDIfCallsProgress.cbSize = sizeof(VDINTERFACEPROGRESS);
207 mVDIfCallsProgress.enmInterface = VDINTERFACETYPE_PROGRESS;
208 mVDIfCallsProgress.pfnProgress = vdProgressCall;
209 int vrc = VDInterfaceAdd(&mVDIfProgress,
210 "Medium::Task::vdInterfaceProgress",
211 VDINTERFACETYPE_PROGRESS,
212 &mVDIfCallsProgress,
213 mProgress,
214 &mVDOperationIfaces);
215 AssertRC(vrc);
216 if (RT_FAILURE(vrc))
217 mRC = E_FAIL;
218 }
219
220 // Make all destructors virtual. Just in case.
221 virtual ~Task()
222 {}
223
224 HRESULT rc() const { return mRC; }
225 bool isOk() const { return SUCCEEDED(rc()); }
226
227 static int fntMediumTask(RTTHREAD aThread, void *pvUser);
228
229 bool isAsync() { return mThread != NIL_RTTHREAD; }
230
231 PVDINTERFACE mVDOperationIfaces;
232
233 // Whether the caller needs to call VirtualBox::saveSettings() after
234 // the task function returns. Only used in synchronous (wait) mode;
235 // otherwise the task will save the settings itself.
236 bool *m_pfNeedsSaveSettings;
237
238 const ComObjPtr<Medium> mMedium;
239 AutoCaller mMediumCaller;
240
241 friend HRESULT Medium::runNow(Medium::Task*, bool*);
242
243protected:
244 HRESULT mRC;
245 RTTHREAD mThread;
246
247private:
248 virtual HRESULT handler() = 0;
249
250 const ComObjPtr<Progress> mProgress;
251
252 static DECLCALLBACK(int) vdProgressCall(void *pvUser, unsigned uPercent);
253
254 VDINTERFACE mVDIfProgress;
255 VDINTERFACEPROGRESS mVDIfCallsProgress;
256};
257
258class Medium::CreateBaseTask : public Medium::Task
259{
260public:
261 CreateBaseTask(Medium *aMedium,
262 Progress *aProgress,
263 uint64_t aSize,
264 MediumVariant_T aVariant)
265 : Medium::Task(aMedium, aProgress),
266 mSize(aSize),
267 mVariant(aVariant)
268 {}
269
270 uint64_t mSize;
271 MediumVariant_T mVariant;
272
273private:
274 virtual HRESULT handler();
275};
276
277class Medium::CreateDiffTask : public Medium::Task
278{
279public:
280 CreateDiffTask(Medium *aMedium,
281 Progress *aProgress,
282 Medium *aTarget,
283 MediumVariant_T aVariant,
284 MediumLockList *aMediumLockList,
285 bool fKeepMediumLockList = false)
286 : Medium::Task(aMedium, aProgress),
287 mpMediumLockList(aMediumLockList),
288 mTarget(aTarget),
289 mVariant(aVariant),
290 mTargetCaller(aTarget),
291 mfKeepMediumLockList(fKeepMediumLockList)
292 {
293 AssertReturnVoidStmt(aTarget != NULL, mRC = E_FAIL);
294 mRC = mTargetCaller.rc();
295 if (FAILED(mRC))
296 return;
297 }
298
299 ~CreateDiffTask()
300 {
301 if (!mfKeepMediumLockList && mpMediumLockList)
302 delete mpMediumLockList;
303 }
304
305 MediumLockList *mpMediumLockList;
306
307 const ComObjPtr<Medium> mTarget;
308 MediumVariant_T mVariant;
309
310private:
311 virtual HRESULT handler();
312
313 AutoCaller mTargetCaller;
314 bool mfKeepMediumLockList;
315};
316
317class Medium::CloneTask : public Medium::Task
318{
319public:
320 CloneTask(Medium *aMedium,
321 Progress *aProgress,
322 Medium *aTarget,
323 MediumVariant_T aVariant,
324 Medium *aParent,
325 MediumLockList *aSourceMediumLockList,
326 MediumLockList *aTargetMediumLockList,
327 bool fKeepSourceMediumLockList = false,
328 bool fKeepTargetMediumLockList = false)
329 : Medium::Task(aMedium, aProgress),
330 mTarget(aTarget),
331 mParent(aParent),
332 mpSourceMediumLockList(aSourceMediumLockList),
333 mpTargetMediumLockList(aTargetMediumLockList),
334 mVariant(aVariant),
335 mTargetCaller(aTarget),
336 mParentCaller(aParent),
337 mfKeepSourceMediumLockList(fKeepSourceMediumLockList),
338 mfKeepTargetMediumLockList(fKeepTargetMediumLockList)
339 {
340 AssertReturnVoidStmt(aTarget != NULL, mRC = E_FAIL);
341 mRC = mTargetCaller.rc();
342 if (FAILED(mRC))
343 return;
344 /* aParent may be NULL */
345 mRC = mParentCaller.rc();
346 if (FAILED(mRC))
347 return;
348 AssertReturnVoidStmt(aSourceMediumLockList != NULL, mRC = E_FAIL);
349 AssertReturnVoidStmt(aTargetMediumLockList != NULL, mRC = E_FAIL);
350 }
351
352 ~CloneTask()
353 {
354 if (!mfKeepSourceMediumLockList && mpSourceMediumLockList)
355 delete mpSourceMediumLockList;
356 if (!mfKeepTargetMediumLockList && mpTargetMediumLockList)
357 delete mpTargetMediumLockList;
358 }
359
360 const ComObjPtr<Medium> mTarget;
361 const ComObjPtr<Medium> mParent;
362 MediumLockList *mpSourceMediumLockList;
363 MediumLockList *mpTargetMediumLockList;
364 MediumVariant_T mVariant;
365
366private:
367 virtual HRESULT handler();
368
369 AutoCaller mTargetCaller;
370 AutoCaller mParentCaller;
371 bool mfKeepSourceMediumLockList;
372 bool mfKeepTargetMediumLockList;
373};
374
375class Medium::CompactTask : public Medium::Task
376{
377public:
378 CompactTask(Medium *aMedium,
379 Progress *aProgress,
380 MediumLockList *aMediumLockList,
381 bool fKeepMediumLockList = false)
382 : Medium::Task(aMedium, aProgress),
383 mpMediumLockList(aMediumLockList),
384 mfKeepMediumLockList(fKeepMediumLockList)
385 {
386 AssertReturnVoidStmt(aMediumLockList != NULL, mRC = E_FAIL);
387 }
388
389 ~CompactTask()
390 {
391 if (!mfKeepMediumLockList && mpMediumLockList)
392 delete mpMediumLockList;
393 }
394
395 MediumLockList *mpMediumLockList;
396
397private:
398 virtual HRESULT handler();
399
400 bool mfKeepMediumLockList;
401};
402
403class Medium::ResetTask : public Medium::Task
404{
405public:
406 ResetTask(Medium *aMedium,
407 Progress *aProgress,
408 MediumLockList *aMediumLockList,
409 bool fKeepMediumLockList = false)
410 : Medium::Task(aMedium, aProgress),
411 mpMediumLockList(aMediumLockList),
412 mfKeepMediumLockList(fKeepMediumLockList)
413 {}
414
415 ~ResetTask()
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::DeleteTask : public Medium::Task
430{
431public:
432 DeleteTask(Medium *aMedium,
433 Progress *aProgress,
434 MediumLockList *aMediumLockList,
435 bool fKeepMediumLockList = false)
436 : Medium::Task(aMedium, aProgress),
437 mpMediumLockList(aMediumLockList),
438 mfKeepMediumLockList(fKeepMediumLockList)
439 {}
440
441 ~DeleteTask()
442 {
443 if (!mfKeepMediumLockList && mpMediumLockList)
444 delete mpMediumLockList;
445 }
446
447 MediumLockList *mpMediumLockList;
448
449private:
450 virtual HRESULT handler();
451
452 bool mfKeepMediumLockList;
453};
454
455class Medium::MergeTask : public Medium::Task
456{
457public:
458 MergeTask(Medium *aMedium,
459 Medium *aTarget,
460 bool fMergeForward,
461 Medium *aParentForTarget,
462 const MediaList &aChildrenToReparent,
463 Progress *aProgress,
464 MediumLockList *aMediumLockList,
465 bool fKeepMediumLockList = false)
466 : Medium::Task(aMedium, aProgress),
467 mTarget(aTarget),
468 mfMergeForward(fMergeForward),
469 mParentForTarget(aParentForTarget),
470 mChildrenToReparent(aChildrenToReparent),
471 mpMediumLockList(aMediumLockList),
472 mTargetCaller(aTarget),
473 mParentForTargetCaller(aParentForTarget),
474 mfChildrenCaller(false),
475 mfKeepMediumLockList(fKeepMediumLockList)
476 {
477 AssertReturnVoidStmt(aMediumLockList != NULL, mRC = E_FAIL);
478 for (MediaList::const_iterator it = mChildrenToReparent.begin();
479 it != mChildrenToReparent.end();
480 ++it)
481 {
482 HRESULT rc2 = (*it)->addCaller();
483 if (FAILED(rc2))
484 {
485 mRC = E_FAIL;
486 for (MediaList::const_iterator it2 = mChildrenToReparent.begin();
487 it2 != it;
488 --it2)
489 {
490 (*it2)->releaseCaller();
491 }
492 return;
493 }
494 }
495 mfChildrenCaller = true;
496 }
497
498 ~MergeTask()
499 {
500 if (!mfKeepMediumLockList && mpMediumLockList)
501 delete mpMediumLockList;
502 if (mfChildrenCaller)
503 {
504 for (MediaList::const_iterator it = mChildrenToReparent.begin();
505 it != mChildrenToReparent.end();
506 ++it)
507 {
508 (*it)->releaseCaller();
509 }
510 }
511 }
512
513 const ComObjPtr<Medium> mTarget;
514 bool mfMergeForward;
515 /* When mChildrenToReparent is empty then mParentForTarget is non-null.
516 * In other words: they are used in different cases. */
517 const ComObjPtr<Medium> mParentForTarget;
518 MediaList mChildrenToReparent;
519 MediumLockList *mpMediumLockList;
520
521private:
522 virtual HRESULT handler();
523
524 AutoCaller mTargetCaller;
525 AutoCaller mParentForTargetCaller;
526 bool mfChildrenCaller;
527 bool mfKeepMediumLockList;
528};
529
530/**
531 * Thread function for time-consuming medium tasks.
532 *
533 * @param pvUser Pointer to the Medium::Task instance.
534 */
535/* static */
536DECLCALLBACK(int) Medium::Task::fntMediumTask(RTTHREAD aThread, void *pvUser)
537{
538 LogFlowFuncEnter();
539 AssertReturn(pvUser, (int)E_INVALIDARG);
540 Medium::Task *pTask = static_cast<Medium::Task *>(pvUser);
541
542 pTask->mThread = aThread;
543
544 HRESULT rc = pTask->handler();
545
546 /* complete the progress if run asynchronously */
547 if (pTask->isAsync())
548 {
549 if (!pTask->mProgress.isNull())
550 pTask->mProgress->notifyComplete(rc);
551 }
552
553 /* pTask is no longer needed, delete it. */
554 delete pTask;
555
556 LogFlowFunc(("rc=%Rhrc\n", rc));
557 LogFlowFuncLeave();
558
559 return (int)rc;
560}
561
562/**
563 * PFNVDPROGRESS callback handler for Task operations.
564 *
565 * @param pvUser Pointer to the Progress instance.
566 * @param uPercent Completetion precentage (0-100).
567 */
568/*static*/
569DECLCALLBACK(int) Medium::Task::vdProgressCall(void *pvUser, unsigned uPercent)
570{
571 Progress *that = static_cast<Progress *>(pvUser);
572
573 if (that != NULL)
574 {
575 /* update the progress object, capping it at 99% as the final percent
576 * is used for additional operations like setting the UUIDs and similar. */
577 HRESULT rc = that->SetCurrentOperationProgress(uPercent * 99 / 100);
578 if (FAILED(rc))
579 {
580 if (rc == E_FAIL)
581 return VERR_CANCELLED;
582 else
583 return VERR_INVALID_STATE;
584 }
585 }
586
587 return VINF_SUCCESS;
588}
589
590/**
591 * Implementation code for the "create base" task.
592 */
593HRESULT Medium::CreateBaseTask::handler()
594{
595 return mMedium->taskCreateBaseHandler(*this);
596}
597
598/**
599 * Implementation code for the "create diff" task.
600 */
601HRESULT Medium::CreateDiffTask::handler()
602{
603 return mMedium->taskCreateDiffHandler(*this);
604}
605
606/**
607 * Implementation code for the "clone" task.
608 */
609HRESULT Medium::CloneTask::handler()
610{
611 return mMedium->taskCloneHandler(*this);
612}
613
614/**
615 * Implementation code for the "compact" task.
616 */
617HRESULT Medium::CompactTask::handler()
618{
619 return mMedium->taskCompactHandler(*this);
620}
621
622/**
623 * Implementation code for the "reset" task.
624 */
625HRESULT Medium::ResetTask::handler()
626{
627 return mMedium->taskResetHandler(*this);
628}
629
630/**
631 * Implementation code for the "delete" task.
632 */
633HRESULT Medium::DeleteTask::handler()
634{
635 return mMedium->taskDeleteHandler(*this);
636}
637
638/**
639 * Implementation code for the "merge" task.
640 */
641HRESULT Medium::MergeTask::handler()
642{
643 return mMedium->taskMergeHandler(*this);
644}
645
646
647////////////////////////////////////////////////////////////////////////////////
648//
649// Medium constructor / destructor
650//
651////////////////////////////////////////////////////////////////////////////////
652
653DEFINE_EMPTY_CTOR_DTOR(Medium)
654
655HRESULT Medium::FinalConstruct()
656{
657 m = new Data;
658
659 /* Initialize the callbacks of the VD error interface */
660 m->vdIfCallsError.cbSize = sizeof(VDINTERFACEERROR);
661 m->vdIfCallsError.enmInterface = VDINTERFACETYPE_ERROR;
662 m->vdIfCallsError.pfnError = vdErrorCall;
663 m->vdIfCallsError.pfnMessage = NULL;
664
665 /* Initialize the callbacks of the VD config interface */
666 m->vdIfCallsConfig.cbSize = sizeof(VDINTERFACECONFIG);
667 m->vdIfCallsConfig.enmInterface = VDINTERFACETYPE_CONFIG;
668 m->vdIfCallsConfig.pfnAreKeysValid = vdConfigAreKeysValid;
669 m->vdIfCallsConfig.pfnQuerySize = vdConfigQuerySize;
670 m->vdIfCallsConfig.pfnQuery = vdConfigQuery;
671
672 /* Initialize the callbacks of the VD TCP interface (we always use the host
673 * IP stack for now) */
674 m->vdIfCallsTcpNet.cbSize = sizeof(VDINTERFACETCPNET);
675 m->vdIfCallsTcpNet.enmInterface = VDINTERFACETYPE_TCPNET;
676 m->vdIfCallsTcpNet.pfnClientConnect = RTTcpClientConnect;
677 m->vdIfCallsTcpNet.pfnClientClose = RTTcpClientClose;
678 m->vdIfCallsTcpNet.pfnSelectOne = RTTcpSelectOne;
679 m->vdIfCallsTcpNet.pfnRead = RTTcpRead;
680 m->vdIfCallsTcpNet.pfnWrite = RTTcpWrite;
681 m->vdIfCallsTcpNet.pfnFlush = RTTcpFlush;
682 m->vdIfCallsTcpNet.pfnGetLocalAddress = RTTcpGetLocalAddress;
683 m->vdIfCallsTcpNet.pfnGetPeerAddress = RTTcpGetPeerAddress;
684
685 /* Initialize the per-disk interface chain */
686 int vrc;
687 vrc = VDInterfaceAdd(&m->vdIfError,
688 "Medium::vdInterfaceError",
689 VDINTERFACETYPE_ERROR,
690 &m->vdIfCallsError, this, &m->vdDiskIfaces);
691 AssertRCReturn(vrc, E_FAIL);
692
693 vrc = VDInterfaceAdd(&m->vdIfConfig,
694 "Medium::vdInterfaceConfig",
695 VDINTERFACETYPE_CONFIG,
696 &m->vdIfCallsConfig, this, &m->vdDiskIfaces);
697 AssertRCReturn(vrc, E_FAIL);
698
699 vrc = VDInterfaceAdd(&m->vdIfTcpNet,
700 "Medium::vdInterfaceTcpNet",
701 VDINTERFACETYPE_TCPNET,
702 &m->vdIfCallsTcpNet, this, &m->vdDiskIfaces);
703 AssertRCReturn(vrc, E_FAIL);
704
705 vrc = RTSemEventMultiCreate(&m->queryInfoSem);
706 AssertRCReturn(vrc, E_FAIL);
707 vrc = RTSemEventMultiSignal(m->queryInfoSem);
708 AssertRCReturn(vrc, E_FAIL);
709
710 return S_OK;
711}
712
713void Medium::FinalRelease()
714{
715 uninit();
716
717 delete m;
718}
719
720/**
721 * Initializes the hard disk object without creating or opening an associated
722 * storage unit.
723 *
724 * For hard disks that don't have the VD_CAP_CREATE_FIXED or
725 * VD_CAP_CREATE_DYNAMIC capability (and therefore cannot be created or deleted
726 * with the means of VirtualBox) the associated storage unit is assumed to be
727 * ready for use so the state of the hard disk object will be set to Created.
728 *
729 * @param aVirtualBox VirtualBox object.
730 * @param aLocation Storage unit location.
731 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
732 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
733 */
734HRESULT Medium::init(VirtualBox *aVirtualBox,
735 CBSTR aFormat,
736 CBSTR aLocation,
737 bool *pfNeedsSaveSettings)
738{
739 AssertReturn(aVirtualBox != NULL, E_FAIL);
740 AssertReturn(aFormat != NULL && *aFormat != '\0', E_FAIL);
741
742 /* Enclose the state transition NotReady->InInit->Ready */
743 AutoInitSpan autoInitSpan(this);
744 AssertReturn(autoInitSpan.isOk(), E_FAIL);
745
746 HRESULT rc = S_OK;
747
748 /* share VirtualBox weakly (parent remains NULL so far) */
749 unconst(m->pVirtualBox) = aVirtualBox;
750
751 /* no storage yet */
752 m->state = MediumState_NotCreated;
753
754 /* cannot be a host drive */
755 m->hostDrive = FALSE;
756
757 /* No storage unit is created yet, no need to queryInfo() */
758
759 rc = setFormat(aFormat);
760 if (FAILED(rc)) return rc;
761
762 if (m->formatObj->capabilities() & MediumFormatCapabilities_File)
763 {
764 rc = setLocation(aLocation);
765 if (FAILED(rc)) return rc;
766 }
767 else
768 {
769 rc = setLocation(aLocation);
770 if (FAILED(rc)) return rc;
771 }
772
773 if (!(m->formatObj->capabilities() & ( MediumFormatCapabilities_CreateFixed
774 | MediumFormatCapabilities_CreateDynamic))
775 )
776 {
777 /* storage for hard disks of this format can neither be explicitly
778 * created by VirtualBox nor deleted, so we place the hard disk to
779 * Created state here and also add it to the registry */
780 m->state = MediumState_Created;
781 unconst(m->id).create();
782
783 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
784 rc = m->pVirtualBox->registerHardDisk(this, pfNeedsSaveSettings);
785 }
786
787 /* Confirm a successful initialization when it's the case */
788 if (SUCCEEDED(rc))
789 autoInitSpan.setSucceeded();
790
791 return rc;
792}
793
794/**
795 * Initializes the medium object by opening the storage unit at the specified
796 * location. The enOpenMode parameter defines whether the image will be opened
797 * read/write or read-only.
798 *
799 * Note that the UUID, format and the parent of this medium will be
800 * determined when reading the medium storage unit, unless new values are
801 * specified by the parameters. If the detected or set parent is
802 * not known to VirtualBox, then this method will fail.
803 *
804 * @param aVirtualBox VirtualBox object.
805 * @param aLocation Storage unit location.
806 * @param enOpenMode Whether to open the image read/write or read-only.
807 * @param aDeviceType Device type of medium.
808 * @param aSetImageId Whether to set the image UUID or not.
809 * @param aImageId New image UUID if @aSetId is true. Empty string means
810 * create a new UUID, and a zero UUID is invalid.
811 * @param aSetParentId Whether to set the parent UUID or not.
812 * @param aParentId New parent UUID if @aSetParentId is true. Empty string
813 * means create a new UUID, and a zero UUID is valid.
814 */
815HRESULT Medium::init(VirtualBox *aVirtualBox,
816 CBSTR aLocation,
817 HDDOpenMode enOpenMode,
818 DeviceType_T aDeviceType,
819 BOOL aSetImageId,
820 const Guid &aImageId,
821 BOOL aSetParentId,
822 const Guid &aParentId)
823{
824 AssertReturn(aVirtualBox, E_INVALIDARG);
825 AssertReturn(aLocation, E_INVALIDARG);
826
827 /* Enclose the state transition NotReady->InInit->Ready */
828 AutoInitSpan autoInitSpan(this);
829 AssertReturn(autoInitSpan.isOk(), E_FAIL);
830
831 HRESULT rc = S_OK;
832
833 /* share VirtualBox weakly (parent remains NULL so far) */
834 unconst(m->pVirtualBox) = aVirtualBox;
835
836 /* there must be a storage unit */
837 m->state = MediumState_Created;
838
839 /* remember device type for correct unregistering later */
840 m->devType = aDeviceType;
841
842 /* cannot be a host drive */
843 m->hostDrive = FALSE;
844
845 /* remember the open mode (defaults to ReadWrite) */
846 m->hddOpenMode = enOpenMode;
847
848 if (aDeviceType == DeviceType_HardDisk)
849 rc = setLocation(aLocation);
850 else
851 rc = setLocation(aLocation, "RAW");
852 if (FAILED(rc)) return rc;
853
854 /* save the new uuid values, will be used by queryInfo() */
855 m->setImageId = aSetImageId;
856 unconst(m->imageId) = aImageId;
857 m->setParentId = aSetParentId;
858 unconst(m->parentId) = aParentId;
859
860 /* get all the information about the medium from the storage unit */
861 rc = queryInfo();
862
863 if (SUCCEEDED(rc))
864 {
865 /* if the storage unit is not accessible, it's not acceptable for the
866 * newly opened media so convert this into an error */
867 if (m->state == MediumState_Inaccessible)
868 {
869 Assert(!m->strLastAccessError.isEmpty());
870 rc = setError(E_FAIL, m->strLastAccessError.c_str());
871 }
872 else
873 {
874 AssertReturn(!m->id.isEmpty(), E_FAIL);
875
876 /* storage format must be detected by queryInfo() if the medium is accessible */
877 AssertReturn(!m->strFormat.isEmpty(), E_FAIL);
878 }
879 }
880
881 /* Confirm a successful initialization when it's the case */
882 if (SUCCEEDED(rc))
883 autoInitSpan.setSucceeded();
884
885 return rc;
886}
887
888/**
889 * Initializes the medium object by loading its data from the given settings
890 * node. In this mode, the image will always be opened read/write.
891 *
892 * @param aVirtualBox VirtualBox object.
893 * @param aParent Parent medium disk or NULL for a root (base) medium.
894 * @param aDeviceType Device type of the medium.
895 * @param aNode Configuration settings.
896 *
897 * @note Locks VirtualBox for writing, the medium tree for writing.
898 */
899HRESULT Medium::init(VirtualBox *aVirtualBox,
900 Medium *aParent,
901 DeviceType_T aDeviceType,
902 const settings::Medium &data)
903{
904 using namespace settings;
905
906 AssertReturn(aVirtualBox, E_INVALIDARG);
907
908 /* Enclose the state transition NotReady->InInit->Ready */
909 AutoInitSpan autoInitSpan(this);
910 AssertReturn(autoInitSpan.isOk(), E_FAIL);
911
912 HRESULT rc = S_OK;
913
914 /* share VirtualBox and parent weakly */
915 unconst(m->pVirtualBox) = aVirtualBox;
916
917 /* register with VirtualBox/parent early, since uninit() will
918 * unconditionally unregister on failure */
919 if (aParent)
920 {
921 // differencing image: add to parent
922 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
923 m->pParent = aParent;
924 aParent->m->llChildren.push_back(this);
925 }
926
927 /* see below why we don't call queryInfo() (and therefore treat the medium
928 * as inaccessible for now */
929 m->state = MediumState_Inaccessible;
930 m->strLastAccessError = tr("Accessibility check was not yet performed");
931
932 /* required */
933 unconst(m->id) = data.uuid;
934
935 /* assume not a host drive */
936 m->hostDrive = FALSE;
937
938 /* optional */
939 m->strDescription = data.strDescription;
940
941 /* required */
942 if (aDeviceType == DeviceType_HardDisk)
943 {
944 AssertReturn(!data.strFormat.isEmpty(), E_FAIL);
945 rc = setFormat(Bstr(data.strFormat));
946 if (FAILED(rc)) return rc;
947 }
948 else
949 {
950 /// @todo handle host drive settings here as well?
951 if (!data.strFormat.isEmpty())
952 rc = setFormat(Bstr(data.strFormat));
953 else
954 rc = setFormat(Bstr("RAW"));
955 if (FAILED(rc)) return rc;
956 }
957
958 /* optional, only for diffs, default is false;
959 * we can only auto-reset diff images, so they
960 * must not have a parent */
961 if (aParent != NULL)
962 m->autoReset = data.fAutoReset;
963 else
964 m->autoReset = false;
965
966 /* properties (after setting the format as it populates the map). Note that
967 * if some properties are not supported but preseint in the settings file,
968 * they will still be read and accessible (for possible backward
969 * compatibility; we can also clean them up from the XML upon next
970 * XML format version change if we wish) */
971 for (settings::PropertiesMap::const_iterator it = data.properties.begin();
972 it != data.properties.end(); ++it)
973 {
974 const Utf8Str &name = it->first;
975 const Utf8Str &value = it->second;
976 m->properties[Bstr(name)] = Bstr(value);
977 }
978
979 /* required */
980 rc = setLocation(data.strLocation);
981 if (FAILED(rc)) return rc;
982
983 if (aDeviceType == DeviceType_HardDisk)
984 {
985 /* type is only for base hard disks */
986 if (m->pParent.isNull())
987 m->type = data.hdType;
988 }
989 else
990 m->type = MediumType_Writethrough;
991
992 /* remember device type for correct unregistering later */
993 m->devType = aDeviceType;
994
995 LogFlowThisFunc(("m->locationFull='%s', m->format=%s, m->id={%RTuuid}\n",
996 m->strLocationFull.raw(), m->strFormat.raw(), m->id.raw()));
997
998 /* Don't call queryInfo() for registered media to prevent the calling
999 * thread (i.e. the VirtualBox server startup thread) from an unexpected
1000 * freeze but mark it as initially inaccessible instead. The vital UUID,
1001 * location and format properties are read from the registry file above; to
1002 * get the actual state and the rest of the data, the user will have to call
1003 * COMGETTER(State). */
1004
1005 AutoWriteLock treeLock(aVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1006
1007 /* load all children */
1008 for (settings::MediaList::const_iterator it = data.llChildren.begin();
1009 it != data.llChildren.end();
1010 ++it)
1011 {
1012 const settings::Medium &med = *it;
1013
1014 ComObjPtr<Medium> pHD;
1015 pHD.createObject();
1016 rc = pHD->init(aVirtualBox,
1017 this, // parent
1018 aDeviceType,
1019 med); // child data
1020 if (FAILED(rc)) break;
1021
1022 rc = m->pVirtualBox->registerHardDisk(pHD, NULL /*pfNeedsSaveSettings*/);
1023 if (FAILED(rc)) break;
1024 }
1025
1026 /* Confirm a successful initialization when it's the case */
1027 if (SUCCEEDED(rc))
1028 autoInitSpan.setSucceeded();
1029
1030 return rc;
1031}
1032
1033/**
1034 * Initializes the medium object by providing the host drive information.
1035 * Not used for anything but the host floppy/host DVD case.
1036 *
1037 * @todo optimize all callers to avoid reconstructing objects with the same
1038 * information over and over again - in the typical case each VM referring to
1039 * a particular host drive has its own instance.
1040 *
1041 * @param aVirtualBox VirtualBox object.
1042 * @param aDeviceType Device type of the medium.
1043 * @param aLocation Location of the host drive.
1044 * @param aDescription Comment for this host drive.
1045 *
1046 * @note Locks VirtualBox lock for writing.
1047 */
1048HRESULT Medium::init(VirtualBox *aVirtualBox,
1049 DeviceType_T aDeviceType,
1050 CBSTR aLocation,
1051 CBSTR aDescription)
1052{
1053 ComAssertRet(aDeviceType == DeviceType_DVD || aDeviceType == DeviceType_Floppy, E_INVALIDARG);
1054 ComAssertRet(aLocation, E_INVALIDARG);
1055
1056 /* Enclose the state transition NotReady->InInit->Ready */
1057 AutoInitSpan autoInitSpan(this);
1058 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1059
1060 /* share VirtualBox weakly (parent remains NULL so far) */
1061 unconst(m->pVirtualBox) = aVirtualBox;
1062
1063 /* fake up a UUID which is unique, but also reproducible */
1064 RTUUID uuid;
1065 RTUuidClear(&uuid);
1066 if (aDeviceType == DeviceType_DVD)
1067 memcpy(&uuid.au8[0], "DVD", 3);
1068 else
1069 memcpy(&uuid.au8[0], "FD", 2);
1070 /* use device name, adjusted to the end of uuid, shortened if necessary */
1071 Utf8Str loc(aLocation);
1072 size_t cbLocation = strlen(loc.raw());
1073 if (cbLocation > 12)
1074 memcpy(&uuid.au8[4], loc.raw() + (cbLocation - 12), 12);
1075 else
1076 memcpy(&uuid.au8[4 + 12 - cbLocation], loc.raw(), cbLocation);
1077 unconst(m->id) = uuid;
1078
1079 m->type = MediumType_Writethrough;
1080 m->devType = aDeviceType;
1081 m->state = MediumState_Created;
1082 m->hostDrive = true;
1083 HRESULT rc = setFormat(Bstr("RAW"));
1084 if (FAILED(rc)) return rc;
1085 rc = setLocation(aLocation);
1086 if (FAILED(rc)) return rc;
1087 m->strDescription = aDescription;
1088
1089/// @todo generate uuid (similarly to host network interface uuid) from location and device type
1090
1091 autoInitSpan.setSucceeded();
1092 return S_OK;
1093}
1094
1095/**
1096 * Uninitializes the instance.
1097 *
1098 * Called either from FinalRelease() or by the parent when it gets destroyed.
1099 *
1100 * @note All children of this hard disk get uninitialized by calling their
1101 * uninit() methods.
1102 *
1103 * @note Caller must hold the tree lock of the medium tree this medium is on.
1104 */
1105void Medium::uninit()
1106{
1107 /* Enclose the state transition Ready->InUninit->NotReady */
1108 AutoUninitSpan autoUninitSpan(this);
1109 if (autoUninitSpan.uninitDone())
1110 return;
1111
1112 if (!m->formatObj.isNull())
1113 {
1114 /* remove the caller reference we added in setFormat() */
1115 m->formatObj->releaseCaller();
1116 m->formatObj.setNull();
1117 }
1118
1119 if (m->state == MediumState_Deleting)
1120 {
1121 /* we are being uninitialized after've been deleted by merge.
1122 * Reparenting has already been done so don't touch it here (we are
1123 * now orphans and removeDependentChild() will assert) */
1124 Assert(m->pParent.isNull());
1125 }
1126 else
1127 {
1128 MediaList::iterator it;
1129 for (it = m->llChildren.begin();
1130 it != m->llChildren.end();
1131 ++it)
1132 {
1133 Medium *pChild = *it;
1134 pChild->m->pParent.setNull();
1135 pChild->uninit();
1136 }
1137 m->llChildren.clear(); // this unsets all the ComPtrs and probably calls delete
1138
1139 if (m->pParent)
1140 {
1141 // this is a differencing disk: then remove it from the parent's children list
1142 deparent();
1143 }
1144 }
1145
1146 RTSemEventMultiSignal(m->queryInfoSem);
1147 RTSemEventMultiDestroy(m->queryInfoSem);
1148 m->queryInfoSem = NIL_RTSEMEVENTMULTI;
1149
1150 unconst(m->pVirtualBox) = NULL;
1151}
1152
1153/**
1154 * Internal helper that removes "this" from the list of children of its
1155 * parent. Used in uninit() and other places when reparenting is necessary.
1156 *
1157 * The caller must hold the hard disk tree lock!
1158 */
1159void Medium::deparent()
1160{
1161 MediaList &llParent = m->pParent->m->llChildren;
1162 for (MediaList::iterator it = llParent.begin();
1163 it != llParent.end();
1164 ++it)
1165 {
1166 Medium *pParentsChild = *it;
1167 if (this == pParentsChild)
1168 {
1169 llParent.erase(it);
1170 break;
1171 }
1172 }
1173 m->pParent.setNull();
1174}
1175
1176////////////////////////////////////////////////////////////////////////////////
1177//
1178// IMedium public methods
1179//
1180////////////////////////////////////////////////////////////////////////////////
1181
1182STDMETHODIMP Medium::COMGETTER(Id)(BSTR *aId)
1183{
1184 CheckComArgOutPointerValid(aId);
1185
1186 AutoCaller autoCaller(this);
1187 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1188
1189 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1190
1191 m->id.toUtf16().cloneTo(aId);
1192
1193 return S_OK;
1194}
1195
1196STDMETHODIMP Medium::COMGETTER(Description)(BSTR *aDescription)
1197{
1198 CheckComArgOutPointerValid(aDescription);
1199
1200 AutoCaller autoCaller(this);
1201 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1202
1203 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1204
1205 m->strDescription.cloneTo(aDescription);
1206
1207 return S_OK;
1208}
1209
1210STDMETHODIMP Medium::COMSETTER(Description)(IN_BSTR aDescription)
1211{
1212 AutoCaller autoCaller(this);
1213 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1214
1215// AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1216
1217 /// @todo update m->description and save the global registry (and local
1218 /// registries of portable VMs referring to this medium), this will also
1219 /// require to add the mRegistered flag to data
1220
1221 NOREF(aDescription);
1222
1223 ReturnComNotImplemented();
1224}
1225
1226STDMETHODIMP Medium::COMGETTER(State)(MediumState_T *aState)
1227{
1228 CheckComArgOutPointerValid(aState);
1229
1230 AutoCaller autoCaller(this);
1231 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1232
1233 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1234 *aState = m->state;
1235
1236 return S_OK;
1237}
1238
1239
1240STDMETHODIMP Medium::COMGETTER(Location)(BSTR *aLocation)
1241{
1242 CheckComArgOutPointerValid(aLocation);
1243
1244 AutoCaller autoCaller(this);
1245 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1246
1247 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1248
1249 m->strLocationFull.cloneTo(aLocation);
1250
1251 return S_OK;
1252}
1253
1254STDMETHODIMP Medium::COMSETTER(Location)(IN_BSTR aLocation)
1255{
1256 CheckComArgStrNotEmptyOrNull(aLocation);
1257
1258 AutoCaller autoCaller(this);
1259 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1260
1261 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1262
1263 /// @todo NEWMEDIA for file names, add the default extension if no extension
1264 /// is present (using the information from the VD backend which also implies
1265 /// that one more parameter should be passed to setLocation() requesting
1266 /// that functionality since it is only allwed when called from this method
1267
1268 /// @todo NEWMEDIA rename the file and set m->location on success, then save
1269 /// the global registry (and local registries of portable VMs referring to
1270 /// this medium), this will also require to add the mRegistered flag to data
1271
1272 ReturnComNotImplemented();
1273}
1274
1275STDMETHODIMP Medium::COMGETTER(Name)(BSTR *aName)
1276{
1277 CheckComArgOutPointerValid(aName);
1278
1279 AutoCaller autoCaller(this);
1280 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1281
1282 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1283
1284 getName().cloneTo(aName);
1285
1286 return S_OK;
1287}
1288
1289STDMETHODIMP Medium::COMGETTER(DeviceType)(DeviceType_T *aDeviceType)
1290{
1291 CheckComArgOutPointerValid(aDeviceType);
1292
1293 AutoCaller autoCaller(this);
1294 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1295
1296 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1297
1298 *aDeviceType = m->devType;
1299
1300 return S_OK;
1301}
1302
1303STDMETHODIMP Medium::COMGETTER(HostDrive)(BOOL *aHostDrive)
1304{
1305 CheckComArgOutPointerValid(aHostDrive);
1306
1307 AutoCaller autoCaller(this);
1308 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1309
1310 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1311
1312 *aHostDrive = m->hostDrive;
1313
1314 return S_OK;
1315}
1316
1317STDMETHODIMP Medium::COMGETTER(Size)(ULONG64 *aSize)
1318{
1319 CheckComArgOutPointerValid(aSize);
1320
1321 AutoCaller autoCaller(this);
1322 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1323
1324 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1325
1326 *aSize = m->size;
1327
1328 return S_OK;
1329}
1330
1331STDMETHODIMP Medium::COMGETTER(Format)(BSTR *aFormat)
1332{
1333 CheckComArgOutPointerValid(aFormat);
1334
1335 AutoCaller autoCaller(this);
1336 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1337
1338 /* no need to lock, m->format is const */
1339 m->strFormat.cloneTo(aFormat);
1340
1341 return S_OK;
1342}
1343
1344STDMETHODIMP Medium::COMGETTER(Type)(MediumType_T *aType)
1345{
1346 CheckComArgOutPointerValid(aType);
1347
1348 AutoCaller autoCaller(this);
1349 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1350
1351 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1352
1353 *aType = m->type;
1354
1355 return S_OK;
1356}
1357
1358STDMETHODIMP Medium::COMSETTER(Type)(MediumType_T aType)
1359{
1360 AutoCaller autoCaller(this);
1361 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1362
1363 // we access mParent and members
1364 AutoMultiWriteLock2 mlock(&m->pVirtualBox->getMediaTreeLockHandle(), this->lockHandle() COMMA_LOCKVAL_SRC_POS);
1365
1366 switch (m->state)
1367 {
1368 case MediumState_Created:
1369 case MediumState_Inaccessible:
1370 break;
1371 default:
1372 return setStateError();
1373 }
1374
1375 if (m->type == aType)
1376 {
1377 /* Nothing to do */
1378 return S_OK;
1379 }
1380
1381 /* cannot change the type of a differencing hard disk */
1382 if (m->pParent)
1383 return setError(E_FAIL,
1384 tr("Cannot change the type of hard disk '%s' because it is a differencing hard disk"),
1385 m->strLocationFull.raw());
1386
1387 /* cannot change the type of a hard disk being in use */
1388 if (m->backRefs.size() != 0)
1389 return setError(E_FAIL,
1390 tr("Cannot change the type of hard disk '%s' because it is attached to %d virtual machines"),
1391 m->strLocationFull.raw(), m->backRefs.size());
1392
1393 switch (aType)
1394 {
1395 case MediumType_Normal:
1396 case MediumType_Immutable:
1397 {
1398 /* normal can be easily converted to immutable and vice versa even
1399 * if they have children as long as they are not attached to any
1400 * machine themselves */
1401 break;
1402 }
1403 case MediumType_Writethrough:
1404 {
1405 /* cannot change to writethrough if there are children */
1406 if (getChildren().size() != 0)
1407 return setError(E_FAIL,
1408 tr("Cannot change type for hard disk '%s' since it has %d child hard disk(s)"),
1409 m->strLocationFull.raw(), getChildren().size());
1410 break;
1411 }
1412 default:
1413 AssertFailedReturn(E_FAIL);
1414 }
1415
1416 m->type = aType;
1417
1418 // saveSettings needs vbox lock
1419 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
1420 mlock.leave();
1421 AutoWriteLock alock(m->pVirtualBox COMMA_LOCKVAL_SRC_POS);
1422
1423 HRESULT rc = pVirtualBox->saveSettings();
1424
1425 return rc;
1426}
1427
1428STDMETHODIMP Medium::COMGETTER(Parent)(IMedium **aParent)
1429{
1430 CheckComArgOutPointerValid(aParent);
1431
1432 AutoCaller autoCaller(this);
1433 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1434
1435 /* we access mParent */
1436 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1437
1438 m->pParent.queryInterfaceTo(aParent);
1439
1440 return S_OK;
1441}
1442
1443STDMETHODIMP Medium::COMGETTER(Children)(ComSafeArrayOut(IMedium *, aChildren))
1444{
1445 CheckComArgOutSafeArrayPointerValid(aChildren);
1446
1447 AutoCaller autoCaller(this);
1448 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1449
1450 /* we access children */
1451 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1452
1453 SafeIfaceArray<IMedium> children(this->getChildren());
1454 children.detachTo(ComSafeArrayOutArg(aChildren));
1455
1456 return S_OK;
1457}
1458
1459STDMETHODIMP Medium::COMGETTER(Base)(IMedium **aBase)
1460{
1461 CheckComArgOutPointerValid(aBase);
1462
1463 /* base() will do callers/locking */
1464
1465 getBase().queryInterfaceTo(aBase);
1466
1467 return S_OK;
1468}
1469
1470STDMETHODIMP Medium::COMGETTER(ReadOnly)(BOOL *aReadOnly)
1471{
1472 CheckComArgOutPointerValid(aReadOnly);
1473
1474 AutoCaller autoCaller(this);
1475 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1476
1477 /* isRadOnly() will do locking */
1478
1479 *aReadOnly = isReadOnly();
1480
1481 return S_OK;
1482}
1483
1484STDMETHODIMP Medium::COMGETTER(LogicalSize)(ULONG64 *aLogicalSize)
1485{
1486 CheckComArgOutPointerValid(aLogicalSize);
1487
1488 {
1489 AutoCaller autoCaller(this);
1490 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1491
1492 /* we access mParent */
1493 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1494
1495 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1496
1497 if (m->pParent.isNull())
1498 {
1499 *aLogicalSize = m->logicalSize;
1500
1501 return S_OK;
1502 }
1503 }
1504
1505 /* We assume that some backend may decide to return a meaningless value in
1506 * response to VDGetSize() for differencing hard disks and therefore
1507 * always ask the base hard disk ourselves. */
1508
1509 /* base() will do callers/locking */
1510
1511 return getBase()->COMGETTER(LogicalSize)(aLogicalSize);
1512}
1513
1514STDMETHODIMP Medium::COMGETTER(AutoReset)(BOOL *aAutoReset)
1515{
1516 CheckComArgOutPointerValid(aAutoReset);
1517
1518 AutoCaller autoCaller(this);
1519 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1520
1521 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1522
1523 if (m->pParent)
1524 *aAutoReset = FALSE;
1525
1526 *aAutoReset = m->autoReset;
1527
1528 return S_OK;
1529}
1530
1531STDMETHODIMP Medium::COMSETTER(AutoReset)(BOOL aAutoReset)
1532{
1533 AutoCaller autoCaller(this);
1534 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1535
1536 /* VirtualBox::saveSettings() needs a write lock */
1537 AutoMultiWriteLock2 alock(m->pVirtualBox, this COMMA_LOCKVAL_SRC_POS);
1538
1539 if (m->pParent.isNull())
1540 return setError(VBOX_E_NOT_SUPPORTED,
1541 tr("Hard disk '%s' is not differencing"),
1542 m->strLocationFull.raw());
1543
1544 if (m->autoReset != aAutoReset)
1545 {
1546 m->autoReset = aAutoReset;
1547
1548 return m->pVirtualBox->saveSettings();
1549 }
1550
1551 return S_OK;
1552}
1553STDMETHODIMP Medium::COMGETTER(LastAccessError)(BSTR *aLastAccessError)
1554{
1555 CheckComArgOutPointerValid(aLastAccessError);
1556
1557 AutoCaller autoCaller(this);
1558 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1559
1560 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1561
1562 m->strLastAccessError.cloneTo(aLastAccessError);
1563
1564 return S_OK;
1565}
1566
1567STDMETHODIMP Medium::COMGETTER(MachineIds)(ComSafeArrayOut(BSTR,aMachineIds))
1568{
1569 CheckComArgOutSafeArrayPointerValid(aMachineIds);
1570
1571 AutoCaller autoCaller(this);
1572 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1573
1574 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1575
1576 com::SafeArray<BSTR> machineIds;
1577
1578 if (m->backRefs.size() != 0)
1579 {
1580 machineIds.reset(m->backRefs.size());
1581
1582 size_t i = 0;
1583 for (BackRefList::const_iterator it = m->backRefs.begin();
1584 it != m->backRefs.end(); ++it, ++i)
1585 {
1586 it->machineId.toUtf16().detachTo(&machineIds[i]);
1587 }
1588 }
1589
1590 machineIds.detachTo(ComSafeArrayOutArg(aMachineIds));
1591
1592 return S_OK;
1593}
1594
1595STDMETHODIMP Medium::RefreshState(MediumState_T *aState)
1596{
1597 CheckComArgOutPointerValid(aState);
1598
1599 AutoCaller autoCaller(this);
1600 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1601
1602 /* queryInfo() locks this for writing. */
1603 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1604
1605 HRESULT rc = S_OK;
1606
1607 switch (m->state)
1608 {
1609 case MediumState_Created:
1610 case MediumState_Inaccessible:
1611 case MediumState_LockedRead:
1612 {
1613 rc = queryInfo();
1614 break;
1615 }
1616 default:
1617 break;
1618 }
1619
1620 *aState = m->state;
1621
1622 return rc;
1623}
1624
1625STDMETHODIMP Medium::GetSnapshotIds(IN_BSTR aMachineId,
1626 ComSafeArrayOut(BSTR, aSnapshotIds))
1627{
1628 CheckComArgExpr(aMachineId, Guid(aMachineId).isEmpty() == false);
1629 CheckComArgOutSafeArrayPointerValid(aSnapshotIds);
1630
1631 AutoCaller autoCaller(this);
1632 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1633
1634 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1635
1636 com::SafeArray<BSTR> snapshotIds;
1637
1638 Guid id(aMachineId);
1639 for (BackRefList::const_iterator it = m->backRefs.begin();
1640 it != m->backRefs.end(); ++it)
1641 {
1642 if (it->machineId == id)
1643 {
1644 size_t size = it->llSnapshotIds.size();
1645
1646 /* if the medium is attached to the machine in the current state, we
1647 * return its ID as the first element of the array */
1648 if (it->fInCurState)
1649 ++size;
1650
1651 if (size > 0)
1652 {
1653 snapshotIds.reset(size);
1654
1655 size_t j = 0;
1656 if (it->fInCurState)
1657 it->machineId.toUtf16().detachTo(&snapshotIds[j++]);
1658
1659 for (BackRef::GuidList::const_iterator jt = it->llSnapshotIds.begin();
1660 jt != it->llSnapshotIds.end();
1661 ++jt, ++j)
1662 {
1663 (*jt).toUtf16().detachTo(&snapshotIds[j]);
1664 }
1665 }
1666
1667 break;
1668 }
1669 }
1670
1671 snapshotIds.detachTo(ComSafeArrayOutArg(aSnapshotIds));
1672
1673 return S_OK;
1674}
1675
1676/**
1677 * @note @a aState may be NULL if the state value is not needed (only for
1678 * in-process calls).
1679 */
1680STDMETHODIMP Medium::LockRead(MediumState_T *aState)
1681{
1682 AutoCaller autoCaller(this);
1683 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1684
1685 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1686
1687 /* Wait for a concurrently running queryInfo() to complete */
1688 while (m->queryInfoRunning)
1689 {
1690 alock.leave();
1691 RTSemEventMultiWait(m->queryInfoSem, RT_INDEFINITE_WAIT);
1692 alock.enter();
1693 }
1694
1695 /* return the current state before */
1696 if (aState)
1697 *aState = m->state;
1698
1699 HRESULT rc = S_OK;
1700
1701 switch (m->state)
1702 {
1703 case MediumState_Created:
1704 case MediumState_Inaccessible:
1705 case MediumState_LockedRead:
1706 {
1707 ++m->readers;
1708
1709 ComAssertMsgBreak(m->readers != 0, ("Counter overflow"), rc = E_FAIL);
1710
1711 /* Remember pre-lock state */
1712 if (m->state != MediumState_LockedRead)
1713 m->preLockState = m->state;
1714
1715 LogFlowThisFunc(("Okay - prev state=%d readers=%d\n", m->state, m->readers));
1716 m->state = MediumState_LockedRead;
1717
1718 break;
1719 }
1720 default:
1721 {
1722 LogFlowThisFunc(("Failing - state=%d\n", m->state));
1723 rc = setStateError();
1724 break;
1725 }
1726 }
1727
1728 return rc;
1729}
1730
1731/**
1732 * @note @a aState may be NULL if the state value is not needed (only for
1733 * in-process calls).
1734 */
1735STDMETHODIMP Medium::UnlockRead(MediumState_T *aState)
1736{
1737 AutoCaller autoCaller(this);
1738 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1739
1740 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1741
1742 HRESULT rc = S_OK;
1743
1744 switch (m->state)
1745 {
1746 case MediumState_LockedRead:
1747 {
1748 Assert(m->readers != 0);
1749 --m->readers;
1750
1751 /* Reset the state after the last reader */
1752 if (m->readers == 0)
1753 m->state = m->preLockState;
1754
1755 LogFlowThisFunc(("new state=%d\n", m->state));
1756 break;
1757 }
1758 default:
1759 {
1760 LogFlowThisFunc(("Failing - state=%d\n", m->state));
1761 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
1762 tr("Medium '%s' is not locked for reading"),
1763 m->strLocationFull.raw());
1764 break;
1765 }
1766 }
1767
1768 /* return the current state after */
1769 if (aState)
1770 *aState = m->state;
1771
1772 return rc;
1773}
1774
1775/**
1776 * @note @a aState may be NULL if the state value is not needed (only for
1777 * in-process calls).
1778 */
1779STDMETHODIMP Medium::LockWrite(MediumState_T *aState)
1780{
1781 AutoCaller autoCaller(this);
1782 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1783
1784 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1785
1786 /* Wait for a concurrently running queryInfo() to complete */
1787 while (m->queryInfoRunning)
1788 {
1789 alock.leave();
1790 RTSemEventMultiWait(m->queryInfoSem, RT_INDEFINITE_WAIT);
1791 alock.enter();
1792 }
1793
1794 /* return the current state before */
1795 if (aState)
1796 *aState = m->state;
1797
1798 HRESULT rc = S_OK;
1799
1800 switch (m->state)
1801 {
1802 case MediumState_Created:
1803 case MediumState_Inaccessible:
1804 {
1805 m->preLockState = m->state;
1806
1807 LogFlowThisFunc(("Okay - prev state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
1808 m->state = MediumState_LockedWrite;
1809 break;
1810 }
1811 default:
1812 {
1813 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
1814 rc = setStateError();
1815 break;
1816 }
1817 }
1818
1819 return rc;
1820}
1821
1822/**
1823 * @note @a aState may be NULL if the state value is not needed (only for
1824 * in-process calls).
1825 */
1826STDMETHODIMP Medium::UnlockWrite(MediumState_T *aState)
1827{
1828 AutoCaller autoCaller(this);
1829 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1830
1831 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1832
1833 HRESULT rc = S_OK;
1834
1835 switch (m->state)
1836 {
1837 case MediumState_LockedWrite:
1838 {
1839 m->state = m->preLockState;
1840 LogFlowThisFunc(("new state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
1841 break;
1842 }
1843 default:
1844 {
1845 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, getLocationFull().c_str()));
1846 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
1847 tr("Medium '%s' is not locked for writing"),
1848 m->strLocationFull.raw());
1849 break;
1850 }
1851 }
1852
1853 /* return the current state after */
1854 if (aState)
1855 *aState = m->state;
1856
1857 return rc;
1858}
1859
1860STDMETHODIMP Medium::Close()
1861{
1862 // we're accessing parent/child and backrefs, so lock the tree first, then ourselves
1863 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
1864 this->lockHandle()
1865 COMMA_LOCKVAL_SRC_POS);
1866
1867 bool wasCreated = true;
1868 bool fNeedsSaveSettings = false;
1869
1870 switch (m->state)
1871 {
1872 case MediumState_NotCreated:
1873 wasCreated = false;
1874 break;
1875 case MediumState_Created:
1876 case MediumState_Inaccessible:
1877 break;
1878 default:
1879 return setStateError();
1880 }
1881
1882 if (m->backRefs.size() != 0)
1883 return setError(VBOX_E_OBJECT_IN_USE,
1884 tr("Medium '%s' is attached to %d virtual machines"),
1885 m->strLocationFull.raw(), m->backRefs.size());
1886
1887 /* perform extra media-dependent close checks */
1888 HRESULT rc = canClose();
1889 if (FAILED(rc)) return rc;
1890
1891 if (wasCreated)
1892 {
1893 /* remove from the list of known media before performing actual
1894 * uninitialization (to keep the media registry consistent on
1895 * failure to do so) */
1896 rc = unregisterWithVirtualBox(&fNeedsSaveSettings);
1897 if (FAILED(rc)) return rc;
1898 }
1899
1900 // make a copy of VirtualBox pointer which gets nulled by uninit()
1901 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
1902
1903 /* Keep the locks held until after uninit, as otherwise the consistency
1904 * of the medium tree cannot be guaranteed. */
1905 uninit();
1906
1907 multilock.release();
1908
1909 if (fNeedsSaveSettings)
1910 {
1911 AutoWriteLock vboxlock(pVirtualBox COMMA_LOCKVAL_SRC_POS);
1912 pVirtualBox->saveSettings();
1913 }
1914
1915 return S_OK;
1916}
1917
1918STDMETHODIMP Medium::GetProperty(IN_BSTR aName, BSTR *aValue)
1919{
1920 CheckComArgStrNotEmptyOrNull(aName);
1921 CheckComArgOutPointerValid(aValue);
1922
1923 AutoCaller autoCaller(this);
1924 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1925
1926 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1927
1928 Data::PropertyMap::const_iterator it = m->properties.find(Bstr(aName));
1929 if (it == m->properties.end())
1930 return setError(VBOX_E_OBJECT_NOT_FOUND,
1931 tr("Property '%ls' does not exist"), aName);
1932
1933 it->second.cloneTo(aValue);
1934
1935 return S_OK;
1936}
1937
1938STDMETHODIMP Medium::SetProperty(IN_BSTR aName, IN_BSTR aValue)
1939{
1940 CheckComArgStrNotEmptyOrNull(aName);
1941
1942 AutoCaller autoCaller(this);
1943 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1944
1945 /* VirtualBox::saveSettings() needs a write lock */
1946 AutoMultiWriteLock2 alock(m->pVirtualBox, this COMMA_LOCKVAL_SRC_POS);
1947
1948 switch (m->state)
1949 {
1950 case MediumState_Created:
1951 case MediumState_Inaccessible:
1952 break;
1953 default:
1954 return setStateError();
1955 }
1956
1957 Data::PropertyMap::iterator it = m->properties.find(Bstr(aName));
1958 if (it == m->properties.end())
1959 return setError(VBOX_E_OBJECT_NOT_FOUND,
1960 tr("Property '%ls' does not exist"),
1961 aName);
1962
1963 if (aValue && !*aValue)
1964 it->second = (const char *)NULL;
1965 else
1966 it->second = aValue;
1967
1968 HRESULT rc = m->pVirtualBox->saveSettings();
1969
1970 return rc;
1971}
1972
1973STDMETHODIMP Medium::GetProperties(IN_BSTR aNames,
1974 ComSafeArrayOut(BSTR, aReturnNames),
1975 ComSafeArrayOut(BSTR, aReturnValues))
1976{
1977 CheckComArgOutSafeArrayPointerValid(aReturnNames);
1978 CheckComArgOutSafeArrayPointerValid(aReturnValues);
1979
1980 AutoCaller autoCaller(this);
1981 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1982
1983 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1984
1985 /// @todo make use of aNames according to the documentation
1986 NOREF(aNames);
1987
1988 com::SafeArray<BSTR> names(m->properties.size());
1989 com::SafeArray<BSTR> values(m->properties.size());
1990 size_t i = 0;
1991
1992 for (Data::PropertyMap::const_iterator it = m->properties.begin();
1993 it != m->properties.end();
1994 ++it)
1995 {
1996 it->first.cloneTo(&names[i]);
1997 it->second.cloneTo(&values[i]);
1998 ++i;
1999 }
2000
2001 names.detachTo(ComSafeArrayOutArg(aReturnNames));
2002 values.detachTo(ComSafeArrayOutArg(aReturnValues));
2003
2004 return S_OK;
2005}
2006
2007STDMETHODIMP Medium::SetProperties(ComSafeArrayIn(IN_BSTR, aNames),
2008 ComSafeArrayIn(IN_BSTR, aValues))
2009{
2010 CheckComArgSafeArrayNotNull(aNames);
2011 CheckComArgSafeArrayNotNull(aValues);
2012
2013 AutoCaller autoCaller(this);
2014 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2015
2016 /* VirtualBox::saveSettings() needs a write lock */
2017 AutoMultiWriteLock2 alock(m->pVirtualBox, this COMMA_LOCKVAL_SRC_POS);
2018
2019 com::SafeArray<IN_BSTR> names(ComSafeArrayInArg(aNames));
2020 com::SafeArray<IN_BSTR> values(ComSafeArrayInArg(aValues));
2021
2022 /* first pass: validate names */
2023 for (size_t i = 0;
2024 i < names.size();
2025 ++i)
2026 {
2027 if (m->properties.find(Bstr(names[i])) == m->properties.end())
2028 return setError(VBOX_E_OBJECT_NOT_FOUND,
2029 tr("Property '%ls' does not exist"), names[i]);
2030 }
2031
2032 /* second pass: assign */
2033 for (size_t i = 0;
2034 i < names.size();
2035 ++i)
2036 {
2037 Data::PropertyMap::iterator it = m->properties.find(Bstr(names[i]));
2038 AssertReturn(it != m->properties.end(), E_FAIL);
2039
2040 if (values[i] && !*values[i])
2041 it->second = (const char *)NULL;
2042 else
2043 it->second = values[i];
2044 }
2045
2046 HRESULT rc = m->pVirtualBox->saveSettings();
2047
2048 return rc;
2049}
2050
2051STDMETHODIMP Medium::CreateBaseStorage(ULONG64 aLogicalSize,
2052 MediumVariant_T aVariant,
2053 IProgress **aProgress)
2054{
2055 CheckComArgOutPointerValid(aProgress);
2056
2057 AutoCaller autoCaller(this);
2058 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2059
2060 HRESULT rc = S_OK;
2061 ComObjPtr <Progress> pProgress;
2062 Medium::Task *pTask = NULL;
2063
2064 try
2065 {
2066 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2067
2068 aVariant = (MediumVariant_T)((unsigned)aVariant & (unsigned)~MediumVariant_Diff);
2069 if ( !(aVariant & MediumVariant_Fixed)
2070 && !(m->formatObj->capabilities() & MediumFormatCapabilities_CreateDynamic))
2071 throw setError(VBOX_E_NOT_SUPPORTED,
2072 tr("Hard disk format '%s' does not support dynamic storage creation"),
2073 m->strFormat.raw());
2074 if ( (aVariant & MediumVariant_Fixed)
2075 && !(m->formatObj->capabilities() & MediumFormatCapabilities_CreateDynamic))
2076 throw setError(VBOX_E_NOT_SUPPORTED,
2077 tr("Hard disk format '%s' does not support fixed storage creation"),
2078 m->strFormat.raw());
2079
2080 if (m->state != MediumState_NotCreated)
2081 throw setStateError();
2082
2083 pProgress.createObject();
2084 rc = pProgress->init(m->pVirtualBox,
2085 static_cast<IMedium*>(this),
2086 (aVariant & MediumVariant_Fixed)
2087 ? BstrFmt(tr("Creating fixed hard disk storage unit '%s'"), m->strLocationFull.raw())
2088 : BstrFmt(tr("Creating dynamic hard disk storage unit '%s'"), m->strLocationFull.raw()),
2089 TRUE /* aCancelable */);
2090 if (FAILED(rc))
2091 throw rc;
2092
2093 /* setup task object to carry out the operation asynchronously */
2094 pTask = new Medium::CreateBaseTask(this, pProgress, aLogicalSize,
2095 aVariant);
2096 rc = pTask->rc();
2097 AssertComRC(rc);
2098 if (FAILED(rc))
2099 throw rc;
2100
2101 m->state = MediumState_Creating;
2102 }
2103 catch (HRESULT aRC) { rc = aRC; }
2104
2105 if (SUCCEEDED(rc))
2106 {
2107 rc = startThread(pTask);
2108
2109 if (SUCCEEDED(rc))
2110 pProgress.queryInterfaceTo(aProgress);
2111 }
2112 else if (pTask != NULL)
2113 delete pTask;
2114
2115 return rc;
2116}
2117
2118STDMETHODIMP Medium::DeleteStorage(IProgress **aProgress)
2119{
2120 CheckComArgOutPointerValid(aProgress);
2121
2122 AutoCaller autoCaller(this);
2123 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2124
2125 ComObjPtr <Progress> pProgress;
2126
2127 HRESULT rc = deleteStorage(&pProgress, false /* aWait */,
2128 NULL /* pfNeedsSaveSettings */);
2129 if (SUCCEEDED(rc))
2130 pProgress.queryInterfaceTo(aProgress);
2131
2132 return rc;
2133}
2134
2135STDMETHODIMP Medium::CreateDiffStorage(IMedium *aTarget,
2136 MediumVariant_T aVariant,
2137 IProgress **aProgress)
2138{
2139 CheckComArgNotNull(aTarget);
2140 CheckComArgOutPointerValid(aProgress);
2141
2142 AutoCaller autoCaller(this);
2143 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2144
2145 ComObjPtr<Medium> diff = static_cast<Medium*>(aTarget);
2146
2147 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2148
2149 if (m->type == MediumType_Writethrough)
2150 return setError(E_FAIL,
2151 tr("Hard disk '%s' is Writethrough"),
2152 m->strLocationFull.raw());
2153
2154 /* Apply the normal locking logic to the entire chain. */
2155 MediumLockList *pMediumLockList(new MediumLockList());
2156 HRESULT rc = diff->createMediumLockList(true, this, *pMediumLockList);
2157 if (FAILED(rc))
2158 {
2159 delete pMediumLockList;
2160 return rc;
2161 }
2162
2163 ComObjPtr <Progress> pProgress;
2164
2165 rc = createDiffStorage(diff, aVariant, pMediumLockList, &pProgress,
2166 false /* aWait */, NULL /* pfNeedsSaveSettings*/);
2167 if (FAILED(rc))
2168 delete pMediumLockList;
2169 else
2170 pProgress.queryInterfaceTo(aProgress);
2171
2172 return rc;
2173}
2174
2175STDMETHODIMP Medium::MergeTo(IMedium *aTarget, IProgress **aProgress)
2176{
2177 CheckComArgNotNull(aTarget);
2178 CheckComArgOutPointerValid(aProgress);
2179 ComAssertRet(aTarget != this, E_INVALIDARG);
2180
2181 AutoCaller autoCaller(this);
2182 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2183
2184 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aTarget);
2185
2186 bool fMergeForward = false;
2187 ComObjPtr<Medium> pParentForTarget;
2188 MediaList childrenToReparent;
2189 MediumLockList *pMediumLockList = NULL;
2190
2191 HRESULT rc = S_OK;
2192
2193 rc = prepareMergeTo(pTarget, NULL, NULL, true, fMergeForward,
2194 pParentForTarget, childrenToReparent, pMediumLockList);
2195 if (FAILED(rc)) return rc;
2196
2197 ComObjPtr <Progress> pProgress;
2198
2199 rc = mergeTo(pTarget, fMergeForward, pParentForTarget, childrenToReparent,
2200 pMediumLockList, &pProgress, false /* aWait */,
2201 NULL /* pfNeedsSaveSettings */);
2202 if (FAILED(rc))
2203 cancelMergeTo(childrenToReparent, pMediumLockList);
2204 else
2205 pProgress.queryInterfaceTo(aProgress);
2206
2207 return rc;
2208}
2209
2210STDMETHODIMP Medium::CloneTo(IMedium *aTarget,
2211 MediumVariant_T aVariant,
2212 IMedium *aParent,
2213 IProgress **aProgress)
2214{
2215 CheckComArgNotNull(aTarget);
2216 CheckComArgOutPointerValid(aProgress);
2217 ComAssertRet(aTarget != this, E_INVALIDARG);
2218
2219 AutoCaller autoCaller(this);
2220 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2221
2222 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aTarget);
2223 ComObjPtr<Medium> pParent;
2224 if (aParent)
2225 pParent = static_cast<Medium*>(aParent);
2226
2227 HRESULT rc = S_OK;
2228 ComObjPtr<Progress> pProgress;
2229 Medium::Task *pTask = NULL;
2230
2231 try
2232 {
2233 // locking: we need the tree lock first because we access parent pointers
2234 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2235 // and we need to write-lock the images involved
2236 AutoMultiWriteLock3 alock(this, pTarget, pParent COMMA_LOCKVAL_SRC_POS);
2237
2238 if ( pTarget->m->state != MediumState_NotCreated
2239 && pTarget->m->state != MediumState_Created)
2240 throw pTarget->setStateError();
2241
2242 /* Build the source lock list. */
2243 MediumLockList *pSourceMediumLockList(new MediumLockList());
2244 rc = createMediumLockList(false, NULL,
2245 *pSourceMediumLockList);
2246 if (FAILED(rc))
2247 {
2248 delete pSourceMediumLockList;
2249 throw rc;
2250 }
2251
2252 /* Build the target lock list (including the to-be parent chain). */
2253 MediumLockList *pTargetMediumLockList(new MediumLockList());
2254 rc = pTarget->createMediumLockList(true, pParent,
2255 *pTargetMediumLockList);
2256 if (FAILED(rc))
2257 {
2258 delete pSourceMediumLockList;
2259 delete pTargetMediumLockList;
2260 throw rc;
2261 }
2262
2263 rc = pSourceMediumLockList->Lock();
2264 if (FAILED(rc))
2265 {
2266 delete pSourceMediumLockList;
2267 delete pTargetMediumLockList;
2268 throw setError(rc,
2269 tr("Failed to lock source media '%ls'"),
2270 getLocationFull().raw());
2271 }
2272 rc = pTargetMediumLockList->Lock();
2273 if (FAILED(rc))
2274 {
2275 delete pSourceMediumLockList;
2276 delete pTargetMediumLockList;
2277 throw setError(rc,
2278 tr("Failed to lock target media '%ls'"),
2279 pTarget->getLocationFull().raw());
2280 }
2281
2282 pProgress.createObject();
2283 rc = pProgress->init(m->pVirtualBox,
2284 static_cast <IMedium *>(this),
2285 BstrFmt(tr("Creating clone hard disk '%s'"), pTarget->m->strLocationFull.raw()),
2286 TRUE /* aCancelable */);
2287 if (FAILED(rc))
2288 {
2289 delete pSourceMediumLockList;
2290 delete pTargetMediumLockList;
2291 throw rc;
2292 }
2293
2294 /* setup task object to carry out the operation asynchronously */
2295 pTask = new Medium::CloneTask(this, pProgress, pTarget, aVariant,
2296 pParent, pSourceMediumLockList,
2297 pTargetMediumLockList);
2298 rc = pTask->rc();
2299 AssertComRC(rc);
2300 if (FAILED(rc))
2301 throw rc;
2302
2303 if (pTarget->m->state == MediumState_NotCreated)
2304 pTarget->m->state = MediumState_Creating;
2305 }
2306 catch (HRESULT aRC) { rc = aRC; }
2307
2308 if (SUCCEEDED(rc))
2309 {
2310 rc = startThread(pTask);
2311
2312 if (SUCCEEDED(rc))
2313 pProgress.queryInterfaceTo(aProgress);
2314 }
2315 else if (pTask != NULL)
2316 delete pTask;
2317
2318 return rc;
2319}
2320
2321STDMETHODIMP Medium::Compact(IProgress **aProgress)
2322{
2323 CheckComArgOutPointerValid(aProgress);
2324
2325 AutoCaller autoCaller(this);
2326 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2327
2328 HRESULT rc = S_OK;
2329 ComObjPtr <Progress> pProgress;
2330 Medium::Task *pTask = NULL;
2331
2332 try
2333 {
2334 /* We need to lock both the current object, and the tree lock (would
2335 * cause a lock order violation otherwise) for createMediumLockList. */
2336 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
2337 this->lockHandle()
2338 COMMA_LOCKVAL_SRC_POS);
2339
2340 /* Build the medium lock list. */
2341 MediumLockList *pMediumLockList(new MediumLockList());
2342 rc = createMediumLockList(true, NULL,
2343 *pMediumLockList);
2344 if (FAILED(rc))
2345 {
2346 delete pMediumLockList;
2347 throw rc;
2348 }
2349
2350 rc = pMediumLockList->Lock();
2351 if (FAILED(rc))
2352 {
2353 delete pMediumLockList;
2354 throw setError(rc,
2355 tr("Failed to lock media when compacting '%ls'"),
2356 getLocationFull().raw());
2357 }
2358
2359 pProgress.createObject();
2360 rc = pProgress->init(m->pVirtualBox,
2361 static_cast <IMedium *>(this),
2362 BstrFmt(tr("Compacting hard disk '%s'"), m->strLocationFull.raw()),
2363 TRUE /* aCancelable */);
2364 if (FAILED(rc))
2365 {
2366 delete pMediumLockList;
2367 throw rc;
2368 }
2369
2370 /* setup task object to carry out the operation asynchronously */
2371 pTask = new Medium::CompactTask(this, pProgress, pMediumLockList);
2372 rc = pTask->rc();
2373 AssertComRC(rc);
2374 if (FAILED(rc))
2375 throw rc;
2376 }
2377 catch (HRESULT aRC) { rc = aRC; }
2378
2379 if (SUCCEEDED(rc))
2380 {
2381 rc = startThread(pTask);
2382
2383 if (SUCCEEDED(rc))
2384 pProgress.queryInterfaceTo(aProgress);
2385 }
2386 else if (pTask != NULL)
2387 delete pTask;
2388
2389 return rc;
2390}
2391
2392STDMETHODIMP Medium::Resize(ULONG64 aLogicalSize, IProgress **aProgress)
2393{
2394 CheckComArgOutPointerValid(aProgress);
2395
2396 AutoCaller autoCaller(this);
2397 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2398
2399 NOREF(aLogicalSize);
2400 NOREF(aProgress);
2401 ReturnComNotImplemented();
2402}
2403
2404STDMETHODIMP Medium::Reset(IProgress **aProgress)
2405{
2406 CheckComArgOutPointerValid(aProgress);
2407
2408 AutoCaller autoCaller(this);
2409 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2410
2411 HRESULT rc = S_OK;
2412 ComObjPtr <Progress> pProgress;
2413 Medium::Task *pTask = NULL;
2414
2415 try
2416 {
2417 /* canClose() needs the tree lock */
2418 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
2419 this->lockHandle()
2420 COMMA_LOCKVAL_SRC_POS);
2421
2422 LogFlowThisFunc(("ENTER for medium %s\n", m->strLocationFull.c_str()));
2423
2424 if (m->pParent.isNull())
2425 throw setError(VBOX_E_NOT_SUPPORTED,
2426 tr("Hard disk '%s' is not differencing"),
2427 m->strLocationFull.raw());
2428
2429 rc = canClose();
2430 if (FAILED(rc))
2431 throw rc;
2432
2433 /* Build the medium lock list. */
2434 MediumLockList *pMediumLockList(new MediumLockList());
2435 rc = createMediumLockList(true, NULL,
2436 *pMediumLockList);
2437 if (FAILED(rc))
2438 {
2439 delete pMediumLockList;
2440 throw rc;
2441 }
2442
2443 rc = pMediumLockList->Lock();
2444 if (FAILED(rc))
2445 {
2446 delete pMediumLockList;
2447 throw setError(rc,
2448 tr("Failed to lock media when resetting '%ls'"),
2449 getLocationFull().raw());
2450 }
2451
2452 pProgress.createObject();
2453 rc = pProgress->init(m->pVirtualBox,
2454 static_cast<IMedium*>(this),
2455 BstrFmt(tr("Resetting differencing hard disk '%s'"), m->strLocationFull.raw()),
2456 FALSE /* aCancelable */);
2457 if (FAILED(rc))
2458 throw rc;
2459
2460 /* setup task object to carry out the operation asynchronously */
2461 pTask = new Medium::ResetTask(this, pProgress, pMediumLockList);
2462 rc = pTask->rc();
2463 AssertComRC(rc);
2464 if (FAILED(rc))
2465 throw rc;
2466 }
2467 catch (HRESULT aRC) { rc = aRC; }
2468
2469 if (SUCCEEDED(rc))
2470 {
2471 rc = startThread(pTask);
2472
2473 if (SUCCEEDED(rc))
2474 pProgress.queryInterfaceTo(aProgress);
2475 }
2476 else
2477 {
2478 /* Note: on success, the task will unlock this */
2479 {
2480 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2481 HRESULT rc2 = UnlockWrite(NULL);
2482 AssertComRC(rc2);
2483 }
2484 if (pTask != NULL)
2485 delete pTask;
2486 }
2487
2488 LogFlowThisFunc(("LEAVE, rc=%Rhrc\n", rc));
2489
2490 return rc;
2491}
2492
2493////////////////////////////////////////////////////////////////////////////////
2494//
2495// Medium internal methods
2496//
2497////////////////////////////////////////////////////////////////////////////////
2498
2499/**
2500 * Internal method to return the medium's parent medium. Must have caller + locking!
2501 * @return
2502 */
2503const ComObjPtr<Medium>& Medium::getParent() const
2504{
2505 return m->pParent;
2506}
2507
2508/**
2509 * Internal method to return the medium's list of child media. Must have caller + locking!
2510 * @return
2511 */
2512const MediaList& Medium::getChildren() const
2513{
2514 return m->llChildren;
2515}
2516
2517/**
2518 * Internal method to return the medium's GUID. Must have caller + locking!
2519 * @return
2520 */
2521const Guid& Medium::getId() const
2522{
2523 return m->id;
2524}
2525
2526/**
2527 * Internal method to return the medium's GUID. Must have caller + locking!
2528 * @return
2529 */
2530MediumState_T Medium::getState() const
2531{
2532 return m->state;
2533}
2534
2535/**
2536 * Internal method to return the medium's location. Must have caller + locking!
2537 * @return
2538 */
2539const Utf8Str& Medium::getLocation() const
2540{
2541 return m->strLocation;
2542}
2543
2544/**
2545 * Internal method to return the medium's full location. Must have caller + locking!
2546 * @return
2547 */
2548const Utf8Str& Medium::getLocationFull() const
2549{
2550 return m->strLocationFull;
2551}
2552
2553/**
2554 * Internal method to return the medium's size. Must have caller + locking!
2555 * @return
2556 */
2557uint64_t Medium::getSize() const
2558{
2559 return m->size;
2560}
2561
2562/**
2563 * Adds the given machine and optionally the snapshot to the list of the objects
2564 * this image is attached to.
2565 *
2566 * @param aMachineId Machine ID.
2567 * @param aSnapshotId Snapshot ID; when non-empty, adds a snapshot attachment.
2568 */
2569HRESULT Medium::attachTo(const Guid &aMachineId,
2570 const Guid &aSnapshotId /*= Guid::Empty*/)
2571{
2572 AssertReturn(!aMachineId.isEmpty(), E_FAIL);
2573
2574 LogFlowThisFunc(("ENTER, aMachineId: {%RTuuid}, aSnapshotId: {%RTuuid}\n", aMachineId.raw(), aSnapshotId.raw()));
2575
2576 AutoCaller autoCaller(this);
2577 AssertComRCReturnRC(autoCaller.rc());
2578
2579 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2580
2581 switch (m->state)
2582 {
2583 case MediumState_Created:
2584 case MediumState_Inaccessible:
2585 case MediumState_LockedRead:
2586 case MediumState_LockedWrite:
2587 break;
2588
2589 default:
2590 return setStateError();
2591 }
2592
2593 if (m->numCreateDiffTasks > 0)
2594 return setError(E_FAIL,
2595 tr("Cannot attach hard disk '%s' {%RTuuid}: %u differencing child hard disk(s) are being created"),
2596 m->strLocationFull.raw(),
2597 m->id.raw(),
2598 m->numCreateDiffTasks);
2599
2600 BackRefList::iterator it = std::find_if(m->backRefs.begin(),
2601 m->backRefs.end(),
2602 BackRef::EqualsTo(aMachineId));
2603 if (it == m->backRefs.end())
2604 {
2605 BackRef ref(aMachineId, aSnapshotId);
2606 m->backRefs.push_back(ref);
2607
2608 return S_OK;
2609 }
2610
2611 // if the caller has not supplied a snapshot ID, then we're attaching
2612 // to a machine a medium which represents the machine's current state,
2613 // so set the flag
2614 if (aSnapshotId.isEmpty())
2615 {
2616 /* sanity: no duplicate attachments */
2617 AssertReturn(!it->fInCurState, E_FAIL);
2618 it->fInCurState = true;
2619
2620 return S_OK;
2621 }
2622
2623 // otherwise: a snapshot medium is being attached
2624
2625 /* sanity: no duplicate attachments */
2626 for (BackRef::GuidList::const_iterator jt = it->llSnapshotIds.begin();
2627 jt != it->llSnapshotIds.end();
2628 ++jt)
2629 {
2630 const Guid &idOldSnapshot = *jt;
2631
2632 if (idOldSnapshot == aSnapshotId)
2633 {
2634#ifdef DEBUG
2635 dumpBackRefs();
2636#endif
2637 return setError(E_FAIL,
2638 tr("Cannot attach medium '%s' {%RTuuid} from snapshot '%RTuuid': medium is already in use by this snapshot!"),
2639 m->strLocationFull.raw(),
2640 m->id.raw(),
2641 aSnapshotId.raw(),
2642 idOldSnapshot.raw());
2643 }
2644 }
2645
2646 it->llSnapshotIds.push_back(aSnapshotId);
2647 it->fInCurState = false;
2648
2649 LogFlowThisFuncLeave();
2650
2651 return S_OK;
2652}
2653
2654/**
2655 * Removes the given machine and optionally the snapshot from the list of the
2656 * objects this image is attached to.
2657 *
2658 * @param aMachineId Machine ID.
2659 * @param aSnapshotId Snapshot ID; when non-empty, removes the snapshot
2660 * attachment.
2661 */
2662HRESULT Medium::detachFrom(const Guid &aMachineId,
2663 const Guid &aSnapshotId /*= Guid::Empty*/)
2664{
2665 AssertReturn(!aMachineId.isEmpty(), E_FAIL);
2666
2667 AutoCaller autoCaller(this);
2668 AssertComRCReturnRC(autoCaller.rc());
2669
2670 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2671
2672 BackRefList::iterator it =
2673 std::find_if(m->backRefs.begin(), m->backRefs.end(),
2674 BackRef::EqualsTo(aMachineId));
2675 AssertReturn(it != m->backRefs.end(), E_FAIL);
2676
2677 if (aSnapshotId.isEmpty())
2678 {
2679 /* remove the current state attachment */
2680 it->fInCurState = false;
2681 }
2682 else
2683 {
2684 /* remove the snapshot attachment */
2685 BackRef::GuidList::iterator jt =
2686 std::find(it->llSnapshotIds.begin(), it->llSnapshotIds.end(), aSnapshotId);
2687
2688 AssertReturn(jt != it->llSnapshotIds.end(), E_FAIL);
2689 it->llSnapshotIds.erase(jt);
2690 }
2691
2692 /* if the backref becomes empty, remove it */
2693 if (it->fInCurState == false && it->llSnapshotIds.size() == 0)
2694 m->backRefs.erase(it);
2695
2696 return S_OK;
2697}
2698
2699/**
2700 * Internal method to return the medium's list of backrefs. Must have caller + locking!
2701 * @return
2702 */
2703const Guid* Medium::getFirstMachineBackrefId() const
2704{
2705 if (!m->backRefs.size())
2706 return NULL;
2707
2708 return &m->backRefs.front().machineId;
2709}
2710
2711const Guid* Medium::getFirstMachineBackrefSnapshotId() const
2712{
2713 if (!m->backRefs.size())
2714 return NULL;
2715
2716 const BackRef &ref = m->backRefs.front();
2717 if (!ref.llSnapshotIds.size())
2718 return NULL;
2719
2720 return &ref.llSnapshotIds.front();
2721}
2722
2723#ifdef DEBUG
2724/**
2725 * Debugging helper that gets called after VirtualBox initialization that writes all
2726 * machine backreferences to the debug log.
2727 */
2728void Medium::dumpBackRefs()
2729{
2730 AutoCaller autoCaller(this);
2731 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2732
2733 LogFlowThisFunc(("Dumping backrefs for medium '%s':\n", m->strLocationFull.raw()));
2734
2735 for (BackRefList::iterator it2 = m->backRefs.begin();
2736 it2 != m->backRefs.end();
2737 ++it2)
2738 {
2739 const BackRef &ref = *it2;
2740 LogFlowThisFunc((" Backref from machine {%RTuuid} (fInCurState: %d)\n", ref.machineId.raw(), ref.fInCurState));
2741
2742 for (BackRef::GuidList::const_iterator jt2 = it2->llSnapshotIds.begin();
2743 jt2 != it2->llSnapshotIds.end();
2744 ++jt2)
2745 {
2746 const Guid &id = *jt2;
2747 LogFlowThisFunc((" Backref from snapshot {%RTuuid}\n", id.raw()));
2748 }
2749 }
2750}
2751#endif
2752
2753/**
2754 * Checks if the given change of \a aOldPath to \a aNewPath affects the location
2755 * of this media and updates it if necessary to reflect the new location.
2756 *
2757 * @param aOldPath Old path (full).
2758 * @param aNewPath New path (full).
2759 *
2760 * @note Locks this object for writing.
2761 */
2762HRESULT Medium::updatePath(const char *aOldPath, const char *aNewPath)
2763{
2764 AssertReturn(aOldPath, E_FAIL);
2765 AssertReturn(aNewPath, E_FAIL);
2766
2767 AutoCaller autoCaller(this);
2768 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2769
2770 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2771
2772 LogFlowThisFunc(("locationFull.before='%s'\n", m->strLocationFull.raw()));
2773
2774 const char *pcszMediumPath = m->strLocationFull.c_str();
2775
2776 if (RTPathStartsWith(pcszMediumPath, aOldPath))
2777 {
2778 Utf8Str newPath = Utf8StrFmt("%s%s",
2779 aNewPath,
2780 pcszMediumPath + strlen(aOldPath));
2781 Utf8Str path = newPath;
2782 m->pVirtualBox->calculateRelativePath(path, path);
2783 unconst(m->strLocationFull) = newPath;
2784 unconst(m->strLocation) = path;
2785
2786 LogFlowThisFunc(("locationFull.after='%s'\n", m->strLocationFull.raw()));
2787 }
2788
2789 return S_OK;
2790}
2791
2792/**
2793 * Checks if the given change of \a aOldPath to \a aNewPath affects the location
2794 * of this hard disk or any its child and updates the paths if necessary to
2795 * reflect the new location.
2796 *
2797 * @param aOldPath Old path (full).
2798 * @param aNewPath New path (full).
2799 *
2800 * @note Locks the medium tree for reading, this object and all children for writing.
2801 */
2802void Medium::updatePaths(const char *aOldPath, const char *aNewPath)
2803{
2804 AssertReturnVoid(aOldPath);
2805 AssertReturnVoid(aNewPath);
2806
2807 AutoCaller autoCaller(this);
2808 AssertComRCReturnVoid(autoCaller.rc());
2809
2810 /* we access children() */
2811 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2812
2813 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2814
2815 updatePath(aOldPath, aNewPath);
2816
2817 /* update paths of all children */
2818 for (MediaList::const_iterator it = getChildren().begin();
2819 it != getChildren().end();
2820 ++it)
2821 {
2822 (*it)->updatePaths(aOldPath, aNewPath);
2823 }
2824}
2825
2826/**
2827 * Returns the base hard disk of the hard disk chain this hard disk is part of.
2828 *
2829 * The base hard disk is found by walking up the parent-child relationship axis.
2830 * If the hard disk doesn't have a parent (i.e. it's a base hard disk), it
2831 * returns itself in response to this method.
2832 *
2833 * @param aLevel Where to store the number of ancestors of this hard disk
2834 * (zero for the base), may be @c NULL.
2835 *
2836 * @note Locks medium tree for reading.
2837 */
2838ComObjPtr<Medium> Medium::getBase(uint32_t *aLevel /*= NULL*/)
2839{
2840 ComObjPtr<Medium> pBase;
2841 uint32_t level;
2842
2843 AutoCaller autoCaller(this);
2844 AssertReturn(autoCaller.isOk(), pBase);
2845
2846 /* we access mParent */
2847 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2848
2849 pBase = this;
2850 level = 0;
2851
2852 if (m->pParent)
2853 {
2854 for (;;)
2855 {
2856 AutoCaller baseCaller(pBase);
2857 AssertReturn(baseCaller.isOk(), pBase);
2858
2859 if (pBase->m->pParent.isNull())
2860 break;
2861
2862 pBase = pBase->m->pParent;
2863 ++level;
2864 }
2865 }
2866
2867 if (aLevel != NULL)
2868 *aLevel = level;
2869
2870 return pBase;
2871}
2872
2873/**
2874 * Returns @c true if this hard disk cannot be modified because it has
2875 * dependants (children) or is part of the snapshot. Related to the hard disk
2876 * type and posterity, not to the current media state.
2877 *
2878 * @note Locks this object and medium tree for reading.
2879 */
2880bool Medium::isReadOnly()
2881{
2882 AutoCaller autoCaller(this);
2883 AssertComRCReturn(autoCaller.rc(), false);
2884
2885 /* we access children */
2886 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2887
2888 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2889
2890 switch (m->type)
2891 {
2892 case MediumType_Normal:
2893 {
2894 if (getChildren().size() != 0)
2895 return true;
2896
2897 for (BackRefList::const_iterator it = m->backRefs.begin();
2898 it != m->backRefs.end(); ++it)
2899 if (it->llSnapshotIds.size() != 0)
2900 return true;
2901
2902 return false;
2903 }
2904 case MediumType_Immutable:
2905 {
2906 return true;
2907 }
2908 case MediumType_Writethrough:
2909 {
2910 return false;
2911 }
2912 default:
2913 break;
2914 }
2915
2916 AssertFailedReturn(false);
2917}
2918
2919/**
2920 * Saves hard disk data by appending a new <HardDisk> child node to the given
2921 * parent node which can be either <HardDisks> or <HardDisk>.
2922 *
2923 * @param data Settings struct to be updated.
2924 *
2925 * @note Locks this object, medium tree and children for reading.
2926 */
2927HRESULT Medium::saveSettings(settings::Medium &data)
2928{
2929 AutoCaller autoCaller(this);
2930 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2931
2932 /* we access mParent */
2933 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2934
2935 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2936
2937 data.uuid = m->id;
2938 data.strLocation = m->strLocation;
2939 data.strFormat = m->strFormat;
2940
2941 /* optional, only for diffs, default is false */
2942 if (m->pParent)
2943 data.fAutoReset = !!m->autoReset;
2944 else
2945 data.fAutoReset = false;
2946
2947 /* optional */
2948 data.strDescription = m->strDescription;
2949
2950 /* optional properties */
2951 data.properties.clear();
2952 for (Data::PropertyMap::const_iterator it = m->properties.begin();
2953 it != m->properties.end();
2954 ++it)
2955 {
2956 /* only save properties that have non-default values */
2957 if (!it->second.isEmpty())
2958 {
2959 Utf8Str name = it->first;
2960 Utf8Str value = it->second;
2961 data.properties[name] = value;
2962 }
2963 }
2964
2965 /* only for base hard disks */
2966 if (m->pParent.isNull())
2967 data.hdType = m->type;
2968
2969 /* save all children */
2970 for (MediaList::const_iterator it = getChildren().begin();
2971 it != getChildren().end();
2972 ++it)
2973 {
2974 settings::Medium med;
2975 HRESULT rc = (*it)->saveSettings(med);
2976 AssertComRCReturnRC(rc);
2977 data.llChildren.push_back(med);
2978 }
2979
2980 return S_OK;
2981}
2982
2983/**
2984 * Compares the location of this hard disk to the given location.
2985 *
2986 * The comparison takes the location details into account. For example, if the
2987 * location is a file in the host's filesystem, a case insensitive comparison
2988 * will be performed for case insensitive filesystems.
2989 *
2990 * @param aLocation Location to compare to (as is).
2991 * @param aResult Where to store the result of comparison: 0 if locations
2992 * are equal, 1 if this object's location is greater than
2993 * the specified location, and -1 otherwise.
2994 */
2995HRESULT Medium::compareLocationTo(const char *aLocation, int &aResult)
2996{
2997 AutoCaller autoCaller(this);
2998 AssertComRCReturnRC(autoCaller.rc());
2999
3000 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3001
3002 Utf8Str locationFull(m->strLocationFull);
3003
3004 /// @todo NEWMEDIA delegate the comparison to the backend?
3005
3006 if (m->formatObj->capabilities() & MediumFormatCapabilities_File)
3007 {
3008 Utf8Str location(aLocation);
3009
3010 /* For locations represented by files, append the default path if
3011 * only the name is given, and then get the full path. */
3012 if (!RTPathHavePath(aLocation))
3013 {
3014 location = Utf8StrFmt("%s%c%s",
3015 m->pVirtualBox->getDefaultHardDiskFolder().raw(),
3016 RTPATH_DELIMITER,
3017 aLocation);
3018 }
3019
3020 int vrc = m->pVirtualBox->calculateFullPath(location, location);
3021 if (RT_FAILURE(vrc))
3022 return setError(E_FAIL,
3023 tr("Invalid hard disk storage file location '%s' (%Rrc)"),
3024 location.raw(),
3025 vrc);
3026
3027 aResult = RTPathCompare(locationFull.c_str(), location.c_str());
3028 }
3029 else
3030 aResult = locationFull.compare(aLocation);
3031
3032 return S_OK;
3033}
3034
3035/**
3036 * Constructs a medium lock list for this medium. The lock is not taken.
3037 *
3038 * @note Locks the medium tree for reading.
3039 *
3040 * @param fMediumWritable Whether to associate a write lock with this medium.
3041 * @param pToBeParent Medium which will become the parent of this medium.
3042 * @param mediumLockList Where to store the resulting list.
3043 */
3044HRESULT Medium::createMediumLockList(bool fMediumWritable,
3045 Medium *pToBeParent,
3046 MediumLockList &mediumLockList)
3047{
3048 HRESULT rc = S_OK;
3049
3050 /* we access parent medium objects */
3051 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3052
3053 /* paranoid sanity checking if the medium has a to-be parent medium */
3054 if (pToBeParent)
3055 {
3056 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3057 ComAssertRet(getParent().isNull(), E_FAIL);
3058 ComAssertRet(getChildren().size() == 0, E_FAIL);
3059 }
3060
3061 ErrorInfoKeeper eik;
3062 MultiResult mrc(S_OK);
3063
3064 ComObjPtr<Medium> pMedium = this;
3065 while (!pMedium.isNull())
3066 {
3067 // need write lock for RefreshState if medium is inaccessible
3068 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
3069
3070 /* Accessibility check must be first, otherwise locking interferes
3071 * with getting the medium state. Lock lists are not created for
3072 * fun, and thus getting the image status is no luxury. */
3073 MediumState_T mediumState = pMedium->getState();
3074 if (mediumState == MediumState_Inaccessible)
3075 {
3076 rc = pMedium->RefreshState(&mediumState);
3077 if (FAILED(rc)) return rc;
3078
3079 if (mediumState == MediumState_Inaccessible)
3080 {
3081 Bstr error;
3082 rc = pMedium->COMGETTER(LastAccessError)(error.asOutParam());
3083 if (FAILED(rc)) return rc;
3084
3085 Bstr loc;
3086 rc = pMedium->COMGETTER(Location)(loc.asOutParam());
3087 if (FAILED(rc)) return rc;
3088
3089 /* collect multiple errors */
3090 eik.restore();
3091
3092 /* be in sync with MediumBase::setStateError() */
3093 Assert(!error.isEmpty());
3094 mrc = setError(E_FAIL,
3095 tr("Medium '%ls' is not accessible. %ls"),
3096 loc.raw(),
3097 error.raw());
3098
3099 eik.fetch();
3100 }
3101 }
3102
3103 if (pMedium == this)
3104 mediumLockList.Prepend(pMedium, fMediumWritable);
3105 else
3106 mediumLockList.Prepend(pMedium, false);
3107
3108 pMedium = pMedium->getParent();
3109 if (pMedium.isNull() && pToBeParent)
3110 {
3111 pMedium = pToBeParent;
3112 pToBeParent = NULL;
3113 }
3114 }
3115
3116 return mrc;
3117}
3118
3119/**
3120 * Returns a preferred format for differencing hard disks.
3121 */
3122Bstr Medium::preferredDiffFormat()
3123{
3124 Utf8Str strFormat;
3125
3126 AutoCaller autoCaller(this);
3127 AssertComRCReturn(autoCaller.rc(), strFormat);
3128
3129 /* m->format is const, no need to lock */
3130 strFormat = m->strFormat;
3131
3132 /* check that our own format supports diffs */
3133 if (!(m->formatObj->capabilities() & MediumFormatCapabilities_Differencing))
3134 {
3135 /* use the default format if not */
3136 AutoReadLock propsLock(m->pVirtualBox->systemProperties() COMMA_LOCKVAL_SRC_POS);
3137 strFormat = m->pVirtualBox->getDefaultHardDiskFormat();
3138 }
3139
3140 return strFormat;
3141}
3142
3143/**
3144 * Returns the medium type. Must have caller + locking!
3145 * @return
3146 */
3147MediumType_T Medium::getType() const
3148{
3149 return m->type;
3150}
3151
3152// private methods
3153////////////////////////////////////////////////////////////////////////////////
3154
3155/**
3156 * Returns a short version of the location attribute.
3157 *
3158 * @note Must be called from under this object's read or write lock.
3159 */
3160Utf8Str Medium::getName()
3161{
3162 Utf8Str name = RTPathFilename(m->strLocationFull.c_str());
3163 return name;
3164}
3165
3166/**
3167 * Sets the value of m->strLocation and calculates the value of m->strLocationFull.
3168 *
3169 * Treats non-FS-path locations specially, and prepends the default hard disk
3170 * folder if the given location string does not contain any path information
3171 * at all.
3172 *
3173 * Also, if the specified location is a file path that ends with '/' then the
3174 * file name part will be generated by this method automatically in the format
3175 * '{<uuid>}.<ext>' where <uuid> is a fresh UUID that this method will generate
3176 * and assign to this medium, and <ext> is the default extension for this
3177 * medium's storage format. Note that this procedure requires the media state to
3178 * be NotCreated and will return a failure otherwise.
3179 *
3180 * @param aLocation Location of the storage unit. If the location is a FS-path,
3181 * then it can be relative to the VirtualBox home directory.
3182 * @param aFormat Optional fallback format if it is an import and the format
3183 * cannot be determined.
3184 *
3185 * @note Must be called from under this object's write lock.
3186 */
3187HRESULT Medium::setLocation(const Utf8Str &aLocation, const Utf8Str &aFormat)
3188{
3189 AssertReturn(!aLocation.isEmpty(), E_FAIL);
3190
3191 AutoCaller autoCaller(this);
3192 AssertComRCReturnRC(autoCaller.rc());
3193
3194 /* formatObj may be null only when initializing from an existing path and
3195 * no format is known yet */
3196 AssertReturn( (!m->strFormat.isEmpty() && !m->formatObj.isNull())
3197 || ( autoCaller.state() == InInit
3198 && m->state != MediumState_NotCreated
3199 && m->id.isEmpty()
3200 && m->strFormat.isEmpty()
3201 && m->formatObj.isNull()),
3202 E_FAIL);
3203
3204 /* are we dealing with a new medium constructed using the existing
3205 * location? */
3206 bool isImport = m->strFormat.isEmpty();
3207
3208 if ( isImport
3209 || ( (m->formatObj->capabilities() & MediumFormatCapabilities_File)
3210 && !m->hostDrive))
3211 {
3212 Guid id;
3213
3214 Utf8Str location(aLocation);
3215
3216 if (m->state == MediumState_NotCreated)
3217 {
3218 /* must be a file (formatObj must be already known) */
3219 Assert(m->formatObj->capabilities() & MediumFormatCapabilities_File);
3220
3221 if (RTPathFilename(location.c_str()) == NULL)
3222 {
3223 /* no file name is given (either an empty string or ends with a
3224 * slash), generate a new UUID + file name if the state allows
3225 * this */
3226
3227 ComAssertMsgRet(!m->formatObj->fileExtensions().empty(),
3228 ("Must be at least one extension if it is MediumFormatCapabilities_File\n"),
3229 E_FAIL);
3230
3231 Bstr ext = m->formatObj->fileExtensions().front();
3232 ComAssertMsgRet(!ext.isEmpty(),
3233 ("Default extension must not be empty\n"),
3234 E_FAIL);
3235
3236 id.create();
3237
3238 location = Utf8StrFmt("%s{%RTuuid}.%ls",
3239 location.raw(), id.raw(), ext.raw());
3240 }
3241 }
3242
3243 /* append the default folder if no path is given */
3244 if (!RTPathHavePath(location.c_str()))
3245 location = Utf8StrFmt("%s%c%s",
3246 m->pVirtualBox->getDefaultHardDiskFolder().raw(),
3247 RTPATH_DELIMITER,
3248 location.raw());
3249
3250 /* get the full file name */
3251 Utf8Str locationFull;
3252 int vrc = m->pVirtualBox->calculateFullPath(location, locationFull);
3253 if (RT_FAILURE(vrc))
3254 return setError(VBOX_E_FILE_ERROR,
3255 tr("Invalid medium storage file location '%s' (%Rrc)"),
3256 location.raw(), vrc);
3257
3258 /* detect the backend from the storage unit if importing */
3259 if (isImport)
3260 {
3261 char *backendName = NULL;
3262
3263 /* is it a file? */
3264 {
3265 RTFILE file;
3266 vrc = RTFileOpen(&file, locationFull.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
3267 if (RT_SUCCESS(vrc))
3268 RTFileClose(file);
3269 }
3270 if (RT_SUCCESS(vrc))
3271 {
3272 vrc = VDGetFormat(NULL, locationFull.c_str(), &backendName);
3273 }
3274 else if (vrc != VERR_FILE_NOT_FOUND && vrc != VERR_PATH_NOT_FOUND)
3275 {
3276 /* assume it's not a file, restore the original location */
3277 location = locationFull = aLocation;
3278 vrc = VDGetFormat(NULL, locationFull.c_str(), &backendName);
3279 }
3280
3281 if (RT_FAILURE(vrc))
3282 {
3283 if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
3284 return setError(VBOX_E_FILE_ERROR,
3285 tr("Could not find file for the medium '%s' (%Rrc)"),
3286 locationFull.raw(), vrc);
3287 else if (aFormat.isEmpty())
3288 return setError(VBOX_E_IPRT_ERROR,
3289 tr("Could not get the storage format of the medium '%s' (%Rrc)"),
3290 locationFull.raw(), vrc);
3291 else
3292 {
3293 HRESULT rc = setFormat(Bstr(aFormat));
3294 /* setFormat() must not fail since we've just used the backend so
3295 * the format object must be there */
3296 AssertComRCReturnRC(rc);
3297 }
3298 }
3299 else
3300 {
3301 ComAssertRet(backendName != NULL && *backendName != '\0', E_FAIL);
3302
3303 HRESULT rc = setFormat(Bstr(backendName));
3304 RTStrFree(backendName);
3305
3306 /* setFormat() must not fail since we've just used the backend so
3307 * the format object must be there */
3308 AssertComRCReturnRC(rc);
3309 }
3310 }
3311
3312 /* is it still a file? */
3313 if (m->formatObj->capabilities() & MediumFormatCapabilities_File)
3314 {
3315 m->strLocation = location;
3316 m->strLocationFull = locationFull;
3317
3318 if (m->state == MediumState_NotCreated)
3319 {
3320 /* assign a new UUID (this UUID will be used when calling
3321 * VDCreateBase/VDCreateDiff as a wanted UUID). Note that we
3322 * also do that if we didn't generate it to make sure it is
3323 * either generated by us or reset to null */
3324 unconst(m->id) = id;
3325 }
3326 }
3327 else
3328 {
3329 m->strLocation = locationFull;
3330 m->strLocationFull = locationFull;
3331 }
3332 }
3333 else
3334 {
3335 m->strLocation = aLocation;
3336 m->strLocationFull = aLocation;
3337 }
3338
3339 return S_OK;
3340}
3341
3342/**
3343 * Queries information from the image file.
3344 *
3345 * As a result of this call, the accessibility state and data members such as
3346 * size and description will be updated with the current information.
3347 *
3348 * @note This method may block during a system I/O call that checks storage
3349 * accessibility.
3350 *
3351 * @note Locks medium tree for reading and writing (for new diff media checked
3352 * for the first time). Locks mParent for reading. Locks this object for
3353 * writing.
3354 */
3355HRESULT Medium::queryInfo()
3356{
3357 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3358
3359 if ( m->state != MediumState_Created
3360 && m->state != MediumState_Inaccessible
3361 && m->state != MediumState_LockedRead)
3362 return E_FAIL;
3363
3364 HRESULT rc = S_OK;
3365
3366 int vrc = VINF_SUCCESS;
3367
3368 /* check if a blocking queryInfo() call is in progress on some other thread,
3369 * and wait for it to finish if so instead of querying data ourselves */
3370 if (m->queryInfoRunning)
3371 {
3372 Assert( m->state == MediumState_LockedRead
3373 || m->state == MediumState_LockedWrite);
3374
3375 alock.leave();
3376
3377 vrc = RTSemEventMultiWait(m->queryInfoSem, RT_INDEFINITE_WAIT);
3378
3379 alock.enter();
3380
3381 AssertRC(vrc);
3382
3383 return S_OK;
3384 }
3385
3386 bool success = false;
3387 Utf8Str lastAccessError;
3388
3389 /* are we dealing with a new medium constructed using the existing
3390 * location? */
3391 bool isImport = m->id.isEmpty();
3392 unsigned flags = VD_OPEN_FLAGS_INFO;
3393
3394 /* Note that we don't use VD_OPEN_FLAGS_READONLY when opening new
3395 * media because that would prevent necessary modifications
3396 * when opening media of some third-party formats for the first
3397 * time in VirtualBox (such as VMDK for which VDOpen() needs to
3398 * generate an UUID if it is missing) */
3399 if ( (m->hddOpenMode == OpenReadOnly)
3400 || !isImport
3401 )
3402 flags |= VD_OPEN_FLAGS_READONLY;
3403
3404 /* Lock the medium, which makes the behavior much more consistent */
3405 if (flags & VD_OPEN_FLAGS_READONLY)
3406 rc = LockRead(NULL);
3407 else
3408 rc = LockWrite(NULL);
3409 if (FAILED(rc)) return rc;
3410
3411 /* Copies of the input state fields which are not read-only,
3412 * as we're dropping the lock. CAUTION: be extremely careful what
3413 * you do with the contents of this medium object, as you will
3414 * create races if there are concurrent changes. */
3415 Utf8Str format(m->strFormat);
3416 Utf8Str location(m->strLocationFull);
3417 ComObjPtr<MediumFormat> formatObj = m->formatObj;
3418
3419 /* "Output" values which can't be set because the lock isn't held
3420 * at the time the values are determined. */
3421 Guid mediumId = m->id;
3422 uint64_t mediumSize = 0;
3423 uint64_t mediumLogicalSize = 0;
3424
3425 /* leave the lock before a lengthy operation */
3426 vrc = RTSemEventMultiReset(m->queryInfoSem);
3427 AssertRCReturn(vrc, E_FAIL);
3428 m->queryInfoRunning = true;
3429 alock.leave();
3430
3431 try
3432 {
3433 /* skip accessibility checks for host drives */
3434 if (m->hostDrive)
3435 {
3436 success = true;
3437 throw S_OK;
3438 }
3439
3440 PVBOXHDD hdd;
3441 vrc = VDCreate(m->vdDiskIfaces, &hdd);
3442 ComAssertRCThrow(vrc, E_FAIL);
3443
3444 try
3445 {
3446 /** @todo This kind of opening of images is assuming that diff
3447 * images can be opened as base images. Should be documented if
3448 * it must work for all medium format backends. */
3449 vrc = VDOpen(hdd,
3450 format.c_str(),
3451 location.c_str(),
3452 flags,
3453 m->vdDiskIfaces);
3454 if (RT_FAILURE(vrc))
3455 {
3456 lastAccessError = Utf8StrFmt(tr("Could not open the medium '%s'%s"),
3457 location.c_str(), vdError(vrc).c_str());
3458 throw S_OK;
3459 }
3460
3461 if (formatObj->capabilities() & MediumFormatCapabilities_Uuid)
3462 {
3463 /* Modify the UUIDs if necessary. The associated fields are
3464 * not modified by other code, so no need to copy. */
3465 if (m->setImageId)
3466 {
3467 vrc = VDSetUuid(hdd, 0, m->imageId);
3468 ComAssertRCThrow(vrc, E_FAIL);
3469 }
3470 if (m->setParentId)
3471 {
3472 vrc = VDSetParentUuid(hdd, 0, m->parentId);
3473 ComAssertRCThrow(vrc, E_FAIL);
3474 }
3475 /* zap the information, these are no long-term members */
3476 m->setImageId = false;
3477 unconst(m->imageId).clear();
3478 m->setParentId = false;
3479 unconst(m->parentId).clear();
3480
3481 /* check the UUID */
3482 RTUUID uuid;
3483 vrc = VDGetUuid(hdd, 0, &uuid);
3484 ComAssertRCThrow(vrc, E_FAIL);
3485
3486 if (isImport)
3487 {
3488 mediumId = uuid;
3489
3490 if (mediumId.isEmpty() && (m->hddOpenMode == OpenReadOnly))
3491 // only when importing a VDMK that has no UUID, create one in memory
3492 mediumId.create();
3493 }
3494 else
3495 {
3496 Assert(!mediumId.isEmpty());
3497
3498 if (mediumId != uuid)
3499 {
3500 lastAccessError = Utf8StrFmt(
3501 tr("UUID {%RTuuid} of the medium '%s' does not match the value {%RTuuid} stored in the media registry ('%s')"),
3502 &uuid,
3503 location.c_str(),
3504 mediumId.raw(),
3505 m->pVirtualBox->settingsFilePath().c_str());
3506 throw S_OK;
3507 }
3508 }
3509 }
3510 else
3511 {
3512 /* the backend does not support storing UUIDs within the
3513 * underlying storage so use what we store in XML */
3514
3515 /* generate an UUID for an imported UUID-less medium */
3516 if (isImport)
3517 {
3518 if (m->setImageId)
3519 mediumId = m->imageId;
3520 else
3521 mediumId.create();
3522 }
3523 }
3524
3525 /* check the type */
3526 unsigned uImageFlags;
3527 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
3528 ComAssertRCThrow(vrc, E_FAIL);
3529
3530 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
3531 {
3532 RTUUID parentId;
3533 vrc = VDGetParentUuid(hdd, 0, &parentId);
3534 ComAssertRCThrow(vrc, E_FAIL);
3535
3536 if (isImport)
3537 {
3538 /* the parent must be known to us. Note that we freely
3539 * call locking methods of mVirtualBox and parent from the
3540 * write lock (breaking the {parent,child} lock order)
3541 * because there may be no concurrent access to the just
3542 * opened hard disk on ther threads yet (and init() will
3543 * fail if this method reporst MediumState_Inaccessible) */
3544
3545 Guid id = parentId;
3546 ComObjPtr<Medium> pParent;
3547 rc = m->pVirtualBox->findHardDisk(&id, NULL,
3548 false /* aSetError */,
3549 &pParent);
3550 if (FAILED(rc))
3551 {
3552 lastAccessError = Utf8StrFmt(
3553 tr("Parent hard disk with UUID {%RTuuid} of the hard disk '%s' is not found in the media registry ('%s')"),
3554 &parentId, location.c_str(),
3555 m->pVirtualBox->settingsFilePath().c_str());
3556 throw S_OK;
3557 }
3558
3559 /* we set mParent & children() */
3560 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3561
3562 Assert(m->pParent.isNull());
3563 m->pParent = pParent;
3564 m->pParent->m->llChildren.push_back(this);
3565 }
3566 else
3567 {
3568 /* we access mParent */
3569 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3570
3571 /* check that parent UUIDs match. Note that there's no need
3572 * for the parent's AutoCaller (our lifetime is bound to
3573 * it) */
3574
3575 if (m->pParent.isNull())
3576 {
3577 lastAccessError = Utf8StrFmt(
3578 tr("Hard disk '%s' is differencing but it is not associated with any parent hard disk in the media registry ('%s')"),
3579 location.c_str(),
3580 m->pVirtualBox->settingsFilePath().c_str());
3581 throw S_OK;
3582 }
3583
3584 AutoReadLock parentLock(m->pParent COMMA_LOCKVAL_SRC_POS);
3585 if ( m->pParent->getState() != MediumState_Inaccessible
3586 && m->pParent->getId() != parentId)
3587 {
3588 lastAccessError = Utf8StrFmt(
3589 tr("Parent UUID {%RTuuid} of the hard disk '%s' does not match UUID {%RTuuid} of its parent hard disk stored in the media registry ('%s')"),
3590 &parentId, location.c_str(),
3591 m->pParent->getId().raw(),
3592 m->pVirtualBox->settingsFilePath().c_str());
3593 throw S_OK;
3594 }
3595
3596 /// @todo NEWMEDIA what to do if the parent is not
3597 /// accessible while the diff is? Probably nothing. The
3598 /// real code will detect the mismatch anyway.
3599 }
3600 }
3601
3602 mediumSize = VDGetFileSize(hdd, 0);
3603 mediumLogicalSize = VDGetSize(hdd, 0) / _1M;
3604
3605 success = true;
3606 }
3607 catch (HRESULT aRC)
3608 {
3609 rc = aRC;
3610 }
3611
3612 VDDestroy(hdd);
3613
3614 }
3615 catch (HRESULT aRC)
3616 {
3617 rc = aRC;
3618 }
3619
3620 alock.enter();
3621
3622 if (isImport)
3623 unconst(m->id) = mediumId;
3624
3625 if (success)
3626 {
3627 m->size = mediumSize;
3628 m->logicalSize = mediumLogicalSize;
3629 m->strLastAccessError.setNull();
3630 }
3631 else
3632 {
3633 m->strLastAccessError = lastAccessError;
3634 LogWarningFunc(("'%s' is not accessible (error='%s', rc=%Rhrc, vrc=%Rrc)\n",
3635 location.c_str(), m->strLastAccessError.c_str(),
3636 rc, vrc));
3637 }
3638
3639 /* inform other callers if there are any */
3640 RTSemEventMultiSignal(m->queryInfoSem);
3641 m->queryInfoRunning = false;
3642
3643 /* Set the proper state according to the result of the check */
3644 if (success)
3645 m->preLockState = MediumState_Created;
3646 else
3647 m->preLockState = MediumState_Inaccessible;
3648
3649 if (flags & VD_OPEN_FLAGS_READONLY)
3650 rc = UnlockRead(NULL);
3651 else
3652 rc = UnlockWrite(NULL);
3653 if (FAILED(rc)) return rc;
3654
3655 return rc;
3656}
3657
3658/**
3659 * Sets the extended error info according to the current media state.
3660 *
3661 * @note Must be called from under this object's write or read lock.
3662 */
3663HRESULT Medium::setStateError()
3664{
3665 HRESULT rc = E_FAIL;
3666
3667 switch (m->state)
3668 {
3669 case MediumState_NotCreated:
3670 {
3671 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3672 tr("Storage for the medium '%s' is not created"),
3673 m->strLocationFull.raw());
3674 break;
3675 }
3676 case MediumState_Created:
3677 {
3678 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3679 tr("Storage for the medium '%s' is already created"),
3680 m->strLocationFull.raw());
3681 break;
3682 }
3683 case MediumState_LockedRead:
3684 {
3685 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3686 tr("Medium '%s' is locked for reading by another task"),
3687 m->strLocationFull.raw());
3688 break;
3689 }
3690 case MediumState_LockedWrite:
3691 {
3692 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3693 tr("Medium '%s' is locked for writing by another task"),
3694 m->strLocationFull.raw());
3695 break;
3696 }
3697 case MediumState_Inaccessible:
3698 {
3699 /* be in sync with Console::powerUpThread() */
3700 if (!m->strLastAccessError.isEmpty())
3701 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3702 tr("Medium '%s' is not accessible. %s"),
3703 m->strLocationFull.raw(), m->strLastAccessError.c_str());
3704 else
3705 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3706 tr("Medium '%s' is not accessible"),
3707 m->strLocationFull.raw());
3708 break;
3709 }
3710 case MediumState_Creating:
3711 {
3712 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3713 tr("Storage for the medium '%s' is being created"),
3714 m->strLocationFull.raw());
3715 break;
3716 }
3717 case MediumState_Deleting:
3718 {
3719 rc = setError(VBOX_E_INVALID_OBJECT_STATE,
3720 tr("Storage for the medium '%s' is being deleted"),
3721 m->strLocationFull.raw());
3722 break;
3723 }
3724 default:
3725 {
3726 AssertFailed();
3727 break;
3728 }
3729 }
3730
3731 return rc;
3732}
3733
3734/**
3735 * Deletes the hard disk storage unit.
3736 *
3737 * If @a aProgress is not NULL but the object it points to is @c null then a new
3738 * progress object will be created and assigned to @a *aProgress on success,
3739 * otherwise the existing progress object is used. If Progress is NULL, then no
3740 * progress object is created/used at all.
3741 *
3742 * When @a aWait is @c false, this method will create a thread to perform the
3743 * delete operation asynchronously and will return immediately. Otherwise, it
3744 * will perform the operation on the calling thread and will not return to the
3745 * caller until the operation is completed. Note that @a aProgress cannot be
3746 * NULL when @a aWait is @c false (this method will assert in this case).
3747 *
3748 * @param aProgress Where to find/store a Progress object to track operation
3749 * completion.
3750 * @param aWait @c true if this method should block instead of creating
3751 * an asynchronous thread.
3752 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
3753 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
3754 * This only works in "wait" mode; otherwise saveSettings gets called automatically by the thread that was created,
3755 * and this parameter is ignored.
3756 *
3757 * @note Locks mVirtualBox and this object for writing. Locks medium tree for
3758 * writing.
3759 */
3760HRESULT Medium::deleteStorage(ComObjPtr<Progress> *aProgress,
3761 bool aWait,
3762 bool *pfNeedsSaveSettings)
3763{
3764 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
3765
3766 HRESULT rc = S_OK;
3767 ComObjPtr<Progress> pProgress;
3768 Medium::Task *pTask = NULL;
3769
3770 try
3771 {
3772 /* we're accessing the media tree, and canClose() needs it too */
3773 AutoMultiWriteLock2 multilock(&m->pVirtualBox->getMediaTreeLockHandle(),
3774 this->lockHandle()
3775 COMMA_LOCKVAL_SRC_POS);
3776 LogFlowThisFunc(("aWait=%RTbool locationFull=%s\n", aWait, getLocationFull().c_str() ));
3777
3778 if ( !(m->formatObj->capabilities() & ( MediumFormatCapabilities_CreateDynamic
3779 | MediumFormatCapabilities_CreateFixed)))
3780 throw setError(VBOX_E_NOT_SUPPORTED,
3781 tr("Hard disk format '%s' does not support storage deletion"),
3782 m->strFormat.raw());
3783
3784 /* Note that we are fine with Inaccessible state too: a) for symmetry
3785 * with create calls and b) because it doesn't really harm to try, if
3786 * it is really inaccessible, the delete operation will fail anyway.
3787 * Accepting Inaccessible state is especially important because all
3788 * registered hard disks are initially Inaccessible upon VBoxSVC
3789 * startup until COMGETTER(RefreshState) is called. Accept Deleting
3790 * state because some callers need to put the image in this state early
3791 * to prevent races. */
3792 switch (m->state)
3793 {
3794 case MediumState_Created:
3795 case MediumState_Deleting:
3796 case MediumState_Inaccessible:
3797 break;
3798 default:
3799 throw setStateError();
3800 }
3801
3802 if (m->backRefs.size() != 0)
3803 {
3804 Utf8Str strMachines;
3805 for (BackRefList::const_iterator it = m->backRefs.begin();
3806 it != m->backRefs.end();
3807 ++it)
3808 {
3809 const BackRef &b = *it;
3810 if (strMachines.length())
3811 strMachines.append(", ");
3812 strMachines.append(b.machineId.toString().c_str());
3813 }
3814#ifdef DEBUG
3815 dumpBackRefs();
3816#endif
3817 throw setError(VBOX_E_OBJECT_IN_USE,
3818 tr("Cannot delete storage: hard disk '%s' is still attached to the following %d virtual machine(s): %s"),
3819 m->strLocationFull.c_str(),
3820 m->backRefs.size(),
3821 strMachines.c_str());
3822 }
3823
3824 rc = canClose();
3825 if (FAILED(rc))
3826 throw rc;
3827
3828 /* go to Deleting state, so that the medium is not actually locked */
3829 rc = markForDeletion();
3830 if (FAILED(rc))
3831 throw rc;
3832
3833 /* Build the medium lock list. */
3834 MediumLockList *pMediumLockList(new MediumLockList());
3835 rc = createMediumLockList(true, NULL,
3836 *pMediumLockList);
3837 if (FAILED(rc))
3838 {
3839 delete pMediumLockList;
3840 throw rc;
3841 }
3842
3843 rc = pMediumLockList->Lock();
3844 if (FAILED(rc))
3845 {
3846 delete pMediumLockList;
3847 throw setError(rc,
3848 tr("Failed to lock media when deleting '%ls'"),
3849 getLocationFull().raw());
3850 }
3851
3852 /* try to remove from the list of known hard disks before performing
3853 * actual deletion (we favor the consistency of the media registry in
3854 * the first place which would have been broken if
3855 * unregisterWithVirtualBox() failed after we successfully deleted the
3856 * storage) */
3857 rc = unregisterWithVirtualBox(pfNeedsSaveSettings);
3858 if (FAILED(rc))
3859 throw rc;
3860
3861 if (aProgress != NULL)
3862 {
3863 /* use the existing progress object... */
3864 pProgress = *aProgress;
3865
3866 /* ...but create a new one if it is null */
3867 if (pProgress.isNull())
3868 {
3869 pProgress.createObject();
3870 rc = pProgress->init(m->pVirtualBox,
3871 static_cast<IMedium*>(this),
3872 BstrFmt(tr("Deleting hard disk storage unit '%s'"), m->strLocationFull.raw()),
3873 FALSE /* aCancelable */);
3874 if (FAILED(rc))
3875 throw rc;
3876 }
3877 }
3878
3879 /* setup task object to carry out the operation sync/async */
3880 pTask = new Medium::DeleteTask(this, pProgress, pMediumLockList);
3881 rc = pTask->rc();
3882 AssertComRC(rc);
3883 if (FAILED(rc))
3884 throw rc;
3885 }
3886 catch (HRESULT aRC) { rc = aRC; }
3887
3888 if (SUCCEEDED(rc))
3889 {
3890 if (aWait)
3891 rc = runNow(pTask, NULL /* pfNeedsSaveSettings*/);
3892 else
3893 rc = startThread(pTask);
3894
3895 if (SUCCEEDED(rc) && aProgress != NULL)
3896 *aProgress = pProgress;
3897
3898 }
3899 else
3900 {
3901 if (pTask)
3902 delete pTask;
3903
3904 /* Undo deleting state if necessary. */
3905 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3906 unmarkForDeletion();
3907 }
3908
3909 return rc;
3910}
3911
3912/**
3913 * Mark a medium for deletion.
3914 *
3915 * @note Caller must hold the write lock on this medium!
3916 */
3917HRESULT Medium::markForDeletion()
3918{
3919 ComAssertRet(this->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
3920 switch (m->state)
3921 {
3922 case MediumState_Created:
3923 case MediumState_Inaccessible:
3924 m->preLockState = m->state;
3925 m->state = MediumState_Deleting;
3926 return S_OK;
3927 default:
3928 return setStateError();
3929 }
3930}
3931
3932/**
3933 * Removes the "mark for deletion".
3934 *
3935 * @note Caller must hold the write lock on this medium!
3936 */
3937HRESULT Medium::unmarkForDeletion()
3938{
3939 ComAssertRet(this->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
3940 switch (m->state)
3941 {
3942 case MediumState_Deleting:
3943 m->state = m->preLockState;
3944 return S_OK;
3945 default:
3946 return setStateError();
3947 }
3948}
3949
3950/**
3951 * Mark a medium for deletion which is in locked state.
3952 *
3953 * @note Caller must hold the write lock on this medium!
3954 */
3955HRESULT Medium::markLockedForDeletion()
3956{
3957 ComAssertRet(this->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
3958 if ( ( m->state == MediumState_LockedRead
3959 || m->state == MediumState_LockedWrite)
3960 && m->preLockState == MediumState_Created)
3961 {
3962 m->preLockState = MediumState_Deleting;
3963 return S_OK;
3964 }
3965 else
3966 return setStateError();
3967}
3968
3969/**
3970 * Removes the "mark for deletion" for a medium in locked state.
3971 *
3972 * @note Caller must hold the write lock on this medium!
3973 */
3974HRESULT Medium::unmarkLockedForDeletion()
3975{
3976 ComAssertRet(this->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
3977 if ( ( m->state == MediumState_LockedRead
3978 || m->state == MediumState_LockedWrite)
3979 && m->preLockState == MediumState_Deleting)
3980 {
3981 m->preLockState = MediumState_Created;
3982 return S_OK;
3983 }
3984 else
3985 return setStateError();
3986}
3987
3988/**
3989 * Creates a new differencing storage unit using the given target hard disk's
3990 * format and the location. Note that @c aTarget must be NotCreated.
3991 *
3992 * The @a aMediumLockList parameter contains the associated medium lock list,
3993 * which must be in locked state. If @a aWait is @c true then the caller is
3994 * responsible for unlocking.
3995 *
3996 * If @a aProgress is not NULL but the object it points to is @c null then a
3997 * new progress object will be created and assigned to @a *aProgress on
3998 * success, otherwise the existing progress object is used. If @a aProgress is
3999 * NULL, then no progress object is created/used at all.
4000 *
4001 * When @a aWait is @c false, this method will create a thread to perform the
4002 * create operation asynchronously and will return immediately. Otherwise, it
4003 * will perform the operation on the calling thread and will not return to the
4004 * caller until the operation is completed. Note that @a aProgress cannot be
4005 * NULL when @a aWait is @c false (this method will assert in this case).
4006 *
4007 * @param aTarget Target hard disk.
4008 * @param aVariant Precise image variant to create.
4009 * @param aMediumLockList List of media which should be locked.
4010 * @param aProgress Where to find/store a Progress object to track
4011 * operation completion.
4012 * @param aWait @c true if this method should block instead of
4013 * creating an asynchronous thread.
4014 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been
4015 * initialized to false and that will be set to true
4016 * by this function if the caller should invoke
4017 * VirtualBox::saveSettings() because the global
4018 * settings have changed. This only works in "wait"
4019 * mode; otherwise saveSettings is called
4020 * automatically by the thread that was created,
4021 * and this parameter is ignored.
4022 *
4023 * @note Locks this object and @a aTarget for writing.
4024 */
4025HRESULT Medium::createDiffStorage(ComObjPtr<Medium> &aTarget,
4026 MediumVariant_T aVariant,
4027 MediumLockList *aMediumLockList,
4028 ComObjPtr<Progress> *aProgress,
4029 bool aWait,
4030 bool *pfNeedsSaveSettings)
4031{
4032 AssertReturn(!aTarget.isNull(), E_FAIL);
4033 AssertReturn(aMediumLockList, E_FAIL);
4034 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
4035
4036 AutoCaller autoCaller(this);
4037 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4038
4039 AutoCaller targetCaller(aTarget);
4040 if (FAILED(targetCaller.rc())) return targetCaller.rc();
4041
4042 HRESULT rc = S_OK;
4043 ComObjPtr<Progress> pProgress;
4044 Medium::Task *pTask = NULL;
4045
4046 try
4047 {
4048 AutoMultiWriteLock2 alock(this, aTarget COMMA_LOCKVAL_SRC_POS);
4049
4050 ComAssertThrow(m->type != MediumType_Writethrough, E_FAIL);
4051 ComAssertThrow(m->state == MediumState_LockedRead, E_FAIL);
4052
4053 if (aTarget->m->state != MediumState_NotCreated)
4054 throw aTarget->setStateError();
4055
4056 /* Check that the hard disk is not attached to the current state of
4057 * any VM referring to it. */
4058 for (BackRefList::const_iterator it = m->backRefs.begin();
4059 it != m->backRefs.end();
4060 ++it)
4061 {
4062 if (it->fInCurState)
4063 {
4064 /* Note: when a VM snapshot is being taken, all normal hard
4065 * disks attached to the VM in the current state will be, as an
4066 * exception, also associated with the snapshot which is about
4067 * to create (see SnapshotMachine::init()) before deassociating
4068 * them from the current state (which takes place only on
4069 * success in Machine::fixupHardDisks()), so that the size of
4070 * snapshotIds will be 1 in this case. The extra condition is
4071 * used to filter out this legal situation. */
4072 if (it->llSnapshotIds.size() == 0)
4073 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4074 tr("Hard disk '%s' is attached to a virtual machine with UUID {%RTuuid}. No differencing hard disks based on it may be created until it is detached"),
4075 m->strLocationFull.raw(), it->machineId.raw());
4076
4077 Assert(it->llSnapshotIds.size() == 1);
4078 }
4079 }
4080
4081 if (aProgress != NULL)
4082 {
4083 /* use the existing progress object... */
4084 pProgress = *aProgress;
4085
4086 /* ...but create a new one if it is null */
4087 if (pProgress.isNull())
4088 {
4089 pProgress.createObject();
4090 rc = pProgress->init(m->pVirtualBox,
4091 static_cast<IMedium*>(this),
4092 BstrFmt(tr("Creating differencing hard disk storage unit '%s'"), aTarget->m->strLocationFull.raw()),
4093 TRUE /* aCancelable */);
4094 if (FAILED(rc))
4095 throw rc;
4096 }
4097 }
4098
4099 /* setup task object to carry out the operation sync/async */
4100 pTask = new Medium::CreateDiffTask(this, pProgress, aTarget, aVariant,
4101 aMediumLockList,
4102 aWait /* fKeepMediumLockList */);
4103 rc = pTask->rc();
4104 AssertComRC(rc);
4105 if (FAILED(rc))
4106 throw rc;
4107
4108 /* register a task (it will deregister itself when done) */
4109 ++m->numCreateDiffTasks;
4110 Assert(m->numCreateDiffTasks != 0); /* overflow? */
4111
4112 aTarget->m->state = MediumState_Creating;
4113 }
4114 catch (HRESULT aRC) { rc = aRC; }
4115
4116 if (SUCCEEDED(rc))
4117 {
4118 if (aWait)
4119 rc = runNow(pTask, pfNeedsSaveSettings);
4120 else
4121 rc = startThread(pTask);
4122
4123 if (SUCCEEDED(rc) && aProgress != NULL)
4124 *aProgress = pProgress;
4125 }
4126 else if (pTask != NULL)
4127 delete pTask;
4128
4129 return rc;
4130}
4131
4132/**
4133 * Prepares this (source) hard disk, target hard disk and all intermediate hard
4134 * disks for the merge operation.
4135 *
4136 * This method is to be called prior to calling the #mergeTo() to perform
4137 * necessary consistency checks and place involved hard disks to appropriate
4138 * states. If #mergeTo() is not called or fails, the state modifications
4139 * performed by this method must be undone by #cancelMergeTo().
4140 *
4141 * See #mergeTo() for more information about merging.
4142 *
4143 * @param pTarget Target hard disk.
4144 * @param aMachineId Allowed machine attachment. NULL means do not check.
4145 * @param aSnapshotId Allowed snapshot attachment. NULL or empty UUID means
4146 * do not check.
4147 * @param fLockMedia Flag whether to lock the medium lock list or not.
4148 * If set to false and the medium lock list locking fails
4149 * later you must call #cancelMergeTo().
4150 * @param fMergeForward Resulting merge direction (out).
4151 * @param pParentForTarget New parent for target medium after merge (out).
4152 * @param aChildrenToReparent List of children of the source which will have
4153 * to be reparented to the target after merge (out).
4154 * @param aMediumLockList Medium locking information (out).
4155 *
4156 * @note Locks medium tree for reading. Locks this object, aTarget and all
4157 * intermediate hard disks for writing.
4158 */
4159HRESULT Medium::prepareMergeTo(const ComObjPtr<Medium> &pTarget,
4160 const Guid *aMachineId,
4161 const Guid *aSnapshotId,
4162 bool fLockMedia,
4163 bool &fMergeForward,
4164 ComObjPtr<Medium> &pParentForTarget,
4165 MediaList &aChildrenToReparent,
4166 MediumLockList * &aMediumLockList)
4167{
4168 AssertReturn(pTarget != NULL, E_FAIL);
4169 AssertReturn(pTarget != this, E_FAIL);
4170
4171 AutoCaller autoCaller(this);
4172 AssertComRCReturnRC(autoCaller.rc());
4173
4174 AutoCaller targetCaller(pTarget);
4175 AssertComRCReturnRC(targetCaller.rc());
4176
4177 HRESULT rc = S_OK;
4178 fMergeForward = false;
4179 pParentForTarget.setNull();
4180 aChildrenToReparent.clear();
4181 Assert(aMediumLockList == NULL);
4182 aMediumLockList = NULL;
4183
4184 try
4185 {
4186 // locking: we need the tree lock first because we access parent pointers
4187 AutoReadLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4188
4189 /* more sanity checking and figuring out the merge direction */
4190 ComObjPtr<Medium> pMedium = getParent();
4191 while (!pMedium.isNull() && pMedium != pTarget)
4192 pMedium = pMedium->getParent();
4193 if (pMedium == pTarget)
4194 fMergeForward = false;
4195 else
4196 {
4197 pMedium = pTarget->getParent();
4198 while (!pMedium.isNull() && pMedium != this)
4199 pMedium = pMedium->getParent();
4200 if (pMedium == this)
4201 fMergeForward = true;
4202 else
4203 {
4204 Utf8Str tgtLoc;
4205 {
4206 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4207 tgtLoc = pTarget->getLocationFull();
4208 }
4209
4210 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4211 throw setError(E_FAIL,
4212 tr("Hard disks '%s' and '%s' are unrelated"),
4213 m->strLocationFull.raw(), tgtLoc.raw());
4214 }
4215 }
4216
4217 /* Build the lock list. */
4218 aMediumLockList = new MediumLockList();
4219 if (fMergeForward)
4220 rc = pTarget->createMediumLockList(true, NULL, *aMediumLockList);
4221 else
4222 rc = createMediumLockList(false, NULL, *aMediumLockList);
4223 if (FAILED(rc))
4224 throw rc;
4225
4226 /* Sanity checking, must be after lock list creation as it depends on
4227 * valid medium states. The medium objects must be accessible. Only
4228 * do this if immediate locking is requested, otherwise it fails when
4229 * we construct a medium lock list for an already running VM. Snapshot
4230 * deletion uses this to simplify its life. */
4231 if (fLockMedia)
4232 {
4233 {
4234 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4235 if (m->state != MediumState_Created)
4236 throw setStateError();
4237 }
4238 {
4239 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4240 if (pTarget->m->state != MediumState_Created)
4241 throw pTarget->setStateError();
4242 }
4243 }
4244
4245 /* check medium attachment and other sanity conditions */
4246 if (fMergeForward)
4247 {
4248 AutoReadLock(this COMMA_LOCKVAL_SRC_POS);
4249 if (getChildren().size() > 1)
4250 {
4251 throw setError(E_FAIL,
4252 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4253 m->strLocationFull.raw(), getChildren().size());
4254 }
4255 /* One backreference is only allowed if the machine ID is not empty
4256 * and it matches the machine the image is attached to (including
4257 * the snapshot ID if not empty). */
4258 if ( m->backRefs.size() != 0
4259 && ( !aMachineId
4260 || m->backRefs.size() != 1
4261 || aMachineId->isEmpty()
4262 || *getFirstMachineBackrefId() != *aMachineId
4263 || ( (!aSnapshotId || !aSnapshotId->isEmpty())
4264 && *getFirstMachineBackrefSnapshotId() != *aSnapshotId)))
4265 throw setError(E_FAIL,
4266 tr("Medium '%s' is attached to %d virtual machines"),
4267 m->strLocationFull.raw(), m->backRefs.size());
4268 if (m->type == MediumType_Immutable)
4269 throw setError(E_FAIL,
4270 tr("Medium '%s' is immutable"),
4271 m->strLocationFull.raw());
4272 }
4273 else
4274 {
4275 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4276 if (pTarget->getChildren().size() > 1)
4277 {
4278 throw setError(E_FAIL,
4279 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4280 pTarget->m->strLocationFull.raw(),
4281 pTarget->getChildren().size());
4282 }
4283 if (pTarget->m->type == MediumType_Immutable)
4284 throw setError(E_FAIL,
4285 tr("Medium '%s' is immutable"),
4286 pTarget->m->strLocationFull.raw());
4287 }
4288 ComObjPtr<Medium> pLast(fMergeForward ? (Medium *)pTarget : this);
4289 ComObjPtr<Medium> pLastIntermediate = pLast->getParent();
4290 for (pLast = pLastIntermediate;
4291 !pLast.isNull() && pLast != pTarget && pLast != this;
4292 pLast = pLast->getParent())
4293 {
4294 AutoReadLock alock(pLast COMMA_LOCKVAL_SRC_POS);
4295 if (pLast->getChildren().size() > 1)
4296 {
4297 throw setError(E_FAIL,
4298 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
4299 pLast->m->strLocationFull.raw(),
4300 pLast->getChildren().size());
4301 }
4302 if (pLast->m->backRefs.size() != 0)
4303 throw setError(E_FAIL,
4304 tr("Medium '%s' is attached to %d virtual machines"),
4305 pLast->m->strLocationFull.raw(),
4306 pLast->m->backRefs.size());
4307
4308 }
4309
4310 /* Update medium states appropriately */
4311 if (m->state == MediumState_Created)
4312 {
4313 rc = markForDeletion();
4314 if (FAILED(rc))
4315 throw rc;
4316 }
4317 else
4318 {
4319 if (fLockMedia)
4320 throw setStateError();
4321 else if ( ( m->state == MediumState_LockedWrite
4322 || m->state == MediumState_LockedRead)
4323 && (m->preLockState == MediumState_Created))
4324 markLockedForDeletion();
4325 else
4326 throw setStateError();
4327 }
4328
4329 if (fMergeForward)
4330 {
4331 /* we will need parent to reparent target */
4332 pParentForTarget = m->pParent;
4333 }
4334 else
4335 {
4336 /* we will need to reparent children of the source */
4337 for (MediaList::const_iterator it = getChildren().begin();
4338 it != getChildren().end();
4339 ++it)
4340 {
4341 pMedium = *it;
4342 rc = pMedium->LockWrite(NULL);
4343 if (FAILED(rc))
4344 throw rc;
4345
4346 aChildrenToReparent.push_back(pMedium);
4347 }
4348 }
4349 for (pLast = pLastIntermediate;
4350 !pLast.isNull() && pLast != pTarget && pLast != this;
4351 pLast = pLast->getParent())
4352 {
4353 AutoWriteLock alock(pLast COMMA_LOCKVAL_SRC_POS);
4354 if (pLast->m->state == MediumState_Created)
4355 {
4356 rc = pLast->markForDeletion();
4357 if (FAILED(rc))
4358 throw rc;
4359 }
4360 else
4361 throw pLast->setStateError();
4362 }
4363
4364 /* Tweak the lock list in the backward merge case, as the target
4365 * isn't marked to be locked for writing yet. */
4366 if (!fMergeForward)
4367 {
4368 MediumLockList::Base::iterator lockListBegin =
4369 aMediumLockList->GetBegin();
4370 MediumLockList::Base::iterator lockListEnd =
4371 aMediumLockList->GetEnd();
4372 lockListEnd--;
4373 for (MediumLockList::Base::iterator it = lockListBegin;
4374 it != lockListEnd;
4375 ++it)
4376 {
4377 MediumLock &mediumLock = *it;
4378 if (mediumLock.GetMedium() == pTarget)
4379 {
4380 HRESULT rc2 = mediumLock.UpdateLock(true);
4381 AssertComRC(rc2);
4382 break;
4383 }
4384 }
4385 }
4386
4387 if (fLockMedia)
4388 {
4389 rc = aMediumLockList->Lock();
4390 if (FAILED(rc))
4391 {
4392 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4393 throw setError(rc,
4394 tr("Failed to lock media when merging to '%ls'"),
4395 pTarget->getLocationFull().raw());
4396 }
4397 }
4398 }
4399 catch (HRESULT aRC) { rc = aRC; }
4400
4401 if (FAILED(rc))
4402 {
4403 delete aMediumLockList;
4404 aMediumLockList = NULL;
4405 }
4406
4407 return rc;
4408}
4409
4410/**
4411 * Merges this hard disk to the specified hard disk which must be either its
4412 * direct ancestor or descendant.
4413 *
4414 * Given this hard disk is SOURCE and the specified hard disk is TARGET, we will
4415 * get two varians of the merge operation:
4416 *
4417 * forward merge
4418 * ------------------------->
4419 * [Extra] <- SOURCE <- Intermediate <- TARGET
4420 * Any Del Del LockWr
4421 *
4422 *
4423 * backward merge
4424 * <-------------------------
4425 * TARGET <- Intermediate <- SOURCE <- [Extra]
4426 * LockWr Del Del LockWr
4427 *
4428 * Each diagram shows the involved hard disks on the hard disk chain where
4429 * SOURCE and TARGET belong. Under each hard disk there is a state value which
4430 * the hard disk must have at a time of the mergeTo() call.
4431 *
4432 * The hard disks in the square braces may be absent (e.g. when the forward
4433 * operation takes place and SOURCE is the base hard disk, or when the backward
4434 * merge operation takes place and TARGET is the last child in the chain) but if
4435 * they present they are involved too as shown.
4436 *
4437 * Nor the source hard disk neither intermediate hard disks may be attached to
4438 * any VM directly or in the snapshot, otherwise this method will assert.
4439 *
4440 * The #prepareMergeTo() method must be called prior to this method to place all
4441 * involved to necessary states and perform other consistency checks.
4442 *
4443 * If @a aWait is @c true then this method will perform the operation on the
4444 * calling thread and will not return to the caller until the operation is
4445 * completed. When this method succeeds, all intermediate hard disk objects in
4446 * the chain will be uninitialized, the state of the target hard disk (and all
4447 * involved extra hard disks) will be restored. @a aMediumLockList will not be
4448 * deleted, whether the operation is successful or not. The caller has to do
4449 * this if appropriate. Note that this (source) hard disk is not uninitialized
4450 * because of possible AutoCaller instances held by the caller of this method
4451 * on the current thread. It's therefore the responsibility of the caller to
4452 * call Medium::uninit() after releasing all callers.
4453 *
4454 * If @a aWait is @c false then this method will create a thread to perform the
4455 * operation asynchronously and will return immediately. If the operation
4456 * succeeds, the thread will uninitialize the source hard disk object and all
4457 * intermediate hard disk objects in the chain, reset the state of the target
4458 * hard disk (and all involved extra hard disks) and delete @a aMediumLockList.
4459 * If the operation fails, the thread will only reset the states of all
4460 * involved hard disks and delete @a aMediumLockList.
4461 *
4462 * When this method fails (regardless of the @a aWait mode), it is a caller's
4463 * responsiblity to undo state changes and delete @a aMediumLockList using
4464 * #cancelMergeTo().
4465 *
4466 * If @a aProgress is not NULL but the object it points to is @c null then a new
4467 * progress object will be created and assigned to @a *aProgress on success,
4468 * otherwise the existing progress object is used. If Progress is NULL, then no
4469 * progress object is created/used at all. Note that @a aProgress cannot be
4470 * NULL when @a aWait is @c false (this method will assert in this case).
4471 *
4472 * @param pTarget Target hard disk.
4473 * @param fMergeForward Merge direction.
4474 * @param pParentForTarget New parent for target medium after merge.
4475 * @param aChildrenToReparent List of children of the source which will have
4476 * to be reparented to the target after merge.
4477 * @param aMediumLockList Medium locking information.
4478 * @param aProgress Where to find/store a Progress object to track operation
4479 * completion.
4480 * @param aWait @c true if this method should block instead of creating
4481 * an asynchronous thread.
4482 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
4483 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
4484 * This only works in "wait" mode; otherwise saveSettings gets called automatically by the thread that was created,
4485 * and this parameter is ignored.
4486 *
4487 * @note Locks the tree lock for writing. Locks the hard disks from the chain
4488 * for writing.
4489 */
4490HRESULT Medium::mergeTo(const ComObjPtr<Medium> &pTarget,
4491 bool fMergeForward,
4492 const ComObjPtr<Medium> &pParentForTarget,
4493 const MediaList &aChildrenToReparent,
4494 MediumLockList *aMediumLockList,
4495 ComObjPtr <Progress> *aProgress,
4496 bool aWait,
4497 bool *pfNeedsSaveSettings)
4498{
4499 AssertReturn(pTarget != NULL, E_FAIL);
4500 AssertReturn(pTarget != this, E_FAIL);
4501 AssertReturn(aMediumLockList != NULL, E_FAIL);
4502 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
4503
4504 AutoCaller autoCaller(this);
4505 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4506
4507 HRESULT rc = S_OK;
4508 ComObjPtr <Progress> pProgress;
4509 Medium::Task *pTask = NULL;
4510
4511 try
4512 {
4513 if (aProgress != NULL)
4514 {
4515 /* use the existing progress object... */
4516 pProgress = *aProgress;
4517
4518 /* ...but create a new one if it is null */
4519 if (pProgress.isNull())
4520 {
4521 Utf8Str tgtName;
4522 {
4523 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
4524 tgtName = pTarget->getName();
4525 }
4526
4527 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4528
4529 pProgress.createObject();
4530 rc = pProgress->init(m->pVirtualBox,
4531 static_cast<IMedium*>(this),
4532 BstrFmt(tr("Merging hard disk '%s' to '%s'"),
4533 getName().raw(),
4534 tgtName.raw()),
4535 TRUE /* aCancelable */);
4536 if (FAILED(rc))
4537 throw rc;
4538 }
4539 }
4540
4541 /* setup task object to carry out the operation sync/async */
4542 pTask = new Medium::MergeTask(this, pTarget, fMergeForward,
4543 pParentForTarget, aChildrenToReparent,
4544 pProgress, aMediumLockList,
4545 aWait /* fKeepMediumLockList */);
4546 rc = pTask->rc();
4547 AssertComRC(rc);
4548 if (FAILED(rc))
4549 throw rc;
4550 }
4551 catch (HRESULT aRC) { rc = aRC; }
4552
4553 if (SUCCEEDED(rc))
4554 {
4555 if (aWait)
4556 rc = runNow(pTask, pfNeedsSaveSettings);
4557 else
4558 rc = startThread(pTask);
4559
4560 if (SUCCEEDED(rc) && aProgress != NULL)
4561 *aProgress = pProgress;
4562 }
4563 else if (pTask != NULL)
4564 delete pTask;
4565
4566 return rc;
4567}
4568
4569/**
4570 * Undoes what #prepareMergeTo() did. Must be called if #mergeTo() is not
4571 * called or fails. Frees memory occupied by @a aMediumLockList and unlocks
4572 * the medium objects in @a aChildrenToReparent.
4573 *
4574 * @param aChildrenToReparent List of children of the source which will have
4575 * to be reparented to the target after merge.
4576 * @param aMediumLockList Medium locking information.
4577 *
4578 * @note Locks the hard disks from the chain for writing.
4579 */
4580void Medium::cancelMergeTo(const MediaList &aChildrenToReparent,
4581 MediumLockList *aMediumLockList)
4582{
4583 AutoCaller autoCaller(this);
4584 AssertComRCReturnVoid(autoCaller.rc());
4585
4586 AssertReturnVoid(aMediumLockList != NULL);
4587
4588 /* Revert media marked for deletion to previous state. */
4589 HRESULT rc;
4590 MediumLockList::Base::const_iterator mediumListBegin =
4591 aMediumLockList->GetBegin();
4592 MediumLockList::Base::const_iterator mediumListEnd =
4593 aMediumLockList->GetEnd();
4594 for (MediumLockList::Base::const_iterator it = mediumListBegin;
4595 it != mediumListEnd;
4596 ++it)
4597 {
4598 const MediumLock &mediumLock = *it;
4599 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
4600 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
4601
4602 if (pMedium->m->state == MediumState_Deleting)
4603 {
4604 rc = pMedium->unmarkForDeletion();
4605 AssertComRC(rc);
4606 }
4607 }
4608
4609 /* the destructor will do the work */
4610 delete aMediumLockList;
4611
4612 /* unlock the children which had to be reparented */
4613 for (MediaList::const_iterator it = aChildrenToReparent.begin();
4614 it != aChildrenToReparent.end();
4615 ++it)
4616 {
4617 const ComObjPtr<Medium> &pMedium = *it;
4618
4619 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
4620 pMedium->UnlockWrite(NULL);
4621 }
4622}
4623
4624/**
4625 * Checks that the format ID is valid and sets it on success.
4626 *
4627 * Note that this method will caller-reference the format object on success!
4628 * This reference must be released somewhere to let the MediumFormat object be
4629 * uninitialized.
4630 *
4631 * @note Must be called from under this object's write lock.
4632 */
4633HRESULT Medium::setFormat(CBSTR aFormat)
4634{
4635 /* get the format object first */
4636 {
4637 AutoReadLock propsLock(m->pVirtualBox->systemProperties() COMMA_LOCKVAL_SRC_POS);
4638
4639 unconst(m->formatObj)
4640 = m->pVirtualBox->systemProperties()->mediumFormat(aFormat);
4641 if (m->formatObj.isNull())
4642 return setError(E_INVALIDARG,
4643 tr("Invalid hard disk storage format '%ls'"),
4644 aFormat);
4645
4646 /* reference the format permanently to prevent its unexpected
4647 * uninitialization */
4648 HRESULT rc = m->formatObj->addCaller();
4649 AssertComRCReturnRC(rc);
4650
4651 /* get properties (preinsert them as keys in the map). Note that the
4652 * map doesn't grow over the object life time since the set of
4653 * properties is meant to be constant. */
4654
4655 Assert(m->properties.empty());
4656
4657 for (MediumFormat::PropertyList::const_iterator it =
4658 m->formatObj->properties().begin();
4659 it != m->formatObj->properties().end();
4660 ++it)
4661 {
4662 m->properties.insert(std::make_pair(it->name, Bstr::Null));
4663 }
4664 }
4665
4666 unconst(m->strFormat) = aFormat;
4667
4668 return S_OK;
4669}
4670
4671/**
4672 * @note Also reused by Medium::Reset().
4673 *
4674 * @note Caller must hold the media tree write lock!
4675 */
4676HRESULT Medium::canClose()
4677{
4678 Assert(m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
4679
4680 if (getChildren().size() != 0)
4681 return setError(E_FAIL,
4682 tr("Cannot close medium '%s' because it has %d child hard disk(s)"),
4683 m->strLocationFull.raw(), getChildren().size());
4684
4685 return S_OK;
4686}
4687
4688/**
4689 * Calls either VirtualBox::unregisterImage or VirtualBox::unregisterHardDisk depending
4690 * on the device type of this medium.
4691 *
4692 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
4693 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
4694 *
4695 * @note Caller must have locked the media tree lock for writing!
4696 */
4697HRESULT Medium::unregisterWithVirtualBox(bool *pfNeedsSaveSettings)
4698{
4699 /* Note that we need to de-associate ourselves from the parent to let
4700 * unregisterHardDisk() properly save the registry */
4701
4702 /* we modify mParent and access children */
4703 Assert(m->pVirtualBox->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
4704
4705 Medium *pParentBackup = m->pParent;
4706 AssertReturn(getChildren().size() == 0, E_FAIL);
4707 if (m->pParent)
4708 deparent();
4709
4710 HRESULT rc = E_FAIL;
4711 switch (m->devType)
4712 {
4713 case DeviceType_DVD:
4714 rc = m->pVirtualBox->unregisterImage(this, DeviceType_DVD, pfNeedsSaveSettings);
4715 break;
4716
4717 case DeviceType_Floppy:
4718 rc = m->pVirtualBox->unregisterImage(this, DeviceType_Floppy, pfNeedsSaveSettings);
4719 break;
4720
4721 case DeviceType_HardDisk:
4722 rc = m->pVirtualBox->unregisterHardDisk(this, pfNeedsSaveSettings);
4723 break;
4724
4725 default:
4726 break;
4727 }
4728
4729 if (FAILED(rc))
4730 {
4731 if (pParentBackup)
4732 {
4733 /* re-associate with the parent as we are still relatives in the
4734 * registry */
4735 m->pParent = pParentBackup;
4736 m->pParent->m->llChildren.push_back(this);
4737 }
4738 }
4739
4740 return rc;
4741}
4742
4743/**
4744 * Returns the last error message collected by the vdErrorCall callback and
4745 * resets it.
4746 *
4747 * The error message is returned prepended with a dot and a space, like this:
4748 * <code>
4749 * ". <error_text> (%Rrc)"
4750 * </code>
4751 * to make it easily appendable to a more general error message. The @c %Rrc
4752 * format string is given @a aVRC as an argument.
4753 *
4754 * If there is no last error message collected by vdErrorCall or if it is a
4755 * null or empty string, then this function returns the following text:
4756 * <code>
4757 * " (%Rrc)"
4758 * </code>
4759 *
4760 * @note Doesn't do any object locking; it is assumed that the caller makes sure
4761 * the callback isn't called by more than one thread at a time.
4762 *
4763 * @param aVRC VBox error code to use when no error message is provided.
4764 */
4765Utf8Str Medium::vdError(int aVRC)
4766{
4767 Utf8Str error;
4768
4769 if (m->vdError.isEmpty())
4770 error = Utf8StrFmt(" (%Rrc)", aVRC);
4771 else
4772 error = Utf8StrFmt(".\n%s", m->vdError.raw());
4773
4774 m->vdError.setNull();
4775
4776 return error;
4777}
4778
4779/**
4780 * Error message callback.
4781 *
4782 * Puts the reported error message to the m->vdError field.
4783 *
4784 * @note Doesn't do any object locking; it is assumed that the caller makes sure
4785 * the callback isn't called by more than one thread at a time.
4786 *
4787 * @param pvUser The opaque data passed on container creation.
4788 * @param rc The VBox error code.
4789 * @param RT_SRC_POS_DECL Use RT_SRC_POS.
4790 * @param pszFormat Error message format string.
4791 * @param va Error message arguments.
4792 */
4793/*static*/
4794DECLCALLBACK(void) Medium::vdErrorCall(void *pvUser, int rc, RT_SRC_POS_DECL,
4795 const char *pszFormat, va_list va)
4796{
4797 NOREF(pszFile); NOREF(iLine); NOREF(pszFunction); /* RT_SRC_POS_DECL */
4798
4799 Medium *that = static_cast<Medium*>(pvUser);
4800 AssertReturnVoid(that != NULL);
4801
4802 if (that->m->vdError.isEmpty())
4803 that->m->vdError =
4804 Utf8StrFmt("%s (%Rrc)", Utf8StrFmtVA(pszFormat, va).raw(), rc);
4805 else
4806 that->m->vdError =
4807 Utf8StrFmt("%s.\n%s (%Rrc)", that->m->vdError.raw(),
4808 Utf8StrFmtVA(pszFormat, va).raw(), rc);
4809}
4810
4811/* static */
4812DECLCALLBACK(bool) Medium::vdConfigAreKeysValid(void *pvUser,
4813 const char * /* pszzValid */)
4814{
4815 Medium *that = static_cast<Medium*>(pvUser);
4816 AssertReturn(that != NULL, false);
4817
4818 /* we always return true since the only keys we have are those found in
4819 * VDBACKENDINFO */
4820 return true;
4821}
4822
4823/* static */
4824DECLCALLBACK(int) Medium::vdConfigQuerySize(void *pvUser, const char *pszName,
4825 size_t *pcbValue)
4826{
4827 AssertReturn(VALID_PTR(pcbValue), VERR_INVALID_POINTER);
4828
4829 Medium *that = static_cast<Medium*>(pvUser);
4830 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
4831
4832 Data::PropertyMap::const_iterator it =
4833 that->m->properties.find(Bstr(pszName));
4834 if (it == that->m->properties.end())
4835 return VERR_CFGM_VALUE_NOT_FOUND;
4836
4837 /* we interpret null values as "no value" in Medium */
4838 if (it->second.isEmpty())
4839 return VERR_CFGM_VALUE_NOT_FOUND;
4840
4841 *pcbValue = it->second.length() + 1 /* include terminator */;
4842
4843 return VINF_SUCCESS;
4844}
4845
4846/* static */
4847DECLCALLBACK(int) Medium::vdConfigQuery(void *pvUser, const char *pszName,
4848 char *pszValue, size_t cchValue)
4849{
4850 AssertReturn(VALID_PTR(pszValue), VERR_INVALID_POINTER);
4851
4852 Medium *that = static_cast<Medium*>(pvUser);
4853 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
4854
4855 Data::PropertyMap::const_iterator it =
4856 that->m->properties.find(Bstr(pszName));
4857 if (it == that->m->properties.end())
4858 return VERR_CFGM_VALUE_NOT_FOUND;
4859
4860 Utf8Str value = it->second;
4861 if (value.length() >= cchValue)
4862 return VERR_CFGM_NOT_ENOUGH_SPACE;
4863
4864 /* we interpret null values as "no value" in Medium */
4865 if (it->second.isEmpty())
4866 return VERR_CFGM_VALUE_NOT_FOUND;
4867
4868 memcpy(pszValue, value.c_str(), value.length() + 1);
4869
4870 return VINF_SUCCESS;
4871}
4872
4873/**
4874 * Starts a new thread driven by the appropriate Medium::Task::handler() method.
4875 *
4876 * @note When the task is executed by this method, IProgress::notifyComplete()
4877 * is automatically called for the progress object associated with this
4878 * task when the task is finished to signal the operation completion for
4879 * other threads asynchronously waiting for it.
4880 */
4881HRESULT Medium::startThread(Medium::Task *pTask)
4882{
4883#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
4884 /* Extreme paranoia: The calling thread should not hold the medium
4885 * tree lock or any medium lock. Since there is no separate lock class
4886 * for medium objects be even more strict: no other object locks. */
4887 Assert(!AutoLockHoldsLocksInClass(LOCKCLASS_LISTOFMEDIA));
4888 Assert(!AutoLockHoldsLocksInClass(getLockingClass()));
4889#endif
4890
4891 /// @todo use a more descriptive task name
4892 int vrc = RTThreadCreate(NULL, Medium::Task::fntMediumTask, pTask,
4893 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
4894 "Medium::Task");
4895 if (RT_FAILURE(vrc))
4896 {
4897 delete pTask;
4898 ComAssertMsgRCRet(vrc,
4899 ("Could not create Medium::Task thread (%Rrc)\n",
4900 vrc),
4901 E_FAIL);
4902 }
4903
4904 return S_OK;
4905}
4906
4907/**
4908 * Runs Medium::Task::handler() on the current thread instead of creating
4909 * a new one.
4910 *
4911 * This call implies that it is made on another temporary thread created for
4912 * some asynchronous task. Avoid calling it from a normal thread since the task
4913 * operations are potentially lengthy and will block the calling thread in this
4914 * case.
4915 *
4916 * @note When the task is executed by this method, IProgress::notifyComplete()
4917 * is not called for the progress object associated with this task when
4918 * the task is finished. Instead, the result of the operation is returned
4919 * by this method directly and it's the caller's responsibility to
4920 * complete the progress object in this case.
4921 */
4922HRESULT Medium::runNow(Medium::Task *pTask,
4923 bool *pfNeedsSaveSettings)
4924{
4925#ifdef VBOX_WITH_MAIN_LOCK_VALIDATION
4926 /* Extreme paranoia: The calling thread should not hold the medium
4927 * tree lock or any medium lock. Since there is no separate lock class
4928 * for medium objects be even more strict: no other object locks. */
4929 Assert(!AutoLockHoldsLocksInClass(LOCKCLASS_LISTOFMEDIA));
4930 Assert(!AutoLockHoldsLocksInClass(getLockingClass()));
4931#endif
4932
4933 pTask->m_pfNeedsSaveSettings = pfNeedsSaveSettings;
4934
4935 /* NIL_RTTHREAD indicates synchronous call. */
4936 return (HRESULT)Medium::Task::fntMediumTask(NIL_RTTHREAD, pTask);
4937}
4938
4939/**
4940 * Implementation code for the "create base" task.
4941 *
4942 * This only gets started from Medium::CreateBaseStorage() and always runs
4943 * asynchronously. As a result, we always save the VirtualBox.xml file when
4944 * we're done here.
4945 *
4946 * @param task
4947 * @return
4948 */
4949HRESULT Medium::taskCreateBaseHandler(Medium::CreateBaseTask &task)
4950{
4951 HRESULT rc = S_OK;
4952
4953 /* these parameters we need after creation */
4954 uint64_t size = 0, logicalSize = 0;
4955 bool fGenerateUuid = false;
4956
4957 try
4958 {
4959 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
4960
4961 /* The object may request a specific UUID (through a special form of
4962 * the setLocation() argument). Otherwise we have to generate it */
4963 Guid id = m->id;
4964 fGenerateUuid = id.isEmpty();
4965 if (fGenerateUuid)
4966 {
4967 id.create();
4968 /* VirtualBox::registerHardDisk() will need UUID */
4969 unconst(m->id) = id;
4970 }
4971
4972 Utf8Str format(m->strFormat);
4973 Utf8Str location(m->strLocationFull);
4974 uint64_t capabilities = m->formatObj->capabilities();
4975 ComAssertThrow(capabilities & ( VD_CAP_CREATE_FIXED
4976 | VD_CAP_CREATE_DYNAMIC), E_FAIL);
4977 Assert(m->state == MediumState_Creating);
4978
4979 PVBOXHDD hdd;
4980 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
4981 ComAssertRCThrow(vrc, E_FAIL);
4982
4983 /* unlock before the potentially lengthy operation */
4984 thisLock.leave();
4985
4986 try
4987 {
4988 /* ensure the directory exists */
4989 rc = VirtualBox::ensureFilePathExists(location);
4990 if (FAILED(rc))
4991 throw rc;
4992
4993 PDMMEDIAGEOMETRY geo = { 0, 0, 0 }; /* auto-detect */
4994
4995 vrc = VDCreateBase(hdd,
4996 format.c_str(),
4997 location.c_str(),
4998 task.mSize * _1M,
4999 task.mVariant,
5000 NULL,
5001 &geo,
5002 &geo,
5003 id.raw(),
5004 VD_OPEN_FLAGS_NORMAL,
5005 NULL,
5006 task.mVDOperationIfaces);
5007 if (RT_FAILURE(vrc))
5008 {
5009 throw setError(E_FAIL,
5010 tr("Could not create the hard disk storage unit '%s'%s"),
5011 location.raw(), vdError(vrc).raw());
5012 }
5013
5014 size = VDGetFileSize(hdd, 0);
5015 logicalSize = VDGetSize(hdd, 0) / _1M;
5016 }
5017 catch (HRESULT aRC) { rc = aRC; }
5018
5019 VDDestroy(hdd);
5020 }
5021 catch (HRESULT aRC) { rc = aRC; }
5022
5023 if (SUCCEEDED(rc))
5024 {
5025 /* register with mVirtualBox as the last step and move to
5026 * Created state only on success (leaving an orphan file is
5027 * better than breaking media registry consistency) */
5028 bool fNeedsSaveSettings = false;
5029 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5030 rc = m->pVirtualBox->registerHardDisk(this, &fNeedsSaveSettings);
5031 treeLock.release();
5032
5033 if (fNeedsSaveSettings)
5034 {
5035 AutoWriteLock vboxlock(m->pVirtualBox COMMA_LOCKVAL_SRC_POS);
5036 m->pVirtualBox->saveSettings();
5037 }
5038 }
5039
5040 // reenter the lock before changing state
5041 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5042
5043 if (SUCCEEDED(rc))
5044 {
5045 m->state = MediumState_Created;
5046
5047 m->size = size;
5048 m->logicalSize = logicalSize;
5049 }
5050 else
5051 {
5052 /* back to NotCreated on failure */
5053 m->state = MediumState_NotCreated;
5054
5055 /* reset UUID to prevent it from being reused next time */
5056 if (fGenerateUuid)
5057 unconst(m->id).clear();
5058 }
5059
5060 return rc;
5061}
5062
5063/**
5064 * Implementation code for the "create diff" task.
5065 *
5066 * This task always gets started from Medium::createDiffStorage() and can run
5067 * synchronously or asynchronously depending on the "wait" parameter passed to
5068 * that function. If we run synchronously, the caller expects the bool
5069 * *pfNeedsSaveSettings to be set before returning; otherwise (in asynchronous
5070 * mode), we save the settings ourselves.
5071 *
5072 * @param task
5073 * @return
5074 */
5075HRESULT Medium::taskCreateDiffHandler(Medium::CreateDiffTask &task)
5076{
5077 HRESULT rc = S_OK;
5078
5079 bool fNeedsSaveSettings = false;
5080
5081 const ComObjPtr<Medium> &pTarget = task.mTarget;
5082
5083 uint64_t size = 0, logicalSize = 0;
5084 bool fGenerateUuid = false;
5085
5086 try
5087 {
5088 /* Lock both in {parent,child} order. */
5089 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
5090
5091 /* The object may request a specific UUID (through a special form of
5092 * the setLocation() argument). Otherwise we have to generate it */
5093 Guid targetId = pTarget->m->id;
5094 fGenerateUuid = targetId.isEmpty();
5095 if (fGenerateUuid)
5096 {
5097 targetId.create();
5098 /* VirtualBox::registerHardDisk() will need UUID */
5099 unconst(pTarget->m->id) = targetId;
5100 }
5101
5102 Guid id = m->id;
5103
5104 Utf8Str targetFormat(pTarget->m->strFormat);
5105 Utf8Str targetLocation(pTarget->m->strLocationFull);
5106 uint64_t capabilities = m->formatObj->capabilities();
5107 ComAssertThrow(capabilities & VD_CAP_CREATE_DYNAMIC, E_FAIL);
5108
5109 Assert(pTarget->m->state == MediumState_Creating);
5110 Assert(m->state == MediumState_LockedRead);
5111
5112 PVBOXHDD hdd;
5113 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5114 ComAssertRCThrow(vrc, E_FAIL);
5115
5116 /* the two media are now protected by their non-default states;
5117 * unlock the media before the potentially lengthy operation */
5118 mediaLock.leave();
5119
5120 try
5121 {
5122 /* Open all hard disk images in the target chain but the last. */
5123 MediumLockList::Base::const_iterator targetListBegin =
5124 task.mpMediumLockList->GetBegin();
5125 MediumLockList::Base::const_iterator targetListEnd =
5126 task.mpMediumLockList->GetEnd();
5127 for (MediumLockList::Base::const_iterator it = targetListBegin;
5128 it != targetListEnd;
5129 ++it)
5130 {
5131 const MediumLock &mediumLock = *it;
5132 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
5133
5134 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
5135
5136 /* Skip over the target diff image */
5137 if (pMedium->m->state == MediumState_Creating)
5138 continue;
5139
5140 /* sanity check */
5141 Assert(pMedium->m->state == MediumState_LockedRead);
5142
5143 /* Open all images in appropriate mode. */
5144 vrc = VDOpen(hdd,
5145 pMedium->m->strFormat.c_str(),
5146 pMedium->m->strLocationFull.c_str(),
5147 VD_OPEN_FLAGS_READONLY,
5148 pMedium->m->vdDiskIfaces);
5149 if (RT_FAILURE(vrc))
5150 throw setError(E_FAIL,
5151 tr("Could not open the hard disk storage unit '%s'%s"),
5152 pMedium->m->strLocationFull.raw(),
5153 vdError(vrc).raw());
5154 }
5155
5156 /* ensure the target directory exists */
5157 rc = VirtualBox::ensureFilePathExists(targetLocation);
5158 if (FAILED(rc))
5159 throw rc;
5160
5161 vrc = VDCreateDiff(hdd,
5162 targetFormat.c_str(),
5163 targetLocation.c_str(),
5164 task.mVariant | VD_IMAGE_FLAGS_DIFF,
5165 NULL,
5166 targetId.raw(),
5167 id.raw(),
5168 VD_OPEN_FLAGS_NORMAL,
5169 pTarget->m->vdDiskIfaces,
5170 task.mVDOperationIfaces);
5171 if (RT_FAILURE(vrc))
5172 throw setError(E_FAIL,
5173 tr("Could not create the differencing hard disk storage unit '%s'%s"),
5174 targetLocation.raw(), vdError(vrc).raw());
5175
5176 size = VDGetFileSize(hdd, 1);
5177 logicalSize = VDGetSize(hdd, 1) / _1M;
5178 }
5179 catch (HRESULT aRC) { rc = aRC; }
5180
5181 VDDestroy(hdd);
5182 }
5183 catch (HRESULT aRC) { rc = aRC; }
5184
5185 if (SUCCEEDED(rc))
5186 {
5187 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5188
5189 Assert(pTarget->m->pParent.isNull());
5190
5191 /* associate the child with the parent */
5192 pTarget->m->pParent = this;
5193 m->llChildren.push_back(pTarget);
5194
5195 /** @todo r=klaus neither target nor base() are locked,
5196 * potential race! */
5197 /* diffs for immutable hard disks are auto-reset by default */
5198 pTarget->m->autoReset = (getBase()->m->type == MediumType_Immutable);
5199
5200 /* register with mVirtualBox as the last step and move to
5201 * Created state only on success (leaving an orphan file is
5202 * better than breaking media registry consistency) */
5203 rc = m->pVirtualBox->registerHardDisk(pTarget, &fNeedsSaveSettings);
5204
5205 if (FAILED(rc))
5206 /* break the parent association on failure to register */
5207 deparent();
5208 }
5209
5210 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
5211
5212 if (SUCCEEDED(rc))
5213 {
5214 pTarget->m->state = MediumState_Created;
5215
5216 pTarget->m->size = size;
5217 pTarget->m->logicalSize = logicalSize;
5218 }
5219 else
5220 {
5221 /* back to NotCreated on failure */
5222 pTarget->m->state = MediumState_NotCreated;
5223
5224 pTarget->m->autoReset = FALSE;
5225
5226 /* reset UUID to prevent it from being reused next time */
5227 if (fGenerateUuid)
5228 unconst(pTarget->m->id).clear();
5229 }
5230
5231 if (task.isAsync())
5232 {
5233 if (fNeedsSaveSettings)
5234 {
5235 mediaLock.leave();
5236 AutoWriteLock vboxlock(m->pVirtualBox COMMA_LOCKVAL_SRC_POS);
5237 m->pVirtualBox->saveSettings();
5238 }
5239 }
5240 else
5241 // synchronous mode: report save settings result to caller
5242 if (task.m_pfNeedsSaveSettings)
5243 *task.m_pfNeedsSaveSettings = fNeedsSaveSettings;
5244
5245 /* deregister the task registered in createDiffStorage() */
5246 Assert(m->numCreateDiffTasks != 0);
5247 --m->numCreateDiffTasks;
5248
5249 /* Note that in sync mode, it's the caller's responsibility to
5250 * unlock the hard disk */
5251
5252 return rc;
5253}
5254
5255/**
5256 * Implementation code for the "merge" task.
5257 *
5258 * This task always gets started from Medium::mergeTo() and can run
5259 * synchronously or asynchrously depending on the "wait" parameter passed to
5260 * that function. If we run synchronously, the caller expects the bool
5261 * *pfNeedsSaveSettings to be set before returning; otherwise (in asynchronous
5262 * mode), we save the settings ourselves.
5263 *
5264 * @param task
5265 * @return
5266 */
5267HRESULT Medium::taskMergeHandler(Medium::MergeTask &task)
5268{
5269 HRESULT rc = S_OK;
5270
5271 const ComObjPtr<Medium> &pTarget = task.mTarget;
5272
5273 try
5274 {
5275 PVBOXHDD hdd;
5276 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5277 ComAssertRCThrow(vrc, E_FAIL);
5278
5279 try
5280 {
5281 unsigned uTargetIdx = VD_LAST_IMAGE;
5282 unsigned uSourceIdx = VD_LAST_IMAGE;
5283 /* Open all hard disks in the chain. */
5284 MediumLockList::Base::iterator lockListBegin =
5285 task.mpMediumLockList->GetBegin();
5286 MediumLockList::Base::iterator lockListEnd =
5287 task.mpMediumLockList->GetEnd();
5288 unsigned i = 0;
5289 for (MediumLockList::Base::iterator it = lockListBegin;
5290 it != lockListEnd;
5291 ++it)
5292 {
5293 MediumLock &mediumLock = *it;
5294 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
5295
5296 if (pMedium == this)
5297 uSourceIdx = i;
5298 else if (pMedium == pTarget)
5299 uTargetIdx = i;
5300
5301 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
5302
5303 /*
5304 * complex sanity (sane complexity)
5305 *
5306 * The current image must be in the Deleting (image is merged)
5307 * or LockedRead (parent image) state if it is not the target.
5308 * If it is the target it must be in the LockedWrite state.
5309 */
5310 Assert( ( pMedium != pTarget
5311 && ( pMedium->m->state == MediumState_Deleting
5312 || pMedium->m->state == MediumState_LockedRead))
5313 || ( pMedium == pTarget
5314 && pMedium->m->state == MediumState_LockedWrite));
5315
5316 /*
5317 * Image must be the target, in the LockedRead state
5318 * or Deleting state where it is not allowed to be attached
5319 * to a virtual machine.
5320 */
5321 Assert( pMedium == pTarget
5322 || pMedium->m->state == MediumState_LockedRead
5323 || ( pMedium->m->backRefs.size() == 0
5324 && pMedium->m->state == MediumState_Deleting));
5325 /* The source medium must be in Deleting state. */
5326 Assert( pMedium != this
5327 || pMedium->m->state == MediumState_Deleting);
5328
5329 unsigned uOpenFlags = 0;
5330
5331 if ( pMedium->m->state == MediumState_LockedRead
5332 || pMedium->m->state == MediumState_Deleting)
5333 uOpenFlags = VD_OPEN_FLAGS_READONLY;
5334
5335 /* Open the image */
5336 vrc = VDOpen(hdd,
5337 pMedium->m->strFormat.c_str(),
5338 pMedium->m->strLocationFull.c_str(),
5339 uOpenFlags,
5340 pMedium->m->vdDiskIfaces);
5341 if (RT_FAILURE(vrc))
5342 throw vrc;
5343
5344 i++;
5345 }
5346
5347 ComAssertThrow( uSourceIdx != VD_LAST_IMAGE
5348 && uTargetIdx != VD_LAST_IMAGE, E_FAIL);
5349
5350 vrc = VDMerge(hdd, uSourceIdx, uTargetIdx,
5351 task.mVDOperationIfaces);
5352 if (RT_FAILURE(vrc))
5353 throw vrc;
5354
5355 /* update parent UUIDs */
5356 if (!task.mfMergeForward)
5357 {
5358 /* we need to update UUIDs of all source's children
5359 * which cannot be part of the container at once so
5360 * add each one in there individually */
5361 if (task.mChildrenToReparent.size() > 0)
5362 {
5363 for (MediaList::const_iterator it = task.mChildrenToReparent.begin();
5364 it != task.mChildrenToReparent.end();
5365 ++it)
5366 {
5367 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
5368 vrc = VDOpen(hdd,
5369 (*it)->m->strFormat.c_str(),
5370 (*it)->m->strLocationFull.c_str(),
5371 VD_OPEN_FLAGS_INFO,
5372 (*it)->m->vdDiskIfaces);
5373 if (RT_FAILURE(vrc))
5374 throw vrc;
5375
5376 vrc = VDSetParentUuid(hdd, 1,
5377 pTarget->m->id);
5378 if (RT_FAILURE(vrc))
5379 throw vrc;
5380
5381 vrc = VDClose(hdd, false /* fDelete */);
5382 if (RT_FAILURE(vrc))
5383 throw vrc;
5384 }
5385 }
5386 }
5387 }
5388 catch (HRESULT aRC) { rc = aRC; }
5389 catch (int aVRC)
5390 {
5391 throw setError(E_FAIL,
5392 tr("Could not merge the hard disk '%s' to '%s'%s"),
5393 m->strLocationFull.raw(), m->strLocationFull.raw(),
5394 vdError(aVRC).raw());
5395 }
5396
5397 VDDestroy(hdd);
5398 }
5399 catch (HRESULT aRC) { rc = aRC; }
5400
5401 HRESULT rc2;
5402
5403 if (SUCCEEDED(rc))
5404 {
5405 /* all hard disks but the target were successfully deleted by
5406 * VDMerge; reparent the last one and uninitialize deleted media. */
5407
5408 AutoWriteLock treeLock(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5409
5410 if (task.mfMergeForward)
5411 {
5412 /* first, unregister the target since it may become a base
5413 * hard disk which needs re-registration */
5414 rc2 = m->pVirtualBox->unregisterHardDisk(pTarget, NULL /*&fNeedsSaveSettings*/);
5415 AssertComRC(rc2);
5416
5417 /* then, reparent it and disconnect the deleted branch at
5418 * both ends (chain->parent() is source's parent) */
5419 pTarget->deparent();
5420 pTarget->m->pParent = task.mParentForTarget;
5421 if (pTarget->m->pParent)
5422 {
5423 pTarget->m->pParent->m->llChildren.push_back(pTarget);
5424 deparent();
5425 }
5426
5427 /* then, register again */
5428 rc2 = m->pVirtualBox->registerHardDisk(pTarget, NULL /*&fNeedsSaveSettings*/);
5429 AssertComRC(rc2);
5430 }
5431 else
5432 {
5433 Assert(pTarget->getChildren().size() == 1);
5434 Medium *targetChild = pTarget->getChildren().front();
5435
5436 /* disconnect the deleted branch at the elder end */
5437 targetChild->deparent();
5438
5439 /* reparent source's chidren and disconnect the deleted
5440 * branch at the younger end m*/
5441 if (task.mChildrenToReparent.size() > 0)
5442 {
5443 /* obey {parent,child} lock order */
5444 AutoWriteLock sourceLock(this COMMA_LOCKVAL_SRC_POS);
5445
5446 for (MediaList::iterator it = task.mChildrenToReparent.begin();
5447 it != task.mChildrenToReparent.end();
5448 it++)
5449 {
5450 Medium *pMedium = *it;
5451 AutoWriteLock childLock(pMedium COMMA_LOCKVAL_SRC_POS);
5452
5453 pMedium->deparent(); // removes pMedium from source
5454 pTarget->m->llChildren.push_back(pMedium);
5455 pMedium->m->pParent = pTarget;
5456 }
5457 }
5458 }
5459
5460 /* unregister and uninitialize all hard disks removed by the merge */
5461 MediumLockList::Base::iterator lockListBegin =
5462 task.mpMediumLockList->GetBegin();
5463 MediumLockList::Base::iterator lockListEnd =
5464 task.mpMediumLockList->GetEnd();
5465 for (MediumLockList::Base::iterator it = lockListBegin;
5466 it != lockListEnd;
5467 )
5468 {
5469 MediumLock &mediumLock = *it;
5470 /* Create a real copy of the medium pointer, as the medium
5471 * lock deletion below would invalidate the referenced object. */
5472 const ComObjPtr<Medium> pMedium = mediumLock.GetMedium();
5473
5474 /* The target and all images not merged (readonly) are skipped */
5475 if ( pMedium == pTarget
5476 || pMedium->m->state == MediumState_LockedRead)
5477 {
5478 ++it;
5479 continue;
5480 }
5481
5482 rc2 = pMedium->m->pVirtualBox->unregisterHardDisk(pMedium,
5483 NULL /*pfNeedsSaveSettings*/);
5484 AssertComRC(rc2);
5485
5486 /* now, uninitialize the deleted hard disk (note that
5487 * due to the Deleting state, uninit() will not touch
5488 * the parent-child relationship so we need to
5489 * uninitialize each disk individually) */
5490
5491 /* note that the operation initiator hard disk (which is
5492 * normally also the source hard disk) is a special case
5493 * -- there is one more caller added by Task to it which
5494 * we must release. Also, if we are in sync mode, the
5495 * caller may still hold an AutoCaller instance for it
5496 * and therefore we cannot uninit() it (it's therefore
5497 * the caller's responsibility) */
5498 if (pMedium == this)
5499 {
5500 Assert(getChildren().size() == 0);
5501 Assert(m->backRefs.size() == 0);
5502 task.mMediumCaller.release();
5503 }
5504
5505 /* Delete the medium lock list entry, which also releases the
5506 * caller added by MergeChain before uninit() and updates the
5507 * iterator to point to the right place. */
5508 rc2 = task.mpMediumLockList->RemoveByIterator(it);
5509 AssertComRC(rc2);
5510
5511 if (task.isAsync() || pMedium != this)
5512 pMedium->uninit();
5513 }
5514 }
5515
5516 if (task.isAsync())
5517 {
5518 // in asynchronous mode, save settings now
5519 AutoWriteLock vboxlock(m->pVirtualBox COMMA_LOCKVAL_SRC_POS);
5520 m->pVirtualBox->saveSettings();
5521 }
5522 else
5523 // synchronous mode: report save settings result to caller
5524 if (task.m_pfNeedsSaveSettings)
5525 *task.m_pfNeedsSaveSettings = true;
5526
5527 if (FAILED(rc))
5528 {
5529 /* Here we come if either VDMerge() failed (in which case we
5530 * assume that it tried to do everything to make a further
5531 * retry possible -- e.g. not deleted intermediate hard disks
5532 * and so on) or VirtualBox::saveSettings() failed (where we
5533 * should have the original tree but with intermediate storage
5534 * units deleted by VDMerge()). We have to only restore states
5535 * (through the MergeChain dtor) unless we are run synchronously
5536 * in which case it's the responsibility of the caller as stated
5537 * in the mergeTo() docs. The latter also implies that we
5538 * don't own the merge chain, so release it in this case. */
5539 if (task.isAsync())
5540 {
5541 Assert(task.mChildrenToReparent.size() == 0);
5542 cancelMergeTo(task.mChildrenToReparent, task.mpMediumLockList);
5543 }
5544 }
5545
5546 return rc;
5547}
5548
5549/**
5550 * Implementation code for the "clone" task.
5551 *
5552 * This only gets started from Medium::CloneTo() and always runs asynchronously.
5553 * As a result, we always save the VirtualBox.xml file when we're done here.
5554 *
5555 * @param task
5556 * @return
5557 */
5558HRESULT Medium::taskCloneHandler(Medium::CloneTask &task)
5559{
5560 HRESULT rc = S_OK;
5561
5562 const ComObjPtr<Medium> &pTarget = task.mTarget;
5563 const ComObjPtr<Medium> &pParent = task.mParent;
5564
5565 bool fCreatingTarget = false;
5566
5567 uint64_t size = 0, logicalSize = 0;
5568 bool fGenerateUuid = false;
5569
5570 try
5571 {
5572 /* Lock all in {parent,child} order. The lock is also used as a
5573 * signal from the task initiator (which releases it only after
5574 * RTThreadCreate()) that we can start the job. */
5575 AutoMultiWriteLock3 thisLock(this, pTarget, pParent COMMA_LOCKVAL_SRC_POS);
5576
5577 fCreatingTarget = pTarget->m->state == MediumState_Creating;
5578
5579 /* The object may request a specific UUID (through a special form of
5580 * the setLocation() argument). Otherwise we have to generate it */
5581 Guid targetId = pTarget->m->id;
5582 fGenerateUuid = targetId.isEmpty();
5583 if (fGenerateUuid)
5584 {
5585 targetId.create();
5586 /* VirtualBox::registerHardDisk() will need UUID */
5587 unconst(pTarget->m->id) = targetId;
5588 }
5589
5590 PVBOXHDD hdd;
5591 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5592 ComAssertRCThrow(vrc, E_FAIL);
5593
5594 try
5595 {
5596 /* Open all hard disk images in the source chain. */
5597 MediumLockList::Base::const_iterator sourceListBegin =
5598 task.mpSourceMediumLockList->GetBegin();
5599 MediumLockList::Base::const_iterator sourceListEnd =
5600 task.mpSourceMediumLockList->GetEnd();
5601 for (MediumLockList::Base::const_iterator it = sourceListBegin;
5602 it != sourceListEnd;
5603 ++it)
5604 {
5605 const MediumLock &mediumLock = *it;
5606 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
5607 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
5608
5609 /* sanity check */
5610 Assert(pMedium->m->state == MediumState_LockedRead);
5611
5612 /** Open all images in read-only mode. */
5613 vrc = VDOpen(hdd,
5614 pMedium->m->strFormat.c_str(),
5615 pMedium->m->strLocationFull.c_str(),
5616 VD_OPEN_FLAGS_READONLY,
5617 pMedium->m->vdDiskIfaces);
5618 if (RT_FAILURE(vrc))
5619 throw setError(E_FAIL,
5620 tr("Could not open the hard disk storage unit '%s'%s"),
5621 pMedium->m->strLocationFull.raw(),
5622 vdError(vrc).raw());
5623 }
5624
5625 Utf8Str targetFormat(pTarget->m->strFormat);
5626 Utf8Str targetLocation(pTarget->m->strLocationFull);
5627
5628 Assert( pTarget->m->state == MediumState_Creating
5629 || pTarget->m->state == MediumState_LockedWrite);
5630 Assert(m->state == MediumState_LockedRead);
5631 Assert(pParent.isNull() || pParent->m->state == MediumState_LockedRead);
5632
5633 /* unlock before the potentially lengthy operation */
5634 thisLock.leave();
5635
5636 /* ensure the target directory exists */
5637 rc = VirtualBox::ensureFilePathExists(targetLocation);
5638 if (FAILED(rc))
5639 throw rc;
5640
5641 PVBOXHDD targetHdd;
5642 vrc = VDCreate(m->vdDiskIfaces, &targetHdd);
5643 ComAssertRCThrow(vrc, E_FAIL);
5644
5645 try
5646 {
5647 /* Open all hard disk images in the target chain. */
5648 MediumLockList::Base::const_iterator targetListBegin =
5649 task.mpTargetMediumLockList->GetBegin();
5650 MediumLockList::Base::const_iterator targetListEnd =
5651 task.mpTargetMediumLockList->GetEnd();
5652 for (MediumLockList::Base::const_iterator it = targetListBegin;
5653 it != targetListEnd;
5654 ++it)
5655 {
5656 const MediumLock &mediumLock = *it;
5657 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
5658
5659 /* If the target medium is not created yet there's no
5660 * reason to open it. */
5661 if (pMedium == pTarget && fCreatingTarget)
5662 continue;
5663
5664 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
5665
5666 /* sanity check */
5667 Assert( pMedium->m->state == MediumState_LockedRead
5668 || pMedium->m->state == MediumState_LockedWrite);
5669
5670 /* Open all images in appropriate mode. */
5671 vrc = VDOpen(targetHdd,
5672 pMedium->m->strFormat.c_str(),
5673 pMedium->m->strLocationFull.c_str(),
5674 (pMedium->m->state == MediumState_LockedWrite) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
5675 pMedium->m->vdDiskIfaces);
5676 if (RT_FAILURE(vrc))
5677 throw setError(E_FAIL,
5678 tr("Could not open the hard disk storage unit '%s'%s"),
5679 pMedium->m->strLocationFull.raw(),
5680 vdError(vrc).raw());
5681 }
5682
5683 /** @todo r=klaus target isn't locked, race getting the state */
5684 vrc = VDCopy(hdd,
5685 VD_LAST_IMAGE,
5686 targetHdd,
5687 targetFormat.c_str(),
5688 (fCreatingTarget) ? targetLocation.raw() : (char *)NULL,
5689 false,
5690 0,
5691 task.mVariant,
5692 targetId.raw(),
5693 NULL,
5694 pTarget->m->vdDiskIfaces,
5695 task.mVDOperationIfaces);
5696 if (RT_FAILURE(vrc))
5697 throw setError(E_FAIL,
5698 tr("Could not create the clone hard disk '%s'%s"),
5699 targetLocation.raw(), vdError(vrc).raw());
5700
5701 size = VDGetFileSize(targetHdd, 0);
5702 logicalSize = VDGetSize(targetHdd, 0) / _1M;
5703 }
5704 catch (HRESULT aRC) { rc = aRC; }
5705
5706 VDDestroy(targetHdd);
5707 }
5708 catch (HRESULT aRC) { rc = aRC; }
5709
5710 VDDestroy(hdd);
5711 }
5712 catch (HRESULT aRC) { rc = aRC; }
5713
5714 /* Only do the parent changes for newly created images. */
5715 if (SUCCEEDED(rc) && fCreatingTarget)
5716 {
5717 /* we set mParent & children() */
5718 AutoWriteLock alock2(m->pVirtualBox->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5719
5720 Assert(pTarget->m->pParent.isNull());
5721
5722 if (pParent)
5723 {
5724 /* associate the clone with the parent and deassociate
5725 * from VirtualBox */
5726 pTarget->m->pParent = pParent;
5727 pParent->m->llChildren.push_back(pTarget);
5728
5729 /* register with mVirtualBox as the last step and move to
5730 * Created state only on success (leaving an orphan file is
5731 * better than breaking media registry consistency) */
5732 rc = pParent->m->pVirtualBox->registerHardDisk(pTarget, NULL /* pfNeedsSaveSettings */);
5733
5734 if (FAILED(rc))
5735 /* break parent association on failure to register */
5736 pTarget->deparent(); // removes target from parent
5737 }
5738 else
5739 {
5740 /* just register */
5741 rc = m->pVirtualBox->registerHardDisk(pTarget, NULL /* pfNeedsSaveSettings */);
5742 }
5743 }
5744
5745 if (fCreatingTarget)
5746 {
5747 AutoWriteLock mLock(pTarget COMMA_LOCKVAL_SRC_POS);
5748
5749 if (SUCCEEDED(rc))
5750 {
5751 pTarget->m->state = MediumState_Created;
5752
5753 pTarget->m->size = size;
5754 pTarget->m->logicalSize = logicalSize;
5755 }
5756 else
5757 {
5758 /* back to NotCreated on failure */
5759 pTarget->m->state = MediumState_NotCreated;
5760
5761 /* reset UUID to prevent it from being reused next time */
5762 if (fGenerateUuid)
5763 unconst(pTarget->m->id).clear();
5764 }
5765 }
5766
5767 // now, at the end of this task (always asynchronous), save the settings
5768 {
5769 AutoWriteLock vboxlock(m->pVirtualBox COMMA_LOCKVAL_SRC_POS);
5770 m->pVirtualBox->saveSettings();
5771 }
5772
5773 /* Everything is explicitly unlocked when the task exits,
5774 * as the task destruction also destroys the source chain. */
5775
5776 /* Make sure the source chain is released early. It could happen
5777 * that we get a deadlock in Appliance::Import when Medium::Close
5778 * is called & the source chain is released at the same time. */
5779 task.mpSourceMediumLockList->Clear();
5780
5781 return rc;
5782}
5783
5784/**
5785 * Implementation code for the "delete" task.
5786 *
5787 * This task always gets started from Medium::deleteStorage() and can run
5788 * synchronously or asynchrously depending on the "wait" parameter passed to
5789 * that function.
5790 *
5791 * @param task
5792 * @return
5793 */
5794HRESULT Medium::taskDeleteHandler(Medium::DeleteTask &task)
5795{
5796 NOREF(task);
5797 HRESULT rc = S_OK;
5798
5799 try
5800 {
5801 /* The lock is also used as a signal from the task initiator (which
5802 * releases it only after RTThreadCreate()) that we can start the job */
5803 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5804
5805 PVBOXHDD hdd;
5806 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5807 ComAssertRCThrow(vrc, E_FAIL);
5808
5809 Utf8Str format(m->strFormat);
5810 Utf8Str location(m->strLocationFull);
5811
5812 /* unlock before the potentially lengthy operation */
5813 Assert(m->state == MediumState_Deleting);
5814 thisLock.release();
5815
5816 try
5817 {
5818 vrc = VDOpen(hdd,
5819 format.c_str(),
5820 location.c_str(),
5821 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
5822 m->vdDiskIfaces);
5823 if (RT_SUCCESS(vrc))
5824 vrc = VDClose(hdd, true /* fDelete */);
5825
5826 if (RT_FAILURE(vrc))
5827 throw setError(E_FAIL,
5828 tr("Could not delete the hard disk storage unit '%s'%s"),
5829 location.raw(), vdError(vrc).raw());
5830
5831 }
5832 catch (HRESULT aRC) { rc = aRC; }
5833
5834 VDDestroy(hdd);
5835 }
5836 catch (HRESULT aRC) { rc = aRC; }
5837
5838 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5839
5840 /* go to the NotCreated state even on failure since the storage
5841 * may have been already partially deleted and cannot be used any
5842 * more. One will be able to manually re-open the storage if really
5843 * needed to re-register it. */
5844 m->state = MediumState_NotCreated;
5845
5846 /* Reset UUID to prevent Create* from reusing it again */
5847 unconst(m->id).clear();
5848
5849 return rc;
5850}
5851
5852/**
5853 * Implementation code for the "reset" task.
5854 *
5855 * This always gets started asynchronously from Medium::Reset().
5856 *
5857 * @param task
5858 * @return
5859 */
5860HRESULT Medium::taskResetHandler(Medium::ResetTask &task)
5861{
5862 HRESULT rc = S_OK;
5863
5864 uint64_t size = 0, logicalSize = 0;
5865
5866 try
5867 {
5868 /* The lock is also used as a signal from the task initiator (which
5869 * releases it only after RTThreadCreate()) that we can start the job */
5870 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5871
5872 /// @todo Below we use a pair of delete/create operations to reset
5873 /// the diff contents but the most efficient way will of course be
5874 /// to add a VDResetDiff() API call
5875
5876 PVBOXHDD hdd;
5877 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5878 ComAssertRCThrow(vrc, E_FAIL);
5879
5880 Guid id = m->id;
5881 Utf8Str format(m->strFormat);
5882 Utf8Str location(m->strLocationFull);
5883
5884 Medium *pParent = m->pParent;
5885 Guid parentId = pParent->m->id;
5886 Utf8Str parentFormat(pParent->m->strFormat);
5887 Utf8Str parentLocation(pParent->m->strLocationFull);
5888
5889 Assert(m->state == MediumState_LockedWrite);
5890
5891 /* unlock before the potentially lengthy operation */
5892 thisLock.release();
5893
5894 try
5895 {
5896 /* first, delete the storage unit */
5897 vrc = VDOpen(hdd,
5898 format.c_str(),
5899 location.c_str(),
5900 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
5901 m->vdDiskIfaces);
5902 if (RT_SUCCESS(vrc))
5903 vrc = VDClose(hdd, true /* fDelete */);
5904
5905 if (RT_FAILURE(vrc))
5906 throw setError(E_FAIL,
5907 tr("Could not delete the hard disk storage unit '%s'%s"),
5908 location.raw(), vdError(vrc).raw());
5909
5910 /* next, create it again */
5911 vrc = VDOpen(hdd,
5912 parentFormat.c_str(),
5913 parentLocation.c_str(),
5914 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
5915 m->vdDiskIfaces);
5916 if (RT_FAILURE(vrc))
5917 throw setError(E_FAIL,
5918 tr("Could not open the hard disk storage unit '%s'%s"),
5919 parentLocation.raw(), vdError(vrc).raw());
5920
5921 vrc = VDCreateDiff(hdd,
5922 format.c_str(),
5923 location.c_str(),
5924 /// @todo use the same image variant as before
5925 VD_IMAGE_FLAGS_NONE,
5926 NULL,
5927 id.raw(),
5928 parentId.raw(),
5929 VD_OPEN_FLAGS_NORMAL,
5930 m->vdDiskIfaces,
5931 task.mVDOperationIfaces);
5932 if (RT_FAILURE(vrc))
5933 throw setError(E_FAIL,
5934 tr("Could not create the differencing hard disk storage unit '%s'%s"),
5935 location.raw(), vdError(vrc).raw());
5936
5937 size = VDGetFileSize(hdd, 1);
5938 logicalSize = VDGetSize(hdd, 1) / _1M;
5939 }
5940 catch (HRESULT aRC) { rc = aRC; }
5941
5942 VDDestroy(hdd);
5943 }
5944 catch (HRESULT aRC) { rc = aRC; }
5945
5946 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5947
5948 m->size = size;
5949 m->logicalSize = logicalSize;
5950
5951 if (task.isAsync())
5952 {
5953 /* unlock ourselves when done */
5954 HRESULT rc2 = UnlockWrite(NULL);
5955 AssertComRC(rc2);
5956 }
5957
5958 /* Note that in sync mode, it's the caller's responsibility to
5959 * unlock the hard disk */
5960
5961 return rc;
5962}
5963
5964/**
5965 * Implementation code for the "compact" task.
5966 *
5967 * @param task
5968 * @return
5969 */
5970HRESULT Medium::taskCompactHandler(Medium::CompactTask &task)
5971{
5972 HRESULT rc = S_OK;
5973
5974 /* Lock all in {parent,child} order. The lock is also used as a
5975 * signal from the task initiator (which releases it only after
5976 * RTThreadCreate()) that we can start the job. */
5977 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
5978
5979 try
5980 {
5981 PVBOXHDD hdd;
5982 int vrc = VDCreate(m->vdDiskIfaces, &hdd);
5983 ComAssertRCThrow(vrc, E_FAIL);
5984
5985 try
5986 {
5987 /* Open all hard disk images in the chain. */
5988 MediumLockList::Base::const_iterator mediumListBegin =
5989 task.mpMediumLockList->GetBegin();
5990 MediumLockList::Base::const_iterator mediumListEnd =
5991 task.mpMediumLockList->GetEnd();
5992 MediumLockList::Base::const_iterator mediumListLast =
5993 mediumListEnd;
5994 mediumListLast--;
5995 for (MediumLockList::Base::const_iterator it = mediumListBegin;
5996 it != mediumListEnd;
5997 ++it)
5998 {
5999 const MediumLock &mediumLock = *it;
6000 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
6001 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
6002
6003 /* sanity check */
6004 if (it == mediumListLast)
6005 Assert(pMedium->m->state == MediumState_LockedWrite);
6006 else
6007 Assert(pMedium->m->state == MediumState_LockedRead);
6008
6009 /** Open all images but last in read-only mode. */
6010 vrc = VDOpen(hdd,
6011 pMedium->m->strFormat.c_str(),
6012 pMedium->m->strLocationFull.c_str(),
6013 (it == mediumListLast) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
6014 pMedium->m->vdDiskIfaces);
6015 if (RT_FAILURE(vrc))
6016 throw setError(E_FAIL,
6017 tr("Could not open the hard disk storage unit '%s'%s"),
6018 pMedium->m->strLocationFull.raw(),
6019 vdError(vrc).raw());
6020 }
6021
6022 Assert(m->state == MediumState_LockedWrite);
6023
6024 Utf8Str location(m->strLocationFull);
6025
6026 /* unlock before the potentially lengthy operation */
6027 thisLock.leave();
6028
6029 vrc = VDCompact(hdd, VD_LAST_IMAGE, task.mVDOperationIfaces);
6030 if (RT_FAILURE(vrc))
6031 {
6032 if (vrc == VERR_NOT_SUPPORTED)
6033 throw setError(VBOX_E_NOT_SUPPORTED,
6034 tr("Compacting is not yet supported for hard disk '%s'"),
6035 location.raw());
6036 else if (vrc == VERR_NOT_IMPLEMENTED)
6037 throw setError(E_NOTIMPL,
6038 tr("Compacting is not implemented, hard disk '%s'"),
6039 location.raw());
6040 else
6041 throw setError(E_FAIL,
6042 tr("Could not compact hard disk '%s'%s"),
6043 location.raw(),
6044 vdError(vrc).raw());
6045 }
6046 }
6047 catch (HRESULT aRC) { rc = aRC; }
6048
6049 VDDestroy(hdd);
6050 }
6051 catch (HRESULT aRC) { rc = aRC; }
6052
6053 /* Everything is explicitly unlocked when the task exits,
6054 * as the task destruction also destroys the image chain. */
6055
6056 return rc;
6057}
6058
6059/* 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