VirtualBox

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

Last change on this file since 33064 was 32934, checked in by vboxsync, 15 years ago

Main/Medium: Automatically repair streamOptimized VMDK files too. They are never created as diff images by normal VirtualBox operations, and forcing them to be base images allows importing OVF appliances created by broken VirtualBox versions.

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

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette