VirtualBox

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

Last change on this file since 104981 was 102590, checked in by vboxsync, 12 months ago

Unattended: implemented quick fix so files are properly cleaned up after interrupted vm execution, bugref:10180

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 397.6 KB
Line 
1/* $Id: MediumImpl.cpp 102590 2023-12-13 04:42:31Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2008-2023 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * SPDX-License-Identifier: GPL-3.0-only
26 */
27
28#define LOG_GROUP LOG_GROUP_MAIN_MEDIUM
29#include "MediumImpl.h"
30#include "MediumIOImpl.h"
31#include "TokenImpl.h"
32#include "ProgressImpl.h"
33#include "SystemPropertiesImpl.h"
34#include "VirtualBoxImpl.h"
35#include "ExtPackManagerImpl.h"
36
37#include "AutoCaller.h"
38#include "Global.h"
39#include "LoggingNew.h"
40#include "ThreadTask.h"
41#include "VBox/com/MultiResult.h"
42#include "VBox/com/ErrorInfo.h"
43
44#include <VBox/err.h>
45#include <VBox/settings.h>
46
47#include <iprt/param.h>
48#include <iprt/path.h>
49#include <iprt/file.h>
50#include <iprt/cpp/utils.h>
51#include <iprt/memsafer.h>
52#include <iprt/base64.h>
53#include <iprt/vfs.h>
54#include <iprt/fsvfs.h>
55
56#include <VBox/vd.h>
57
58#include <algorithm>
59#include <list>
60#include <set>
61#include <map>
62
63
64typedef std::list<Guid> GuidList;
65
66
67#ifdef VBOX_WITH_EXTPACK
68static const char g_szVDPlugin[] = "VDPluginCrypt";
69#endif
70
71
72////////////////////////////////////////////////////////////////////////////////
73//
74// Medium data definition
75//
76////////////////////////////////////////////////////////////////////////////////
77#if __cplusplus < 201700 && RT_GNUC_PREREQ(11,0) /* gcc/libstdc++ 12.1.1 errors out here because unary_function is deprecated */
78# pragma GCC diagnostic push
79# pragma GCC diagnostic ignored "-Wdeprecated-declarations"
80#endif
81
82struct SnapshotRef
83{
84 /** Equality predicate for stdc++. */
85 struct EqualsTo
86#if __cplusplus < 201700 /* deprecated in C++11, removed in C++17. */
87 : public std::unary_function <SnapshotRef, bool>
88#endif
89 {
90 explicit EqualsTo(const Guid &aSnapshotId) : snapshotId(aSnapshotId) {}
91
92#if __cplusplus < 201700 /* deprecated in C++11, removed in C++17. */
93 bool operator()(const argument_type &aThat) const
94#else
95 bool operator()(const SnapshotRef &aThat) const
96#endif
97 {
98 return aThat.snapshotId == snapshotId;
99 }
100
101 const Guid snapshotId;
102 };
103
104 SnapshotRef(const Guid &aSnapshotId,
105 const int &aRefCnt = 1)
106 : snapshotId(aSnapshotId),
107 iRefCnt(aRefCnt) {}
108
109 Guid snapshotId;
110 /*
111 * The number of attachments of the medium in the same snapshot.
112 * Used for MediumType_Readonly. It is always equal to 1 for other types.
113 * Usual int is used because any changes in the BackRef are guarded by
114 * AutoWriteLock.
115 */
116 int iRefCnt;
117};
118
119/** Describes how a machine refers to this medium. */
120struct BackRef
121{
122 /** Equality predicate for stdc++. */
123 struct EqualsTo
124#if __cplusplus < 201700 /* deprecated in C++11, removed in C++17. */
125 : public std::unary_function <BackRef, bool>
126#endif
127 {
128 explicit EqualsTo(const Guid &aMachineId) : machineId(aMachineId) {}
129
130#if __cplusplus < 201700 /* deprecated in C++11, removed in C++17. */
131 bool operator()(const argument_type &aThat) const
132#else
133 bool operator()(const BackRef &aThat) const
134#endif
135 {
136 return aThat.machineId == machineId;
137 }
138
139 const Guid machineId;
140 };
141
142 BackRef(const Guid &aMachineId,
143 const Guid &aSnapshotId = Guid::Empty)
144 : machineId(aMachineId),
145 iRefCnt(1),
146 fInCurState(aSnapshotId.isZero())
147 {
148 if (aSnapshotId.isValid() && !aSnapshotId.isZero())
149 llSnapshotIds.push_back(SnapshotRef(aSnapshotId));
150 }
151
152 Guid machineId;
153 /*
154 * The number of attachments of the medium in the same machine.
155 * Used for MediumType_Readonly. It is always equal to 1 for other types.
156 * Usual int is used because any changes in the BackRef are guarded by
157 * AutoWriteLock.
158 */
159 int iRefCnt;
160 bool fInCurState : 1;
161 std::list<SnapshotRef> llSnapshotIds;
162};
163
164typedef std::list<BackRef> BackRefList;
165
166#if __cplusplus < 201700 && RT_GNUC_PREREQ(11,0)
167# pragma GCC diagnostic pop
168#endif
169
170
171struct Medium::Data
172{
173 Data()
174 : pVirtualBox(NULL),
175 state(MediumState_NotCreated),
176 variant(MediumVariant_Standard),
177 size(0),
178 readers(0),
179 preLockState(MediumState_NotCreated),
180 queryInfoSem(LOCKCLASS_MEDIUMQUERY),
181 queryInfoRunning(false),
182 type(MediumType_Normal),
183 devType(DeviceType_HardDisk),
184 logicalSize(0),
185 hddOpenMode(OpenReadWrite),
186 autoReset(false),
187 hostDrive(false),
188 implicit(false),
189 fClosing(false),
190 uOpenFlagsDef(VD_OPEN_FLAGS_IGNORE_FLUSH),
191 numCreateDiffTasks(0),
192 vdDiskIfaces(NULL),
193 vdImageIfaces(NULL),
194 fMoveThisMedium(false)
195 { }
196
197 /** weak VirtualBox parent */
198 VirtualBox * const pVirtualBox;
199
200 // pParent and llChildren are protected by VirtualBox::i_getMediaTreeLockHandle()
201 ComObjPtr<Medium> pParent;
202 MediaList llChildren; // to add a child, just call push_back; to remove
203 // a child, call child->deparent() which does a lookup
204
205 GuidList llRegistryIDs; // media registries in which this medium is listed
206
207 const Guid id;
208 Utf8Str strDescription;
209 MediumState_T state;
210 MediumVariant_T variant;
211 Utf8Str strLocationFull;
212 uint64_t size;
213 Utf8Str strLastAccessError;
214
215 BackRefList backRefs;
216
217 size_t readers;
218 MediumState_T preLockState;
219
220 /** Special synchronization for operations which must wait for
221 * Medium::i_queryInfo in another thread to complete. Using a SemRW is
222 * not quite ideal, but at least it is subject to the lock validator,
223 * unlike the SemEventMulti which we had here for many years. Catching
224 * possible deadlocks is more important than a tiny bit of efficiency. */
225 RWLockHandle queryInfoSem;
226 bool queryInfoRunning : 1;
227
228 const Utf8Str strFormat;
229 ComObjPtr<MediumFormat> formatObj;
230
231 MediumType_T type;
232 DeviceType_T devType;
233 uint64_t logicalSize;
234
235 HDDOpenMode hddOpenMode;
236
237 bool autoReset : 1;
238
239 /** New UUID to be set on the next Medium::i_queryInfo call. */
240 const Guid uuidImage;
241 /** New parent UUID to be set on the next Medium::i_queryInfo call. */
242 const Guid uuidParentImage;
243
244/** @todo r=bird: the boolean bitfields are pointless if they're not grouped! */
245 bool hostDrive : 1;
246
247 settings::StringsMap mapProperties;
248
249 bool implicit : 1;
250 /** Flag whether the medium is in the process of being closed. */
251 bool fClosing: 1;
252
253 /** Default flags passed to VDOpen(). */
254 unsigned uOpenFlagsDef;
255
256 uint32_t numCreateDiffTasks;
257
258 Utf8Str vdError; /*< Error remembered by the VD error callback. */
259
260 VDINTERFACEERROR vdIfError;
261
262 VDINTERFACECONFIG vdIfConfig;
263
264 /** The handle to the default VD TCP/IP interface. */
265 VDIFINST hTcpNetInst;
266
267 PVDINTERFACE vdDiskIfaces;
268 PVDINTERFACE vdImageIfaces;
269
270 /** Flag if the medium is going to move to a new
271 * location. */
272 bool fMoveThisMedium;
273 /** new location path */
274 Utf8Str strNewLocationFull;
275};
276
277typedef struct VDSOCKETINT
278{
279 /** Socket handle. */
280 RTSOCKET hSocket;
281} VDSOCKETINT, *PVDSOCKETINT;
282
283////////////////////////////////////////////////////////////////////////////////
284//
285// Globals
286//
287////////////////////////////////////////////////////////////////////////////////
288
289/**
290 * Medium::Task class for asynchronous operations.
291 *
292 * @note Instances of this class must be created using new() because the
293 * task thread function will delete them when the task is complete.
294 *
295 * @note The constructor of this class adds a caller on the managed Medium
296 * object which is automatically released upon destruction.
297 */
298class Medium::Task : public ThreadTask
299{
300public:
301 Task(Medium *aMedium, Progress *aProgress, bool fNotifyAboutChanges = true)
302 : ThreadTask("Medium::Task"),
303 mVDOperationIfaces(NULL),
304 mMedium(aMedium),
305 mMediumCaller(aMedium),
306 mProgress(aProgress),
307 mVirtualBoxCaller(NULL),
308 mNotifyAboutChanges(fNotifyAboutChanges)
309 {
310 AssertReturnVoidStmt(aMedium, mHrc = E_FAIL);
311 mHrc = mMediumCaller.hrc();
312 if (FAILED(mHrc))
313 return;
314
315 /* Get strong VirtualBox reference, see below. */
316 VirtualBox *pVirtualBox = aMedium->m->pVirtualBox;
317 mVirtualBox = pVirtualBox;
318 mVirtualBoxCaller.attach(pVirtualBox);
319 mHrc = mVirtualBoxCaller.hrc();
320 if (FAILED(mHrc))
321 return;
322
323 /* Set up a per-operation progress interface, can be used freely (for
324 * binary operations you can use it either on the source or target). */
325 if (mProgress)
326 {
327 mVDIfProgress.pfnProgress = aProgress->i_vdProgressCallback;
328 int vrc = VDInterfaceAdd(&mVDIfProgress.Core,
329 "Medium::Task::vdInterfaceProgress",
330 VDINTERFACETYPE_PROGRESS,
331 mProgress,
332 sizeof(mVDIfProgress),
333 &mVDOperationIfaces);
334 AssertRC(vrc);
335 if (RT_FAILURE(vrc))
336 mHrc = E_FAIL;
337 }
338 }
339
340 // Make all destructors virtual. Just in case.
341 virtual ~Task()
342 {
343 /* send the notification of completion.*/
344 if ( isAsync()
345 && !mProgress.isNull())
346 mProgress->i_notifyComplete(mHrc);
347 }
348
349 HRESULT hrc() const { return mHrc; }
350 bool isOk() const { return SUCCEEDED(hrc()); }
351 bool NotifyAboutChanges() const { return mNotifyAboutChanges; }
352
353 const ComPtr<Progress>& GetProgressObject() const {return mProgress;}
354
355 /**
356 * Runs Medium::Task::executeTask() on the current thread
357 * instead of creating a new one.
358 */
359 HRESULT runNow()
360 {
361 LogFlowFuncEnter();
362
363 mHrc = executeTask();
364
365 LogFlowFunc(("hrc=%Rhrc\n", mHrc));
366 LogFlowFuncLeave();
367 return mHrc;
368 }
369
370 /**
371 * Implementation code for the "create base" task.
372 * Used as function for execution from a standalone thread.
373 */
374 void handler()
375 {
376 LogFlowFuncEnter();
377 try
378 {
379 mHrc = executeTask(); /* (destructor picks up mHrc, see above) */
380 LogFlowFunc(("hrc=%Rhrc\n", mHrc));
381 }
382 catch (...)
383 {
384 LogRel(("Some exception in the function Medium::Task:handler()\n"));
385 }
386
387 LogFlowFuncLeave();
388 }
389
390 PVDINTERFACE mVDOperationIfaces;
391
392 const ComObjPtr<Medium> mMedium;
393 AutoCaller mMediumCaller;
394
395protected:
396 HRESULT mHrc;
397
398private:
399 virtual HRESULT executeTask() = 0;
400
401 const ComObjPtr<Progress> mProgress;
402
403 VDINTERFACEPROGRESS mVDIfProgress;
404
405 /* Must have a strong VirtualBox reference during a task otherwise the
406 * reference count might drop to 0 while a task is still running. This
407 * would result in weird behavior, including deadlocks due to uninit and
408 * locking order issues. The deadlock often is not detectable because the
409 * uninit uses event semaphores which sabotages deadlock detection. */
410 ComObjPtr<VirtualBox> mVirtualBox;
411 AutoCaller mVirtualBoxCaller;
412 bool mNotifyAboutChanges;
413};
414
415class Medium::CreateBaseTask : public Medium::Task
416{
417public:
418 CreateBaseTask(Medium *aMedium,
419 Progress *aProgress,
420 uint64_t aSize,
421 MediumVariant_T aVariant,
422 bool fNotifyAboutChanges = true)
423 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
424 mSize(aSize),
425 mVariant(aVariant)
426 {
427 m_strTaskName = "createBase";
428 }
429
430 uint64_t mSize;
431 MediumVariant_T mVariant;
432
433private:
434 HRESULT executeTask()
435 {
436 return mMedium->i_taskCreateBaseHandler(*this);
437 }
438};
439
440class Medium::CreateDiffTask : public Medium::Task
441{
442public:
443 CreateDiffTask(Medium *aMedium,
444 Progress *aProgress,
445 Medium *aTarget,
446 MediumVariant_T aVariant,
447 MediumLockList *aMediumLockList,
448 bool fKeepMediumLockList = false,
449 bool fNotifyAboutChanges = true)
450 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
451 mpMediumLockList(aMediumLockList),
452 mTarget(aTarget),
453 mVariant(aVariant),
454 mTargetCaller(aTarget),
455 mfKeepMediumLockList(fKeepMediumLockList)
456 {
457 AssertReturnVoidStmt(aTarget != NULL, mHrc = E_FAIL);
458 mHrc = mTargetCaller.hrc();
459 if (FAILED(mHrc))
460 return;
461 m_strTaskName = "createDiff";
462 }
463
464 ~CreateDiffTask()
465 {
466 if (!mfKeepMediumLockList && mpMediumLockList)
467 delete mpMediumLockList;
468 }
469
470 MediumLockList *mpMediumLockList;
471
472 const ComObjPtr<Medium> mTarget;
473 MediumVariant_T mVariant;
474
475private:
476 HRESULT executeTask()
477 {
478 return mMedium->i_taskCreateDiffHandler(*this);
479 }
480
481 AutoCaller mTargetCaller;
482 bool mfKeepMediumLockList;
483};
484
485class Medium::CloneTask : public Medium::Task
486{
487public:
488 CloneTask(Medium *aMedium,
489 Progress *aProgress,
490 Medium *aTarget,
491 MediumVariant_T aVariant,
492 Medium *aParent,
493 uint32_t idxSrcImageSame,
494 uint32_t idxDstImageSame,
495 MediumLockList *aSourceMediumLockList,
496 MediumLockList *aTargetMediumLockList,
497 bool fKeepSourceMediumLockList = false,
498 bool fKeepTargetMediumLockList = false,
499 bool fNotifyAboutChanges = true,
500 uint64_t aTargetLogicalSize = 0)
501 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
502 mTarget(aTarget),
503 mParent(aParent),
504 mTargetLogicalSize(aTargetLogicalSize),
505 mpSourceMediumLockList(aSourceMediumLockList),
506 mpTargetMediumLockList(aTargetMediumLockList),
507 mVariant(aVariant),
508 midxSrcImageSame(idxSrcImageSame),
509 midxDstImageSame(idxDstImageSame),
510 mTargetCaller(aTarget),
511 mParentCaller(aParent),
512 mfKeepSourceMediumLockList(fKeepSourceMediumLockList),
513 mfKeepTargetMediumLockList(fKeepTargetMediumLockList)
514 {
515 AssertReturnVoidStmt(aTarget != NULL, mHrc = E_FAIL);
516 mHrc = mTargetCaller.hrc();
517 if (FAILED(mHrc))
518 return;
519 /* aParent may be NULL */
520 mHrc = mParentCaller.hrc();
521 if (FAILED(mHrc))
522 return;
523 AssertReturnVoidStmt(aSourceMediumLockList != NULL, mHrc = E_FAIL);
524 AssertReturnVoidStmt(aTargetMediumLockList != NULL, mHrc = E_FAIL);
525 m_strTaskName = "createClone";
526 }
527
528 ~CloneTask()
529 {
530 if (!mfKeepSourceMediumLockList && mpSourceMediumLockList)
531 delete mpSourceMediumLockList;
532 if (!mfKeepTargetMediumLockList && mpTargetMediumLockList)
533 delete mpTargetMediumLockList;
534 }
535
536 const ComObjPtr<Medium> mTarget;
537 const ComObjPtr<Medium> mParent;
538 uint64_t mTargetLogicalSize;
539 MediumLockList *mpSourceMediumLockList;
540 MediumLockList *mpTargetMediumLockList;
541 MediumVariant_T mVariant;
542 uint32_t midxSrcImageSame;
543 uint32_t midxDstImageSame;
544
545private:
546 HRESULT executeTask()
547 {
548 return mMedium->i_taskCloneHandler(*this);
549 }
550
551 AutoCaller mTargetCaller;
552 AutoCaller mParentCaller;
553 bool mfKeepSourceMediumLockList;
554 bool mfKeepTargetMediumLockList;
555};
556
557class Medium::MoveTask : public Medium::Task
558{
559public:
560 MoveTask(Medium *aMedium,
561 Progress *aProgress,
562 MediumVariant_T aVariant,
563 MediumLockList *aMediumLockList,
564 bool fKeepMediumLockList = false,
565 bool fNotifyAboutChanges = true)
566 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
567 mpMediumLockList(aMediumLockList),
568 mVariant(aVariant),
569 mfKeepMediumLockList(fKeepMediumLockList)
570 {
571 AssertReturnVoidStmt(aMediumLockList != NULL, mHrc = E_FAIL);
572 m_strTaskName = "createMove";
573 }
574
575 ~MoveTask()
576 {
577 if (!mfKeepMediumLockList && mpMediumLockList)
578 delete mpMediumLockList;
579 }
580
581 MediumLockList *mpMediumLockList;
582 MediumVariant_T mVariant;
583
584private:
585 HRESULT executeTask()
586 {
587 return mMedium->i_taskMoveHandler(*this);
588 }
589
590 bool mfKeepMediumLockList;
591};
592
593class Medium::CompactTask : public Medium::Task
594{
595public:
596 CompactTask(Medium *aMedium,
597 Progress *aProgress,
598 MediumLockList *aMediumLockList,
599 bool fKeepMediumLockList = false,
600 bool fNotifyAboutChanges = true)
601 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
602 mpMediumLockList(aMediumLockList),
603 mfKeepMediumLockList(fKeepMediumLockList)
604 {
605 AssertReturnVoidStmt(aMediumLockList != NULL, mHrc = E_FAIL);
606 m_strTaskName = "createCompact";
607 }
608
609 ~CompactTask()
610 {
611 if (!mfKeepMediumLockList && mpMediumLockList)
612 delete mpMediumLockList;
613 }
614
615 MediumLockList *mpMediumLockList;
616
617private:
618 HRESULT executeTask()
619 {
620 return mMedium->i_taskCompactHandler(*this);
621 }
622
623 bool mfKeepMediumLockList;
624};
625
626class Medium::ResizeTask : public Medium::Task
627{
628public:
629 ResizeTask(Medium *aMedium,
630 uint64_t aSize,
631 Progress *aProgress,
632 MediumLockList *aMediumLockList,
633 bool fKeepMediumLockList = false,
634 bool fNotifyAboutChanges = true)
635 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
636 mSize(aSize),
637 mpMediumLockList(aMediumLockList),
638 mfKeepMediumLockList(fKeepMediumLockList)
639 {
640 AssertReturnVoidStmt(aMediumLockList != NULL, mHrc = E_FAIL);
641 m_strTaskName = "createResize";
642 }
643
644 ~ResizeTask()
645 {
646 if (!mfKeepMediumLockList && mpMediumLockList)
647 delete mpMediumLockList;
648 }
649
650 uint64_t mSize;
651 MediumLockList *mpMediumLockList;
652
653private:
654 HRESULT executeTask()
655 {
656 return mMedium->i_taskResizeHandler(*this);
657 }
658
659 bool mfKeepMediumLockList;
660};
661
662class Medium::ResetTask : public Medium::Task
663{
664public:
665 ResetTask(Medium *aMedium,
666 Progress *aProgress,
667 MediumLockList *aMediumLockList,
668 bool fKeepMediumLockList = false,
669 bool fNotifyAboutChanges = true)
670 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
671 mpMediumLockList(aMediumLockList),
672 mfKeepMediumLockList(fKeepMediumLockList)
673 {
674 m_strTaskName = "createReset";
675 }
676
677 ~ResetTask()
678 {
679 if (!mfKeepMediumLockList && mpMediumLockList)
680 delete mpMediumLockList;
681 }
682
683 MediumLockList *mpMediumLockList;
684
685private:
686 HRESULT executeTask()
687 {
688 return mMedium->i_taskResetHandler(*this);
689 }
690
691 bool mfKeepMediumLockList;
692};
693
694class Medium::DeleteTask : public Medium::Task
695{
696public:
697 DeleteTask(Medium *aMedium,
698 Progress *aProgress,
699 MediumLockList *aMediumLockList,
700 bool fKeepMediumLockList = false,
701 bool fNotifyAboutChanges = true)
702 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
703 mpMediumLockList(aMediumLockList),
704 mfKeepMediumLockList(fKeepMediumLockList)
705 {
706 m_strTaskName = "createDelete";
707 }
708
709 ~DeleteTask()
710 {
711 if (!mfKeepMediumLockList && mpMediumLockList)
712 delete mpMediumLockList;
713 }
714
715 MediumLockList *mpMediumLockList;
716
717private:
718 HRESULT executeTask()
719 {
720 return mMedium->i_taskDeleteHandler(*this);
721 }
722
723 bool mfKeepMediumLockList;
724};
725
726class Medium::MergeTask : public Medium::Task
727{
728public:
729 MergeTask(Medium *aMedium,
730 Medium *aTarget,
731 bool fMergeForward,
732 Medium *aParentForTarget,
733 MediumLockList *aChildrenToReparent,
734 Progress *aProgress,
735 MediumLockList *aMediumLockList,
736 bool fKeepMediumLockList = false,
737 bool fNotifyAboutChanges = true)
738 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
739 mTarget(aTarget),
740 mfMergeForward(fMergeForward),
741 mParentForTarget(aParentForTarget),
742 mpChildrenToReparent(aChildrenToReparent),
743 mpMediumLockList(aMediumLockList),
744 mTargetCaller(aTarget),
745 mParentForTargetCaller(aParentForTarget),
746 mfKeepMediumLockList(fKeepMediumLockList)
747 {
748 AssertReturnVoidStmt(aMediumLockList != NULL, mHrc = E_FAIL);
749 m_strTaskName = "createMerge";
750 }
751
752 ~MergeTask()
753 {
754 if (!mfKeepMediumLockList && mpMediumLockList)
755 delete mpMediumLockList;
756 if (mpChildrenToReparent)
757 delete mpChildrenToReparent;
758 }
759
760 const ComObjPtr<Medium> mTarget;
761 bool mfMergeForward;
762 /* When mpChildrenToReparent is null then mParentForTarget is non-null and
763 * vice versa. In other words: they are used in different cases. */
764 const ComObjPtr<Medium> mParentForTarget;
765 MediumLockList *mpChildrenToReparent;
766 MediumLockList *mpMediumLockList;
767
768private:
769 HRESULT executeTask()
770 {
771 return mMedium->i_taskMergeHandler(*this);
772 }
773
774 AutoCaller mTargetCaller;
775 AutoCaller mParentForTargetCaller;
776 bool mfKeepMediumLockList;
777};
778
779class Medium::ImportTask : public Medium::Task
780{
781public:
782 ImportTask(Medium *aMedium,
783 Progress *aProgress,
784 const char *aFilename,
785 MediumFormat *aFormat,
786 MediumVariant_T aVariant,
787 RTVFSIOSTREAM aVfsIosSrc,
788 Medium *aParent,
789 MediumLockList *aTargetMediumLockList,
790 bool fKeepTargetMediumLockList = false,
791 bool fNotifyAboutChanges = true)
792 : Medium::Task(aMedium, aProgress, fNotifyAboutChanges),
793 mFilename(aFilename),
794 mFormat(aFormat),
795 mVariant(aVariant),
796 mParent(aParent),
797 mpTargetMediumLockList(aTargetMediumLockList),
798 mpVfsIoIf(NULL),
799 mParentCaller(aParent),
800 mfKeepTargetMediumLockList(fKeepTargetMediumLockList)
801 {
802 AssertReturnVoidStmt(aTargetMediumLockList != NULL, mHrc = E_FAIL);
803 /* aParent may be NULL */
804 mHrc = mParentCaller.hrc();
805 if (FAILED(mHrc))
806 return;
807
808 mVDImageIfaces = aMedium->m->vdImageIfaces;
809
810 int vrc = VDIfCreateFromVfsStream(aVfsIosSrc, RTFILE_O_READ, &mpVfsIoIf);
811 AssertRCReturnVoidStmt(vrc, mHrc = E_FAIL);
812
813 vrc = VDInterfaceAdd(&mpVfsIoIf->Core, "Medium::ImportTaskVfsIos",
814 VDINTERFACETYPE_IO, mpVfsIoIf,
815 sizeof(VDINTERFACEIO), &mVDImageIfaces);
816 AssertRCReturnVoidStmt(vrc, mHrc = E_FAIL);
817 m_strTaskName = "createImport";
818 }
819
820 ~ImportTask()
821 {
822 if (!mfKeepTargetMediumLockList && mpTargetMediumLockList)
823 delete mpTargetMediumLockList;
824 if (mpVfsIoIf)
825 {
826 VDIfDestroyFromVfsStream(mpVfsIoIf);
827 mpVfsIoIf = NULL;
828 }
829 }
830
831 Utf8Str mFilename;
832 ComObjPtr<MediumFormat> mFormat;
833 MediumVariant_T mVariant;
834 const ComObjPtr<Medium> mParent;
835 MediumLockList *mpTargetMediumLockList;
836 PVDINTERFACE mVDImageIfaces;
837 PVDINTERFACEIO mpVfsIoIf; /**< Pointer to the VFS I/O stream to VD I/O interface wrapper. */
838
839private:
840 HRESULT executeTask()
841 {
842 return mMedium->i_taskImportHandler(*this);
843 }
844
845 AutoCaller mParentCaller;
846 bool mfKeepTargetMediumLockList;
847};
848
849class Medium::EncryptTask : public Medium::Task
850{
851public:
852 EncryptTask(Medium *aMedium,
853 const com::Utf8Str &strNewPassword,
854 const com::Utf8Str &strCurrentPassword,
855 const com::Utf8Str &strCipher,
856 const com::Utf8Str &strNewPasswordId,
857 Progress *aProgress,
858 MediumLockList *aMediumLockList)
859 : Medium::Task(aMedium, aProgress, false),
860 mstrNewPassword(strNewPassword),
861 mstrCurrentPassword(strCurrentPassword),
862 mstrCipher(strCipher),
863 mstrNewPasswordId(strNewPasswordId),
864 mpMediumLockList(aMediumLockList)
865 {
866 AssertReturnVoidStmt(aMediumLockList != NULL, mHrc = E_FAIL);
867 /* aParent may be NULL */
868 mHrc = mParentCaller.hrc();
869 if (FAILED(mHrc))
870 return;
871
872 mVDImageIfaces = aMedium->m->vdImageIfaces;
873 m_strTaskName = "createEncrypt";
874 }
875
876 ~EncryptTask()
877 {
878 if (mstrNewPassword.length())
879 RTMemWipeThoroughly(mstrNewPassword.mutableRaw(), mstrNewPassword.length(), 10 /* cPasses */);
880 if (mstrCurrentPassword.length())
881 RTMemWipeThoroughly(mstrCurrentPassword.mutableRaw(), mstrCurrentPassword.length(), 10 /* cPasses */);
882
883 /* Keep any errors which might be set when deleting the lock list. */
884 ErrorInfoKeeper eik;
885 delete mpMediumLockList;
886 }
887
888 Utf8Str mstrNewPassword;
889 Utf8Str mstrCurrentPassword;
890 Utf8Str mstrCipher;
891 Utf8Str mstrNewPasswordId;
892 MediumLockList *mpMediumLockList;
893 PVDINTERFACE mVDImageIfaces;
894
895private:
896 HRESULT executeTask()
897 {
898 return mMedium->i_taskEncryptHandler(*this);
899 }
900
901 AutoCaller mParentCaller;
902};
903
904
905
906/**
907 * Converts the Medium device type to the VD type.
908 */
909static const char *getVDTypeName(VDTYPE enmType)
910{
911 switch (enmType)
912 {
913 case VDTYPE_HDD: return "HDD";
914 case VDTYPE_OPTICAL_DISC: return "DVD";
915 case VDTYPE_FLOPPY: return "floppy";
916 case VDTYPE_INVALID: return "invalid";
917 default:
918 AssertFailedReturn("unknown");
919 }
920}
921
922/**
923 * Converts the Medium device type to the VD type.
924 */
925static const char *getDeviceTypeName(DeviceType_T enmType)
926{
927 switch (enmType)
928 {
929 case DeviceType_HardDisk: return "HDD";
930 case DeviceType_DVD: return "DVD";
931 case DeviceType_Floppy: return "floppy";
932 case DeviceType_Null: return "null";
933 case DeviceType_Network: return "network";
934 case DeviceType_USB: return "USB";
935 case DeviceType_SharedFolder: return "shared folder";
936 case DeviceType_Graphics3D: return "graphics 3d";
937 default:
938 AssertFailedReturn("unknown");
939 }
940}
941
942
943
944////////////////////////////////////////////////////////////////////////////////
945//
946// Medium constructor / destructor
947//
948////////////////////////////////////////////////////////////////////////////////
949
950DEFINE_EMPTY_CTOR_DTOR(Medium)
951
952HRESULT Medium::FinalConstruct()
953{
954 m = new Data;
955
956 /* Initialize the callbacks of the VD error interface */
957 m->vdIfError.pfnError = i_vdErrorCall;
958 m->vdIfError.pfnMessage = NULL;
959
960 /* Initialize the callbacks of the VD config interface */
961 m->vdIfConfig.pfnAreKeysValid = i_vdConfigAreKeysValid;
962 m->vdIfConfig.pfnQuerySize = i_vdConfigQuerySize;
963 m->vdIfConfig.pfnQuery = i_vdConfigQuery;
964 m->vdIfConfig.pfnUpdate = i_vdConfigUpdate;
965 m->vdIfConfig.pfnQueryBytes = NULL;
966
967 /* Initialize the per-disk interface chain (could be done more globally,
968 * but it's not wasting much time or space so it's not worth it). */
969 int vrc;
970 vrc = VDInterfaceAdd(&m->vdIfError.Core,
971 "Medium::vdInterfaceError",
972 VDINTERFACETYPE_ERROR, this,
973 sizeof(VDINTERFACEERROR), &m->vdDiskIfaces);
974 AssertRCReturn(vrc, E_FAIL);
975
976 /* Initialize the per-image interface chain */
977 vrc = VDInterfaceAdd(&m->vdIfConfig.Core,
978 "Medium::vdInterfaceConfig",
979 VDINTERFACETYPE_CONFIG, this,
980 sizeof(VDINTERFACECONFIG), &m->vdImageIfaces);
981 AssertRCReturn(vrc, E_FAIL);
982
983 /* Initialize the callbacks of the VD TCP interface (we always use the host
984 * IP stack for now) */
985 vrc = VDIfTcpNetInstDefaultCreate(&m->hTcpNetInst, &m->vdImageIfaces);
986 AssertRCReturn(vrc, E_FAIL);
987
988 return BaseFinalConstruct();
989}
990
991void Medium::FinalRelease()
992{
993 uninit();
994
995 VDIfTcpNetInstDefaultDestroy(m->hTcpNetInst);
996 delete m;
997
998 BaseFinalRelease();
999}
1000
1001/**
1002 * Initializes an empty hard disk object without creating or opening an associated
1003 * storage unit.
1004 *
1005 * This gets called by VirtualBox::CreateMedium() in which case uuidMachineRegistry
1006 * is empty since starting with VirtualBox 4.0, we no longer add opened media to a
1007 * registry automatically (this is deferred until the medium is attached to a machine).
1008 *
1009 * This also gets called when VirtualBox creates diff images; in this case uuidMachineRegistry
1010 * is set to the registry of the parent image to make sure they all end up in the same
1011 * file.
1012 *
1013 * For hard disks that don't have the MediumFormatCapabilities_CreateFixed or
1014 * MediumFormatCapabilities_CreateDynamic capability (and therefore cannot be created or deleted
1015 * with the means of VirtualBox) the associated storage unit is assumed to be
1016 * ready for use so the state of the hard disk object will be set to Created.
1017 *
1018 * @param aVirtualBox VirtualBox object.
1019 * @param aFormat
1020 * @param aLocation Storage unit location.
1021 * @param uuidMachineRegistry The registry to which this medium should be added
1022 * (global registry UUID or machine UUID or empty if none).
1023 * @param aDeviceType Device Type.
1024 */
1025HRESULT Medium::init(VirtualBox *aVirtualBox,
1026 const Utf8Str &aFormat,
1027 const Utf8Str &aLocation,
1028 const Guid &uuidMachineRegistry,
1029 const DeviceType_T aDeviceType)
1030{
1031 AssertReturn(aVirtualBox != NULL, E_FAIL);
1032 AssertReturn(!aFormat.isEmpty(), E_FAIL);
1033
1034 /* Enclose the state transition NotReady->InInit->Ready */
1035 AutoInitSpan autoInitSpan(this);
1036 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1037
1038 HRESULT hrc = S_OK;
1039
1040 unconst(m->pVirtualBox) = aVirtualBox;
1041
1042 if (uuidMachineRegistry.isValid() && !uuidMachineRegistry.isZero())
1043 m->llRegistryIDs.push_back(uuidMachineRegistry);
1044
1045 /* no storage yet */
1046 m->state = MediumState_NotCreated;
1047
1048 /* cannot be a host drive */
1049 m->hostDrive = false;
1050
1051 m->devType = aDeviceType;
1052
1053 /* No storage unit is created yet, no need to call Medium::i_queryInfo */
1054
1055 hrc = i_setFormat(aFormat);
1056 if (FAILED(hrc)) return hrc;
1057
1058 hrc = i_setLocation(aLocation);
1059 if (FAILED(hrc)) return hrc;
1060
1061 if (!(m->formatObj->i_getCapabilities() & ( MediumFormatCapabilities_CreateFixed
1062 | MediumFormatCapabilities_CreateDynamic
1063 | MediumFormatCapabilities_File))
1064 )
1065 {
1066 /* Storage for media of this format can neither be explicitly
1067 * created by VirtualBox nor deleted, so we place the medium to
1068 * Inaccessible state here and also add it to the registry. The
1069 * state means that one has to use RefreshState() to update the
1070 * medium format specific fields. */
1071 m->state = MediumState_Inaccessible;
1072 // create new UUID
1073 unconst(m->id).create();
1074
1075 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1076 ComObjPtr<Medium> pMedium;
1077
1078 /*
1079 * Check whether the UUID is taken already and create a new one
1080 * if required.
1081 * Try this only a limited amount of times in case the PRNG is broken
1082 * in some way to prevent an endless loop.
1083 */
1084 for (unsigned i = 0; i < 5; i++)
1085 {
1086 bool fInUse;
1087
1088 fInUse = m->pVirtualBox->i_isMediaUuidInUse(m->id, aDeviceType);
1089 if (fInUse)
1090 {
1091 // create new UUID
1092 unconst(m->id).create();
1093 }
1094 else
1095 break;
1096 }
1097
1098 hrc = m->pVirtualBox->i_registerMedium(this, &pMedium, treeLock);
1099 Assert(this == pMedium || FAILED(hrc));
1100 }
1101
1102 /* Confirm a successful initialization when it's the case */
1103 if (SUCCEEDED(hrc))
1104 autoInitSpan.setSucceeded();
1105
1106 return hrc;
1107}
1108
1109/**
1110 * Initializes the medium object by opening the storage unit at the specified
1111 * location. The enOpenMode parameter defines whether the medium will be opened
1112 * read/write or read-only.
1113 *
1114 * This gets called by VirtualBox::OpenMedium() and also by
1115 * Machine::AttachDevice() and createImplicitDiffs() when new diff
1116 * images are created.
1117 *
1118 * There is no registry for this case since starting with VirtualBox 4.0, we
1119 * no longer add opened media to a registry automatically (this is deferred
1120 * until the medium is attached to a machine).
1121 *
1122 * For hard disks, the UUID, format and the parent of this medium will be
1123 * determined when reading the medium storage unit. For DVD and floppy images,
1124 * which have no UUIDs in their storage units, new UUIDs are created.
1125 * If the detected or set parent is not known to VirtualBox, then this method
1126 * will fail.
1127 *
1128 * @param aVirtualBox VirtualBox object.
1129 * @param aLocation Storage unit location.
1130 * @param enOpenMode Whether to open the medium read/write or read-only.
1131 * @param fForceNewUuid Whether a new UUID should be set to avoid duplicates.
1132 * @param aDeviceType Device type of medium.
1133 */
1134HRESULT Medium::init(VirtualBox *aVirtualBox,
1135 const Utf8Str &aLocation,
1136 HDDOpenMode enOpenMode,
1137 bool fForceNewUuid,
1138 DeviceType_T aDeviceType)
1139{
1140 AssertReturn(aVirtualBox, E_INVALIDARG);
1141 AssertReturn(!aLocation.isEmpty(), E_INVALIDARG);
1142
1143 HRESULT hrc = S_OK;
1144
1145 {
1146 /* Enclose the state transition NotReady->InInit->Ready */
1147 AutoInitSpan autoInitSpan(this);
1148 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1149
1150 unconst(m->pVirtualBox) = aVirtualBox;
1151
1152 /* there must be a storage unit */
1153 m->state = MediumState_Created;
1154
1155 /* remember device type for correct unregistering later */
1156 m->devType = aDeviceType;
1157
1158 /* cannot be a host drive */
1159 m->hostDrive = false;
1160
1161 /* remember the open mode (defaults to ReadWrite) */
1162 m->hddOpenMode = enOpenMode;
1163
1164 if (aDeviceType == DeviceType_DVD)
1165 m->type = MediumType_Readonly;
1166 else if (aDeviceType == DeviceType_Floppy)
1167 m->type = MediumType_Writethrough;
1168
1169 hrc = i_setLocation(aLocation);
1170 if (FAILED(hrc)) return hrc;
1171
1172 /* get all the information about the medium from the storage unit */
1173 if (fForceNewUuid)
1174 unconst(m->uuidImage).create();
1175
1176 m->state = MediumState_Inaccessible;
1177 m->strLastAccessError = tr("Accessibility check was not yet performed");
1178
1179 /* Confirm a successful initialization before the call to i_queryInfo.
1180 * Otherwise we can end up with a AutoCaller deadlock because the
1181 * medium becomes visible but is not marked as initialized. Causes
1182 * locking trouble (e.g. trying to save media registries) which is
1183 * hard to solve. */
1184 autoInitSpan.setSucceeded();
1185 }
1186
1187 /* we're normal code from now on, no longer init */
1188 AutoCaller autoCaller(this);
1189 if (FAILED(autoCaller.hrc()))
1190 return autoCaller.hrc();
1191
1192 /* need to call i_queryInfo immediately to correctly place the medium in
1193 * the respective media tree and update other information such as uuid */
1194 hrc = i_queryInfo(fForceNewUuid /* fSetImageId */, false /* fSetParentId */, autoCaller);
1195 if (SUCCEEDED(hrc))
1196 {
1197 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1198
1199 /* if the storage unit is not accessible, it's not acceptable for the
1200 * newly opened media so convert this into an error */
1201 if (m->state == MediumState_Inaccessible)
1202 {
1203 Assert(!m->strLastAccessError.isEmpty());
1204 hrc = setError(E_FAIL, "%s", m->strLastAccessError.c_str());
1205 alock.release();
1206 autoCaller.release();
1207 uninit();
1208 }
1209 else
1210 {
1211 AssertStmt(!m->id.isZero(),
1212 alock.release(); autoCaller.release(); uninit(); return E_FAIL);
1213
1214 /* storage format must be detected by Medium::i_queryInfo if the
1215 * medium is accessible */
1216 AssertStmt(!m->strFormat.isEmpty(),
1217 alock.release(); autoCaller.release(); uninit(); return E_FAIL);
1218 }
1219 }
1220 else
1221 {
1222 /* opening this image failed, mark the object as dead */
1223 autoCaller.release();
1224 uninit();
1225 }
1226
1227 return hrc;
1228}
1229
1230/**
1231 * Initializes the medium object by loading its data from the given settings
1232 * node. The medium will always be opened read/write.
1233 *
1234 * In this case, since we're loading from a registry, uuidMachineRegistry is
1235 * always set: it's either the global registry UUID or a machine UUID when
1236 * loading from a per-machine registry.
1237 *
1238 * @param aParent Parent medium disk or NULL for a root (base) medium.
1239 * @param aDeviceType Device type of the medium.
1240 * @param uuidMachineRegistry The registry to which this medium should be
1241 * added (global registry UUID or machine UUID).
1242 * @param data Configuration settings.
1243 * @param strMachineFolder The machine folder with which to resolve relative paths;
1244 * if empty, then we use the VirtualBox home directory
1245 *
1246 * @note Locks the medium tree for writing.
1247 */
1248HRESULT Medium::initOne(Medium *aParent,
1249 DeviceType_T aDeviceType,
1250 const Guid &uuidMachineRegistry,
1251 const Utf8Str &strMachineFolder,
1252 const settings::Medium &data)
1253{
1254 HRESULT hrc;
1255
1256 if (uuidMachineRegistry.isValid() && !uuidMachineRegistry.isZero())
1257 m->llRegistryIDs.push_back(uuidMachineRegistry);
1258
1259 /* register with VirtualBox/parent early, since uninit() will
1260 * unconditionally unregister on failure */
1261 if (aParent)
1262 {
1263 // differencing medium: add to parent
1264 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1265 // no need to check maximum depth as settings reading did it
1266 i_setParent(aParent);
1267 }
1268
1269 /* see below why we don't call Medium::i_queryInfo (and therefore treat
1270 * the medium as inaccessible for now */
1271 m->state = MediumState_Inaccessible;
1272 m->strLastAccessError = tr("Accessibility check was not yet performed");
1273
1274 /* required */
1275 unconst(m->id) = data.uuid;
1276
1277 /* assume not a host drive */
1278 m->hostDrive = false;
1279
1280 /* optional */
1281 m->strDescription = data.strDescription;
1282
1283 /* required */
1284 if (aDeviceType == DeviceType_HardDisk)
1285 {
1286 AssertReturn(!data.strFormat.isEmpty(), E_FAIL);
1287 hrc = i_setFormat(data.strFormat);
1288 if (FAILED(hrc)) return hrc;
1289 }
1290 else
1291 {
1292 /// @todo handle host drive settings here as well?
1293 if (!data.strFormat.isEmpty())
1294 hrc = i_setFormat(data.strFormat);
1295 else
1296 hrc = i_setFormat("RAW");
1297 if (FAILED(hrc)) return hrc;
1298 }
1299
1300 /* optional, only for diffs, default is false; we can only auto-reset
1301 * diff media so they must have a parent */
1302 if (aParent != NULL)
1303 m->autoReset = data.fAutoReset;
1304 else
1305 m->autoReset = false;
1306
1307 /* properties (after setting the format as it populates the map). Note that
1308 * if some properties are not supported but present in the settings file,
1309 * they will still be read and accessible (for possible backward
1310 * compatibility; we can also clean them up from the XML upon next
1311 * XML format version change if we wish) */
1312 for (settings::StringsMap::const_iterator it = data.properties.begin();
1313 it != data.properties.end();
1314 ++it)
1315 {
1316 const Utf8Str &name = it->first;
1317 const Utf8Str &value = it->second;
1318 m->mapProperties[name] = value;
1319 }
1320
1321 /* try to decrypt an optional iSCSI initiator secret */
1322 settings::StringsMap::const_iterator itCph = data.properties.find("InitiatorSecretEncrypted");
1323 if ( itCph != data.properties.end()
1324 && !itCph->second.isEmpty())
1325 {
1326 Utf8Str strPlaintext;
1327 int vrc = m->pVirtualBox->i_decryptSetting(&strPlaintext, itCph->second);
1328 if (RT_SUCCESS(vrc))
1329 m->mapProperties["InitiatorSecret"] = strPlaintext;
1330 }
1331
1332 Utf8Str strFull;
1333 if (m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File)
1334 {
1335 // compose full path of the medium, if it's not fully qualified...
1336 // slightly convoluted logic here. If the caller has given us a
1337 // machine folder, then a relative path will be relative to that:
1338 if ( !strMachineFolder.isEmpty()
1339 && !RTPathStartsWithRoot(data.strLocation.c_str())
1340 )
1341 {
1342 strFull = strMachineFolder;
1343 strFull += RTPATH_SLASH;
1344 strFull += data.strLocation;
1345 }
1346 else
1347 {
1348 // Otherwise use the old VirtualBox "make absolute path" logic:
1349 int vrc = m->pVirtualBox->i_calculateFullPath(data.strLocation, strFull);
1350 if (RT_FAILURE(vrc))
1351 return Global::vboxStatusCodeToCOM(vrc);
1352 }
1353 }
1354 else
1355 strFull = data.strLocation;
1356
1357 hrc = i_setLocation(strFull);
1358 if (FAILED(hrc)) return hrc;
1359
1360 if (aDeviceType == DeviceType_HardDisk)
1361 {
1362 /* type is only for base hard disks */
1363 if (m->pParent.isNull())
1364 m->type = data.hdType;
1365 }
1366 else if (aDeviceType == DeviceType_DVD)
1367 m->type = MediumType_Readonly;
1368 else
1369 m->type = MediumType_Writethrough;
1370
1371 /* remember device type for correct unregistering later */
1372 m->devType = aDeviceType;
1373
1374 LogFlowThisFunc(("m->strLocationFull='%s', m->strFormat=%s, m->id={%RTuuid}\n",
1375 m->strLocationFull.c_str(), m->strFormat.c_str(), m->id.raw()));
1376
1377 return S_OK;
1378}
1379
1380/**
1381 * Initializes and registers the medium object and its children by loading its
1382 * data from the given settings node. The medium will always be opened
1383 * read/write.
1384 *
1385 * @todo r=bird: What's that stuff about 'always be opened read/write'?
1386 *
1387 * In this case, since we're loading from a registry, uuidMachineRegistry is
1388 * always set: it's either the global registry UUID or a machine UUID when
1389 * loading from a per-machine registry.
1390 *
1391 * The only caller is currently VirtualBox::initMedia().
1392 *
1393 * @param aVirtualBox VirtualBox object.
1394 * @param aDeviceType Device type of the medium.
1395 * @param uuidMachineRegistry The registry to which this medium should be added
1396 * (global registry UUID or machine UUID).
1397 * @param strMachineFolder The machine folder with which to resolve relative
1398 * paths; if empty, then we use the VirtualBox home directory
1399 * @param data Configuration settings.
1400 * @param mediaTreeLock Autolock.
1401 * @param uIdsForNotify List to be updated with newly registered media.
1402 *
1403 * @note Assumes that the medium tree lock is held for writing. May release
1404 * and lock it again. At the end it is always held.
1405 */
1406/* static */
1407HRESULT Medium::initFromSettings(VirtualBox *aVirtualBox,
1408 DeviceType_T aDeviceType,
1409 const Guid &uuidMachineRegistry,
1410 const Utf8Str &strMachineFolder,
1411 const settings::Medium &data,
1412 AutoWriteLock &mediaTreeLock,
1413 std::list<std::pair<Guid, DeviceType_T> > &uIdsForNotify)
1414{
1415 Assert(aVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
1416 AssertReturn(aVirtualBox, E_INVALIDARG);
1417
1418 HRESULT hrc = S_OK;
1419
1420 MediaList llMediaTocleanup;
1421
1422 std::list<const settings::Medium *> llSettingsTodo;
1423 llSettingsTodo.push_back(&data);
1424 MediaList llParentsTodo;
1425 llParentsTodo.push_back(NULL);
1426
1427 while (!llSettingsTodo.empty())
1428 {
1429 const settings::Medium *current = llSettingsTodo.front();
1430 llSettingsTodo.pop_front();
1431 ComObjPtr<Medium> pParent = llParentsTodo.front();
1432 llParentsTodo.pop_front();
1433
1434 bool fReleasedMediaTreeLock = false;
1435 ComObjPtr<Medium> pMedium;
1436 hrc = pMedium.createObject();
1437 if (FAILED(hrc))
1438 break;
1439 ComObjPtr<Medium> pActualMedium(pMedium);
1440
1441 {
1442 AutoInitSpan autoInitSpan(pMedium);
1443 AssertBreakStmt(autoInitSpan.isOk(), hrc = E_FAIL);
1444
1445 unconst(pMedium->m->pVirtualBox) = aVirtualBox;
1446 hrc = pMedium->initOne(pParent, aDeviceType, uuidMachineRegistry, strMachineFolder, *current);
1447 if (FAILED(hrc))
1448 break;
1449 hrc = aVirtualBox->i_registerMedium(pActualMedium, &pActualMedium, mediaTreeLock, true /*fCalledFromMediumInit*/);
1450 if (SUCCEEDED(hrc) && pActualMedium == pMedium)
1451 {
1452 /* It is a truly new medium, remember details for cleanup. */
1453 autoInitSpan.setSucceeded();
1454 llMediaTocleanup.push_front(pMedium);
1455 }
1456 else
1457 {
1458 /* Since the newly created medium was replaced by an already
1459 * known one when merging medium trees, we can immediately mark
1460 * it as failed. */
1461 autoInitSpan.setFailed();
1462 mediaTreeLock.release();
1463 fReleasedMediaTreeLock = true;
1464 }
1465 }
1466 if (fReleasedMediaTreeLock)
1467 {
1468 /* With the InitSpan out of the way it's safe to let the refcount
1469 * drop to 0 without causing uninit trouble. */
1470 pMedium.setNull();
1471 mediaTreeLock.acquire();
1472
1473 if (FAILED(hrc))
1474 break;
1475 }
1476
1477 /* create all children */
1478 std::list<settings::Medium>::const_iterator itBegin = current->llChildren.begin();
1479 std::list<settings::Medium>::const_iterator itEnd = current->llChildren.end();
1480 for (std::list<settings::Medium>::const_iterator it = itBegin; it != itEnd; ++it)
1481 {
1482 llSettingsTodo.push_back(&*it);
1483 llParentsTodo.push_back(pActualMedium);
1484 }
1485 }
1486
1487 if (SUCCEEDED(hrc))
1488 {
1489 /* Check for consistency. */
1490 Assert(llSettingsTodo.size() == 0);
1491 Assert(llParentsTodo.size() == 0);
1492 /* Create the list of notifications, parent first. */
1493 for (MediaList::const_reverse_iterator it = llMediaTocleanup.rbegin(); it != llMediaTocleanup.rend(); ++it)
1494 {
1495 ComObjPtr<Medium> pMedium = *it;
1496 AutoCaller mediumCaller(pMedium);
1497 if (mediumCaller.isOk())
1498 {
1499 const Guid &id = pMedium->i_getId();
1500 uIdsForNotify.push_back(std::pair<Guid, DeviceType_T>(id, aDeviceType));
1501 }
1502 }
1503 }
1504 else
1505 {
1506 /* Forget state of the settings processing. */
1507 llSettingsTodo.clear();
1508 llParentsTodo.clear();
1509 /* Unregister all accumulated medium objects in the right order (last
1510 * created to first created, avoiding config leftovers). */
1511 MediaList::const_iterator itBegin = llMediaTocleanup.begin();
1512 MediaList::const_iterator itEnd = llMediaTocleanup.end();
1513 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
1514 {
1515 ComObjPtr<Medium> pMedium = *it;
1516 pMedium->i_unregisterWithVirtualBox();
1517 }
1518 /* Forget the only references to all newly created medium objects,
1519 * triggering freeing (uninit happened in unregistering above). */
1520 mediaTreeLock.release();
1521 llMediaTocleanup.clear();
1522 mediaTreeLock.acquire();
1523 }
1524
1525 return hrc;
1526}
1527
1528/**
1529 * Initializes the medium object by providing the host drive information.
1530 * Not used for anything but the host floppy/host DVD case.
1531 *
1532 * There is no registry for this case.
1533 *
1534 * @param aVirtualBox VirtualBox object.
1535 * @param aDeviceType Device type of the medium.
1536 * @param aLocation Location of the host drive.
1537 * @param aDescription Comment for this host drive.
1538 *
1539 * @note Locks VirtualBox lock for writing.
1540 */
1541HRESULT Medium::init(VirtualBox *aVirtualBox,
1542 DeviceType_T aDeviceType,
1543 const Utf8Str &aLocation,
1544 const Utf8Str &aDescription /* = Utf8Str::Empty */)
1545{
1546 ComAssertRet(aDeviceType == DeviceType_DVD || aDeviceType == DeviceType_Floppy, E_INVALIDARG);
1547 ComAssertRet(!aLocation.isEmpty(), E_INVALIDARG);
1548
1549 /* Enclose the state transition NotReady->InInit->Ready */
1550 AutoInitSpan autoInitSpan(this);
1551 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1552
1553 unconst(m->pVirtualBox) = aVirtualBox;
1554
1555 // We do not store host drives in VirtualBox.xml or anywhere else, so if we want
1556 // host drives to be identifiable by UUID and not give the drive a different UUID
1557 // every time VirtualBox starts, we need to fake a reproducible UUID here:
1558 RTUUID uuid;
1559 RTUuidClear(&uuid);
1560 if (aDeviceType == DeviceType_DVD)
1561 memcpy(&uuid.au8[0], "DVD", 3);
1562 else
1563 memcpy(&uuid.au8[0], "FD", 2);
1564 /* use device name, adjusted to the end of uuid, shortened if necessary */
1565 size_t lenLocation = aLocation.length();
1566 if (lenLocation > 12)
1567 memcpy(&uuid.au8[4], aLocation.c_str() + (lenLocation - 12), 12);
1568 else
1569 memcpy(&uuid.au8[4 + 12 - lenLocation], aLocation.c_str(), lenLocation);
1570 unconst(m->id) = uuid;
1571
1572 if (aDeviceType == DeviceType_DVD)
1573 m->type = MediumType_Readonly;
1574 else
1575 m->type = MediumType_Writethrough;
1576 m->devType = aDeviceType;
1577 m->state = MediumState_Created;
1578 m->hostDrive = true;
1579 HRESULT hrc = i_setFormat("RAW");
1580 if (FAILED(hrc)) return hrc;
1581 hrc = i_setLocation(aLocation);
1582 if (FAILED(hrc)) return hrc;
1583 m->strDescription = aDescription;
1584
1585 autoInitSpan.setSucceeded();
1586 return S_OK;
1587}
1588
1589/**
1590 * Uninitializes the instance.
1591 *
1592 * Called either from FinalRelease() or by the parent when it gets destroyed.
1593 *
1594 * @note All children of this medium get uninitialized, too, in a stack
1595 * friendly manner.
1596 */
1597void Medium::uninit()
1598{
1599 /* It is possible that some previous/concurrent uninit has already cleared
1600 * the pVirtualBox reference, and in this case we don't need to continue.
1601 * Normally this would be handled through the AutoUninitSpan magic, however
1602 * this cannot be done at this point as the media tree must be locked
1603 * before reaching the AutoUninitSpan, otherwise deadlocks can happen.
1604 *
1605 * NOTE: The tree lock is higher priority than the medium caller and medium
1606 * object locks, i.e. the medium caller may have to be released and be
1607 * re-acquired in the right place later. See Medium::getParent() for sample
1608 * code how to do this safely. */
1609 VirtualBox *pVirtualBox = m->pVirtualBox;
1610 if (!pVirtualBox)
1611 return;
1612
1613 /* Caller must not hold the object (checked below) or media tree lock. */
1614 Assert(!pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
1615
1616 AutoWriteLock treeLock(pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1617
1618 /* Must use a list without refcounting help since "this" might already have
1619 * reached 0, and then the refcount must not be increased again since it
1620 * would otherwise trigger a double free. For all other list entries this
1621 * needs manual refcount updating, to make sure the refcount for children
1622 * does not drop to 0 too early. */
1623 std::list<Medium *> llMediaTodo;
1624 llMediaTodo.push_back(this);
1625
1626 while (!llMediaTodo.empty())
1627 {
1628 Medium *pMedium = llMediaTodo.front();
1629 llMediaTodo.pop_front();
1630
1631 /* Enclose the state transition Ready->InUninit->NotReady */
1632 AutoUninitSpan autoUninitSpan(pMedium);
1633 if (autoUninitSpan.uninitDone())
1634 {
1635 if (pMedium != this)
1636 pMedium->Release();
1637 continue;
1638 }
1639
1640 Assert(!pMedium->isWriteLockOnCurrentThread());
1641#ifdef DEBUG
1642 if (!pMedium->m->backRefs.empty())
1643 pMedium->i_dumpBackRefs();
1644#endif
1645 Assert(pMedium->m->backRefs.empty());
1646
1647 pMedium->m->formatObj.setNull();
1648
1649 if (pMedium->m->state == MediumState_Deleting)
1650 {
1651 /* This medium has been already deleted (directly or as part of a
1652 * merge). Reparenting has already been done. */
1653 Assert(pMedium->m->pParent.isNull());
1654 Assert(pMedium->m->llChildren.empty());
1655 if (pMedium != this)
1656 pMedium->Release();
1657 continue;
1658 }
1659
1660 //Assert(!pMedium->m->pParent);
1661 /** @todo r=klaus Should not be necessary, since the caller should be
1662 * doing the deparenting. No time right now to test everything. */
1663 if (pMedium == this && pMedium->m->pParent)
1664 pMedium->i_deparent();
1665
1666 /* Process all children */
1667 MediaList::const_iterator itBegin = pMedium->m->llChildren.begin();
1668 MediaList::const_iterator itEnd = pMedium->m->llChildren.end();
1669 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
1670 {
1671 Medium *pChild = *it;
1672 pChild->m->pParent.setNull();
1673 pChild->AddRef();
1674 llMediaTodo.push_back(pChild);
1675 }
1676
1677 /* Children information obsolete, will be processed anyway. */
1678 pMedium->m->llChildren.clear();
1679
1680 unconst(pMedium->m->pVirtualBox) = NULL;
1681
1682 if (pMedium != this)
1683 pMedium->Release();
1684
1685 autoUninitSpan.setSucceeded();
1686 }
1687}
1688
1689/**
1690 * Internal helper that removes "this" from the list of children of its
1691 * parent. Used in uninit() and other places when reparenting is necessary.
1692 *
1693 * The caller must hold the medium tree lock!
1694 */
1695void Medium::i_deparent()
1696{
1697 MediaList &llParent = m->pParent->m->llChildren;
1698 for (MediaList::iterator it = llParent.begin();
1699 it != llParent.end();
1700 ++it)
1701 {
1702 Medium *pParentsChild = *it;
1703 if (this == pParentsChild)
1704 {
1705 llParent.erase(it);
1706 break;
1707 }
1708 }
1709 m->pParent.setNull();
1710}
1711
1712/**
1713 * Internal helper that removes "this" from the list of children of its
1714 * parent. Used in uninit() and other places when reparenting is necessary.
1715 *
1716 * The caller must hold the medium tree lock!
1717 */
1718void Medium::i_setParent(const ComObjPtr<Medium> &pParent)
1719{
1720 m->pParent = pParent;
1721 if (pParent)
1722 pParent->m->llChildren.push_back(this);
1723}
1724
1725
1726////////////////////////////////////////////////////////////////////////////////
1727//
1728// IMedium public methods
1729//
1730////////////////////////////////////////////////////////////////////////////////
1731
1732HRESULT Medium::getId(com::Guid &aId)
1733{
1734 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1735
1736 aId = m->id;
1737
1738 return S_OK;
1739}
1740
1741HRESULT Medium::getDescription(AutoCaller &autoCaller, com::Utf8Str &aDescription)
1742{
1743 NOREF(autoCaller);
1744 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1745
1746 aDescription = m->strDescription;
1747
1748 return S_OK;
1749}
1750
1751HRESULT Medium::setDescription(AutoCaller &autoCaller, const com::Utf8Str &aDescription)
1752{
1753 /// @todo update m->strDescription and save the global registry (and local
1754 /// registries of portable VMs referring to this medium), this will also
1755 /// require to add the mRegistered flag to data
1756
1757 HRESULT hrc = S_OK;
1758
1759 MediumLockList *pMediumLockList(new MediumLockList());
1760
1761 try
1762 {
1763 autoCaller.release();
1764
1765 // to avoid redundant locking, which just takes a time, just call required functions.
1766 // the error will be just stored and will be reported after locks will be acquired again
1767
1768 const char *pszError = NULL;
1769
1770
1771 /* Build the lock list. */
1772 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
1773 this /* pToLockWrite */,
1774 true /* fMediumLockWriteAll */,
1775 NULL,
1776 *pMediumLockList);
1777 if (FAILED(hrc))
1778 pszError = tr("Failed to create medium lock list for '%s'");
1779 else
1780 {
1781 hrc = pMediumLockList->Lock();
1782 if (FAILED(hrc))
1783 pszError = tr("Failed to lock media '%s'");
1784 }
1785
1786 // locking: we need the tree lock first because we access parent pointers
1787 // and we need to write-lock the media involved
1788 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1789
1790 autoCaller.add();
1791 AssertComRCThrowRC(autoCaller.hrc());
1792
1793 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1794
1795 if (FAILED(hrc))
1796 throw setError(hrc, pszError, i_getLocationFull().c_str());
1797
1798 /* Set a new description */
1799 m->strDescription = aDescription;
1800
1801 // save the settings
1802 alock.release();
1803 autoCaller.release();
1804 treeLock.release();
1805 i_markRegistriesModified();
1806 m->pVirtualBox->i_saveModifiedRegistries();
1807 m->pVirtualBox->i_onMediumConfigChanged(this);
1808 }
1809 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
1810
1811 delete pMediumLockList;
1812
1813 return hrc;
1814}
1815
1816HRESULT Medium::getState(MediumState_T *aState)
1817{
1818 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1819 *aState = m->state;
1820
1821 return S_OK;
1822}
1823
1824HRESULT Medium::getVariant(std::vector<MediumVariant_T> &aVariant)
1825{
1826 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1827
1828 const size_t cBits = sizeof(MediumVariant_T) * 8;
1829 aVariant.resize(cBits);
1830 for (size_t i = 0; i < cBits; ++i)
1831 aVariant[i] = (MediumVariant_T)(m->variant & RT_BIT(i));
1832
1833 return S_OK;
1834}
1835
1836HRESULT Medium::getLocation(com::Utf8Str &aLocation)
1837{
1838 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1839
1840 aLocation = m->strLocationFull;
1841
1842 return S_OK;
1843}
1844
1845HRESULT Medium::getName(com::Utf8Str &aName)
1846{
1847 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1848
1849 aName = i_getName();
1850
1851 return S_OK;
1852}
1853
1854HRESULT Medium::getDeviceType(DeviceType_T *aDeviceType)
1855{
1856 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1857
1858 *aDeviceType = m->devType;
1859
1860 return S_OK;
1861}
1862
1863HRESULT Medium::getHostDrive(BOOL *aHostDrive)
1864{
1865 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1866
1867 *aHostDrive = m->hostDrive;
1868
1869 return S_OK;
1870}
1871
1872HRESULT Medium::getSize(LONG64 *aSize)
1873{
1874 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1875
1876 *aSize = (LONG64)m->size;
1877
1878 return S_OK;
1879}
1880
1881HRESULT Medium::getFormat(com::Utf8Str &aFormat)
1882{
1883 /* no need to lock, m->strFormat is const */
1884
1885 aFormat = m->strFormat;
1886 return S_OK;
1887}
1888
1889HRESULT Medium::getMediumFormat(ComPtr<IMediumFormat> &aMediumFormat)
1890{
1891 /* no need to lock, m->formatObj is const */
1892 m->formatObj.queryInterfaceTo(aMediumFormat.asOutParam());
1893
1894 return S_OK;
1895}
1896
1897HRESULT Medium::getType(AutoCaller &autoCaller, MediumType_T *aType)
1898{
1899 NOREF(autoCaller);
1900 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1901
1902 *aType = m->type;
1903
1904 return S_OK;
1905}
1906
1907HRESULT Medium::setType(AutoCaller &autoCaller, MediumType_T aType)
1908{
1909 autoCaller.release();
1910
1911 /* It is possible that some previous/concurrent uninit has already cleared
1912 * the pVirtualBox reference, see #uninit(). */
1913 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
1914
1915 // we access m->pParent
1916 AutoReadLock treeLock(!pVirtualBox.isNull() ? &pVirtualBox->i_getMediaTreeLockHandle() : NULL COMMA_LOCKVAL_SRC_POS);
1917
1918 autoCaller.add();
1919 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
1920
1921 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
1922
1923 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
1924 while (m->queryInfoRunning)
1925 {
1926 mlock.release();
1927 autoCaller.release();
1928 treeLock.release();
1929 /* Must not hold the media tree lock, as Medium::i_queryInfo needs
1930 * this lock and thus we would run into a deadlock here. */
1931 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
1932 /* must not hold the object lock now */
1933 Assert(!isWriteLockOnCurrentThread());
1934 {
1935 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
1936 }
1937 treeLock.acquire();
1938 autoCaller.add();
1939 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
1940 mlock.acquire();
1941 }
1942
1943 switch (m->state)
1944 {
1945 case MediumState_Created:
1946 case MediumState_Inaccessible:
1947 break;
1948 default:
1949 return i_setStateError();
1950 }
1951
1952 if (m->type == aType)
1953 {
1954 /* Nothing to do */
1955 return S_OK;
1956 }
1957
1958 DeviceType_T devType = i_getDeviceType();
1959 // DVD media can only be readonly.
1960 if (devType == DeviceType_DVD && aType != MediumType_Readonly)
1961 return setError(VBOX_E_INVALID_OBJECT_STATE,
1962 tr("Cannot change the type of DVD medium '%s'"),
1963 m->strLocationFull.c_str());
1964 // Floppy media can only be writethrough or readonly.
1965 if ( devType == DeviceType_Floppy
1966 && aType != MediumType_Writethrough
1967 && aType != MediumType_Readonly)
1968 return setError(VBOX_E_INVALID_OBJECT_STATE,
1969 tr("Cannot change the type of floppy medium '%s'"),
1970 m->strLocationFull.c_str());
1971
1972 /* cannot change the type of a differencing medium */
1973 if (m->pParent)
1974 return setError(VBOX_E_INVALID_OBJECT_STATE,
1975 tr("Cannot change the type of medium '%s' because it is a differencing medium"),
1976 m->strLocationFull.c_str());
1977
1978 /* Cannot change the type of a medium being in use by more than one VM.
1979 * If the change is to Immutable or MultiAttach then it must not be
1980 * directly attached to any VM, otherwise the assumptions about indirect
1981 * attachment elsewhere are violated and the VM becomes inaccessible.
1982 * Attaching an immutable medium triggers the diff creation, and this is
1983 * vital for the correct operation. */
1984 if ( m->backRefs.size() > 1
1985 || ( ( aType == MediumType_Immutable
1986 || aType == MediumType_MultiAttach)
1987 && m->backRefs.size() > 0))
1988 return setError(VBOX_E_INVALID_OBJECT_STATE,
1989 tr("Cannot change the type of medium '%s' because it is attached to %d virtual machines",
1990 "", m->backRefs.size()),
1991 m->strLocationFull.c_str(), m->backRefs.size());
1992
1993 switch (aType)
1994 {
1995 case MediumType_Normal:
1996 case MediumType_Immutable:
1997 case MediumType_MultiAttach:
1998 {
1999 /* normal can be easily converted to immutable and vice versa even
2000 * if they have children as long as they are not attached to any
2001 * machine themselves */
2002 break;
2003 }
2004 case MediumType_Writethrough:
2005 case MediumType_Shareable:
2006 case MediumType_Readonly:
2007 {
2008 /* cannot change to writethrough, shareable or readonly
2009 * if there are children */
2010 if (i_getChildren().size() != 0)
2011 return setError(VBOX_E_OBJECT_IN_USE,
2012 tr("Cannot change type for medium '%s' since it has %d child media", "", i_getChildren().size()),
2013 m->strLocationFull.c_str(), i_getChildren().size());
2014 if (aType == MediumType_Shareable)
2015 {
2016 MediumVariant_T variant = i_getVariant();
2017 if (!(variant & MediumVariant_Fixed))
2018 return setError(VBOX_E_INVALID_OBJECT_STATE,
2019 tr("Cannot change type for medium '%s' to 'Shareable' since it is a dynamic medium storage unit"),
2020 m->strLocationFull.c_str());
2021 }
2022 else if (aType == MediumType_Readonly && devType == DeviceType_HardDisk)
2023 {
2024 // Readonly hard disks are not allowed, this medium type is reserved for
2025 // DVDs and floppy images at the moment. Later we might allow readonly hard
2026 // disks, but that's extremely unusual and many guest OSes will have trouble.
2027 return setError(VBOX_E_INVALID_OBJECT_STATE,
2028 tr("Cannot change type for medium '%s' to 'Readonly' since it is a hard disk"),
2029 m->strLocationFull.c_str());
2030 }
2031 break;
2032 }
2033 default:
2034 AssertFailedReturn(E_FAIL);
2035 }
2036
2037 if (aType == MediumType_MultiAttach)
2038 {
2039 // This type is new with VirtualBox 4.0 and therefore requires settings
2040 // version 1.11 in the settings backend. Unfortunately it is not enough to do
2041 // the usual routine in MachineConfigFile::bumpSettingsVersionIfNeeded() for
2042 // two reasons: The medium type is a property of the media registry tree, which
2043 // can reside in the global config file (for pre-4.0 media); we would therefore
2044 // possibly need to bump the global config version. We don't want to do that though
2045 // because that might make downgrading to pre-4.0 impossible.
2046 // As a result, we can only use these two new types if the medium is NOT in the
2047 // global registry:
2048 const Guid &uuidGlobalRegistry = m->pVirtualBox->i_getGlobalRegistryId();
2049 if (i_isInRegistry(uuidGlobalRegistry))
2050 return setError(VBOX_E_INVALID_OBJECT_STATE,
2051 tr("Cannot change type for medium '%s': the media type 'MultiAttach' can only be used "
2052 "on media registered with a machine that was created with VirtualBox 4.0 or later"),
2053 m->strLocationFull.c_str());
2054 }
2055
2056 m->type = aType;
2057
2058 // save the settings
2059 mlock.release();
2060 autoCaller.release();
2061 treeLock.release();
2062 i_markRegistriesModified();
2063 m->pVirtualBox->i_saveModifiedRegistries();
2064 m->pVirtualBox->i_onMediumConfigChanged(this);
2065
2066 return S_OK;
2067}
2068
2069HRESULT Medium::getAllowedTypes(std::vector<MediumType_T> &aAllowedTypes)
2070{
2071 NOREF(aAllowedTypes);
2072 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2073
2074 ReturnComNotImplemented();
2075}
2076
2077HRESULT Medium::getParent(AutoCaller &autoCaller, ComPtr<IMedium> &aParent)
2078{
2079 autoCaller.release();
2080
2081 /* It is possible that some previous/concurrent uninit has already cleared
2082 * the pVirtualBox reference, see #uninit(). */
2083 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
2084
2085 /* we access m->pParent */
2086 AutoReadLock treeLock(!pVirtualBox.isNull() ? &pVirtualBox->i_getMediaTreeLockHandle() : NULL COMMA_LOCKVAL_SRC_POS);
2087
2088 autoCaller.add();
2089 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
2090
2091 m->pParent.queryInterfaceTo(aParent.asOutParam());
2092
2093 return S_OK;
2094}
2095
2096HRESULT Medium::getChildren(AutoCaller &autoCaller, std::vector<ComPtr<IMedium> > &aChildren)
2097{
2098 autoCaller.release();
2099
2100 /* It is possible that some previous/concurrent uninit has already cleared
2101 * the pVirtualBox reference, see #uninit(). */
2102 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
2103
2104 /* we access children */
2105 AutoReadLock treeLock(!pVirtualBox.isNull() ? &pVirtualBox->i_getMediaTreeLockHandle() : NULL COMMA_LOCKVAL_SRC_POS);
2106
2107 autoCaller.add();
2108 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
2109
2110 MediaList children(this->i_getChildren());
2111 aChildren.resize(children.size());
2112 size_t i = 0;
2113 for (MediaList::const_iterator it = children.begin(); it != children.end(); ++it, ++i)
2114 (*it).queryInterfaceTo(aChildren[i].asOutParam());
2115 return S_OK;
2116}
2117
2118HRESULT Medium::getBase(AutoCaller &autoCaller, ComPtr<IMedium> &aBase)
2119{
2120 autoCaller.release();
2121
2122 /* i_getBase() will do callers/locking */
2123 i_getBase().queryInterfaceTo(aBase.asOutParam());
2124
2125 return S_OK;
2126}
2127
2128HRESULT Medium::getReadOnly(AutoCaller &autoCaller, BOOL *aReadOnly)
2129{
2130 autoCaller.release();
2131
2132 /* isReadOnly() will do locking */
2133 *aReadOnly = i_isReadOnly();
2134
2135 return S_OK;
2136}
2137
2138HRESULT Medium::getLogicalSize(LONG64 *aLogicalSize)
2139{
2140 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2141
2142 *aLogicalSize = (LONG64)m->logicalSize;
2143
2144 return S_OK;
2145}
2146
2147HRESULT Medium::getAutoReset(BOOL *aAutoReset)
2148{
2149 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2150
2151 if (m->pParent.isNull())
2152 *aAutoReset = FALSE;
2153 else
2154 *aAutoReset = m->autoReset;
2155
2156 return S_OK;
2157}
2158
2159HRESULT Medium::setAutoReset(BOOL aAutoReset)
2160{
2161 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
2162
2163 if (m->pParent.isNull())
2164 return setError(VBOX_E_NOT_SUPPORTED,
2165 tr("Medium '%s' is not differencing"),
2166 m->strLocationFull.c_str());
2167
2168 if (m->autoReset != !!aAutoReset)
2169 {
2170 m->autoReset = !!aAutoReset;
2171
2172 // save the settings
2173 mlock.release();
2174 i_markRegistriesModified();
2175 m->pVirtualBox->i_saveModifiedRegistries();
2176 m->pVirtualBox->i_onMediumConfigChanged(this);
2177 }
2178
2179 return S_OK;
2180}
2181
2182HRESULT Medium::getLastAccessError(com::Utf8Str &aLastAccessError)
2183{
2184 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2185
2186 aLastAccessError = m->strLastAccessError;
2187
2188 return S_OK;
2189}
2190
2191HRESULT Medium::getMachineIds(std::vector<com::Guid> &aMachineIds)
2192{
2193 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2194
2195 if (m->backRefs.size() != 0)
2196 {
2197 BackRefList brlist(m->backRefs);
2198 aMachineIds.resize(brlist.size());
2199 size_t i = 0;
2200 for (BackRefList::const_iterator it = brlist.begin(); it != brlist.end(); ++it, ++i)
2201 aMachineIds[i] = it->machineId;
2202 }
2203
2204 return S_OK;
2205}
2206
2207HRESULT Medium::setIds(AutoCaller &autoCaller,
2208 BOOL aSetImageId,
2209 const com::Guid &aImageId,
2210 BOOL aSetParentId,
2211 const com::Guid &aParentId)
2212{
2213 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2214
2215 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
2216 if (m->queryInfoRunning)
2217 {
2218 /* Must not hold the media tree lock, as Medium::i_queryInfo needs this
2219 * lock and thus we would run into a deadlock here. */
2220 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2221 while (m->queryInfoRunning)
2222 {
2223 alock.release();
2224 /* must not hold the object lock now */
2225 Assert(!isWriteLockOnCurrentThread());
2226 {
2227 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2228 }
2229 alock.acquire();
2230 }
2231 }
2232
2233 switch (m->state)
2234 {
2235 case MediumState_Created:
2236 break;
2237 default:
2238 return i_setStateError();
2239 }
2240
2241 Guid imageId, parentId;
2242 if (aSetImageId)
2243 {
2244 if (aImageId.isZero())
2245 imageId.create();
2246 else
2247 {
2248 imageId = aImageId;
2249 if (!imageId.isValid())
2250 return setError(E_INVALIDARG, tr("Argument %s is invalid"), "aImageId");
2251 }
2252 }
2253 if (aSetParentId)
2254 {
2255 if (aParentId.isZero())
2256 parentId.create();
2257 else
2258 parentId = aParentId;
2259 }
2260
2261 const Guid uPrevImage = m->uuidImage;
2262 unconst(m->uuidImage) = imageId;
2263 ComObjPtr<Medium> pPrevParent = i_getParent();
2264 unconst(m->uuidParentImage) = parentId;
2265
2266 // must not hold any locks before calling Medium::i_queryInfo
2267 alock.release();
2268
2269 HRESULT hrc = i_queryInfo(!!aSetImageId /* fSetImageId */,
2270 !!aSetParentId /* fSetParentId */,
2271 autoCaller);
2272
2273 AutoReadLock arlock(this COMMA_LOCKVAL_SRC_POS);
2274 const Guid uCurrImage = m->uuidImage;
2275 ComObjPtr<Medium> pCurrParent = i_getParent();
2276 arlock.release();
2277
2278 if (SUCCEEDED(hrc))
2279 {
2280 if (uCurrImage != uPrevImage)
2281 m->pVirtualBox->i_onMediumConfigChanged(this);
2282 if (pPrevParent != pCurrParent)
2283 {
2284 if (pPrevParent)
2285 m->pVirtualBox->i_onMediumConfigChanged(pPrevParent);
2286 if (pCurrParent)
2287 m->pVirtualBox->i_onMediumConfigChanged(pCurrParent);
2288 }
2289 }
2290
2291 return hrc;
2292}
2293
2294HRESULT Medium::refreshState(AutoCaller &autoCaller, MediumState_T *aState)
2295{
2296 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2297
2298 HRESULT hrc = S_OK;
2299
2300 switch (m->state)
2301 {
2302 case MediumState_Created:
2303 case MediumState_Inaccessible:
2304 case MediumState_LockedRead:
2305 {
2306 // must not hold any locks before calling Medium::i_queryInfo
2307 alock.release();
2308
2309 hrc = i_queryInfo(false /* fSetImageId */, false /* fSetParentId */, autoCaller);
2310
2311 alock.acquire();
2312 break;
2313 }
2314 default:
2315 break;
2316 }
2317
2318 *aState = m->state;
2319
2320 return hrc;
2321}
2322
2323HRESULT Medium::getSnapshotIds(const com::Guid &aMachineId,
2324 std::vector<com::Guid> &aSnapshotIds)
2325{
2326 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2327
2328 for (BackRefList::const_iterator it = m->backRefs.begin();
2329 it != m->backRefs.end(); ++it)
2330 {
2331 if (it->machineId == aMachineId)
2332 {
2333 size_t size = it->llSnapshotIds.size();
2334
2335 /* if the medium is attached to the machine in the current state, we
2336 * return its ID as the first element of the array */
2337 if (it->fInCurState)
2338 ++size;
2339
2340 if (size > 0)
2341 {
2342 aSnapshotIds.resize(size);
2343
2344 size_t j = 0;
2345 if (it->fInCurState)
2346 aSnapshotIds[j++] = it->machineId.toUtf16();
2347
2348 for(std::list<SnapshotRef>::const_iterator jt = it->llSnapshotIds.begin(); jt != it->llSnapshotIds.end(); ++jt, ++j)
2349 aSnapshotIds[j] = jt->snapshotId;
2350 }
2351
2352 break;
2353 }
2354 }
2355
2356 return S_OK;
2357}
2358
2359HRESULT Medium::lockRead(ComPtr<IToken> &aToken)
2360{
2361 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2362
2363 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
2364 if (m->queryInfoRunning)
2365 {
2366 /* Must not hold the media tree lock, as Medium::i_queryInfo needs this
2367 * lock and thus we would run into a deadlock here. */
2368 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2369 while (m->queryInfoRunning)
2370 {
2371 alock.release();
2372 /* must not hold the object lock now */
2373 Assert(!isWriteLockOnCurrentThread());
2374 {
2375 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2376 }
2377 alock.acquire();
2378 }
2379 }
2380
2381 HRESULT hrc = S_OK;
2382
2383 switch (m->state)
2384 {
2385 case MediumState_Created:
2386 case MediumState_Inaccessible:
2387 case MediumState_LockedRead:
2388 {
2389 ++m->readers;
2390
2391 ComAssertMsgBreak(m->readers != 0, (tr("Counter overflow")), hrc = E_FAIL);
2392
2393 /* Remember pre-lock state */
2394 if (m->state != MediumState_LockedRead)
2395 m->preLockState = m->state;
2396
2397 LogFlowThisFunc(("Okay - prev state=%d readers=%d\n", m->state, m->readers));
2398 m->state = MediumState_LockedRead;
2399
2400 ComObjPtr<MediumLockToken> pToken;
2401 hrc = pToken.createObject();
2402 if (SUCCEEDED(hrc))
2403 hrc = pToken->init(this, false /* fWrite */);
2404 if (FAILED(hrc))
2405 {
2406 --m->readers;
2407 if (m->readers == 0)
2408 m->state = m->preLockState;
2409 return hrc;
2410 }
2411
2412 pToken.queryInterfaceTo(aToken.asOutParam());
2413 break;
2414 }
2415 default:
2416 {
2417 LogFlowThisFunc(("Failing - state=%d\n", m->state));
2418 hrc = i_setStateError();
2419 break;
2420 }
2421 }
2422
2423 return hrc;
2424}
2425
2426/**
2427 * @note @a aState may be NULL if the state value is not needed (only for
2428 * in-process calls).
2429 */
2430HRESULT Medium::i_unlockRead(MediumState_T *aState)
2431{
2432 AutoCaller autoCaller(this);
2433 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
2434
2435 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2436
2437 HRESULT hrc = S_OK;
2438
2439 switch (m->state)
2440 {
2441 case MediumState_LockedRead:
2442 {
2443 ComAssertMsgBreak(m->readers != 0, (tr("Counter underflow")), hrc = E_FAIL);
2444 --m->readers;
2445
2446 /* Reset the state after the last reader */
2447 if (m->readers == 0)
2448 {
2449 m->state = m->preLockState;
2450 /* There are cases where we inject the deleting state into
2451 * a medium locked for reading. Make sure #unmarkForDeletion()
2452 * gets the right state afterwards. */
2453 if (m->preLockState == MediumState_Deleting)
2454 m->preLockState = MediumState_Created;
2455 }
2456
2457 LogFlowThisFunc(("new state=%d\n", m->state));
2458 break;
2459 }
2460 default:
2461 {
2462 LogFlowThisFunc(("Failing - state=%d\n", m->state));
2463 hrc = setError(VBOX_E_INVALID_OBJECT_STATE, tr("Medium '%s' is not locked for reading"), m->strLocationFull.c_str());
2464 break;
2465 }
2466 }
2467
2468 /* return the current state after */
2469 if (aState)
2470 *aState = m->state;
2471
2472 return hrc;
2473}
2474HRESULT Medium::lockWrite(ComPtr<IToken> &aToken)
2475{
2476 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2477
2478 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
2479 if (m->queryInfoRunning)
2480 {
2481 /* Must not hold the media tree lock, as Medium::i_queryInfo needs this
2482 * lock and thus we would run into a deadlock here. */
2483 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2484 while (m->queryInfoRunning)
2485 {
2486 alock.release();
2487 /* must not hold the object lock now */
2488 Assert(!isWriteLockOnCurrentThread());
2489 {
2490 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2491 }
2492 alock.acquire();
2493 }
2494 }
2495
2496 HRESULT hrc = S_OK;
2497
2498 switch (m->state)
2499 {
2500 case MediumState_Created:
2501 case MediumState_Inaccessible:
2502 {
2503 m->preLockState = m->state;
2504
2505 LogFlowThisFunc(("Okay - prev state=%d locationFull=%s\n", m->state, i_getLocationFull().c_str()));
2506 m->state = MediumState_LockedWrite;
2507
2508 ComObjPtr<MediumLockToken> pToken;
2509 hrc = pToken.createObject();
2510 if (SUCCEEDED(hrc))
2511 hrc = pToken->init(this, true /* fWrite */);
2512 if (FAILED(hrc))
2513 {
2514 m->state = m->preLockState;
2515 return hrc;
2516 }
2517
2518 pToken.queryInterfaceTo(aToken.asOutParam());
2519 break;
2520 }
2521 default:
2522 {
2523 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, i_getLocationFull().c_str()));
2524 hrc = i_setStateError();
2525 break;
2526 }
2527 }
2528
2529 return hrc;
2530}
2531
2532/**
2533 * @note @a aState may be NULL if the state value is not needed (only for
2534 * in-process calls).
2535 */
2536HRESULT Medium::i_unlockWrite(MediumState_T *aState)
2537{
2538 AutoCaller autoCaller(this);
2539 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
2540
2541 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2542
2543 HRESULT hrc = S_OK;
2544
2545 switch (m->state)
2546 {
2547 case MediumState_LockedWrite:
2548 {
2549 m->state = m->preLockState;
2550 /* There are cases where we inject the deleting state into
2551 * a medium locked for writing. Make sure #unmarkForDeletion()
2552 * gets the right state afterwards. */
2553 if (m->preLockState == MediumState_Deleting)
2554 m->preLockState = MediumState_Created;
2555 LogFlowThisFunc(("new state=%d locationFull=%s\n", m->state, i_getLocationFull().c_str()));
2556 break;
2557 }
2558 default:
2559 {
2560 LogFlowThisFunc(("Failing - state=%d locationFull=%s\n", m->state, i_getLocationFull().c_str()));
2561 hrc = setError(VBOX_E_INVALID_OBJECT_STATE, tr("Medium '%s' is not locked for writing"), m->strLocationFull.c_str());
2562 break;
2563 }
2564 }
2565
2566 /* return the current state after */
2567 if (aState)
2568 *aState = m->state;
2569
2570 return hrc;
2571}
2572
2573HRESULT Medium::close(AutoCaller &aAutoCaller)
2574{
2575 // make a copy of VirtualBox pointer which gets nulled by uninit()
2576 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
2577
2578 Guid uId = i_getId();
2579 DeviceType_T devType = i_getDeviceType();
2580 MultiResult mrc = i_close(aAutoCaller);
2581
2582 pVirtualBox->i_saveModifiedRegistries();
2583
2584 if (SUCCEEDED(mrc) && uId.isValid() && !uId.isZero())
2585 pVirtualBox->i_onMediumRegistered(uId, devType, FALSE);
2586
2587 return mrc;
2588}
2589
2590HRESULT Medium::getProperty(const com::Utf8Str &aName,
2591 com::Utf8Str &aValue)
2592{
2593 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2594
2595 settings::StringsMap::const_iterator it = m->mapProperties.find(aName);
2596 if (it == m->mapProperties.end())
2597 {
2598 if (!aName.startsWith("Special/"))
2599 return setError(VBOX_E_OBJECT_NOT_FOUND,
2600 tr("Property '%s' does not exist"), aName.c_str());
2601 else
2602 /* be more silent here */
2603 return VBOX_E_OBJECT_NOT_FOUND;
2604 }
2605
2606 aValue = it->second;
2607
2608 return S_OK;
2609}
2610
2611HRESULT Medium::setProperty(const com::Utf8Str &aName,
2612 const com::Utf8Str &aValue)
2613{
2614 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
2615
2616 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
2617 if (m->queryInfoRunning)
2618 {
2619 /* Must not hold the media tree lock, as Medium::i_queryInfo needs this
2620 * lock and thus we would run into a deadlock here. */
2621 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2622 while (m->queryInfoRunning)
2623 {
2624 mlock.release();
2625 /* must not hold the object lock now */
2626 Assert(!isWriteLockOnCurrentThread());
2627 {
2628 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
2629 }
2630 mlock.acquire();
2631 }
2632 }
2633
2634 switch (m->state)
2635 {
2636 case MediumState_NotCreated:
2637 case MediumState_Created:
2638 case MediumState_Inaccessible:
2639 break;
2640 default:
2641 return i_setStateError();
2642 }
2643
2644 settings::StringsMap::iterator it = m->mapProperties.find(aName);
2645 if ( !aName.startsWith("Special/")
2646 && !i_isPropertyForFilter(aName))
2647 {
2648 if (it == m->mapProperties.end())
2649 return setError(VBOX_E_OBJECT_NOT_FOUND,
2650 tr("Property '%s' does not exist"),
2651 aName.c_str());
2652 it->second = aValue;
2653 }
2654 else
2655 {
2656 if (it == m->mapProperties.end())
2657 {
2658 if (!aValue.isEmpty())
2659 m->mapProperties[aName] = aValue;
2660 }
2661 else
2662 {
2663 if (!aValue.isEmpty())
2664 it->second = aValue;
2665 else
2666 m->mapProperties.erase(it);
2667 }
2668 }
2669
2670 // save the settings
2671 mlock.release();
2672 i_markRegistriesModified();
2673 m->pVirtualBox->i_saveModifiedRegistries();
2674 m->pVirtualBox->i_onMediumConfigChanged(this);
2675
2676 return S_OK;
2677}
2678
2679HRESULT Medium::getProperties(const com::Utf8Str &aNames,
2680 std::vector<com::Utf8Str> &aReturnNames,
2681 std::vector<com::Utf8Str> &aReturnValues)
2682{
2683 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2684
2685 /// @todo make use of aNames according to the documentation
2686 NOREF(aNames);
2687
2688 aReturnNames.resize(m->mapProperties.size());
2689 aReturnValues.resize(m->mapProperties.size());
2690 size_t i = 0;
2691 for (settings::StringsMap::const_iterator it = m->mapProperties.begin();
2692 it != m->mapProperties.end();
2693 ++it, ++i)
2694 {
2695 aReturnNames[i] = it->first;
2696 aReturnValues[i] = it->second;
2697 }
2698 return S_OK;
2699}
2700
2701HRESULT Medium::setProperties(const std::vector<com::Utf8Str> &aNames,
2702 const std::vector<com::Utf8Str> &aValues)
2703{
2704 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
2705
2706 /* first pass: validate names */
2707 for (size_t i = 0;
2708 i < aNames.size();
2709 ++i)
2710 {
2711 Utf8Str strName(aNames[i]);
2712 if ( !strName.startsWith("Special/")
2713 && !i_isPropertyForFilter(strName)
2714 && m->mapProperties.find(strName) == m->mapProperties.end())
2715 return setError(VBOX_E_OBJECT_NOT_FOUND,
2716 tr("Property '%s' does not exist"), strName.c_str());
2717 }
2718
2719 /* second pass: assign */
2720 for (size_t i = 0;
2721 i < aNames.size();
2722 ++i)
2723 {
2724 Utf8Str strName(aNames[i]);
2725 Utf8Str strValue(aValues[i]);
2726 settings::StringsMap::iterator it = m->mapProperties.find(strName);
2727 if ( !strName.startsWith("Special/")
2728 && !i_isPropertyForFilter(strName))
2729 {
2730 AssertReturn(it != m->mapProperties.end(), E_FAIL);
2731 it->second = strValue;
2732 }
2733 else
2734 {
2735 if (it == m->mapProperties.end())
2736 {
2737 if (!strValue.isEmpty())
2738 m->mapProperties[strName] = strValue;
2739 }
2740 else
2741 {
2742 if (!strValue.isEmpty())
2743 it->second = strValue;
2744 else
2745 m->mapProperties.erase(it);
2746 }
2747 }
2748 }
2749
2750 // save the settings
2751 mlock.release();
2752 i_markRegistriesModified();
2753 m->pVirtualBox->i_saveModifiedRegistries();
2754 m->pVirtualBox->i_onMediumConfigChanged(this);
2755
2756 return S_OK;
2757}
2758
2759HRESULT Medium::createBaseStorage(LONG64 aLogicalSize,
2760 const std::vector<MediumVariant_T> &aVariant,
2761 ComPtr<IProgress> &aProgress)
2762{
2763 if (aLogicalSize < 0)
2764 return setError(E_INVALIDARG, tr("The medium size argument (%lld) is negative"), aLogicalSize);
2765
2766 HRESULT hrc = S_OK;
2767 ComObjPtr<Progress> pProgress;
2768 Medium::Task *pTask = NULL;
2769
2770 try
2771 {
2772 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2773
2774 ULONG mediumVariantFlags = 0;
2775
2776 if (aVariant.size())
2777 {
2778 for (size_t i = 0; i < aVariant.size(); i++)
2779 mediumVariantFlags |= (ULONG)aVariant[i];
2780 }
2781
2782 mediumVariantFlags &= ((unsigned)~MediumVariant_Diff);
2783
2784 if ( !(mediumVariantFlags & MediumVariant_Fixed)
2785 && !(m->formatObj->i_getCapabilities() & MediumFormatCapabilities_CreateDynamic))
2786 throw setError(VBOX_E_NOT_SUPPORTED,
2787 tr("Medium format '%s' does not support dynamic storage creation"),
2788 m->strFormat.c_str());
2789
2790 if ( (mediumVariantFlags & MediumVariant_Fixed)
2791 && !(m->formatObj->i_getCapabilities() & MediumFormatCapabilities_CreateFixed))
2792 throw setError(VBOX_E_NOT_SUPPORTED,
2793 tr("Medium format '%s' does not support fixed storage creation"),
2794 m->strFormat.c_str());
2795
2796 if ( (mediumVariantFlags & MediumVariant_Formatted)
2797 && i_getDeviceType() != DeviceType_Floppy)
2798 throw setError(VBOX_E_NOT_SUPPORTED,
2799 tr("Medium variant 'formatted' applies to floppy images only"));
2800
2801 if (m->state != MediumState_NotCreated)
2802 throw i_setStateError();
2803
2804 pProgress.createObject();
2805 hrc = pProgress->init(m->pVirtualBox,
2806 static_cast<IMedium*>(this),
2807 mediumVariantFlags & MediumVariant_Fixed
2808 ? BstrFmt(tr("Creating fixed medium storage unit '%s'"), m->strLocationFull.c_str()).raw()
2809 : BstrFmt(tr("Creating dynamic medium storage unit '%s'"), m->strLocationFull.c_str()).raw(),
2810 TRUE /* aCancelable */);
2811 if (FAILED(hrc))
2812 throw hrc;
2813
2814 /* setup task object to carry out the operation asynchronously */
2815 pTask = new Medium::CreateBaseTask(this, pProgress, (uint64_t)aLogicalSize,
2816 (MediumVariant_T)mediumVariantFlags);
2817 hrc = pTask->hrc();
2818 AssertComRC(hrc);
2819 if (FAILED(hrc))
2820 throw hrc;
2821
2822 m->state = MediumState_Creating;
2823 }
2824 catch (HRESULT hrcXcpt)
2825 {
2826 hrc = hrcXcpt;
2827 }
2828
2829 if (SUCCEEDED(hrc))
2830 {
2831 hrc = pTask->createThread();
2832 pTask = NULL;
2833
2834 if (SUCCEEDED(hrc))
2835 pProgress.queryInterfaceTo(aProgress.asOutParam());
2836 }
2837 else if (pTask != NULL)
2838 delete pTask;
2839
2840 return hrc;
2841}
2842
2843HRESULT Medium::deleteStorage(ComPtr<IProgress> &aProgress)
2844{
2845 ComObjPtr<Progress> pProgress;
2846
2847 MultiResult mrc = i_deleteStorage(&pProgress,
2848 false /* aWait */,
2849 true /* aNotify */);
2850 /* Must save the registries in any case, since an entry was removed. */
2851 m->pVirtualBox->i_saveModifiedRegistries();
2852
2853 if (SUCCEEDED(mrc))
2854 pProgress.queryInterfaceTo(aProgress.asOutParam());
2855
2856 return mrc;
2857}
2858
2859HRESULT Medium::createDiffStorage(AutoCaller &autoCaller,
2860 const ComPtr<IMedium> &aTarget,
2861 const std::vector<MediumVariant_T> &aVariant,
2862 ComPtr<IProgress> &aProgress)
2863{
2864 IMedium *aT = aTarget;
2865 ComObjPtr<Medium> diff = static_cast<Medium*>(aT);
2866
2867 autoCaller.release();
2868
2869 /* It is possible that some previous/concurrent uninit has already cleared
2870 * the pVirtualBox reference, see #uninit(). */
2871 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
2872
2873 // we access m->pParent
2874 AutoReadLock treeLock(!pVirtualBox.isNull() ? &pVirtualBox->i_getMediaTreeLockHandle() : NULL COMMA_LOCKVAL_SRC_POS);
2875
2876 autoCaller.add();
2877 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
2878
2879 AutoMultiWriteLock2 alock(this, diff COMMA_LOCKVAL_SRC_POS);
2880
2881 if (m->type == MediumType_Writethrough)
2882 return setError(VBOX_E_INVALID_OBJECT_STATE,
2883 tr("Medium type of '%s' is Writethrough"),
2884 m->strLocationFull.c_str());
2885 else if (m->type == MediumType_Shareable)
2886 return setError(VBOX_E_INVALID_OBJECT_STATE,
2887 tr("Medium type of '%s' is Shareable"),
2888 m->strLocationFull.c_str());
2889 else if (m->type == MediumType_Readonly)
2890 return setError(VBOX_E_INVALID_OBJECT_STATE,
2891 tr("Medium type of '%s' is Readonly"),
2892 m->strLocationFull.c_str());
2893
2894 /* Apply the normal locking logic to the entire chain. */
2895 MediumLockList *pMediumLockList(new MediumLockList());
2896 alock.release();
2897 autoCaller.release();
2898 treeLock.release();
2899 HRESULT hrc = diff->i_createMediumLockList(true /* fFailIfInaccessible */,
2900 diff /* pToLockWrite */,
2901 false /* fMediumLockWriteAll */,
2902 this,
2903 *pMediumLockList);
2904 treeLock.acquire();
2905 autoCaller.add();
2906 if (FAILED(autoCaller.hrc()))
2907 hrc = autoCaller.hrc();
2908 alock.acquire();
2909 if (FAILED(hrc))
2910 {
2911 delete pMediumLockList;
2912 return hrc;
2913 }
2914
2915 alock.release();
2916 autoCaller.release();
2917 treeLock.release();
2918 hrc = pMediumLockList->Lock();
2919 treeLock.acquire();
2920 autoCaller.add();
2921 if (FAILED(autoCaller.hrc()))
2922 hrc = autoCaller.hrc();
2923 alock.acquire();
2924 if (FAILED(hrc))
2925 {
2926 delete pMediumLockList;
2927
2928 return setError(hrc, tr("Could not lock medium when creating diff '%s'"),
2929 diff->i_getLocationFull().c_str());
2930 }
2931
2932 Guid parentMachineRegistry;
2933 if (i_getFirstRegistryMachineId(parentMachineRegistry))
2934 {
2935 /* since this medium has been just created it isn't associated yet */
2936 diff->m->llRegistryIDs.push_back(parentMachineRegistry);
2937 alock.release();
2938 autoCaller.release();
2939 treeLock.release();
2940 diff->i_markRegistriesModified();
2941 treeLock.acquire();
2942 autoCaller.add();
2943 alock.acquire();
2944 }
2945
2946 alock.release();
2947 autoCaller.release();
2948 treeLock.release();
2949
2950 ComObjPtr<Progress> pProgress;
2951
2952 ULONG mediumVariantFlags = 0;
2953
2954 if (aVariant.size())
2955 {
2956 for (size_t i = 0; i < aVariant.size(); i++)
2957 mediumVariantFlags |= (ULONG)aVariant[i];
2958 }
2959
2960 if (mediumVariantFlags & MediumVariant_Formatted)
2961 {
2962 delete pMediumLockList;
2963 return setError(VBOX_E_NOT_SUPPORTED,
2964 tr("Medium variant 'formatted' applies to floppy images only"));
2965 }
2966
2967 hrc = i_createDiffStorage(diff, (MediumVariant_T)mediumVariantFlags, pMediumLockList,
2968 &pProgress, false /* aWait */, true /* aNotify */);
2969 if (FAILED(hrc))
2970 delete pMediumLockList;
2971 else
2972 pProgress.queryInterfaceTo(aProgress.asOutParam());
2973
2974 return hrc;
2975}
2976
2977HRESULT Medium::mergeTo(const ComPtr<IMedium> &aTarget,
2978 ComPtr<IProgress> &aProgress)
2979{
2980 IMedium *aT = aTarget;
2981
2982 ComAssertRet(aT != this, E_INVALIDARG);
2983
2984 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aT);
2985
2986 bool fMergeForward = false;
2987 ComObjPtr<Medium> pParentForTarget;
2988 MediumLockList *pChildrenToReparent = NULL;
2989 MediumLockList *pMediumLockList = NULL;
2990
2991 HRESULT hrc = i_prepareMergeTo(pTarget, NULL, NULL, true, fMergeForward,
2992 pParentForTarget, pChildrenToReparent, pMediumLockList);
2993 if (FAILED(hrc)) return hrc;
2994
2995 ComObjPtr<Progress> pProgress;
2996
2997 hrc = i_mergeTo(pTarget, fMergeForward, pParentForTarget, pChildrenToReparent,
2998 pMediumLockList, &pProgress, false /* aWait */, true /* aNotify */);
2999 if (FAILED(hrc))
3000 i_cancelMergeTo(pChildrenToReparent, pMediumLockList);
3001 else
3002 pProgress.queryInterfaceTo(aProgress.asOutParam());
3003
3004 return hrc;
3005}
3006
3007HRESULT Medium::cloneToBase(const ComPtr<IMedium> &aTarget,
3008 const std::vector<MediumVariant_T> &aVariant,
3009 ComPtr<IProgress> &aProgress)
3010{
3011 return cloneTo(aTarget, aVariant, NULL, aProgress);
3012}
3013
3014HRESULT Medium::cloneTo(const ComPtr<IMedium> &aTarget,
3015 const std::vector<MediumVariant_T> &aVariant,
3016 const ComPtr<IMedium> &aParent,
3017 ComPtr<IProgress> &aProgress)
3018{
3019 return resizeAndCloneTo(aTarget, 0, aVariant, aParent, aProgress);
3020}
3021
3022/**
3023 * This is a helper function that combines the functionality of
3024 * Medium::cloneTo() and Medium::resize(). The target medium will take the
3025 * contents of the calling medium.
3026 *
3027 * @param aTarget Medium to resize and clone to
3028 * @param aLogicalSize Desired size for targer medium
3029 * @param aVariant
3030 * @param aParent
3031 * @param aProgress
3032 * @return HRESULT
3033 */
3034HRESULT Medium::resizeAndCloneTo(const ComPtr<IMedium> &aTarget,
3035 LONG64 aLogicalSize,
3036 const std::vector<MediumVariant_T> &aVariant,
3037 const ComPtr<IMedium> &aParent,
3038 ComPtr<IProgress> &aProgress)
3039{
3040 /* Check for valid args */
3041 ComAssertRet(aTarget != this, E_INVALIDARG);
3042 CheckComArgExpr(aLogicalSize, aLogicalSize >= 0);
3043
3044 /* Convert args to usable/needed types */
3045 IMedium *aT = aTarget;
3046 ComObjPtr<Medium> pTarget = static_cast<Medium*>(aT);
3047 ComObjPtr<Medium> pParent;
3048 if (aParent)
3049 {
3050 IMedium *aP = aParent;
3051 pParent = static_cast<Medium*>(aP);
3052 }
3053
3054 /* Set up variables. Fetch needed data in lockable blocks */
3055 HRESULT hrc = S_OK;
3056 Medium::Task *pTask = NULL;
3057
3058 Utf8Str strSourceName;
3059 {
3060 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3061 strSourceName = i_getName();
3062 }
3063
3064 uint64_t uTargetExistingSize = 0;
3065 Utf8Str strTargetName;
3066 {
3067 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
3068 uTargetExistingSize = pTarget->i_getLogicalSize();
3069 strTargetName = pTarget->i_getName();
3070 }
3071
3072 /* Set up internal multi-subprocess progress object */
3073 ComObjPtr<Progress> pProgress;
3074 pProgress.createObject();
3075 hrc = pProgress->init(m->pVirtualBox,
3076 static_cast<IMedium*>(this),
3077 BstrFmt(tr("Resizing medium and cloning into it")).raw(),
3078 TRUE, /* aCancelable */
3079 2, /* Number of opearations */
3080 BstrFmt(tr("Resizing medium before clone")).raw()
3081 );
3082 if (FAILED(hrc))
3083 return hrc;
3084
3085 /** @todo r=jack: check below block for correct lock handling */
3086 /* If target does not exist, handle resize. */
3087 if (pTarget->m->state != MediumState_NotCreated && aLogicalSize > 0)
3088 {
3089 if ((LONG64)uTargetExistingSize != aLogicalSize) {
3090 if (!i_isMediumFormatFile())
3091 {
3092 hrc = setError(VBOX_E_NOT_SUPPORTED,
3093 tr("Sizes of '%s' and '%s' are different and medium format does not support resing"),
3094 strSourceName.c_str(), strTargetName.c_str());
3095 return hrc;
3096 }
3097
3098 /**
3099 * Need to lock the target medium as i_resize does do so
3100 * automatically.
3101 */
3102
3103 ComPtr<IToken> pToken;
3104 hrc = pTarget->LockWrite(pToken.asOutParam());
3105
3106 if (FAILED(hrc)) return hrc;
3107
3108 /**
3109 * Have to make own lock list, because "resize" method resizes only
3110 * last image in the lock chain.
3111 */
3112
3113 MediumLockList* pMediumLockListForResize = new MediumLockList();
3114 pMediumLockListForResize->Append(pTarget, pTarget->m->state == MediumState_LockedWrite);
3115
3116 hrc = pMediumLockListForResize->Lock(true /* fSkipOverLockedMedia */);
3117 if (FAILED(hrc))
3118 {
3119 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3120 hrc = setError(hrc,
3121 tr("Failed to lock the medium '%s' to resize before merge"),
3122 strTargetName.c_str());
3123 delete pMediumLockListForResize;
3124 return hrc;
3125 }
3126
3127
3128 hrc = pTarget->i_resize((uint64_t)aLogicalSize, pMediumLockListForResize, &pProgress, true, false);
3129 if (FAILED(hrc))
3130 {
3131 /* No need to setError becasue i_resize and i_taskResizeHandler handle this automatically. */
3132 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3133 delete pMediumLockListForResize;
3134 return hrc;
3135 }
3136
3137 delete pMediumLockListForResize;
3138
3139 pTarget->m->logicalSize = (uint64_t)aLogicalSize;
3140
3141 pToken->Abandon();
3142 pToken.setNull();
3143 }
3144 }
3145
3146 /* Report progress to supplied progress argument */
3147 if (SUCCEEDED(hrc))
3148 {
3149 pProgress.queryInterfaceTo(aProgress.asOutParam());
3150 }
3151
3152 try
3153 {
3154 // locking: we need the tree lock first because we access parent pointers
3155 // and we need to write-lock the media involved
3156 uint32_t cHandles = 3;
3157 LockHandle* pHandles[4] = { &m->pVirtualBox->i_getMediaTreeLockHandle(),
3158 this->lockHandle(),
3159 pTarget->lockHandle() };
3160 /* Only add parent to the lock if it is not null */
3161 if (!pParent.isNull())
3162 pHandles[cHandles++] = pParent->lockHandle();
3163 AutoWriteLock alock(cHandles,
3164 pHandles
3165 COMMA_LOCKVAL_SRC_POS);
3166
3167 if ( pTarget->m->state != MediumState_NotCreated
3168 && pTarget->m->state != MediumState_Created)
3169 throw pTarget->i_setStateError();
3170
3171 /* Determine if cloning within the list */
3172 bool cloningWithinList = false;
3173 ComObjPtr<Medium> pMedium = i_getParent();
3174 while (!pMedium.isNull() && pMedium != pTarget)
3175 pMedium = pMedium->i_getParent();
3176 if (pMedium == pTarget)
3177 cloningWithinList = true;
3178
3179 /* Build the source lock list. */
3180 MediumLockList *pSourceMediumLockList(new MediumLockList());
3181 alock.release();
3182 if (!cloningWithinList)
3183 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
3184 NULL /* pToLockWrite */,
3185 false /* fMediumLockWriteAll */,
3186 NULL,
3187 *pSourceMediumLockList);
3188 else
3189 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
3190 pTarget /* pToLockWrite */,
3191 false /* fMediumLockWriteAll */,
3192 NULL,
3193 *pSourceMediumLockList);
3194 alock.acquire();
3195 if (FAILED(hrc))
3196 {
3197 delete pSourceMediumLockList;
3198 throw hrc;
3199 }
3200
3201 /* Build the target lock list (including the to-be parent chain). */
3202 MediumLockList *pTargetMediumLockList(new MediumLockList());
3203 alock.release();
3204 if (!cloningWithinList)
3205 hrc = pTarget->i_createMediumLockList(true /* fFailIfInaccessible */,
3206 pTarget /* pToLockWrite */,
3207 false /* fMediumLockWriteAll */,
3208 pParent,
3209 *pTargetMediumLockList);
3210 alock.acquire();
3211 if (FAILED(hrc))
3212 {
3213 delete pSourceMediumLockList;
3214 delete pTargetMediumLockList;
3215 throw hrc;
3216 }
3217
3218 alock.release();
3219 hrc = pSourceMediumLockList->Lock();
3220 alock.acquire();
3221 if (FAILED(hrc))
3222 {
3223 delete pSourceMediumLockList;
3224 delete pTargetMediumLockList;
3225 throw setError(hrc,
3226 tr("Failed to lock source media '%s'"),
3227 i_getLocationFull().c_str());
3228 }
3229 alock.release();
3230 hrc = pTargetMediumLockList->Lock();
3231 alock.acquire();
3232 if (FAILED(hrc))
3233 {
3234 delete pSourceMediumLockList;
3235 delete pTargetMediumLockList;
3236 throw setError(hrc,
3237 tr("Failed to lock target media '%s'"),
3238 pTarget->i_getLocationFull().c_str());
3239 }
3240
3241 ULONG mediumVariantFlags = 0;
3242
3243 if (aVariant.size())
3244 {
3245 for (size_t i = 0; i < aVariant.size(); i++)
3246 mediumVariantFlags |= (ULONG)aVariant[i];
3247 }
3248
3249 if (mediumVariantFlags & MediumVariant_Formatted)
3250 {
3251 delete pSourceMediumLockList;
3252 delete pTargetMediumLockList;
3253 throw setError(VBOX_E_NOT_SUPPORTED,
3254 tr("Medium variant 'formatted' applies to floppy images only"));
3255 }
3256
3257 if (pTarget->m->state != MediumState_NotCreated || aLogicalSize == 0)
3258 {
3259 /* setup task object to carry out the operation asynchronously */
3260 pTask = new Medium::CloneTask(this, pProgress, pTarget,
3261 (MediumVariant_T)mediumVariantFlags,
3262 pParent, UINT32_MAX, UINT32_MAX,
3263 pSourceMediumLockList, pTargetMediumLockList,
3264 false, false, true, 0);
3265 }
3266 else
3267 {
3268 /* setup task object to carry out the operation asynchronously */
3269 pTask = new Medium::CloneTask(this, pProgress, pTarget,
3270 (MediumVariant_T)mediumVariantFlags,
3271 pParent, UINT32_MAX, UINT32_MAX,
3272 pSourceMediumLockList, pTargetMediumLockList,
3273 false, false, true, (uint64_t)aLogicalSize);
3274 }
3275
3276 hrc = pTask->hrc();
3277 AssertComRC(hrc);
3278 if (FAILED(hrc))
3279 throw hrc;
3280
3281 if (pTarget->m->state == MediumState_NotCreated)
3282 pTarget->m->state = MediumState_Creating;
3283 }
3284 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3285
3286 if (SUCCEEDED(hrc))
3287 {
3288 hrc = pTask->createThread();
3289 pTask = NULL;
3290 if (SUCCEEDED(hrc))
3291 pProgress.queryInterfaceTo(aProgress.asOutParam());
3292 }
3293 else if (pTask != NULL)
3294 delete pTask;
3295
3296 return hrc;
3297}
3298
3299HRESULT Medium::moveTo(AutoCaller &autoCaller, const com::Utf8Str &aLocation, ComPtr<IProgress> &aProgress)
3300{
3301 ComObjPtr<Medium> pParent;
3302 ComObjPtr<Progress> pProgress;
3303 HRESULT hrc = S_OK;
3304 Medium::Task *pTask = NULL;
3305
3306 try
3307 {
3308 /// @todo NEWMEDIA for file names, add the default extension if no extension
3309 /// is present (using the information from the VD backend which also implies
3310 /// that one more parameter should be passed to moveTo() requesting
3311 /// that functionality since it is only allowed when called from this method
3312
3313 /// @todo NEWMEDIA rename the file and set m->location on success, then save
3314 /// the global registry (and local registries of portable VMs referring to
3315 /// this medium), this will also require to add the mRegistered flag to data
3316
3317 autoCaller.release();
3318
3319 // locking: we need the tree lock first because we access parent pointers
3320 // and we need to write-lock the media involved
3321 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3322
3323 autoCaller.add();
3324 AssertComRCThrowRC(autoCaller.hrc());
3325
3326 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3327
3328 /* play with locations */
3329 {
3330 /* get source path and filename */
3331 Utf8Str sourcePath = i_getLocationFull();
3332 Utf8Str sourceFName = i_getName();
3333
3334 if (aLocation.isEmpty())
3335 {
3336 hrc = setErrorVrc(VERR_PATH_ZERO_LENGTH,
3337 tr("Medium '%s' can't be moved. Destination path is empty."),
3338 i_getLocationFull().c_str());
3339 throw hrc;
3340 }
3341
3342 /* extract destination path and filename */
3343 Utf8Str destPath(aLocation);
3344 Utf8Str destFName(destPath);
3345 destFName.stripPath();
3346
3347 if (destFName.isNotEmpty() && !RTPathHasSuffix(destFName.c_str()))
3348 {
3349 /*
3350 * The target path has no filename: Either "/path/to/new/location" or
3351 * just "newname" (no trailing backslash or there is no filename extension).
3352 */
3353 if (destPath.equals(destFName))
3354 {
3355 /* new path contains only "newname", no path, no extension */
3356 destFName.append(RTPathSuffix(sourceFName.c_str()));
3357 destPath = destFName;
3358 }
3359 else
3360 {
3361 /* new path looks like "/path/to/new/location" */
3362 destFName.setNull();
3363 destPath.append(RTPATH_SLASH);
3364 }
3365 }
3366
3367 if (destFName.isEmpty())
3368 {
3369 /* No target name */
3370 destPath.append(sourceFName);
3371 }
3372 else
3373 {
3374 if (destPath.equals(destFName))
3375 {
3376 /*
3377 * The target path contains of only a filename without a directory.
3378 * Move the medium within the source directory to the new name
3379 * (actually rename operation).
3380 * Scratches sourcePath!
3381 */
3382 destPath = sourcePath.stripFilename().append(RTPATH_SLASH).append(destFName);
3383 }
3384
3385 const char *pszSuffix = RTPathSuffix(sourceFName.c_str());
3386
3387 /* Suffix is empty and one is deduced from the medium format */
3388 if (pszSuffix == NULL)
3389 {
3390 Utf8Str strExt = i_getFormat();
3391 if (strExt.compare("RAW", Utf8Str::CaseInsensitive) == 0)
3392 {
3393 DeviceType_T devType = i_getDeviceType();
3394 switch (devType)
3395 {
3396 case DeviceType_DVD:
3397 strExt = "iso";
3398 break;
3399 case DeviceType_Floppy:
3400 strExt = "img";
3401 break;
3402 default:
3403 hrc = setErrorVrc(VERR_NOT_A_FILE, /** @todo r=bird: Mixing status codes again. */
3404 tr("Medium '%s' has RAW type. \"Move\" operation isn't supported for this type."),
3405 i_getLocationFull().c_str());
3406 throw hrc;
3407 }
3408 }
3409 else if (strExt.compare("Parallels", Utf8Str::CaseInsensitive) == 0)
3410 {
3411 strExt = "hdd";
3412 }
3413
3414 /* Set the target extension like on the source. Any conversions are prohibited */
3415 strExt.toLower();
3416 destPath.stripSuffix().append('.').append(strExt);
3417 }
3418 else
3419 destPath.stripSuffix().append(pszSuffix);
3420 }
3421
3422 /* Simple check for existence */
3423 if (RTFileExists(destPath.c_str()))
3424 throw setError(VBOX_E_FILE_ERROR,
3425 tr("The given path '%s' is an existing file. Delete or rename this file."),
3426 destPath.c_str());
3427
3428 if (!i_isMediumFormatFile())
3429 throw setErrorVrc(VERR_NOT_A_FILE,
3430 tr("Medium '%s' isn't a file object. \"Move\" operation isn't supported."),
3431 i_getLocationFull().c_str());
3432 /* Path must be absolute */
3433 if (!RTPathStartsWithRoot(destPath.c_str()))
3434 throw setError(VBOX_E_FILE_ERROR,
3435 tr("The given path '%s' is not fully qualified"),
3436 destPath.c_str());
3437 /* Check path for a new file object */
3438 hrc = VirtualBox::i_ensureFilePathExists(destPath, true);
3439 if (FAILED(hrc))
3440 throw hrc;
3441
3442 /* Set needed variables for "moving" procedure. It'll be used later in separate thread task */
3443 hrc = i_preparationForMoving(destPath);
3444 if (FAILED(hrc))
3445 throw setErrorVrc(VERR_NO_CHANGE,
3446 tr("Medium '%s' is already in the correct location"),
3447 i_getLocationFull().c_str());
3448 }
3449
3450 /* Check VMs which have this medium attached to*/
3451 std::vector<com::Guid> aMachineIds;
3452 hrc = getMachineIds(aMachineIds);
3453 std::vector<com::Guid>::const_iterator currMachineID = aMachineIds.begin();
3454 std::vector<com::Guid>::const_iterator lastMachineID = aMachineIds.end();
3455
3456 while (currMachineID != lastMachineID)
3457 {
3458 Guid id(*currMachineID);
3459 ComObjPtr<Machine> aMachine;
3460
3461 alock.release();
3462 autoCaller.release();
3463 treeLock.release();
3464 hrc = m->pVirtualBox->i_findMachine(id, false, true, &aMachine);
3465 treeLock.acquire();
3466 autoCaller.add();
3467 AssertComRCThrowRC(autoCaller.hrc());
3468 alock.acquire();
3469
3470 if (SUCCEEDED(hrc))
3471 {
3472 ComObjPtr<SessionMachine> sm;
3473 ComPtr<IInternalSessionControl> ctl;
3474
3475 alock.release();
3476 autoCaller.release();
3477 treeLock.release();
3478 bool ses = aMachine->i_isSessionOpenVM(sm, &ctl);
3479 treeLock.acquire();
3480 autoCaller.add();
3481 AssertComRCThrowRC(autoCaller.hrc());
3482 alock.acquire();
3483
3484 if (ses)
3485 throw setError(VBOX_E_INVALID_VM_STATE,
3486 tr("At least the VM '%s' to whom this medium '%s' attached has currently an opened session. Stop all VMs before relocating this medium"),
3487 id.toString().c_str(), i_getLocationFull().c_str());
3488 }
3489 ++currMachineID;
3490 }
3491
3492 /* Build the source lock list. */
3493 MediumLockList *pMediumLockList(new MediumLockList());
3494 alock.release();
3495 autoCaller.release();
3496 treeLock.release();
3497 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
3498 this /* pToLockWrite */,
3499 true /* fMediumLockWriteAll */,
3500 NULL,
3501 *pMediumLockList);
3502 treeLock.acquire();
3503 autoCaller.add();
3504 AssertComRCThrowRC(autoCaller.hrc());
3505 alock.acquire();
3506 if (FAILED(hrc))
3507 {
3508 delete pMediumLockList;
3509 throw setError(hrc, tr("Failed to create medium lock list for '%s'"), i_getLocationFull().c_str());
3510 }
3511 alock.release();
3512 autoCaller.release();
3513 treeLock.release();
3514 hrc = pMediumLockList->Lock();
3515 treeLock.acquire();
3516 autoCaller.add();
3517 AssertComRCThrowRC(autoCaller.hrc());
3518 alock.acquire();
3519 if (FAILED(hrc))
3520 {
3521 delete pMediumLockList;
3522 throw setError(hrc, tr("Failed to lock media '%s'"), i_getLocationFull().c_str());
3523 }
3524
3525 pProgress.createObject();
3526 hrc = pProgress->init(m->pVirtualBox,
3527 static_cast <IMedium *>(this),
3528 BstrFmt(tr("Moving medium '%s'"), m->strLocationFull.c_str()).raw(),
3529 TRUE /* aCancelable */);
3530
3531 /* Do the disk moving. */
3532 if (SUCCEEDED(hrc))
3533 {
3534 ULONG mediumVariantFlags = i_getVariant();
3535
3536 /* setup task object to carry out the operation asynchronously */
3537 pTask = new Medium::MoveTask(this, pProgress,
3538 (MediumVariant_T)mediumVariantFlags,
3539 pMediumLockList);
3540 hrc = pTask->hrc();
3541 AssertComRC(hrc);
3542 if (FAILED(hrc))
3543 throw hrc;
3544 }
3545
3546 }
3547 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3548
3549 if (SUCCEEDED(hrc))
3550 {
3551 hrc = pTask->createThread();
3552 pTask = NULL;
3553 if (SUCCEEDED(hrc))
3554 pProgress.queryInterfaceTo(aProgress.asOutParam());
3555 }
3556 else
3557 {
3558 if (pTask)
3559 delete pTask;
3560 }
3561
3562 return hrc;
3563}
3564
3565HRESULT Medium::setLocation(const com::Utf8Str &aLocation)
3566{
3567 HRESULT hrc = S_OK;
3568
3569 try
3570 {
3571 // locking: we need the tree lock first because we access parent pointers
3572 // and we need to write-lock the media involved
3573 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
3574
3575 AutoCaller autoCaller(this);
3576 AssertComRCThrowRC(autoCaller.hrc());
3577
3578 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3579
3580 Utf8Str destPath(aLocation);
3581
3582 // some check for file based medium
3583 if (i_isMediumFormatFile())
3584 {
3585 /* Path must be absolute */
3586 if (!RTPathStartsWithRoot(destPath.c_str()))
3587 throw setError(VBOX_E_FILE_ERROR, tr("The given path '%s' is not fully qualified"), destPath.c_str());
3588
3589 /* Simple check for existence */
3590 if (!RTFileExists(destPath.c_str()))
3591 throw setError(VBOX_E_FILE_ERROR,
3592 tr("The given path '%s' is not an existing file. New location is invalid."),
3593 destPath.c_str());
3594 }
3595
3596 /* Check VMs which have this medium attached to*/
3597 std::vector<com::Guid> aMachineIds;
3598 hrc = getMachineIds(aMachineIds);
3599
3600 // switch locks only if there are machines with this medium attached
3601 if (!aMachineIds.empty())
3602 {
3603 std::vector<com::Guid>::const_iterator currMachineID = aMachineIds.begin();
3604 std::vector<com::Guid>::const_iterator lastMachineID = aMachineIds.end();
3605
3606 alock.release();
3607 autoCaller.release();
3608 treeLock.release();
3609
3610 while (currMachineID != lastMachineID)
3611 {
3612 Guid id(*currMachineID);
3613 ComObjPtr<Machine> aMachine;
3614 hrc = m->pVirtualBox->i_findMachine(id, false, true, &aMachine);
3615 if (SUCCEEDED(hrc))
3616 {
3617 ComObjPtr<SessionMachine> sm;
3618 ComPtr<IInternalSessionControl> ctl;
3619
3620 bool ses = aMachine->i_isSessionOpenVM(sm, &ctl);
3621 if (ses)
3622 {
3623 treeLock.acquire();
3624 autoCaller.add();
3625 AssertComRCThrowRC(autoCaller.hrc());
3626 alock.acquire();
3627
3628 throw setError(VBOX_E_INVALID_VM_STATE,
3629 tr("At least the VM '%s' to whom this medium '%s' attached has currently an opened session. Stop all VMs before set location for this medium"),
3630 id.toString().c_str(),
3631 i_getLocationFull().c_str());
3632 }
3633 }
3634 ++currMachineID;
3635 }
3636
3637 treeLock.acquire();
3638 autoCaller.add();
3639 AssertComRCThrowRC(autoCaller.hrc());
3640 alock.acquire();
3641 }
3642
3643 m->strLocationFull = destPath;
3644
3645 // save the settings
3646 alock.release();
3647 autoCaller.release();
3648 treeLock.release();
3649
3650 i_markRegistriesModified();
3651 m->pVirtualBox->i_saveModifiedRegistries();
3652
3653 MediumState_T mediumState;
3654 refreshState(autoCaller, &mediumState);
3655 m->pVirtualBox->i_onMediumConfigChanged(this);
3656 }
3657 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3658
3659 return hrc;
3660}
3661
3662HRESULT Medium::compact(ComPtr<IProgress> &aProgress)
3663{
3664 HRESULT hrc = S_OK;
3665 ComObjPtr<Progress> pProgress;
3666 Medium::Task *pTask = NULL;
3667
3668 try
3669 {
3670 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3671
3672 /* Build the medium lock list. */
3673 MediumLockList *pMediumLockList(new MediumLockList());
3674 alock.release();
3675 hrc = i_createMediumLockList(true /* fFailIfInaccessible */ ,
3676 this /* pToLockWrite */,
3677 false /* fMediumLockWriteAll */,
3678 NULL,
3679 *pMediumLockList);
3680 alock.acquire();
3681 if (FAILED(hrc))
3682 {
3683 delete pMediumLockList;
3684 throw hrc;
3685 }
3686
3687 alock.release();
3688 hrc = pMediumLockList->Lock();
3689 alock.acquire();
3690 if (FAILED(hrc))
3691 {
3692 delete pMediumLockList;
3693 throw setError(hrc,
3694 tr("Failed to lock media when compacting '%s'"),
3695 i_getLocationFull().c_str());
3696 }
3697
3698 pProgress.createObject();
3699 hrc = pProgress->init(m->pVirtualBox,
3700 static_cast <IMedium *>(this),
3701 BstrFmt(tr("Compacting medium '%s'"), m->strLocationFull.c_str()).raw(),
3702 TRUE /* aCancelable */);
3703 if (FAILED(hrc))
3704 {
3705 delete pMediumLockList;
3706 throw hrc;
3707 }
3708
3709 /* setup task object to carry out the operation asynchronously */
3710 pTask = new Medium::CompactTask(this, pProgress, pMediumLockList);
3711 hrc = pTask->hrc();
3712 AssertComRC(hrc);
3713 if (FAILED(hrc))
3714 throw hrc;
3715 }
3716 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3717
3718 if (SUCCEEDED(hrc))
3719 {
3720 hrc = pTask->createThread();
3721 pTask = NULL;
3722 if (SUCCEEDED(hrc))
3723 pProgress.queryInterfaceTo(aProgress.asOutParam());
3724 }
3725 else if (pTask != NULL)
3726 delete pTask;
3727
3728 return hrc;
3729}
3730
3731HRESULT Medium::resize(LONG64 aLogicalSize,
3732 ComPtr<IProgress> &aProgress)
3733{
3734 CheckComArgExpr(aLogicalSize, aLogicalSize > 0);
3735 HRESULT hrc = S_OK;
3736 ComObjPtr<Progress> pProgress;
3737
3738 /* Build the medium lock list. */
3739 MediumLockList *pMediumLockList(new MediumLockList());
3740
3741 try
3742 {
3743 const char *pszError = NULL;
3744
3745 hrc = i_createMediumLockList(true /* fFailIfInaccessible */ ,
3746 this /* pToLockWrite */,
3747 false /* fMediumLockWriteAll */,
3748 NULL,
3749 *pMediumLockList);
3750 if (FAILED(hrc))
3751 pszError = tr("Failed to create medium lock list when resizing '%s'");
3752 else
3753 {
3754 hrc = pMediumLockList->Lock();
3755 if (FAILED(hrc))
3756 pszError = tr("Failed to lock media when resizing '%s'");
3757 }
3758
3759
3760 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3761
3762 if (FAILED(hrc))
3763 {
3764 throw setError(hrc, pszError, i_getLocationFull().c_str());
3765 }
3766
3767 pProgress.createObject();
3768 hrc = pProgress->init(m->pVirtualBox,
3769 static_cast <IMedium *>(this),
3770 BstrFmt(tr("Resizing medium '%s'"), m->strLocationFull.c_str()).raw(),
3771 TRUE /* aCancelable */);
3772 if (FAILED(hrc))
3773 {
3774 throw hrc;
3775 }
3776 }
3777 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3778
3779 if (SUCCEEDED(hrc))
3780 hrc = i_resize((uint64_t)aLogicalSize, pMediumLockList, &pProgress, false /* aWait */, true /* aNotify */);
3781
3782 if (SUCCEEDED(hrc))
3783 pProgress.queryInterfaceTo(aProgress.asOutParam());
3784 else
3785 delete pMediumLockList;
3786
3787 return hrc;
3788}
3789
3790HRESULT Medium::reset(AutoCaller &autoCaller, ComPtr<IProgress> &aProgress)
3791{
3792 HRESULT hrc = S_OK;
3793 ComObjPtr<Progress> pProgress;
3794 Medium::Task *pTask = NULL;
3795
3796 try
3797 {
3798 autoCaller.release();
3799
3800 /* It is possible that some previous/concurrent uninit has already
3801 * cleared the pVirtualBox reference, see #uninit(). */
3802 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
3803
3804 /* i_canClose() needs the tree lock */
3805 AutoMultiWriteLock2 multilock(!pVirtualBox.isNull() ? &pVirtualBox->i_getMediaTreeLockHandle() : NULL,
3806 this->lockHandle()
3807 COMMA_LOCKVAL_SRC_POS);
3808
3809 autoCaller.add();
3810 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
3811
3812 LogFlowThisFunc(("ENTER for medium %s\n", m->strLocationFull.c_str()));
3813
3814 if (m->pParent.isNull())
3815 throw setError(VBOX_E_NOT_SUPPORTED,
3816 tr("Medium type of '%s' is not differencing"),
3817 m->strLocationFull.c_str());
3818
3819 hrc = i_canClose();
3820 if (FAILED(hrc))
3821 throw hrc;
3822
3823 /* Build the medium lock list. */
3824 MediumLockList *pMediumLockList(new MediumLockList());
3825 multilock.release();
3826 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
3827 this /* pToLockWrite */,
3828 false /* fMediumLockWriteAll */,
3829 NULL,
3830 *pMediumLockList);
3831 multilock.acquire();
3832 if (FAILED(hrc))
3833 {
3834 delete pMediumLockList;
3835 throw hrc;
3836 }
3837
3838 multilock.release();
3839 hrc = pMediumLockList->Lock();
3840 multilock.acquire();
3841 if (FAILED(hrc))
3842 {
3843 delete pMediumLockList;
3844 throw setError(hrc,
3845 tr("Failed to lock media when resetting '%s'"),
3846 i_getLocationFull().c_str());
3847 }
3848
3849 pProgress.createObject();
3850 hrc = pProgress->init(m->pVirtualBox,
3851 static_cast<IMedium*>(this),
3852 BstrFmt(tr("Resetting differencing medium '%s'"), m->strLocationFull.c_str()).raw(),
3853 FALSE /* aCancelable */);
3854 if (FAILED(hrc))
3855 throw hrc;
3856
3857 /* setup task object to carry out the operation asynchronously */
3858 pTask = new Medium::ResetTask(this, pProgress, pMediumLockList);
3859 hrc = pTask->hrc();
3860 AssertComRC(hrc);
3861 if (FAILED(hrc))
3862 throw hrc;
3863 }
3864 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
3865
3866 if (SUCCEEDED(hrc))
3867 {
3868 hrc = pTask->createThread();
3869 pTask = NULL;
3870 if (SUCCEEDED(hrc))
3871 pProgress.queryInterfaceTo(aProgress.asOutParam());
3872 }
3873 else if (pTask != NULL)
3874 delete pTask;
3875
3876 LogFlowThisFunc(("LEAVE, hrc=%Rhrc\n", hrc));
3877
3878 return hrc;
3879}
3880
3881HRESULT Medium::changeEncryption(const com::Utf8Str &aCurrentPassword, const com::Utf8Str &aCipher,
3882 const com::Utf8Str &aNewPassword, const com::Utf8Str &aNewPasswordId,
3883 ComPtr<IProgress> &aProgress)
3884{
3885 HRESULT hrc = S_OK;
3886 ComObjPtr<Progress> pProgress;
3887 Medium::Task *pTask = NULL;
3888
3889 try
3890 {
3891 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3892
3893 DeviceType_T devType = i_getDeviceType();
3894 /* Cannot encrypt DVD or floppy images so far. */
3895 if ( devType == DeviceType_DVD
3896 || devType == DeviceType_Floppy)
3897 return setError(VBOX_E_INVALID_OBJECT_STATE,
3898 tr("Cannot encrypt DVD or Floppy medium '%s'"),
3899 m->strLocationFull.c_str());
3900
3901 /* Cannot encrypt media which are attached to more than one virtual machine. */
3902 if (m->backRefs.size() > 1)
3903 return setError(VBOX_E_INVALID_OBJECT_STATE,
3904 tr("Cannot encrypt medium '%s' because it is attached to %d virtual machines", "", m->backRefs.size()),
3905 m->strLocationFull.c_str(), m->backRefs.size());
3906
3907 if (i_getChildren().size() != 0)
3908 return setError(VBOX_E_INVALID_OBJECT_STATE,
3909 tr("Cannot encrypt medium '%s' because it has %d children", "", i_getChildren().size()),
3910 m->strLocationFull.c_str(), i_getChildren().size());
3911
3912 /* Build the medium lock list. */
3913 MediumLockList *pMediumLockList(new MediumLockList());
3914 alock.release();
3915 hrc = i_createMediumLockList(true /* fFailIfInaccessible */ ,
3916 this /* pToLockWrite */,
3917 true /* fMediumLockAllWrite */,
3918 NULL,
3919 *pMediumLockList);
3920 alock.acquire();
3921 if (FAILED(hrc))
3922 {
3923 delete pMediumLockList;
3924 throw hrc;
3925 }
3926
3927 alock.release();
3928 hrc = pMediumLockList->Lock();
3929 alock.acquire();
3930 if (FAILED(hrc))
3931 {
3932 delete pMediumLockList;
3933 throw setError(hrc,
3934 tr("Failed to lock media for encryption '%s'"),
3935 i_getLocationFull().c_str());
3936 }
3937
3938 /*
3939 * Check all media in the chain to not contain any branches or references to
3940 * other virtual machines, we support encrypting only a list of differencing media at the moment.
3941 */
3942 MediumLockList::Base::const_iterator mediumListBegin = pMediumLockList->GetBegin();
3943 MediumLockList::Base::const_iterator mediumListEnd = pMediumLockList->GetEnd();
3944 for (MediumLockList::Base::const_iterator it = mediumListBegin;
3945 it != mediumListEnd;
3946 ++it)
3947 {
3948 const MediumLock &mediumLock = *it;
3949 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
3950 AutoReadLock mediumReadLock(pMedium COMMA_LOCKVAL_SRC_POS);
3951
3952 Assert(pMedium->m->state == MediumState_LockedWrite);
3953
3954 if (pMedium->m->backRefs.size() > 1)
3955 {
3956 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
3957 tr("Cannot encrypt medium '%s' because it is attached to %d virtual machines", "",
3958 pMedium->m->backRefs.size()),
3959 pMedium->m->strLocationFull.c_str(), pMedium->m->backRefs.size());
3960 break;
3961 }
3962 else if (pMedium->i_getChildren().size() > 1)
3963 {
3964 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
3965 tr("Cannot encrypt medium '%s' because it has %d children", "", pMedium->i_getChildren().size()),
3966 pMedium->m->strLocationFull.c_str(), pMedium->i_getChildren().size());
3967 break;
3968 }
3969 }
3970
3971 if (FAILED(hrc))
3972 {
3973 delete pMediumLockList;
3974 throw hrc;
3975 }
3976
3977 const char *pszAction = tr("Encrypting medium");
3978 if ( aCurrentPassword.isNotEmpty()
3979 && aCipher.isEmpty())
3980 pszAction = tr("Decrypting medium");
3981
3982 pProgress.createObject();
3983 hrc = pProgress->init(m->pVirtualBox,
3984 static_cast <IMedium *>(this),
3985 BstrFmt("%s '%s'", pszAction, m->strLocationFull.c_str()).raw(),
3986 TRUE /* aCancelable */);
3987 if (FAILED(hrc))
3988 {
3989 delete pMediumLockList;
3990 throw hrc;
3991 }
3992
3993 /* setup task object to carry out the operation asynchronously */
3994 pTask = new Medium::EncryptTask(this, aNewPassword, aCurrentPassword,
3995 aCipher, aNewPasswordId, pProgress, pMediumLockList);
3996 hrc = pTask->hrc();
3997 AssertComRC(hrc);
3998 if (FAILED(hrc))
3999 throw hrc;
4000 }
4001 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
4002
4003 if (SUCCEEDED(hrc))
4004 {
4005 hrc = pTask->createThread();
4006 pTask = NULL;
4007 if (SUCCEEDED(hrc))
4008 pProgress.queryInterfaceTo(aProgress.asOutParam());
4009 }
4010 else if (pTask != NULL)
4011 delete pTask;
4012
4013 return hrc;
4014}
4015
4016HRESULT Medium::getEncryptionSettings(AutoCaller &autoCaller, com::Utf8Str &aCipher, com::Utf8Str &aPasswordId)
4017{
4018#ifndef VBOX_WITH_EXTPACK
4019 RT_NOREF(aCipher, aPasswordId);
4020#endif
4021 HRESULT hrc = S_OK;
4022
4023 try
4024 {
4025 autoCaller.release();
4026 ComObjPtr<Medium> pBase = i_getBase();
4027 autoCaller.add();
4028 if (FAILED(autoCaller.hrc()))
4029 throw hrc;
4030 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4031
4032 /* Check whether encryption is configured for this medium. */
4033 settings::StringsMap::iterator it = pBase->m->mapProperties.find("CRYPT/KeyStore");
4034 if (it == pBase->m->mapProperties.end())
4035 throw VBOX_E_NOT_SUPPORTED;
4036
4037# ifdef VBOX_WITH_EXTPACK
4038 ExtPackManager *pExtPackManager = m->pVirtualBox->i_getExtPackManager();
4039 if (pExtPackManager->i_isExtPackUsable(ORACLE_PUEL_EXTPACK_NAME))
4040 {
4041 /* Load the plugin */
4042 Utf8Str strPlugin;
4043 hrc = pExtPackManager->i_getLibraryPathForExtPack(g_szVDPlugin, ORACLE_PUEL_EXTPACK_NAME, &strPlugin);
4044 if (SUCCEEDED(hrc))
4045 {
4046 int vrc = VDPluginLoadFromFilename(strPlugin.c_str());
4047 if (RT_FAILURE(vrc))
4048 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
4049 tr("Retrieving encryption settings of the image failed because the encryption plugin could not be loaded (%s)"),
4050 i_vdError(vrc).c_str());
4051 }
4052 else
4053 throw setError(VBOX_E_NOT_SUPPORTED,
4054 tr("Encryption is not supported because the extension pack '%s' is missing the encryption plugin (old extension pack installed?)"),
4055 ORACLE_PUEL_EXTPACK_NAME);
4056 }
4057 else
4058 throw setError(VBOX_E_NOT_SUPPORTED,
4059 tr("Encryption is not supported because the extension pack '%s' is missing"),
4060 ORACLE_PUEL_EXTPACK_NAME);
4061
4062 PVDISK pDisk = NULL;
4063 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &pDisk);
4064 ComAssertRCThrow(vrc, E_FAIL);
4065
4066 MediumCryptoFilterSettings CryptoSettings;
4067
4068 i_taskEncryptSettingsSetup(&CryptoSettings, NULL, it->second.c_str(), NULL, false /* fCreateKeyStore */);
4069 vrc = VDFilterAdd(pDisk, "CRYPT", VD_FILTER_FLAGS_READ | VD_FILTER_FLAGS_INFO, CryptoSettings.vdFilterIfaces);
4070 if (RT_FAILURE(vrc))
4071 throw setErrorBoth(VBOX_E_INVALID_OBJECT_STATE, vrc,
4072 tr("Failed to load the encryption filter: %s"),
4073 i_vdError(vrc).c_str());
4074
4075 it = pBase->m->mapProperties.find("CRYPT/KeyId");
4076 if (it == pBase->m->mapProperties.end())
4077 throw setError(VBOX_E_INVALID_OBJECT_STATE,
4078 tr("Image is configured for encryption but doesn't has a KeyId set"));
4079
4080 aPasswordId = it->second.c_str();
4081 aCipher = CryptoSettings.pszCipherReturned;
4082 RTStrFree(CryptoSettings.pszCipherReturned);
4083
4084 VDDestroy(pDisk);
4085# else
4086 throw setError(VBOX_E_NOT_SUPPORTED,
4087 tr("Encryption is not supported because extension pack support is not built in"));
4088# endif
4089 }
4090 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
4091
4092 return hrc;
4093}
4094
4095HRESULT Medium::checkEncryptionPassword(const com::Utf8Str &aPassword)
4096{
4097 HRESULT hrc = S_OK;
4098
4099 try
4100 {
4101 ComObjPtr<Medium> pBase = i_getBase();
4102 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4103
4104 settings::StringsMap::iterator it = pBase->m->mapProperties.find("CRYPT/KeyStore");
4105 if (it == pBase->m->mapProperties.end())
4106 throw setError(VBOX_E_NOT_SUPPORTED,
4107 tr("The image is not configured for encryption"));
4108
4109 if (aPassword.isEmpty())
4110 throw setError(E_INVALIDARG,
4111 tr("The given password must not be empty"));
4112
4113# ifdef VBOX_WITH_EXTPACK
4114 ExtPackManager *pExtPackManager = m->pVirtualBox->i_getExtPackManager();
4115 if (pExtPackManager->i_isExtPackUsable(ORACLE_PUEL_EXTPACK_NAME))
4116 {
4117 /* Load the plugin */
4118 Utf8Str strPlugin;
4119 hrc = pExtPackManager->i_getLibraryPathForExtPack(g_szVDPlugin, ORACLE_PUEL_EXTPACK_NAME, &strPlugin);
4120 if (SUCCEEDED(hrc))
4121 {
4122 int vrc = VDPluginLoadFromFilename(strPlugin.c_str());
4123 if (RT_FAILURE(vrc))
4124 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
4125 tr("Retrieving encryption settings of the image failed because the encryption plugin could not be loaded (%s)"),
4126 i_vdError(vrc).c_str());
4127 }
4128 else
4129 throw setError(VBOX_E_NOT_SUPPORTED,
4130 tr("Encryption is not supported because the extension pack '%s' is missing the encryption plugin (old extension pack installed?)"),
4131 ORACLE_PUEL_EXTPACK_NAME);
4132 }
4133 else
4134 throw setError(VBOX_E_NOT_SUPPORTED,
4135 tr("Encryption is not supported because the extension pack '%s' is missing"),
4136 ORACLE_PUEL_EXTPACK_NAME);
4137
4138 PVDISK pDisk = NULL;
4139 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &pDisk);
4140 ComAssertRCThrow(vrc, E_FAIL);
4141
4142 MediumCryptoFilterSettings CryptoSettings;
4143
4144 i_taskEncryptSettingsSetup(&CryptoSettings, NULL, it->second.c_str(), aPassword.c_str(),
4145 false /* fCreateKeyStore */);
4146 vrc = VDFilterAdd(pDisk, "CRYPT", VD_FILTER_FLAGS_READ, CryptoSettings.vdFilterIfaces);
4147 if (vrc == VERR_VD_PASSWORD_INCORRECT)
4148 throw setError(VBOX_E_PASSWORD_INCORRECT,
4149 tr("The given password is incorrect"));
4150 else if (RT_FAILURE(vrc))
4151 throw setErrorBoth(VBOX_E_INVALID_OBJECT_STATE, vrc,
4152 tr("Failed to load the encryption filter: %s"),
4153 i_vdError(vrc).c_str());
4154
4155 VDDestroy(pDisk);
4156# else
4157 throw setError(VBOX_E_NOT_SUPPORTED,
4158 tr("Encryption is not supported because extension pack support is not built in"));
4159# endif
4160 }
4161 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
4162
4163 return hrc;
4164}
4165
4166HRESULT Medium::openForIO(BOOL aWritable, com::Utf8Str const &aPassword, ComPtr<IMediumIO> &aMediumIO)
4167{
4168 /*
4169 * Input validation.
4170 */
4171 if (aWritable && i_isReadOnly())
4172 return setError(E_ACCESSDENIED, tr("Write access denied: read-only"));
4173
4174 com::Utf8Str const strKeyId = i_getKeyId();
4175 if (strKeyId.isEmpty() && aPassword.isNotEmpty())
4176 return setError(E_INVALIDARG, tr("Password given for unencrypted medium"));
4177 if (strKeyId.isNotEmpty() && aPassword.isEmpty())
4178 return setError(E_INVALIDARG, tr("Password needed for encrypted medium"));
4179
4180 /*
4181 * Create IO object and return it.
4182 */
4183 ComObjPtr<MediumIO> ptrIO;
4184 HRESULT hrc = ptrIO.createObject();
4185 if (SUCCEEDED(hrc))
4186 {
4187 hrc = ptrIO->initForMedium(this, m->pVirtualBox, aWritable != FALSE, strKeyId, aPassword);
4188 if (SUCCEEDED(hrc))
4189 ptrIO.queryInterfaceTo(aMediumIO.asOutParam());
4190 }
4191 return hrc;
4192}
4193
4194
4195////////////////////////////////////////////////////////////////////////////////
4196//
4197// Medium public internal methods
4198//
4199////////////////////////////////////////////////////////////////////////////////
4200
4201/**
4202 * Internal method to return the medium's parent medium. Must have caller + locking!
4203 * @return
4204 */
4205const ComObjPtr<Medium>& Medium::i_getParent() const
4206{
4207 return m->pParent;
4208}
4209
4210/**
4211 * Internal method to return the medium's list of child media. Must have caller + locking!
4212 * @return
4213 */
4214const MediaList& Medium::i_getChildren() const
4215{
4216 return m->llChildren;
4217}
4218
4219/**
4220 * Internal method to return the medium's GUID. Must have caller + locking!
4221 * @return
4222 */
4223const Guid& Medium::i_getId() const
4224{
4225 return m->id;
4226}
4227
4228/**
4229 * Internal method to return the medium's state. Must have caller + locking!
4230 * @return
4231 */
4232MediumState_T Medium::i_getState() const
4233{
4234 return m->state;
4235}
4236
4237/**
4238 * Internal method to return the medium's variant. Must have caller + locking!
4239 * @return
4240 */
4241MediumVariant_T Medium::i_getVariant() const
4242{
4243 return m->variant;
4244}
4245
4246/**
4247 * Internal method which returns true if this medium represents a host drive.
4248 * @return
4249 */
4250bool Medium::i_isHostDrive() const
4251{
4252 return m->hostDrive;
4253}
4254
4255/**
4256 * Internal method which returns true if this medium is in the process of being closed.
4257 * @return
4258 */
4259bool Medium::i_isClosing() const
4260{
4261 return m->fClosing;
4262}
4263
4264/**
4265 * Internal method to return the medium's full location. Must have caller + locking!
4266 * @return
4267 */
4268const Utf8Str& Medium::i_getLocationFull() const
4269{
4270 return m->strLocationFull;
4271}
4272
4273/**
4274 * Internal method to return the medium's format string. Must have caller + locking!
4275 * @return
4276 */
4277const Utf8Str& Medium::i_getFormat() const
4278{
4279 return m->strFormat;
4280}
4281
4282/**
4283 * Internal method to return the medium's format object. Must have caller + locking!
4284 * @return
4285 */
4286const ComObjPtr<MediumFormat>& Medium::i_getMediumFormat() const
4287{
4288 return m->formatObj;
4289}
4290
4291/**
4292 * Internal method that returns true if the medium is represented by a file on the host disk
4293 * (and not iSCSI or something).
4294 * @return
4295 */
4296bool Medium::i_isMediumFormatFile() const
4297{
4298 if ( m->formatObj
4299 && (m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File)
4300 )
4301 return true;
4302 return false;
4303}
4304
4305/**
4306 * Internal method to return the medium's size. Must have caller + locking!
4307 * @return
4308 */
4309uint64_t Medium::i_getSize() const
4310{
4311 return m->size;
4312}
4313
4314/**
4315 * Internal method to return the medium's size. Must have caller + locking!
4316 * @return
4317 */
4318uint64_t Medium::i_getLogicalSize() const
4319{
4320 return m->logicalSize;
4321}
4322
4323/**
4324 * Returns the medium device type. Must have caller + locking!
4325 * @return
4326 */
4327DeviceType_T Medium::i_getDeviceType() const
4328{
4329 return m->devType;
4330}
4331
4332/**
4333 * Returns the medium type. Must have caller + locking!
4334 * @return
4335 */
4336MediumType_T Medium::i_getType() const
4337{
4338 return m->type;
4339}
4340
4341/**
4342 * Returns a short version of the location attribute.
4343 *
4344 * @note Must be called from under this object's read or write lock.
4345 */
4346Utf8Str Medium::i_getName()
4347{
4348 Utf8Str name = RTPathFilename(m->strLocationFull.c_str());
4349 return name;
4350}
4351
4352/**
4353 * Same as i_addRegistry() except that we don't check the object state, making
4354 * it safe to call with initFromSettings() on the call stack.
4355 */
4356bool Medium::i_addRegistryNoCallerCheck(const Guid &id)
4357{
4358 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4359
4360 bool fAdd = true;
4361
4362 // hard disks cannot be in more than one registry
4363 if ( m->devType == DeviceType_HardDisk
4364 && m->llRegistryIDs.size() > 0)
4365 fAdd = false;
4366
4367 // no need to add the UUID twice
4368 if (fAdd)
4369 {
4370 for (GuidList::const_iterator it = m->llRegistryIDs.begin();
4371 it != m->llRegistryIDs.end();
4372 ++it)
4373 {
4374 if ((*it) == id)
4375 {
4376 fAdd = false;
4377 break;
4378 }
4379 }
4380 }
4381
4382 if (fAdd)
4383 m->llRegistryIDs.push_back(id);
4384
4385 return fAdd;
4386}
4387
4388/**
4389 * This adds the given UUID to the list of media registries in which this
4390 * medium should be registered. The UUID can either be a machine UUID,
4391 * to add a machine registry, or the global registry UUID as returned by
4392 * VirtualBox::getGlobalRegistryId().
4393 *
4394 * Note that for hard disks, this method does nothing if the medium is
4395 * already in another registry to avoid having hard disks in more than
4396 * one registry, which causes trouble with keeping diff images in sync.
4397 * See getFirstRegistryMachineId() for details.
4398 *
4399 * @param id
4400 * @return true if the registry was added; false if the given id was already on the list.
4401 */
4402bool Medium::i_addRegistry(const Guid &id)
4403{
4404 AutoCaller autoCaller(this);
4405 if (FAILED(autoCaller.hrc()))
4406 return false;
4407 return i_addRegistryNoCallerCheck(id);
4408}
4409
4410/**
4411 * This adds the given UUID to the list of media registries in which this
4412 * medium should be registered. The UUID can either be a machine UUID,
4413 * to add a machine registry, or the global registry UUID as returned by
4414 * VirtualBox::getGlobalRegistryId(). Thisis applied to all children.
4415 *
4416 * Note that for hard disks, this method does nothing if the medium is
4417 * already in another registry to avoid having hard disks in more than
4418 * one registry, which causes trouble with keeping diff images in sync.
4419 * See getFirstRegistryMachineId() for details.
4420 *
4421 * @note the caller must hold the media tree lock for reading.
4422 *
4423 * @param id
4424 * @return true if the registry was added; false if the given id was already on the list.
4425 */
4426bool Medium::i_addRegistryAll(const Guid &id)
4427{
4428 MediaList llMediaTodo;
4429 llMediaTodo.push_back(this);
4430
4431 bool fAdd = false;
4432
4433 while (!llMediaTodo.empty())
4434 {
4435 ComObjPtr<Medium> pMedium = llMediaTodo.front();
4436 llMediaTodo.pop_front();
4437
4438 AutoCaller mediumCaller(pMedium);
4439 if (FAILED(mediumCaller.hrc())) continue;
4440
4441 fAdd |= pMedium->i_addRegistryNoCallerCheck(id);
4442
4443 // protected by the medium tree lock held by our original caller
4444 MediaList::const_iterator itBegin = pMedium->i_getChildren().begin();
4445 MediaList::const_iterator itEnd = pMedium->i_getChildren().end();
4446 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
4447 llMediaTodo.push_back(*it);
4448 }
4449
4450 return fAdd;
4451}
4452
4453/**
4454 * Removes the given UUID from the list of media registry UUIDs of this medium.
4455 *
4456 * @param id
4457 * @return true if the UUID was found or false if not.
4458 */
4459bool Medium::i_removeRegistry(const Guid &id)
4460{
4461 AutoCaller autoCaller(this);
4462 if (FAILED(autoCaller.hrc()))
4463 return false;
4464 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4465
4466 bool fRemove = false;
4467
4468 /// @todo r=klaus eliminate this code, replace it by using find.
4469 for (GuidList::iterator it = m->llRegistryIDs.begin();
4470 it != m->llRegistryIDs.end();
4471 ++it)
4472 {
4473 if ((*it) == id)
4474 {
4475 // getting away with this as the iterator isn't used after
4476 m->llRegistryIDs.erase(it);
4477 fRemove = true;
4478 break;
4479 }
4480 }
4481
4482 return fRemove;
4483}
4484
4485/**
4486 * Removes the given UUID from the list of media registry UUIDs, for this
4487 * medium and all its children.
4488 *
4489 * @note the caller must hold the media tree lock for reading.
4490 *
4491 * @param id
4492 * @return true if the UUID was found or false if not.
4493 */
4494bool Medium::i_removeRegistryAll(const Guid &id)
4495{
4496 MediaList llMediaTodo;
4497 llMediaTodo.push_back(this);
4498
4499 bool fRemove = false;
4500
4501 while (!llMediaTodo.empty())
4502 {
4503 ComObjPtr<Medium> pMedium = llMediaTodo.front();
4504 llMediaTodo.pop_front();
4505
4506 AutoCaller mediumCaller(pMedium);
4507 if (FAILED(mediumCaller.hrc())) continue;
4508
4509 fRemove |= pMedium->i_removeRegistry(id);
4510
4511 // protected by the medium tree lock held by our original caller
4512 MediaList::const_iterator itBegin = pMedium->i_getChildren().begin();
4513 MediaList::const_iterator itEnd = pMedium->i_getChildren().end();
4514 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
4515 llMediaTodo.push_back(*it);
4516 }
4517
4518 return fRemove;
4519}
4520
4521/**
4522 * Returns true if id is in the list of media registries for this medium.
4523 *
4524 * Must have caller + read locking!
4525 *
4526 * @param id
4527 * @return
4528 */
4529bool Medium::i_isInRegistry(const Guid &id)
4530{
4531 /// @todo r=klaus eliminate this code, replace it by using find.
4532 for (GuidList::const_iterator it = m->llRegistryIDs.begin();
4533 it != m->llRegistryIDs.end();
4534 ++it)
4535 {
4536 if (*it == id)
4537 return true;
4538 }
4539
4540 return false;
4541}
4542
4543/**
4544 * Internal method to return the medium's first registry machine (i.e. the machine in whose
4545 * machine XML this medium is listed).
4546 *
4547 * Every attached medium must now (4.0) reside in at least one media registry, which is identified
4548 * by a UUID. This is either a machine UUID if the machine is from 4.0 or newer, in which case
4549 * machines have their own media registries, or it is the pseudo-UUID of the VirtualBox
4550 * object if the machine is old and still needs the global registry in VirtualBox.xml.
4551 *
4552 * By definition, hard disks may only be in one media registry, in which all its children
4553 * will be stored as well. Otherwise we run into problems with having keep multiple registries
4554 * in sync. (This is the "cloned VM" case in which VM1 may link to the disks of VM2; in this
4555 * case, only VM2's registry is used for the disk in question.)
4556 *
4557 * If there is no medium registry, particularly if the medium has not been attached yet, this
4558 * does not modify uuid and returns false.
4559 *
4560 * ISOs and RAWs, by contrast, can be in more than one repository to make things easier for
4561 * the user.
4562 *
4563 * Must have caller + locking!
4564 *
4565 * @param uuid Receives first registry machine UUID, if available.
4566 * @return true if uuid was set.
4567 */
4568bool Medium::i_getFirstRegistryMachineId(Guid &uuid) const
4569{
4570 if (m->llRegistryIDs.size())
4571 {
4572 uuid = m->llRegistryIDs.front();
4573 return true;
4574 }
4575 return false;
4576}
4577
4578/**
4579 * Marks all the registries in which this medium is registered as modified.
4580 */
4581void Medium::i_markRegistriesModified()
4582{
4583 AutoCaller autoCaller(this);
4584 if (FAILED(autoCaller.hrc())) return;
4585
4586 // Get local copy, as keeping the lock over VirtualBox::markRegistryModified
4587 // causes trouble with the lock order
4588 GuidList llRegistryIDs;
4589 {
4590 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4591 llRegistryIDs = m->llRegistryIDs;
4592 }
4593
4594 autoCaller.release();
4595
4596 /* Save the error information now, the implicit restore when this goes
4597 * out of scope will throw away spurious additional errors created below. */
4598 ErrorInfoKeeper eik;
4599 for (GuidList::const_iterator it = llRegistryIDs.begin();
4600 it != llRegistryIDs.end();
4601 ++it)
4602 {
4603 m->pVirtualBox->i_markRegistryModified(*it);
4604 }
4605}
4606
4607/**
4608 * Adds the given machine and optionally the snapshot to the list of the objects
4609 * this medium is attached to.
4610 *
4611 * @param aMachineId Machine ID.
4612 * @param aSnapshotId Snapshot ID; when non-empty, adds a snapshot attachment.
4613 */
4614HRESULT Medium::i_addBackReference(const Guid &aMachineId,
4615 const Guid &aSnapshotId /*= Guid::Empty*/)
4616{
4617 AssertReturn(aMachineId.isValid(), E_FAIL);
4618
4619 LogFlowThisFunc(("ENTER, aMachineId: {%RTuuid}, aSnapshotId: {%RTuuid}\n", aMachineId.raw(), aSnapshotId.raw()));
4620
4621 AutoCaller autoCaller(this);
4622 AssertComRCReturnRC(autoCaller.hrc());
4623
4624 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4625
4626 switch (m->state)
4627 {
4628 case MediumState_Created:
4629 case MediumState_Inaccessible:
4630 case MediumState_LockedRead:
4631 case MediumState_LockedWrite:
4632 break;
4633
4634 default:
4635 return i_setStateError();
4636 }
4637
4638 if (m->numCreateDiffTasks > 0)
4639 return setError(VBOX_E_OBJECT_IN_USE,
4640 tr("Cannot attach medium '%s' {%RTuuid}: %u differencing child media are being created", "",
4641 m->numCreateDiffTasks),
4642 m->strLocationFull.c_str(),
4643 m->id.raw(),
4644 m->numCreateDiffTasks);
4645
4646 BackRefList::iterator it = std::find_if(m->backRefs.begin(),
4647 m->backRefs.end(),
4648 BackRef::EqualsTo(aMachineId));
4649 if (it == m->backRefs.end())
4650 {
4651 BackRef ref(aMachineId, aSnapshotId);
4652 m->backRefs.push_back(ref);
4653
4654 return S_OK;
4655 }
4656 bool fDvd = false;
4657 {
4658 AutoReadLock arlock(this COMMA_LOCKVAL_SRC_POS);
4659 /*
4660 * Check the medium is DVD and readonly. It's for the case if DVD
4661 * will be able to be writable sometime in the future.
4662 */
4663 fDvd = m->type == MediumType_Readonly && m->devType == DeviceType_DVD;
4664 }
4665
4666 // if the caller has not supplied a snapshot ID, then we're attaching
4667 // to a machine a medium which represents the machine's current state,
4668 // so set the flag
4669
4670 if (aSnapshotId.isZero())
4671 {
4672 // Allow DVD having MediumType_Readonly to be attached twice.
4673 // (the medium already had been added to back reference)
4674 if (fDvd)
4675 {
4676 it->iRefCnt++;
4677 return S_OK;
4678 }
4679
4680 /* sanity: no duplicate attachments */
4681 if (it->fInCurState)
4682 return setError(VBOX_E_OBJECT_IN_USE,
4683 tr("Cannot attach medium '%s' {%RTuuid}: medium is already associated with the current state of machine uuid {%RTuuid}!"),
4684 m->strLocationFull.c_str(),
4685 m->id.raw(),
4686 aMachineId.raw());
4687 it->fInCurState = true;
4688
4689 return S_OK;
4690 }
4691
4692 // otherwise: a snapshot medium is being attached
4693
4694 /* sanity: no duplicate attachments */
4695 for (std::list<SnapshotRef>::iterator jt = it->llSnapshotIds.begin();
4696 jt != it->llSnapshotIds.end();
4697 ++jt)
4698 {
4699 const Guid &idOldSnapshot = jt->snapshotId;
4700
4701 if (idOldSnapshot == aSnapshotId)
4702 {
4703 if (fDvd)
4704 {
4705 jt->iRefCnt++;
4706 return S_OK;
4707 }
4708#ifdef DEBUG
4709 i_dumpBackRefs();
4710#endif
4711 return setError(VBOX_E_OBJECT_IN_USE,
4712 tr("Cannot attach medium '%s' {%RTuuid} from snapshot '%RTuuid': medium is already in use by this snapshot!"),
4713 m->strLocationFull.c_str(),
4714 m->id.raw(),
4715 aSnapshotId.raw());
4716 }
4717 }
4718
4719 it->llSnapshotIds.push_back(SnapshotRef(aSnapshotId));
4720 // Do not touch fInCurState, as the image may be attached to the current
4721 // state *and* a snapshot, otherwise we lose the current state association!
4722
4723 LogFlowThisFuncLeave();
4724
4725 return S_OK;
4726}
4727
4728/**
4729 * Removes the given machine and optionally the snapshot from the list of the
4730 * objects this medium is attached to.
4731 *
4732 * @param aMachineId Machine ID.
4733 * @param aSnapshotId Snapshot ID; when non-empty, removes the snapshot
4734 * attachment.
4735 */
4736HRESULT Medium::i_removeBackReference(const Guid &aMachineId,
4737 const Guid &aSnapshotId /*= Guid::Empty*/)
4738{
4739 AssertReturn(aMachineId.isValid(), E_FAIL);
4740
4741 AutoCaller autoCaller(this);
4742 AssertComRCReturnRC(autoCaller.hrc());
4743
4744 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4745
4746 BackRefList::iterator it =
4747 std::find_if(m->backRefs.begin(), m->backRefs.end(),
4748 BackRef::EqualsTo(aMachineId));
4749 AssertReturn(it != m->backRefs.end(), E_FAIL);
4750
4751 if (aSnapshotId.isZero())
4752 {
4753 it->iRefCnt--;
4754 if (it->iRefCnt > 0)
4755 return S_OK;
4756
4757 /* remove the current state attachment */
4758 it->fInCurState = false;
4759 }
4760 else
4761 {
4762 /* remove the snapshot attachment */
4763 std::list<SnapshotRef>::iterator jt =
4764 std::find_if(it->llSnapshotIds.begin(),
4765 it->llSnapshotIds.end(),
4766 SnapshotRef::EqualsTo(aSnapshotId));
4767
4768 AssertReturn(jt != it->llSnapshotIds.end(), E_FAIL);
4769
4770 jt->iRefCnt--;
4771 if (jt->iRefCnt > 0)
4772 return S_OK;
4773
4774 it->llSnapshotIds.erase(jt);
4775 }
4776
4777 /* if the backref becomes empty, remove it */
4778 if (it->fInCurState == false && it->llSnapshotIds.size() == 0)
4779 m->backRefs.erase(it);
4780
4781 return S_OK;
4782}
4783
4784/**
4785 * Internal method to return the medium's list of backrefs. Must have caller + locking!
4786 * @return
4787 */
4788const Guid* Medium::i_getFirstMachineBackrefId() const
4789{
4790 if (!m->backRefs.size())
4791 return NULL;
4792
4793 return &m->backRefs.front().machineId;
4794}
4795
4796/**
4797 * Internal method which returns a machine that either this medium or one of its children
4798 * is attached to. This is used for finding a replacement media registry when an existing
4799 * media registry is about to be deleted in VirtualBox::unregisterMachine().
4800 *
4801 * Must have caller + locking, *and* caller must hold the media tree lock!
4802 * @param aId Id to ignore when looking for backrefs.
4803 * @return
4804 */
4805const Guid* Medium::i_getAnyMachineBackref(const Guid &aId) const
4806{
4807 std::list<const Medium *> llMediaTodo;
4808 llMediaTodo.push_back(this);
4809
4810 while (!llMediaTodo.empty())
4811 {
4812 const Medium *pMedium = llMediaTodo.front();
4813 llMediaTodo.pop_front();
4814
4815 if (pMedium->m->backRefs.size())
4816 {
4817 if (pMedium->m->backRefs.front().machineId != aId)
4818 return &pMedium->m->backRefs.front().machineId;
4819 if (pMedium->m->backRefs.size() > 1)
4820 {
4821 BackRefList::const_iterator it = pMedium->m->backRefs.begin();
4822 ++it;
4823 return &it->machineId;
4824 }
4825 }
4826
4827 MediaList::const_iterator itBegin = pMedium->i_getChildren().begin();
4828 MediaList::const_iterator itEnd = pMedium->i_getChildren().end();
4829 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
4830 llMediaTodo.push_back(*it);
4831 }
4832
4833 return NULL;
4834}
4835
4836const Guid* Medium::i_getFirstMachineBackrefSnapshotId() const
4837{
4838 if (!m->backRefs.size())
4839 return NULL;
4840
4841 const BackRef &ref = m->backRefs.front();
4842 if (ref.llSnapshotIds.empty())
4843 return NULL;
4844
4845 return &ref.llSnapshotIds.front().snapshotId;
4846}
4847
4848size_t Medium::i_getMachineBackRefCount() const
4849{
4850 return m->backRefs.size();
4851}
4852
4853#ifdef DEBUG
4854/**
4855 * Debugging helper that gets called after VirtualBox initialization that writes all
4856 * machine backreferences to the debug log.
4857 */
4858void Medium::i_dumpBackRefs()
4859{
4860 AutoCaller autoCaller(this);
4861 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4862
4863 LogFlowThisFunc(("Dumping backrefs for medium '%s':\n", m->strLocationFull.c_str()));
4864
4865 for (BackRefList::iterator it2 = m->backRefs.begin();
4866 it2 != m->backRefs.end();
4867 ++it2)
4868 {
4869 const BackRef &ref = *it2;
4870 LogFlowThisFunc((" Backref from machine {%RTuuid} (fInCurState: %d, iRefCnt: %d)\n", ref.machineId.raw(), ref.fInCurState, ref.iRefCnt));
4871
4872 for (std::list<SnapshotRef>::const_iterator jt2 = it2->llSnapshotIds.begin();
4873 jt2 != it2->llSnapshotIds.end();
4874 ++jt2)
4875 {
4876 const Guid &id = jt2->snapshotId;
4877 LogFlowThisFunc((" Backref from snapshot {%RTuuid} (iRefCnt = %d)\n", id.raw(), jt2->iRefCnt));
4878 }
4879 }
4880}
4881#endif
4882
4883/**
4884 * Checks if the given change of \a aOldPath to \a aNewPath affects the location
4885 * of this media and updates it if necessary to reflect the new location.
4886 *
4887 * @param strOldPath Old path (full).
4888 * @param strNewPath New path (full).
4889 *
4890 * @note Locks this object for writing.
4891 */
4892HRESULT Medium::i_updatePath(const Utf8Str &strOldPath, const Utf8Str &strNewPath)
4893{
4894 AssertReturn(!strOldPath.isEmpty(), E_FAIL);
4895 AssertReturn(!strNewPath.isEmpty(), E_FAIL);
4896
4897 AutoCaller autoCaller(this);
4898 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
4899
4900 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4901
4902 LogFlowThisFunc(("locationFull.before='%s'\n", m->strLocationFull.c_str()));
4903
4904 const char *pcszMediumPath = m->strLocationFull.c_str();
4905
4906 if (RTPathStartsWith(pcszMediumPath, strOldPath.c_str()))
4907 {
4908 Utf8Str newPath(strNewPath);
4909 newPath.append(pcszMediumPath + strOldPath.length());
4910 unconst(m->strLocationFull) = newPath;
4911
4912 m->pVirtualBox->i_onMediumConfigChanged(this);
4913
4914 LogFlowThisFunc(("locationFull.after='%s'\n", m->strLocationFull.c_str()));
4915 // we changed something
4916 return S_OK;
4917 }
4918
4919 // no change was necessary, signal error which the caller needs to interpret
4920 return VBOX_E_FILE_ERROR;
4921}
4922
4923/**
4924 * Returns the base medium of the media chain this medium is part of.
4925 *
4926 * The base medium is found by walking up the parent-child relationship axis.
4927 * If the medium doesn't have a parent (i.e. it's a base medium), it
4928 * returns itself in response to this method.
4929 *
4930 * @param aLevel Where to store the number of ancestors of this medium
4931 * (zero for the base), may be @c NULL.
4932 *
4933 * @note Locks medium tree for reading.
4934 */
4935ComObjPtr<Medium> Medium::i_getBase(uint32_t *aLevel /*= NULL*/)
4936{
4937 ComObjPtr<Medium> pBase;
4938
4939 /* it is possible that some previous/concurrent uninit has already cleared
4940 * the pVirtualBox reference, and in this case we don't need to continue */
4941 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
4942 if (!pVirtualBox)
4943 return pBase;
4944
4945 /* we access m->pParent */
4946 AutoReadLock treeLock(pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4947
4948 AutoCaller autoCaller(this);
4949 AssertReturn(autoCaller.isOk(), pBase);
4950
4951 pBase = this;
4952 uint32_t level = 0;
4953
4954 if (m->pParent)
4955 {
4956 for (;;)
4957 {
4958 AutoCaller baseCaller(pBase);
4959 AssertReturn(baseCaller.isOk(), pBase);
4960
4961 if (pBase->m->pParent.isNull())
4962 break;
4963
4964 pBase = pBase->m->pParent;
4965 ++level;
4966 }
4967 }
4968
4969 if (aLevel != NULL)
4970 *aLevel = level;
4971
4972 return pBase;
4973}
4974
4975/**
4976 * Returns the depth of this medium in the media chain.
4977 *
4978 * @note Locks medium tree for reading.
4979 */
4980uint32_t Medium::i_getDepth()
4981{
4982 /* it is possible that some previous/concurrent uninit has already cleared
4983 * the pVirtualBox reference, and in this case we don't need to continue */
4984 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
4985 if (!pVirtualBox)
4986 return 1;
4987
4988 /* we access m->pParent */
4989 AutoReadLock treeLock(pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4990
4991 uint32_t cDepth = 0;
4992 ComObjPtr<Medium> pMedium(this);
4993 while (!pMedium.isNull())
4994 {
4995 AutoCaller autoCaller(this);
4996 AssertReturn(autoCaller.isOk(), cDepth + 1);
4997
4998 pMedium = pMedium->m->pParent;
4999 cDepth++;
5000 }
5001
5002 return cDepth;
5003}
5004
5005/**
5006 * Returns @c true if this medium cannot be modified because it has
5007 * dependents (children) or is part of the snapshot. Related to the medium
5008 * type and posterity, not to the current media state.
5009 *
5010 * @note Locks this object and medium tree for reading.
5011 */
5012bool Medium::i_isReadOnly()
5013{
5014 /* it is possible that some previous/concurrent uninit has already cleared
5015 * the pVirtualBox reference, and in this case we don't need to continue */
5016 ComObjPtr<VirtualBox> pVirtualBox(m->pVirtualBox);
5017 if (!pVirtualBox)
5018 return false;
5019
5020 /* we access children */
5021 AutoReadLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5022
5023 AutoCaller autoCaller(this);
5024 AssertComRCReturn(autoCaller.hrc(), false);
5025
5026 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5027
5028 switch (m->type)
5029 {
5030 case MediumType_Normal:
5031 {
5032 if (i_getChildren().size() != 0)
5033 return true;
5034
5035 for (BackRefList::const_iterator it = m->backRefs.begin();
5036 it != m->backRefs.end(); ++it)
5037 if (it->llSnapshotIds.size() != 0)
5038 return true;
5039
5040 if (m->variant & MediumVariant_VmdkStreamOptimized)
5041 return true;
5042
5043 return false;
5044 }
5045 case MediumType_Immutable:
5046 case MediumType_MultiAttach:
5047 return true;
5048 case MediumType_Writethrough:
5049 case MediumType_Shareable:
5050 case MediumType_Readonly: /* explicit readonly media has no diffs */
5051 return false;
5052 default:
5053 break;
5054 }
5055
5056 AssertFailedReturn(false);
5057}
5058
5059/**
5060 * Internal method to update the medium's id. Must have caller + locking!
5061 */
5062void Medium::i_updateId(const Guid &id)
5063{
5064 unconst(m->id) = id;
5065}
5066
5067/**
5068 * Saves the settings of one medium.
5069 *
5070 * @note Caller MUST take care of the medium tree lock and caller.
5071 *
5072 * @param data Settings struct to be updated.
5073 * @param strHardDiskFolder Folder for which paths should be relative.
5074 */
5075void Medium::i_saveSettingsOne(settings::Medium &data, const Utf8Str &strHardDiskFolder)
5076{
5077 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5078
5079 data.uuid = m->id;
5080
5081 // make path relative if needed
5082 if ( !strHardDiskFolder.isEmpty()
5083 && RTPathStartsWith(m->strLocationFull.c_str(), strHardDiskFolder.c_str())
5084 )
5085 data.strLocation = m->strLocationFull.substr(strHardDiskFolder.length() + 1);
5086 else
5087 data.strLocation = m->strLocationFull;
5088 data.strFormat = m->strFormat;
5089
5090 /* optional, only for diffs, default is false */
5091 if (m->pParent)
5092 data.fAutoReset = m->autoReset;
5093 else
5094 data.fAutoReset = false;
5095
5096 /* optional */
5097 data.strDescription = m->strDescription;
5098
5099 /* optional properties */
5100 data.properties.clear();
5101
5102 /* handle iSCSI initiator secrets transparently */
5103 bool fHaveInitiatorSecretEncrypted = false;
5104 Utf8Str strCiphertext;
5105 settings::StringsMap::const_iterator itPln = m->mapProperties.find("InitiatorSecret");
5106 if ( itPln != m->mapProperties.end()
5107 && !itPln->second.isEmpty())
5108 {
5109 /* Encrypt the plain secret. If that does not work (i.e. no or wrong settings key
5110 * specified), just use the encrypted secret (if there is any). */
5111 int vrc = m->pVirtualBox->i_encryptSetting(itPln->second, &strCiphertext);
5112 if (RT_SUCCESS(vrc))
5113 fHaveInitiatorSecretEncrypted = true;
5114 }
5115 for (settings::StringsMap::const_iterator it = m->mapProperties.begin();
5116 it != m->mapProperties.end();
5117 ++it)
5118 {
5119 /* only save properties that have non-default values */
5120 if (!it->second.isEmpty())
5121 {
5122 const Utf8Str &name = it->first;
5123 const Utf8Str &value = it->second;
5124 bool fCreateOnly = false;
5125 for (MediumFormat::PropertyArray::const_iterator itf = m->formatObj->i_getProperties().begin();
5126 itf != m->formatObj->i_getProperties().end();
5127 ++itf)
5128 {
5129 if ( itf->strName.equals(name)
5130 && (itf->flags & VD_CFGKEY_CREATEONLY))
5131 {
5132 fCreateOnly = true;
5133 break;
5134 }
5135 }
5136 if (!fCreateOnly)
5137 /* do NOT store the plain InitiatorSecret */
5138 if ( !fHaveInitiatorSecretEncrypted
5139 || !name.equals("InitiatorSecret"))
5140 data.properties[name] = value;
5141 }
5142 }
5143 if (fHaveInitiatorSecretEncrypted)
5144 data.properties["InitiatorSecretEncrypted"] = strCiphertext;
5145
5146 /* only for base media */
5147 if (m->pParent.isNull())
5148 data.hdType = m->type;
5149}
5150
5151/**
5152 * Saves medium data by putting it into the provided data structure.
5153 * The settings of all children is saved, too.
5154 *
5155 * @param data Settings struct to be updated.
5156 * @param strHardDiskFolder Folder for which paths should be relative.
5157 *
5158 * @note Locks this object, medium tree and children for reading.
5159 */
5160HRESULT Medium::i_saveSettings(settings::Medium &data,
5161 const Utf8Str &strHardDiskFolder)
5162{
5163 /* we access m->pParent */
5164 AutoReadLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5165
5166 AutoCaller autoCaller(this);
5167 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
5168
5169 MediaList llMediaTodo;
5170 llMediaTodo.push_back(this);
5171 std::list<settings::Medium *> llSettingsTodo;
5172 llSettingsTodo.push_back(&data);
5173
5174 while (!llMediaTodo.empty())
5175 {
5176 ComObjPtr<Medium> pMedium = llMediaTodo.front();
5177 llMediaTodo.pop_front();
5178 settings::Medium *current = llSettingsTodo.front();
5179 llSettingsTodo.pop_front();
5180
5181 AutoCaller mediumCaller(pMedium);
5182 if (FAILED(mediumCaller.hrc())) return mediumCaller.hrc();
5183
5184 pMedium->i_saveSettingsOne(*current, strHardDiskFolder);
5185
5186 /* save all children */
5187 MediaList::const_iterator itBegin = pMedium->i_getChildren().begin();
5188 MediaList::const_iterator itEnd = pMedium->i_getChildren().end();
5189 for (MediaList::const_iterator it = itBegin; it != itEnd; ++it)
5190 {
5191 llMediaTodo.push_back(*it);
5192 current->llChildren.push_back(settings::Medium::Empty);
5193 llSettingsTodo.push_back(&current->llChildren.back());
5194 }
5195 }
5196
5197 return S_OK;
5198}
5199
5200/**
5201 * Constructs a medium lock list for this medium. The lock is not taken.
5202 *
5203 * @note Caller MUST NOT hold the media tree or medium lock.
5204 *
5205 * @param fFailIfInaccessible If true, this fails with an error if a medium is inaccessible. If false,
5206 * inaccessible media are silently skipped and not locked (i.e. their state remains "Inaccessible");
5207 * this is necessary for a VM's removable media VM startup for which we do not want to fail.
5208 * @param pToLockWrite If not NULL, associate a write lock with this medium object.
5209 * @param fMediumLockWriteAll Whether to associate a write lock to all other media too.
5210 * @param pToBeParent Medium which will become the parent of this medium.
5211 * @param mediumLockList Where to store the resulting list.
5212 */
5213HRESULT Medium::i_createMediumLockList(bool fFailIfInaccessible,
5214 Medium *pToLockWrite,
5215 bool fMediumLockWriteAll,
5216 Medium *pToBeParent,
5217 MediumLockList &mediumLockList)
5218{
5219 /** @todo r=klaus this needs to be reworked, as the code below uses
5220 * i_getParent without holding the tree lock, and changing this is
5221 * a significant amount of effort. */
5222 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5223 Assert(!isWriteLockOnCurrentThread());
5224
5225 AutoCaller autoCaller(this);
5226 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
5227
5228 HRESULT hrc = S_OK;
5229
5230 /* paranoid sanity checking if the medium has a to-be parent medium */
5231 if (pToBeParent)
5232 {
5233 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5234 ComAssertRet(i_getParent().isNull(), E_FAIL);
5235 ComAssertRet(i_getChildren().size() == 0, E_FAIL);
5236 }
5237
5238 ErrorInfoKeeper eik;
5239 MultiResult mrc(S_OK);
5240
5241 ComObjPtr<Medium> pMedium = this;
5242 while (!pMedium.isNull())
5243 {
5244 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
5245
5246 /* Accessibility check must be first, otherwise locking interferes
5247 * with getting the medium state. Lock lists are not created for
5248 * fun, and thus getting the medium status is no luxury. */
5249 MediumState_T mediumState = pMedium->i_getState();
5250 if (mediumState == MediumState_Inaccessible)
5251 {
5252 alock.release();
5253 hrc = pMedium->i_queryInfo(false /* fSetImageId */, false /* fSetParentId */, autoCaller);
5254 alock.acquire();
5255 if (FAILED(hrc)) return hrc;
5256
5257 mediumState = pMedium->i_getState();
5258 if (mediumState == MediumState_Inaccessible)
5259 {
5260 // ignore inaccessible ISO media and silently return S_OK,
5261 // otherwise VM startup (esp. restore) may fail without good reason
5262 if (!fFailIfInaccessible)
5263 return S_OK;
5264
5265 // otherwise report an error
5266 Bstr error;
5267 hrc = pMedium->COMGETTER(LastAccessError)(error.asOutParam());
5268 if (FAILED(hrc)) return hrc;
5269
5270 /* collect multiple errors */
5271 eik.restore();
5272 Assert(!error.isEmpty());
5273 mrc = setError(E_FAIL,
5274 "%ls",
5275 error.raw());
5276 // error message will be something like
5277 // "Could not open the medium ... VD: error VERR_FILE_NOT_FOUND opening image file ... (VERR_FILE_NOT_FOUND).
5278 eik.fetch();
5279 }
5280 }
5281
5282 if (pMedium == pToLockWrite)
5283 mediumLockList.Prepend(pMedium, true);
5284 else
5285 mediumLockList.Prepend(pMedium, fMediumLockWriteAll);
5286
5287 pMedium = pMedium->i_getParent();
5288 if (pMedium.isNull() && pToBeParent)
5289 {
5290 pMedium = pToBeParent;
5291 pToBeParent = NULL;
5292 }
5293 }
5294
5295 return mrc;
5296}
5297
5298/**
5299 * Creates a new differencing storage unit using the format of the given target
5300 * medium and the location. Note that @c aTarget must be NotCreated.
5301 *
5302 * The @a aMediumLockList parameter contains the associated medium lock list,
5303 * which must be in locked state. If @a aWait is @c true then the caller is
5304 * responsible for unlocking.
5305 *
5306 * If @a aProgress is not NULL but the object it points to is @c null then a
5307 * new progress object will be created and assigned to @a *aProgress on
5308 * success, otherwise the existing progress object is used. If @a aProgress is
5309 * NULL, then no progress object is created/used at all.
5310 *
5311 * When @a aWait is @c false, this method will create a thread to perform the
5312 * create operation asynchronously and will return immediately. Otherwise, it
5313 * will perform the operation on the calling thread and will not return to the
5314 * caller until the operation is completed. Note that @a aProgress cannot be
5315 * NULL when @a aWait is @c false (this method will assert in this case).
5316 *
5317 * @param aTarget Target medium.
5318 * @param aVariant Precise medium variant to create.
5319 * @param aMediumLockList List of media which should be locked.
5320 * @param aProgress Where to find/store a Progress object to track
5321 * operation completion.
5322 * @param aWait @c true if this method should block instead of
5323 * creating an asynchronous thread.
5324 * @param aNotify Notify about media for which metadata is changed
5325 * during execution of the function.
5326 *
5327 * @note Locks this object and @a aTarget for writing.
5328 */
5329HRESULT Medium::i_createDiffStorage(ComObjPtr<Medium> &aTarget,
5330 MediumVariant_T aVariant,
5331 MediumLockList *aMediumLockList,
5332 ComObjPtr<Progress> *aProgress,
5333 bool aWait,
5334 bool aNotify)
5335{
5336 AssertReturn(!aTarget.isNull(), E_FAIL);
5337 AssertReturn(aMediumLockList, E_FAIL);
5338 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
5339
5340 AutoCaller autoCaller(this);
5341 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
5342
5343 AutoCaller targetCaller(aTarget);
5344 if (FAILED(targetCaller.hrc())) return targetCaller.hrc();
5345
5346 HRESULT hrc = S_OK;
5347 ComObjPtr<Progress> pProgress;
5348 Medium::Task *pTask = NULL;
5349
5350 try
5351 {
5352 AutoMultiWriteLock2 alock(this, aTarget COMMA_LOCKVAL_SRC_POS);
5353
5354 ComAssertThrow( m->type != MediumType_Writethrough
5355 && m->type != MediumType_Shareable
5356 && m->type != MediumType_Readonly, E_FAIL);
5357 ComAssertThrow(m->state == MediumState_LockedRead, E_FAIL);
5358
5359 if (aTarget->m->state != MediumState_NotCreated)
5360 throw aTarget->i_setStateError();
5361
5362 /* Check that the medium is not attached to the current state of
5363 * any VM referring to it. */
5364 for (BackRefList::const_iterator it = m->backRefs.begin();
5365 it != m->backRefs.end();
5366 ++it)
5367 {
5368 if (it->fInCurState)
5369 {
5370 /* Note: when a VM snapshot is being taken, all normal media
5371 * attached to the VM in the current state will be, as an
5372 * exception, also associated with the snapshot which is about
5373 * to create (see SnapshotMachine::init()) before deassociating
5374 * them from the current state (which takes place only on
5375 * success in Machine::fixupHardDisks()), so that the size of
5376 * snapshotIds will be 1 in this case. The extra condition is
5377 * used to filter out this legal situation. */
5378 if (it->llSnapshotIds.size() == 0)
5379 throw setError(VBOX_E_INVALID_OBJECT_STATE,
5380 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"),
5381 m->strLocationFull.c_str(), it->machineId.raw());
5382
5383 Assert(it->llSnapshotIds.size() == 1);
5384 }
5385 }
5386
5387 if (aProgress != NULL)
5388 {
5389 /* use the existing progress object... */
5390 pProgress = *aProgress;
5391
5392 /* ...but create a new one if it is null */
5393 if (pProgress.isNull())
5394 {
5395 pProgress.createObject();
5396 hrc = pProgress->init(m->pVirtualBox,
5397 static_cast<IMedium*>(this),
5398 BstrFmt(tr("Creating differencing medium storage unit '%s'"),
5399 aTarget->m->strLocationFull.c_str()).raw(),
5400 TRUE /* aCancelable */);
5401 if (FAILED(hrc))
5402 throw hrc;
5403 }
5404 }
5405
5406 /* setup task object to carry out the operation sync/async */
5407 pTask = new Medium::CreateDiffTask(this, pProgress, aTarget, aVariant,
5408 aMediumLockList,
5409 aWait /* fKeepMediumLockList */,
5410 aNotify);
5411 hrc = pTask->hrc();
5412 AssertComRC(hrc);
5413 if (FAILED(hrc))
5414 throw hrc;
5415
5416 /* register a task (it will deregister itself when done) */
5417 ++m->numCreateDiffTasks;
5418 Assert(m->numCreateDiffTasks != 0); /* overflow? */
5419
5420 aTarget->m->state = MediumState_Creating;
5421 }
5422 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
5423
5424 if (SUCCEEDED(hrc))
5425 {
5426 if (aWait)
5427 {
5428 hrc = pTask->runNow();
5429 delete pTask;
5430 }
5431 else
5432 hrc = pTask->createThread();
5433 pTask = NULL;
5434 if (SUCCEEDED(hrc) && aProgress != NULL)
5435 *aProgress = pProgress;
5436 }
5437 else if (pTask != NULL)
5438 delete pTask;
5439
5440 return hrc;
5441}
5442
5443/**
5444 * Returns a preferred format for differencing media.
5445 */
5446Utf8Str Medium::i_getPreferredDiffFormat()
5447{
5448 AutoCaller autoCaller(this);
5449 AssertComRCReturn(autoCaller.hrc(), Utf8Str::Empty);
5450
5451 /* check that our own format supports diffs */
5452 if (!(m->formatObj->i_getCapabilities() & MediumFormatCapabilities_Differencing))
5453 {
5454 /* use the default format if not */
5455 Utf8Str tmp;
5456 m->pVirtualBox->i_getDefaultHardDiskFormat(tmp);
5457 return tmp;
5458 }
5459
5460 /* m->strFormat is const, no need to lock */
5461 return m->strFormat;
5462}
5463
5464/**
5465 * Returns a preferred variant for differencing media.
5466 */
5467MediumVariant_T Medium::i_getPreferredDiffVariant()
5468{
5469 AutoCaller autoCaller(this);
5470 AssertComRCReturn(autoCaller.hrc(), MediumVariant_Standard);
5471
5472 /* check that our own format supports diffs */
5473 if (!(m->formatObj->i_getCapabilities() & MediumFormatCapabilities_Differencing))
5474 return MediumVariant_Standard;
5475
5476 /* m->variant is const, no need to lock */
5477 ULONG mediumVariantFlags = (ULONG)m->variant;
5478 mediumVariantFlags &= ~(ULONG)(MediumVariant_Fixed | MediumVariant_VmdkStreamOptimized | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk);
5479 mediumVariantFlags |= MediumVariant_Diff;
5480 return (MediumVariant_T)mediumVariantFlags;
5481}
5482
5483/**
5484 * Implementation for the public Medium::Close() with the exception of calling
5485 * VirtualBox::saveRegistries(), in case someone wants to call this for several
5486 * media.
5487 *
5488 * After this returns with success, uninit() has been called on the medium, and
5489 * the object is no longer usable ("not ready" state).
5490 *
5491 * @param autoCaller AutoCaller instance which must have been created on the caller's
5492 * stack for this medium. This gets released hereupon
5493 * which the Medium instance gets uninitialized.
5494 * @return
5495 */
5496HRESULT Medium::i_close(AutoCaller &autoCaller)
5497{
5498 // must temporarily drop the caller, need the tree lock first
5499 autoCaller.release();
5500
5501 // we're accessing parent/child and backrefs, so lock the tree first, then ourselves
5502 AutoMultiWriteLock2 multilock(&m->pVirtualBox->i_getMediaTreeLockHandle(),
5503 this->lockHandle()
5504 COMMA_LOCKVAL_SRC_POS);
5505
5506 autoCaller.add();
5507 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
5508
5509 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
5510 while (m->queryInfoRunning)
5511 {
5512 autoCaller.release();
5513 multilock.release();
5514 /* Must not hold the media tree lock, as Medium::i_queryInfo needs
5515 * this lock and thus we would run into a deadlock here. */
5516 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5517 /* must not hold the object lock now */
5518 Assert(!isWriteLockOnCurrentThread());
5519 {
5520 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
5521 }
5522 multilock.acquire();
5523 autoCaller.add();
5524 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
5525 }
5526
5527 LogFlowFunc(("ENTER for %s\n", i_getLocationFull().c_str()));
5528
5529 bool wasCreated = true;
5530
5531 switch (m->state)
5532 {
5533 case MediumState_NotCreated:
5534 wasCreated = false;
5535 break;
5536 case MediumState_Created:
5537 case MediumState_Inaccessible:
5538 break;
5539 default:
5540 return i_setStateError();
5541 }
5542
5543 if (m->fClosing)
5544 return setError(VBOX_E_OBJECT_IN_USE,
5545 tr("Medium '%s' cannot be closed because it is already in the process of being closed", ""),
5546 m->strLocationFull.c_str());
5547
5548 if (m->backRefs.size() != 0)
5549 return setError(VBOX_E_OBJECT_IN_USE,
5550 tr("Medium '%s' cannot be closed because it is still attached to %d virtual machines", "",
5551 m->backRefs.size()),
5552 m->strLocationFull.c_str(), m->backRefs.size());
5553
5554 // perform extra media-dependent close checks
5555 HRESULT hrc = i_canClose();
5556 if (FAILED(hrc)) return hrc;
5557
5558 m->fClosing = true;
5559
5560 if (wasCreated)
5561 {
5562 // remove from the list of known media before performing actual
5563 // uninitialization (to keep the media registry consistent on
5564 // failure to do so)
5565 hrc = i_unregisterWithVirtualBox();
5566 if (FAILED(hrc)) return hrc;
5567
5568 multilock.release();
5569 // Release the AutoCaller now, as otherwise uninit() will simply hang.
5570 // Needs to be done before mark the registries as modified and saving
5571 // the registry, as otherwise there may be a deadlock with someone else
5572 // closing this object while we're in i_saveModifiedRegistries(), which
5573 // needs the media tree lock, which the other thread holds until after
5574 // uninit() below.
5575 autoCaller.release();
5576 i_markRegistriesModified();
5577 m->pVirtualBox->i_saveModifiedRegistries();
5578 }
5579 else
5580 {
5581 multilock.release();
5582 // release the AutoCaller, as otherwise uninit() will simply hang
5583 autoCaller.release();
5584 }
5585
5586 uninit();
5587
5588 LogFlowFuncLeave();
5589
5590 return hrc;
5591}
5592
5593/**
5594 * Deletes the medium storage unit.
5595 *
5596 * If @a aProgress is not NULL but the object it points to is @c null then a new
5597 * progress object will be created and assigned to @a *aProgress on success,
5598 * otherwise the existing progress object is used. If Progress is NULL, then no
5599 * progress object is created/used at all.
5600 *
5601 * When @a aWait is @c false, this method will create a thread to perform the
5602 * delete operation asynchronously and will return immediately. Otherwise, it
5603 * will perform the operation on the calling thread and will not return to the
5604 * caller until the operation is completed. Note that @a aProgress cannot be
5605 * NULL when @a aWait is @c false (this method will assert in this case).
5606 *
5607 * @param aProgress Where to find/store a Progress object to track operation
5608 * completion.
5609 * @param aWait @c true if this method should block instead of creating
5610 * an asynchronous thread.
5611 * @param aNotify Notify about media for which metadata is changed
5612 * during execution of the function.
5613 *
5614 * @note Locks mVirtualBox and this object for writing. Locks medium tree for
5615 * writing.
5616 */
5617HRESULT Medium::i_deleteStorage(ComObjPtr<Progress> *aProgress,
5618 bool aWait, bool aNotify)
5619{
5620 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
5621
5622 HRESULT hrc = S_OK;
5623 ComObjPtr<Progress> pProgress;
5624 Medium::Task *pTask = NULL;
5625
5626 try
5627 {
5628 /* we're accessing the media tree, and i_canClose() needs it too */
5629 AutoWriteLock treelock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5630
5631 AutoCaller autoCaller(this);
5632 AssertComRCThrowRC(autoCaller.hrc());
5633
5634 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5635
5636 LogFlowThisFunc(("aWait=%RTbool locationFull=%s\n", aWait, i_getLocationFull().c_str() ));
5637
5638 if ( !(m->formatObj->i_getCapabilities() & ( MediumFormatCapabilities_CreateDynamic
5639 | MediumFormatCapabilities_CreateFixed
5640 | MediumFormatCapabilities_File)))
5641 throw setError(VBOX_E_NOT_SUPPORTED,
5642 tr("Medium format '%s' does not support storage deletion"),
5643 m->strFormat.c_str());
5644
5645 /* Wait for a concurrently running Medium::i_queryInfo to complete. */
5646 /** @todo r=klaus would be great if this could be moved to the async
5647 * part of the operation as it can take quite a while */
5648 while (m->queryInfoRunning)
5649 {
5650 alock.release();
5651 autoCaller.release();
5652 treelock.release();
5653 /* Must not hold the media tree lock or the object lock, as
5654 * Medium::i_queryInfo needs this lock and thus we would run
5655 * into a deadlock here. */
5656 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
5657 Assert(!isWriteLockOnCurrentThread());
5658 {
5659 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
5660 }
5661 treelock.acquire();
5662 autoCaller.add();
5663 AssertComRCThrowRC(autoCaller.hrc());
5664 alock.acquire();
5665 }
5666
5667 /* Note that we are fine with Inaccessible state too: a) for symmetry
5668 * with create calls and b) because it doesn't really harm to try, if
5669 * it is really inaccessible, the delete operation will fail anyway.
5670 * Accepting Inaccessible state is especially important because all
5671 * registered media are initially Inaccessible upon VBoxSVC startup
5672 * until COMGETTER(RefreshState) is called. Accept Deleting state
5673 * because some callers need to put the medium in this state early
5674 * to prevent races. */
5675 switch (m->state)
5676 {
5677 case MediumState_Created:
5678 case MediumState_Deleting:
5679 case MediumState_Inaccessible:
5680 break;
5681 default:
5682 throw i_setStateError();
5683 }
5684
5685 if (m->backRefs.size() != 0)
5686 {
5687 Utf8Str strMachines;
5688 for (BackRefList::const_iterator it = m->backRefs.begin();
5689 it != m->backRefs.end();
5690 ++it)
5691 {
5692 const BackRef &b = *it;
5693 if (strMachines.length())
5694 strMachines.append(", ");
5695 strMachines.append(b.machineId.toString().c_str());
5696 }
5697#ifdef DEBUG
5698 i_dumpBackRefs();
5699#endif
5700 throw setError(VBOX_E_OBJECT_IN_USE,
5701 tr("Cannot delete storage: medium '%s' is still attached to the following %d virtual machine(s): %s",
5702 "", m->backRefs.size()),
5703 m->strLocationFull.c_str(),
5704 m->backRefs.size(),
5705 strMachines.c_str());
5706 }
5707
5708 hrc = i_canClose();
5709 if (FAILED(hrc))
5710 throw hrc;
5711
5712 /* go to Deleting state, so that the medium is not actually locked */
5713 if (m->state != MediumState_Deleting)
5714 {
5715 hrc = i_markForDeletion();
5716 if (FAILED(hrc))
5717 throw hrc;
5718 }
5719
5720 /* Build the medium lock list. */
5721 MediumLockList *pMediumLockList(new MediumLockList());
5722 alock.release();
5723 autoCaller.release();
5724 treelock.release();
5725 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
5726 this /* pToLockWrite */,
5727 false /* fMediumLockWriteAll */,
5728 NULL,
5729 *pMediumLockList);
5730 treelock.acquire();
5731 autoCaller.add();
5732 AssertComRCThrowRC(autoCaller.hrc());
5733 alock.acquire();
5734 if (FAILED(hrc))
5735 {
5736 delete pMediumLockList;
5737 throw hrc;
5738 }
5739
5740 alock.release();
5741 autoCaller.release();
5742 treelock.release();
5743 hrc = pMediumLockList->Lock();
5744 treelock.acquire();
5745 autoCaller.add();
5746 AssertComRCThrowRC(autoCaller.hrc());
5747 alock.acquire();
5748 if (FAILED(hrc))
5749 {
5750 delete pMediumLockList;
5751 throw setError(hrc,
5752 tr("Failed to lock media when deleting '%s'"),
5753 i_getLocationFull().c_str());
5754 }
5755
5756 /* try to remove from the list of known media before performing
5757 * actual deletion (we favor the consistency of the media registry
5758 * which would have been broken if unregisterWithVirtualBox() failed
5759 * after we successfully deleted the storage) */
5760 hrc = i_unregisterWithVirtualBox();
5761 if (FAILED(hrc))
5762 throw hrc;
5763 // no longer need lock
5764 alock.release();
5765 autoCaller.release();
5766 treelock.release();
5767 i_markRegistriesModified();
5768
5769 if (aProgress != NULL)
5770 {
5771 /* use the existing progress object... */
5772 pProgress = *aProgress;
5773
5774 /* ...but create a new one if it is null */
5775 if (pProgress.isNull())
5776 {
5777 pProgress.createObject();
5778 hrc = pProgress->init(m->pVirtualBox,
5779 static_cast<IMedium*>(this),
5780 BstrFmt(tr("Deleting medium storage unit '%s'"), m->strLocationFull.c_str()).raw(),
5781 FALSE /* aCancelable */);
5782 if (FAILED(hrc))
5783 throw hrc;
5784 }
5785 }
5786
5787 /* setup task object to carry out the operation sync/async */
5788 pTask = new Medium::DeleteTask(this, pProgress, pMediumLockList, false, aNotify);
5789 hrc = pTask->hrc();
5790 AssertComRC(hrc);
5791 if (FAILED(hrc))
5792 throw hrc;
5793 }
5794 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
5795
5796 if (SUCCEEDED(hrc))
5797 {
5798 if (aWait)
5799 {
5800 hrc = pTask->runNow();
5801 delete pTask;
5802 }
5803 else
5804 hrc = pTask->createThread();
5805 pTask = NULL;
5806 if (SUCCEEDED(hrc) && aProgress != NULL)
5807 *aProgress = pProgress;
5808 }
5809 else
5810 {
5811 if (pTask)
5812 delete pTask;
5813
5814 /* Undo deleting state if necessary. */
5815 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5816 /* Make sure that any error signalled by unmarkForDeletion() is not
5817 * ending up in the error list (if the caller uses MultiResult). It
5818 * usually is spurious, as in most cases the medium hasn't been marked
5819 * for deletion when the error was thrown above. */
5820 ErrorInfoKeeper eik;
5821 i_unmarkForDeletion();
5822 }
5823
5824 return hrc;
5825}
5826
5827/**
5828 * Mark a medium for deletion.
5829 *
5830 * @note Caller must hold the write lock on this medium!
5831 */
5832HRESULT Medium::i_markForDeletion()
5833{
5834 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
5835 switch (m->state)
5836 {
5837 case MediumState_Created:
5838 case MediumState_Inaccessible:
5839 m->preLockState = m->state;
5840 m->state = MediumState_Deleting;
5841 return S_OK;
5842 default:
5843 return i_setStateError();
5844 }
5845}
5846
5847/**
5848 * Removes the "mark for deletion".
5849 *
5850 * @note Caller must hold the write lock on this medium!
5851 */
5852HRESULT Medium::i_unmarkForDeletion()
5853{
5854 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
5855 switch (m->state)
5856 {
5857 case MediumState_Deleting:
5858 m->state = m->preLockState;
5859 return S_OK;
5860 default:
5861 return i_setStateError();
5862 }
5863}
5864
5865/**
5866 * Mark a medium for deletion which is in locked state.
5867 *
5868 * @note Caller must hold the write lock on this medium!
5869 */
5870HRESULT Medium::i_markLockedForDeletion()
5871{
5872 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
5873 if ( ( m->state == MediumState_LockedRead
5874 || m->state == MediumState_LockedWrite)
5875 && m->preLockState == MediumState_Created)
5876 {
5877 m->preLockState = MediumState_Deleting;
5878 return S_OK;
5879 }
5880 else
5881 return i_setStateError();
5882}
5883
5884/**
5885 * Removes the "mark for deletion" for a medium in locked state.
5886 *
5887 * @note Caller must hold the write lock on this medium!
5888 */
5889HRESULT Medium::i_unmarkLockedForDeletion()
5890{
5891 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
5892 if ( ( m->state == MediumState_LockedRead
5893 || m->state == MediumState_LockedWrite)
5894 && m->preLockState == MediumState_Deleting)
5895 {
5896 m->preLockState = MediumState_Created;
5897 return S_OK;
5898 }
5899 else
5900 return i_setStateError();
5901}
5902
5903/**
5904 * Queries the preferred merge direction from this to the other medium, i.e.
5905 * the one which requires the least amount of I/O and therefore time and
5906 * disk consumption.
5907 *
5908 * @returns Status code.
5909 * @retval E_FAIL in case determining the merge direction fails for some reason,
5910 * for example if getting the size of the media fails. There is no
5911 * error set though and the caller is free to continue to find out
5912 * what was going wrong later. Leaves fMergeForward unset.
5913 * @retval VBOX_E_INVALID_OBJECT_STATE if both media are not related to each other
5914 * An error is set.
5915 * @param pOther The other medium to merge with.
5916 * @param fMergeForward Resulting preferred merge direction (out).
5917 */
5918HRESULT Medium::i_queryPreferredMergeDirection(const ComObjPtr<Medium> &pOther,
5919 bool &fMergeForward)
5920{
5921 AssertReturn(pOther != NULL, E_FAIL);
5922 AssertReturn(pOther != this, E_FAIL);
5923
5924 HRESULT hrc = S_OK;
5925 bool fThisParent = false; /**<< Flag whether this medium is the parent of pOther. */
5926
5927 try
5928 {
5929 // locking: we need the tree lock first because we access parent pointers
5930 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
5931
5932 AutoCaller autoCaller(this);
5933 AssertComRCThrowRC(autoCaller.hrc());
5934
5935 AutoCaller otherCaller(pOther);
5936 AssertComRCThrowRC(otherCaller.hrc());
5937
5938 /* more sanity checking and figuring out the current merge direction */
5939 ComObjPtr<Medium> pMedium = i_getParent();
5940 while (!pMedium.isNull() && pMedium != pOther)
5941 pMedium = pMedium->i_getParent();
5942 if (pMedium == pOther)
5943 fThisParent = false;
5944 else
5945 {
5946 pMedium = pOther->i_getParent();
5947 while (!pMedium.isNull() && pMedium != this)
5948 pMedium = pMedium->i_getParent();
5949 if (pMedium == this)
5950 fThisParent = true;
5951 else
5952 {
5953 Utf8Str tgtLoc;
5954 {
5955 AutoReadLock alock(pOther COMMA_LOCKVAL_SRC_POS);
5956 tgtLoc = pOther->i_getLocationFull();
5957 }
5958
5959 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5960 throw setError(VBOX_E_INVALID_OBJECT_STATE,
5961 tr("Media '%s' and '%s' are unrelated"),
5962 m->strLocationFull.c_str(), tgtLoc.c_str());
5963 }
5964 }
5965
5966 /*
5967 * Figure out the preferred merge direction. The current way is to
5968 * get the current sizes of file based images and select the merge
5969 * direction depending on the size.
5970 *
5971 * Can't use the VD API to get current size here as the media might
5972 * be write locked by a running VM. Resort to RTFileQuerySize().
5973 */
5974 int vrc = VINF_SUCCESS;
5975 uint64_t cbMediumThis = 0;
5976 uint64_t cbMediumOther = 0;
5977
5978 if (i_isMediumFormatFile() && pOther->i_isMediumFormatFile())
5979 {
5980 vrc = RTFileQuerySizeByPath(this->i_getLocationFull().c_str(), &cbMediumThis);
5981 if (RT_SUCCESS(vrc))
5982 {
5983 vrc = RTFileQuerySizeByPath(pOther->i_getLocationFull().c_str(),
5984 &cbMediumOther);
5985 }
5986
5987 if (RT_FAILURE(vrc))
5988 hrc = E_FAIL;
5989 else
5990 {
5991 /*
5992 * Check which merge direction might be more optimal.
5993 * This method is not bullet proof of course as there might
5994 * be overlapping blocks in the images so the file size is
5995 * not the best indicator but it is good enough for our purpose
5996 * and everything else is too complicated, especially when the
5997 * media are used by a running VM.
5998 */
5999
6000 uint32_t mediumVariants = MediumVariant_Fixed | MediumVariant_VmdkStreamOptimized;
6001 uint32_t mediumCaps = MediumFormatCapabilities_CreateDynamic | MediumFormatCapabilities_File;
6002
6003 bool fDynamicOther = pOther->i_getMediumFormat()->i_getCapabilities() & mediumCaps
6004 && pOther->i_getVariant() & ~mediumVariants;
6005 bool fDynamicThis = i_getMediumFormat()->i_getCapabilities() & mediumCaps
6006 && i_getVariant() & ~mediumVariants;
6007 bool fMergeIntoThis = (fDynamicThis && !fDynamicOther)
6008 || (fDynamicThis == fDynamicOther && cbMediumThis > cbMediumOther);
6009 fMergeForward = fMergeIntoThis != fThisParent;
6010 }
6011 }
6012 }
6013 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6014
6015 return hrc;
6016}
6017
6018/**
6019 * Prepares this (source) medium, target medium and all intermediate media
6020 * for the merge operation.
6021 *
6022 * This method is to be called prior to calling the #mergeTo() to perform
6023 * necessary consistency checks and place involved media to appropriate
6024 * states. If #mergeTo() is not called or fails, the state modifications
6025 * performed by this method must be undone by #i_cancelMergeTo().
6026 *
6027 * See #mergeTo() for more information about merging.
6028 *
6029 * @param pTarget Target medium.
6030 * @param aMachineId Allowed machine attachment. NULL means do not check.
6031 * @param aSnapshotId Allowed snapshot attachment. NULL or empty UUID means
6032 * do not check.
6033 * @param fLockMedia Flag whether to lock the medium lock list or not.
6034 * If set to false and the medium lock list locking fails
6035 * later you must call #i_cancelMergeTo().
6036 * @param fMergeForward Resulting merge direction (out).
6037 * @param pParentForTarget New parent for target medium after merge (out).
6038 * @param aChildrenToReparent Medium lock list containing all children of the
6039 * source which will have to be reparented to the target
6040 * after merge (out).
6041 * @param aMediumLockList Medium locking information (out).
6042 *
6043 * @note Locks medium tree for reading. Locks this object, aTarget and all
6044 * intermediate media for writing.
6045 */
6046HRESULT Medium::i_prepareMergeTo(const ComObjPtr<Medium> &pTarget,
6047 const Guid *aMachineId,
6048 const Guid *aSnapshotId,
6049 bool fLockMedia,
6050 bool &fMergeForward,
6051 ComObjPtr<Medium> &pParentForTarget,
6052 MediumLockList * &aChildrenToReparent,
6053 MediumLockList * &aMediumLockList)
6054{
6055 AssertReturn(pTarget != NULL, E_FAIL);
6056 AssertReturn(pTarget != this, E_FAIL);
6057
6058 HRESULT hrc = S_OK;
6059 fMergeForward = false;
6060 pParentForTarget.setNull();
6061 Assert(aChildrenToReparent == NULL);
6062 aChildrenToReparent = NULL;
6063 Assert(aMediumLockList == NULL);
6064 aMediumLockList = NULL;
6065
6066 try
6067 {
6068 // locking: we need the tree lock first because we access parent pointers
6069 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
6070
6071 AutoCaller autoCaller(this);
6072 AssertComRCThrowRC(autoCaller.hrc());
6073
6074 AutoCaller targetCaller(pTarget);
6075 AssertComRCThrowRC(targetCaller.hrc());
6076
6077 /* more sanity checking and figuring out the merge direction */
6078 ComObjPtr<Medium> pMedium = i_getParent();
6079 while (!pMedium.isNull() && pMedium != pTarget)
6080 pMedium = pMedium->i_getParent();
6081 if (pMedium == pTarget)
6082 fMergeForward = false;
6083 else
6084 {
6085 pMedium = pTarget->i_getParent();
6086 while (!pMedium.isNull() && pMedium != this)
6087 pMedium = pMedium->i_getParent();
6088 if (pMedium == this)
6089 fMergeForward = true;
6090 else
6091 {
6092 Utf8Str tgtLoc;
6093 {
6094 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
6095 tgtLoc = pTarget->i_getLocationFull();
6096 }
6097
6098 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6099 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6100 tr("Media '%s' and '%s' are unrelated"),
6101 m->strLocationFull.c_str(), tgtLoc.c_str());
6102 }
6103 }
6104
6105 /* Build the lock list. */
6106 aMediumLockList = new MediumLockList();
6107 targetCaller.release();
6108 autoCaller.release();
6109 treeLock.release();
6110 if (fMergeForward)
6111 hrc = pTarget->i_createMediumLockList(true /* fFailIfInaccessible */,
6112 pTarget /* pToLockWrite */,
6113 false /* fMediumLockWriteAll */,
6114 NULL,
6115 *aMediumLockList);
6116 else
6117 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
6118 pTarget /* pToLockWrite */,
6119 false /* fMediumLockWriteAll */,
6120 NULL,
6121 *aMediumLockList);
6122 treeLock.acquire();
6123 autoCaller.add();
6124 AssertComRCThrowRC(autoCaller.hrc());
6125 targetCaller.add();
6126 AssertComRCThrowRC(targetCaller.hrc());
6127 if (FAILED(hrc))
6128 throw hrc;
6129
6130 /* Sanity checking, must be after lock list creation as it depends on
6131 * valid medium states. The medium objects must be accessible. Only
6132 * do this if immediate locking is requested, otherwise it fails when
6133 * we construct a medium lock list for an already running VM. Snapshot
6134 * deletion uses this to simplify its life. */
6135 if (fLockMedia)
6136 {
6137 {
6138 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6139 if (m->state != MediumState_Created)
6140 throw i_setStateError();
6141 }
6142 {
6143 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
6144 if (pTarget->m->state != MediumState_Created)
6145 throw pTarget->i_setStateError();
6146 }
6147 }
6148
6149 /* check medium attachment and other sanity conditions */
6150 if (fMergeForward)
6151 {
6152 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6153 if (i_getChildren().size() > 1)
6154 {
6155 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6156 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
6157 m->strLocationFull.c_str(), i_getChildren().size());
6158 }
6159 /* One backreference is only allowed if the machine ID is not empty
6160 * and it matches the machine the medium is attached to (including
6161 * the snapshot ID if not empty). */
6162 if ( m->backRefs.size() != 0
6163 && ( !aMachineId
6164 || m->backRefs.size() != 1
6165 || aMachineId->isZero()
6166 || *i_getFirstMachineBackrefId() != *aMachineId
6167 || ( (!aSnapshotId || !aSnapshotId->isZero())
6168 && *i_getFirstMachineBackrefSnapshotId() != *aSnapshotId)))
6169 throw setError(VBOX_E_OBJECT_IN_USE,
6170 tr("Medium '%s' is attached to %d virtual machines", "", m->backRefs.size()),
6171 m->strLocationFull.c_str(), m->backRefs.size());
6172 if (m->type == MediumType_Immutable)
6173 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6174 tr("Medium '%s' is immutable"),
6175 m->strLocationFull.c_str());
6176 if (m->type == MediumType_MultiAttach)
6177 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6178 tr("Medium '%s' is multi-attach"),
6179 m->strLocationFull.c_str());
6180 }
6181 else
6182 {
6183 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
6184 if (pTarget->i_getChildren().size() > 1)
6185 {
6186 throw setError(VBOX_E_OBJECT_IN_USE,
6187 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
6188 pTarget->m->strLocationFull.c_str(),
6189 pTarget->i_getChildren().size());
6190 }
6191 if (pTarget->m->type == MediumType_Immutable)
6192 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6193 tr("Medium '%s' is immutable"),
6194 pTarget->m->strLocationFull.c_str());
6195 if (pTarget->m->type == MediumType_MultiAttach)
6196 throw setError(VBOX_E_INVALID_OBJECT_STATE,
6197 tr("Medium '%s' is multi-attach"),
6198 pTarget->m->strLocationFull.c_str());
6199 }
6200 ComObjPtr<Medium> pLast(fMergeForward ? (Medium *)pTarget : this);
6201 ComObjPtr<Medium> pLastIntermediate = pLast->i_getParent();
6202 for (pLast = pLastIntermediate;
6203 !pLast.isNull() && pLast != pTarget && pLast != this;
6204 pLast = pLast->i_getParent())
6205 {
6206 AutoReadLock alock(pLast COMMA_LOCKVAL_SRC_POS);
6207 if (pLast->i_getChildren().size() > 1)
6208 {
6209 throw setError(VBOX_E_OBJECT_IN_USE,
6210 tr("Medium '%s' involved in the merge operation has more than one child medium (%d)"),
6211 pLast->m->strLocationFull.c_str(),
6212 pLast->i_getChildren().size());
6213 }
6214 if (pLast->m->backRefs.size() != 0)
6215 throw setError(VBOX_E_OBJECT_IN_USE,
6216 tr("Medium '%s' is attached to %d virtual machines", "", pLast->m->backRefs.size()),
6217 pLast->m->strLocationFull.c_str(),
6218 pLast->m->backRefs.size());
6219
6220 }
6221
6222 /* Update medium states appropriately */
6223 {
6224 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6225
6226 if (m->state == MediumState_Created)
6227 {
6228 hrc = i_markForDeletion();
6229 if (FAILED(hrc))
6230 throw hrc;
6231 }
6232 else
6233 {
6234 if (fLockMedia)
6235 throw i_setStateError();
6236 else if ( m->state == MediumState_LockedWrite
6237 || m->state == MediumState_LockedRead)
6238 {
6239 /* Either mark it for deletion in locked state or allow
6240 * others to have done so. */
6241 if (m->preLockState == MediumState_Created)
6242 i_markLockedForDeletion();
6243 else if (m->preLockState != MediumState_Deleting)
6244 throw i_setStateError();
6245 }
6246 else
6247 throw i_setStateError();
6248 }
6249 }
6250
6251 if (fMergeForward)
6252 {
6253 /* we will need parent to reparent target */
6254 pParentForTarget = i_getParent();
6255 }
6256 else
6257 {
6258 /* we will need to reparent children of the source */
6259 aChildrenToReparent = new MediumLockList();
6260 for (MediaList::const_iterator it = i_getChildren().begin();
6261 it != i_getChildren().end();
6262 ++it)
6263 {
6264 pMedium = *it;
6265 aChildrenToReparent->Append(pMedium, true /* fLockWrite */);
6266 }
6267 if (fLockMedia && aChildrenToReparent)
6268 {
6269 targetCaller.release();
6270 autoCaller.release();
6271 treeLock.release();
6272 hrc = aChildrenToReparent->Lock();
6273 treeLock.acquire();
6274 autoCaller.add();
6275 AssertComRCThrowRC(autoCaller.hrc());
6276 targetCaller.add();
6277 AssertComRCThrowRC(targetCaller.hrc());
6278 if (FAILED(hrc))
6279 throw hrc;
6280 }
6281 }
6282 for (pLast = pLastIntermediate;
6283 !pLast.isNull() && pLast != pTarget && pLast != this;
6284 pLast = pLast->i_getParent())
6285 {
6286 AutoWriteLock alock(pLast COMMA_LOCKVAL_SRC_POS);
6287 if (pLast->m->state == MediumState_Created)
6288 {
6289 hrc = pLast->i_markForDeletion();
6290 if (FAILED(hrc))
6291 throw hrc;
6292 }
6293 else
6294 throw pLast->i_setStateError();
6295 }
6296
6297 /* Tweak the lock list in the backward merge case, as the target
6298 * isn't marked to be locked for writing yet. */
6299 if (!fMergeForward)
6300 {
6301 MediumLockList::Base::iterator lockListBegin =
6302 aMediumLockList->GetBegin();
6303 MediumLockList::Base::iterator lockListEnd =
6304 aMediumLockList->GetEnd();
6305 ++lockListEnd;
6306 for (MediumLockList::Base::iterator it = lockListBegin;
6307 it != lockListEnd;
6308 ++it)
6309 {
6310 MediumLock &mediumLock = *it;
6311 if (mediumLock.GetMedium() == pTarget)
6312 {
6313 HRESULT hrc2 = mediumLock.UpdateLock(true);
6314 AssertComRC(hrc2);
6315 break;
6316 }
6317 }
6318 }
6319
6320 if (fLockMedia)
6321 {
6322 targetCaller.release();
6323 autoCaller.release();
6324 treeLock.release();
6325 hrc = aMediumLockList->Lock();
6326 treeLock.acquire();
6327 autoCaller.add();
6328 AssertComRCThrowRC(autoCaller.hrc());
6329 targetCaller.add();
6330 AssertComRCThrowRC(targetCaller.hrc());
6331 if (FAILED(hrc))
6332 {
6333 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
6334 throw setError(hrc,
6335 tr("Failed to lock media when merging to '%s'"),
6336 pTarget->i_getLocationFull().c_str());
6337 }
6338 }
6339 }
6340 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6341
6342 if (FAILED(hrc))
6343 {
6344 if (aMediumLockList)
6345 {
6346 delete aMediumLockList;
6347 aMediumLockList = NULL;
6348 }
6349 if (aChildrenToReparent)
6350 {
6351 delete aChildrenToReparent;
6352 aChildrenToReparent = NULL;
6353 }
6354 }
6355
6356 return hrc;
6357}
6358
6359/**
6360 * Merges this medium to the specified medium which must be either its
6361 * direct ancestor or descendant.
6362 *
6363 * Given this medium is SOURCE and the specified medium is TARGET, we will
6364 * get two variants of the merge operation:
6365 *
6366 * forward merge
6367 * ------------------------->
6368 * [Extra] <- SOURCE <- Intermediate <- TARGET
6369 * Any Del Del LockWr
6370 *
6371 *
6372 * backward merge
6373 * <-------------------------
6374 * TARGET <- Intermediate <- SOURCE <- [Extra]
6375 * LockWr Del Del LockWr
6376 *
6377 * Each diagram shows the involved media on the media chain where
6378 * SOURCE and TARGET belong. Under each medium there is a state value which
6379 * the medium must have at a time of the mergeTo() call.
6380 *
6381 * The media in the square braces may be absent (e.g. when the forward
6382 * operation takes place and SOURCE is the base medium, or when the backward
6383 * merge operation takes place and TARGET is the last child in the chain) but if
6384 * they present they are involved too as shown.
6385 *
6386 * Neither the source medium nor intermediate media may be attached to
6387 * any VM directly or in the snapshot, otherwise this method will assert.
6388 *
6389 * The #i_prepareMergeTo() method must be called prior to this method to place
6390 * all involved to necessary states and perform other consistency checks.
6391 *
6392 * If @a aWait is @c true then this method will perform the operation on the
6393 * calling thread and will not return to the caller until the operation is
6394 * completed. When this method succeeds, all intermediate medium objects in
6395 * the chain will be uninitialized, the state of the target medium (and all
6396 * involved extra media) will be restored. @a aMediumLockList will not be
6397 * deleted, whether the operation is successful or not. The caller has to do
6398 * this if appropriate. Note that this (source) medium is not uninitialized
6399 * because of possible AutoCaller instances held by the caller of this method
6400 * on the current thread. It's therefore the responsibility of the caller to
6401 * call Medium::uninit() after releasing all callers.
6402 *
6403 * If @a aWait is @c false then this method will create a thread to perform the
6404 * operation asynchronously and will return immediately. If the operation
6405 * succeeds, the thread will uninitialize the source medium object and all
6406 * intermediate medium objects in the chain, reset the state of the target
6407 * medium (and all involved extra media) and delete @a aMediumLockList.
6408 * If the operation fails, the thread will only reset the states of all
6409 * involved media and delete @a aMediumLockList.
6410 *
6411 * When this method fails (regardless of the @a aWait mode), it is a caller's
6412 * responsibility to undo state changes and delete @a aMediumLockList using
6413 * #i_cancelMergeTo().
6414 *
6415 * If @a aProgress is not NULL but the object it points to is @c null then a new
6416 * progress object will be created and assigned to @a *aProgress on success,
6417 * otherwise the existing progress object is used. If Progress is NULL, then no
6418 * progress object is created/used at all. Note that @a aProgress cannot be
6419 * NULL when @a aWait is @c false (this method will assert in this case).
6420 *
6421 * @param pTarget Target medium.
6422 * @param fMergeForward Merge direction.
6423 * @param pParentForTarget New parent for target medium after merge.
6424 * @param aChildrenToReparent List of children of the source which will have
6425 * to be reparented to the target after merge.
6426 * @param aMediumLockList Medium locking information.
6427 * @param aProgress Where to find/store a Progress object to track operation
6428 * completion.
6429 * @param aWait @c true if this method should block instead of creating
6430 * an asynchronous thread.
6431 * @param aNotify Notify about media for which metadata is changed
6432 * during execution of the function.
6433 *
6434 * @note Locks the tree lock for writing. Locks the media from the chain
6435 * for writing.
6436 */
6437HRESULT Medium::i_mergeTo(const ComObjPtr<Medium> &pTarget,
6438 bool fMergeForward,
6439 const ComObjPtr<Medium> &pParentForTarget,
6440 MediumLockList *aChildrenToReparent,
6441 MediumLockList *aMediumLockList,
6442 ComObjPtr<Progress> *aProgress,
6443 bool aWait, bool aNotify)
6444{
6445 AssertReturn(pTarget != NULL, E_FAIL);
6446 AssertReturn(pTarget != this, E_FAIL);
6447 AssertReturn(aMediumLockList != NULL, E_FAIL);
6448 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
6449
6450 AutoCaller autoCaller(this);
6451 AssertComRCReturnRC(autoCaller.hrc());
6452
6453 AutoCaller targetCaller(pTarget);
6454 AssertComRCReturnRC(targetCaller.hrc());
6455
6456 HRESULT hrc = S_OK;
6457 ComObjPtr<Progress> pProgress;
6458 Medium::Task *pTask = NULL;
6459
6460 try
6461 {
6462 if (aProgress != NULL)
6463 {
6464 /* use the existing progress object... */
6465 pProgress = *aProgress;
6466
6467 /* ...but create a new one if it is null */
6468 if (pProgress.isNull())
6469 {
6470 Utf8Str tgtName;
6471 {
6472 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
6473 tgtName = pTarget->i_getName();
6474 }
6475
6476 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6477
6478 pProgress.createObject();
6479 hrc = pProgress->init(m->pVirtualBox,
6480 static_cast<IMedium*>(this),
6481 BstrFmt(tr("Merging medium '%s' to '%s'"),
6482 i_getName().c_str(),
6483 tgtName.c_str()).raw(),
6484 TRUE, /* aCancelable */
6485 2, /* Number of opearations */
6486 BstrFmt(tr("Resizing medium '%s' before merge"),
6487 tgtName.c_str()).raw()
6488 );
6489 if (FAILED(hrc))
6490 throw hrc;
6491 }
6492 }
6493
6494 /* setup task object to carry out the operation sync/async */
6495 pTask = new Medium::MergeTask(this, pTarget, fMergeForward,
6496 pParentForTarget, aChildrenToReparent,
6497 pProgress, aMediumLockList,
6498 aWait /* fKeepMediumLockList */,
6499 aNotify);
6500 hrc = pTask->hrc();
6501 AssertComRC(hrc);
6502 if (FAILED(hrc))
6503 throw hrc;
6504 }
6505 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6506
6507 if (SUCCEEDED(hrc))
6508 {
6509 if (aWait)
6510 {
6511 hrc = pTask->runNow();
6512 delete pTask;
6513 }
6514 else
6515 hrc = pTask->createThread();
6516 pTask = NULL;
6517 if (SUCCEEDED(hrc) && aProgress != NULL)
6518 *aProgress = pProgress;
6519 }
6520 else if (pTask != NULL)
6521 delete pTask;
6522
6523 return hrc;
6524}
6525
6526/**
6527 * Undoes what #i_prepareMergeTo() did. Must be called if #mergeTo() is not
6528 * called or fails. Frees memory occupied by @a aMediumLockList and unlocks
6529 * the medium objects in @a aChildrenToReparent.
6530 *
6531 * @param aChildrenToReparent List of children of the source which will have
6532 * to be reparented to the target after merge.
6533 * @param aMediumLockList Medium locking information.
6534 *
6535 * @note Locks the tree lock for writing. Locks the media from the chain
6536 * for writing.
6537 */
6538void Medium::i_cancelMergeTo(MediumLockList *aChildrenToReparent,
6539 MediumLockList *aMediumLockList)
6540{
6541 AutoCaller autoCaller(this);
6542 AssertComRCReturnVoid(autoCaller.hrc());
6543
6544 AssertReturnVoid(aMediumLockList != NULL);
6545
6546 /* Revert media marked for deletion to previous state. */
6547 MediumLockList::Base::const_iterator mediumListBegin =
6548 aMediumLockList->GetBegin();
6549 MediumLockList::Base::const_iterator mediumListEnd =
6550 aMediumLockList->GetEnd();
6551 for (MediumLockList::Base::const_iterator it = mediumListBegin;
6552 it != mediumListEnd;
6553 ++it)
6554 {
6555 const MediumLock &mediumLock = *it;
6556 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
6557 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
6558
6559 if (pMedium->m->state == MediumState_Deleting)
6560 {
6561 HRESULT hrc = pMedium->i_unmarkForDeletion();
6562 AssertComRC(hrc);
6563 }
6564 else if ( ( pMedium->m->state == MediumState_LockedWrite
6565 || pMedium->m->state == MediumState_LockedRead)
6566 && pMedium->m->preLockState == MediumState_Deleting)
6567 {
6568 HRESULT hrc = pMedium->i_unmarkLockedForDeletion();
6569 AssertComRC(hrc);
6570 }
6571 }
6572
6573 /* the destructor will do the work */
6574 delete aMediumLockList;
6575
6576 /* unlock the children which had to be reparented, the destructor will do
6577 * the work */
6578 if (aChildrenToReparent)
6579 delete aChildrenToReparent;
6580}
6581
6582/**
6583 * Resizes the media.
6584 *
6585 * If @a aWait is @c true then this method will perform the operation on the
6586 * calling thread and will not return to the caller until the operation is
6587 * completed. When this method succeeds, the state of the target medium (and all
6588 * involved extra media) will be restored. @a aMediumLockList will not be
6589 * deleted, whether the operation is successful or not. The caller has to do
6590 * this if appropriate.
6591 *
6592 * If @a aWait is @c false then this method will create a thread to perform the
6593 * operation asynchronously and will return immediately. The thread will reset
6594 * the state of the target medium (and all involved extra media) and delete
6595 * @a aMediumLockList.
6596 *
6597 * When this method fails (regardless of the @a aWait mode), it is a caller's
6598 * responsibility to undo state changes and delete @a aMediumLockList.
6599 *
6600 * If @a aProgress is not NULL but the object it points to is @c null then a new
6601 * progress object will be created and assigned to @a *aProgress on success,
6602 * otherwise the existing progress object is used. If Progress is NULL, then no
6603 * progress object is created/used at all. Note that @a aProgress cannot be
6604 * NULL when @a aWait is @c false (this method will assert in this case).
6605 *
6606 * @param aLogicalSize New nominal capacity of the medium in bytes.
6607 * @param aMediumLockList Medium locking information.
6608 * @param aProgress Where to find/store a Progress object to track operation
6609 * completion.
6610 * @param aWait @c true if this method should block instead of creating
6611 * an asynchronous thread.
6612 * @param aNotify Notify about media for which metadata is changed
6613 * during execution of the function.
6614 *
6615 * @note Locks the media from the chain for writing.
6616 */
6617
6618HRESULT Medium::i_resize(uint64_t aLogicalSize,
6619 MediumLockList *aMediumLockList,
6620 ComObjPtr<Progress> *aProgress,
6621 bool aWait,
6622 bool aNotify)
6623{
6624 AssertReturn(aMediumLockList != NULL, E_FAIL);
6625 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
6626
6627 AutoCaller autoCaller(this);
6628 AssertComRCReturnRC(autoCaller.hrc());
6629
6630 HRESULT hrc = S_OK;
6631 ComObjPtr<Progress> pProgress;
6632 Medium::Task *pTask = NULL;
6633
6634 try
6635 {
6636 if (aProgress != NULL)
6637 {
6638 /* use the existing progress object... */
6639 pProgress = *aProgress;
6640
6641 /* ...but create a new one if it is null */
6642 if (pProgress.isNull())
6643 {
6644 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6645
6646 pProgress.createObject();
6647 hrc = pProgress->init(m->pVirtualBox,
6648 static_cast <IMedium *>(this),
6649 BstrFmt(tr("Resizing medium '%s'"), m->strLocationFull.c_str()).raw(),
6650 TRUE /* aCancelable */);
6651 if (FAILED(hrc))
6652 throw hrc;
6653 }
6654 }
6655
6656 /* setup task object to carry out the operation asynchronously */
6657 pTask = new Medium::ResizeTask(this,
6658 aLogicalSize,
6659 pProgress,
6660 aMediumLockList,
6661 aWait /* fKeepMediumLockList */,
6662 aNotify);
6663 hrc = pTask->hrc();
6664 AssertComRC(hrc);
6665 if (FAILED(hrc))
6666 throw hrc;
6667 }
6668 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6669
6670 if (SUCCEEDED(hrc))
6671 {
6672 if (aWait)
6673 {
6674 hrc = pTask->runNow();
6675 delete pTask;
6676 }
6677 else
6678 hrc = pTask->createThread();
6679 pTask = NULL;
6680 if (SUCCEEDED(hrc) && aProgress != NULL)
6681 *aProgress = pProgress;
6682 }
6683 else if (pTask != NULL)
6684 delete pTask;
6685
6686 return hrc;
6687}
6688
6689/**
6690 * Fix the parent UUID of all children to point to this medium as their
6691 * parent.
6692 */
6693HRESULT Medium::i_fixParentUuidOfChildren(MediumLockList *pChildrenToReparent)
6694{
6695 /** @todo r=klaus The code below needs to be double checked with regard
6696 * to lock order violations, it probably causes lock order issues related
6697 * to the AutoCaller usage. Likewise the code using this method seems
6698 * problematic. */
6699 Assert(!isWriteLockOnCurrentThread());
6700 Assert(!m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
6701 MediumLockList mediumLockList;
6702 HRESULT hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
6703 NULL /* pToLockWrite */,
6704 false /* fMediumLockWriteAll */,
6705 this,
6706 mediumLockList);
6707 AssertComRCReturnRC(hrc);
6708
6709 try
6710 {
6711 PVDISK hdd;
6712 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
6713 ComAssertRCThrow(vrc, E_FAIL);
6714
6715 try
6716 {
6717 MediumLockList::Base::iterator lockListBegin =
6718 mediumLockList.GetBegin();
6719 MediumLockList::Base::iterator lockListEnd =
6720 mediumLockList.GetEnd();
6721 for (MediumLockList::Base::iterator it = lockListBegin;
6722 it != lockListEnd;
6723 ++it)
6724 {
6725 MediumLock &mediumLock = *it;
6726 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
6727 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
6728
6729 // open the medium
6730 vrc = VDOpen(hdd,
6731 pMedium->m->strFormat.c_str(),
6732 pMedium->m->strLocationFull.c_str(),
6733 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
6734 pMedium->m->vdImageIfaces);
6735 if (RT_FAILURE(vrc))
6736 throw vrc;
6737 }
6738
6739 MediumLockList::Base::iterator childrenBegin = pChildrenToReparent->GetBegin();
6740 MediumLockList::Base::iterator childrenEnd = pChildrenToReparent->GetEnd();
6741 for (MediumLockList::Base::iterator it = childrenBegin;
6742 it != childrenEnd;
6743 ++it)
6744 {
6745 Medium *pMedium = it->GetMedium();
6746 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
6747 vrc = VDOpen(hdd,
6748 pMedium->m->strFormat.c_str(),
6749 pMedium->m->strLocationFull.c_str(),
6750 VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
6751 pMedium->m->vdImageIfaces);
6752 if (RT_FAILURE(vrc))
6753 throw vrc;
6754
6755 vrc = VDSetParentUuid(hdd, VD_LAST_IMAGE, m->id.raw());
6756 if (RT_FAILURE(vrc))
6757 throw vrc;
6758
6759 vrc = VDClose(hdd, false /* fDelete */);
6760 if (RT_FAILURE(vrc))
6761 throw vrc;
6762 }
6763 }
6764 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6765 catch (int vrcXcpt)
6766 {
6767 hrc = setErrorBoth(E_FAIL, vrcXcpt,
6768 tr("Could not update medium UUID references to parent '%s' (%s)"),
6769 m->strLocationFull.c_str(),
6770 i_vdError(vrcXcpt).c_str());
6771 }
6772
6773 VDDestroy(hdd);
6774 }
6775 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
6776
6777 return hrc;
6778}
6779
6780/**
6781 *
6782 * @note Similar code exists in i_taskExportHandler.
6783 */
6784HRESULT Medium::i_addRawToFss(const char *aFilename, SecretKeyStore *pKeyStore, RTVFSFSSTREAM hVfsFssDst,
6785 const ComObjPtr<Progress> &aProgress, bool fSparse)
6786{
6787 AutoCaller autoCaller(this);
6788 HRESULT hrc = autoCaller.hrc();
6789 if (SUCCEEDED(hrc))
6790 {
6791 /*
6792 * Get a readonly hdd for this medium.
6793 */
6794 MediumCryptoFilterSettings CryptoSettingsRead;
6795 MediumLockList SourceMediumLockList;
6796 PVDISK pHdd;
6797 hrc = i_openForIO(false /*fWritable*/, pKeyStore, &pHdd, &SourceMediumLockList, &CryptoSettingsRead);
6798 if (SUCCEEDED(hrc))
6799 {
6800 /*
6801 * Create a VFS file interface to the HDD and attach a progress wrapper
6802 * that monitors the progress reading of the raw image. The image will
6803 * be read twice if hVfsFssDst does sparse processing.
6804 */
6805 RTVFSFILE hVfsFileDisk = NIL_RTVFSFILE;
6806 int vrc = VDCreateVfsFileFromDisk(pHdd, 0 /*fFlags*/, &hVfsFileDisk);
6807 if (RT_SUCCESS(vrc))
6808 {
6809 RTVFSFILE hVfsFileProgress = NIL_RTVFSFILE;
6810 vrc = RTVfsCreateProgressForFile(hVfsFileDisk, aProgress->i_iprtProgressCallback, &*aProgress,
6811 RTVFSPROGRESS_F_CANCELABLE | RTVFSPROGRESS_F_FORWARD_SEEK_AS_READ,
6812 VDGetSize(pHdd, VD_LAST_IMAGE) * (fSparse ? 2 : 1) /*cbExpectedRead*/,
6813 0 /*cbExpectedWritten*/, &hVfsFileProgress);
6814 RTVfsFileRelease(hVfsFileDisk);
6815 if (RT_SUCCESS(vrc))
6816 {
6817 RTVFSOBJ hVfsObj = RTVfsObjFromFile(hVfsFileProgress);
6818 RTVfsFileRelease(hVfsFileProgress);
6819
6820 vrc = RTVfsFsStrmAdd(hVfsFssDst, aFilename, hVfsObj, 0 /*fFlags*/);
6821 RTVfsObjRelease(hVfsObj);
6822 if (RT_FAILURE(vrc))
6823 hrc = setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Failed to add '%s' to output (%Rrc)"), aFilename, vrc);
6824 }
6825 else
6826 hrc = setErrorBoth(VBOX_E_FILE_ERROR, vrc,
6827 tr("RTVfsCreateProgressForFile failed when processing '%s' (%Rrc)"), aFilename, vrc);
6828 }
6829 else
6830 hrc = setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("VDCreateVfsFileFromDisk failed for '%s' (%Rrc)"), aFilename, vrc);
6831 VDDestroy(pHdd);
6832 }
6833 }
6834 return hrc;
6835}
6836
6837/**
6838 * Used by IAppliance to export disk images.
6839 *
6840 * @param aFilename Filename to create (UTF8).
6841 * @param aFormat Medium format for creating @a aFilename.
6842 * @param aVariant Which exact image format variant to use for the
6843 * destination image.
6844 * @param pKeyStore The optional key store for decrypting the data for
6845 * encrypted media during the export.
6846 * @param hVfsIosDst The destination I/O stream object.
6847 * @param aProgress Progress object to use.
6848 * @return
6849 *
6850 * @note The source format is defined by the Medium instance.
6851 */
6852HRESULT Medium::i_exportFile(const char *aFilename,
6853 const ComObjPtr<MediumFormat> &aFormat,
6854 MediumVariant_T aVariant,
6855 SecretKeyStore *pKeyStore,
6856 RTVFSIOSTREAM hVfsIosDst,
6857 const ComObjPtr<Progress> &aProgress)
6858{
6859 AssertPtrReturn(aFilename, E_INVALIDARG);
6860 AssertReturn(aFormat.isNotNull(), E_INVALIDARG);
6861 AssertReturn(aProgress.isNotNull(), E_INVALIDARG);
6862
6863 AutoCaller autoCaller(this);
6864 HRESULT hrc = autoCaller.hrc();
6865 if (SUCCEEDED(hrc))
6866 {
6867 /*
6868 * Setup VD interfaces.
6869 */
6870 PVDINTERFACE pVDImageIfaces = m->vdImageIfaces;
6871 PVDINTERFACEIO pVfsIoIf;
6872 int vrc = VDIfCreateFromVfsStream(hVfsIosDst, RTFILE_O_WRITE, &pVfsIoIf);
6873 if (RT_SUCCESS(vrc))
6874 {
6875 vrc = VDInterfaceAdd(&pVfsIoIf->Core, "Medium::ExportTaskVfsIos", VDINTERFACETYPE_IO,
6876 pVfsIoIf, sizeof(VDINTERFACEIO), &pVDImageIfaces);
6877 if (RT_SUCCESS(vrc))
6878 {
6879 /*
6880 * Get a readonly hdd for this medium (source).
6881 */
6882 MediumCryptoFilterSettings CryptoSettingsRead;
6883 MediumLockList SourceMediumLockList;
6884 PVDISK pSrcHdd;
6885 hrc = i_openForIO(false /*fWritable*/, pKeyStore, &pSrcHdd, &SourceMediumLockList, &CryptoSettingsRead);
6886 if (SUCCEEDED(hrc))
6887 {
6888 /*
6889 * Create the target medium.
6890 */
6891 Utf8Str strDstFormat(aFormat->i_getId());
6892
6893 /* ensure the target directory exists */
6894 uint64_t fDstCapabilities = aFormat->i_getCapabilities();
6895 if (fDstCapabilities & MediumFormatCapabilities_File)
6896 {
6897 Utf8Str strDstLocation(aFilename);
6898 hrc = VirtualBox::i_ensureFilePathExists(strDstLocation.c_str(),
6899 !(aVariant & MediumVariant_NoCreateDir) /* fCreate */);
6900 }
6901 if (SUCCEEDED(hrc))
6902 {
6903 PVDISK pDstHdd;
6904 vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &pDstHdd);
6905 if (RT_SUCCESS(vrc))
6906 {
6907 /*
6908 * Create an interface for getting progress callbacks.
6909 */
6910 VDINTERFACEPROGRESS ProgressIf = VDINTERFACEPROGRESS_INITALIZER(aProgress->i_vdProgressCallback);
6911 PVDINTERFACE pProgress = NULL;
6912 vrc = VDInterfaceAdd(&ProgressIf.Core, "export-progress", VDINTERFACETYPE_PROGRESS,
6913 &*aProgress, sizeof(ProgressIf), &pProgress);
6914 AssertRC(vrc);
6915
6916 /*
6917 * Do the exporting.
6918 */
6919 vrc = VDCopy(pSrcHdd,
6920 VD_LAST_IMAGE,
6921 pDstHdd,
6922 strDstFormat.c_str(),
6923 aFilename,
6924 false /* fMoveByRename */,
6925 0 /* cbSize */,
6926 aVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
6927 NULL /* pDstUuid */,
6928 VD_OPEN_FLAGS_NORMAL | VD_OPEN_FLAGS_SEQUENTIAL,
6929 pProgress,
6930 pVDImageIfaces,
6931 NULL);
6932 if (RT_SUCCESS(vrc))
6933 hrc = S_OK;
6934 else
6935 hrc = setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Could not create the exported medium '%s'%s"),
6936 aFilename, i_vdError(vrc).c_str());
6937 VDDestroy(pDstHdd);
6938 }
6939 else
6940 hrc = setErrorVrc(vrc);
6941 }
6942 }
6943 VDDestroy(pSrcHdd);
6944 }
6945 else
6946 hrc = setErrorVrc(vrc, "VDInterfaceAdd -> %Rrc", vrc);
6947 VDIfDestroyFromVfsStream(pVfsIoIf);
6948 }
6949 else
6950 hrc = setErrorVrc(vrc, "VDIfCreateFromVfsStream -> %Rrc", vrc);
6951 }
6952 return hrc;
6953}
6954
6955/**
6956 * Used by IAppliance to import disk images.
6957 *
6958 * @param aFilename Filename to read (UTF8).
6959 * @param aFormat Medium format for reading @a aFilename.
6960 * @param aVariant Which exact image format variant to use
6961 * for the destination image.
6962 * @param aVfsIosSrc Handle to the source I/O stream.
6963 * @param aParent Parent medium. May be NULL.
6964 * @param aProgress Progress object to use.
6965 * @param aNotify Notify about media for which metadata is changed
6966 * during execution of the function.
6967 * @return
6968 * @note The destination format is defined by the Medium instance.
6969 *
6970 * @todo The only consumer of this method (Appliance::i_importOneDiskImage) is
6971 * already on a worker thread, so perhaps consider bypassing the thread
6972 * here and run in the task synchronously? VBoxSVC has enough threads as
6973 * it is...
6974 */
6975HRESULT Medium::i_importFile(const char *aFilename,
6976 const ComObjPtr<MediumFormat> &aFormat,
6977 MediumVariant_T aVariant,
6978 RTVFSIOSTREAM aVfsIosSrc,
6979 const ComObjPtr<Medium> &aParent,
6980 const ComObjPtr<Progress> &aProgress,
6981 bool aNotify)
6982{
6983 /** @todo r=klaus The code below needs to be double checked with regard
6984 * to lock order violations, it probably causes lock order issues related
6985 * to the AutoCaller usage. */
6986 AssertPtrReturn(aFilename, E_INVALIDARG);
6987 AssertReturn(!aFormat.isNull(), E_INVALIDARG);
6988 AssertReturn(!aProgress.isNull(), E_INVALIDARG);
6989
6990 AutoCaller autoCaller(this);
6991 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
6992
6993 HRESULT hrc = S_OK;
6994 Medium::Task *pTask = NULL;
6995
6996 try
6997 {
6998 // locking: we need the tree lock first because we access parent pointers
6999 // and we need to write-lock the media involved
7000 uint32_t cHandles = 2;
7001 LockHandle* pHandles[3] = { &m->pVirtualBox->i_getMediaTreeLockHandle(),
7002 this->lockHandle() };
7003 /* Only add parent to the lock if it is not null */
7004 if (!aParent.isNull())
7005 pHandles[cHandles++] = aParent->lockHandle();
7006 AutoWriteLock alock(cHandles,
7007 pHandles
7008 COMMA_LOCKVAL_SRC_POS);
7009
7010 if ( m->state != MediumState_NotCreated
7011 && m->state != MediumState_Created)
7012 throw i_setStateError();
7013
7014 /* Build the target lock list. */
7015 MediumLockList *pTargetMediumLockList(new MediumLockList());
7016 alock.release();
7017 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
7018 this /* pToLockWrite */,
7019 false /* fMediumLockWriteAll */,
7020 aParent,
7021 *pTargetMediumLockList);
7022 alock.acquire();
7023 if (FAILED(hrc))
7024 {
7025 delete pTargetMediumLockList;
7026 throw hrc;
7027 }
7028
7029 alock.release();
7030 hrc = pTargetMediumLockList->Lock();
7031 alock.acquire();
7032 if (FAILED(hrc))
7033 {
7034 delete pTargetMediumLockList;
7035 throw setError(hrc,
7036 tr("Failed to lock target media '%s'"),
7037 i_getLocationFull().c_str());
7038 }
7039
7040 /* setup task object to carry out the operation asynchronously */
7041 pTask = new Medium::ImportTask(this, aProgress, aFilename, aFormat, aVariant,
7042 aVfsIosSrc, aParent, pTargetMediumLockList, false, aNotify);
7043 hrc = pTask->hrc();
7044 AssertComRC(hrc);
7045 if (FAILED(hrc))
7046 throw hrc;
7047
7048 if (m->state == MediumState_NotCreated)
7049 m->state = MediumState_Creating;
7050 }
7051 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
7052
7053 if (SUCCEEDED(hrc))
7054 {
7055 hrc = pTask->createThread();
7056 pTask = NULL;
7057 }
7058 else if (pTask != NULL)
7059 delete pTask;
7060
7061 return hrc;
7062}
7063
7064/**
7065 * Internal version of the public CloneTo API which allows to enable certain
7066 * optimizations to improve speed during VM cloning.
7067 *
7068 * @param aTarget Target medium
7069 * @param aVariant Which exact image format variant to use
7070 * for the destination image.
7071 * @param aParent Parent medium. May be NULL.
7072 * @param aProgress Progress object to use.
7073 * @param idxSrcImageSame The last image in the source chain which has the
7074 * same content as the given image in the destination
7075 * chain. Use UINT32_MAX to disable this optimization.
7076 * @param idxDstImageSame The last image in the destination chain which has the
7077 * same content as the given image in the source chain.
7078 * Use UINT32_MAX to disable this optimization.
7079 * @param aNotify Notify about media for which metadata is changed
7080 * during execution of the function.
7081 * @return
7082 */
7083HRESULT Medium::i_cloneToEx(const ComObjPtr<Medium> &aTarget, MediumVariant_T aVariant,
7084 const ComObjPtr<Medium> &aParent, IProgress **aProgress,
7085 uint32_t idxSrcImageSame, uint32_t idxDstImageSame, bool aNotify)
7086{
7087 /** @todo r=klaus The code below needs to be double checked with regard
7088 * to lock order violations, it probably causes lock order issues related
7089 * to the AutoCaller usage. */
7090 CheckComArgNotNull(aTarget);
7091 CheckComArgOutPointerValid(aProgress);
7092 ComAssertRet(aTarget != this, E_INVALIDARG);
7093
7094 AutoCaller autoCaller(this);
7095 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
7096
7097 HRESULT hrc = S_OK;
7098 ComObjPtr<Progress> pProgress;
7099 Medium::Task *pTask = NULL;
7100
7101 try
7102 {
7103 // locking: we need the tree lock first because we access parent pointers
7104 // and we need to write-lock the media involved
7105 uint32_t cHandles = 3;
7106 LockHandle* pHandles[4] = { &m->pVirtualBox->i_getMediaTreeLockHandle(),
7107 this->lockHandle(),
7108 aTarget->lockHandle() };
7109 /* Only add parent to the lock if it is not null */
7110 if (!aParent.isNull())
7111 pHandles[cHandles++] = aParent->lockHandle();
7112 AutoWriteLock alock(cHandles,
7113 pHandles
7114 COMMA_LOCKVAL_SRC_POS);
7115
7116 if ( aTarget->m->state != MediumState_NotCreated
7117 && aTarget->m->state != MediumState_Created)
7118 throw aTarget->i_setStateError();
7119
7120 /* Build the source lock list. */
7121 MediumLockList *pSourceMediumLockList(new MediumLockList());
7122 alock.release();
7123 hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
7124 NULL /* pToLockWrite */,
7125 false /* fMediumLockWriteAll */,
7126 NULL,
7127 *pSourceMediumLockList);
7128 alock.acquire();
7129 if (FAILED(hrc))
7130 {
7131 delete pSourceMediumLockList;
7132 throw hrc;
7133 }
7134
7135 /* Build the target lock list (including the to-be parent chain). */
7136 MediumLockList *pTargetMediumLockList(new MediumLockList());
7137 alock.release();
7138 hrc = aTarget->i_createMediumLockList(true /* fFailIfInaccessible */,
7139 aTarget /* pToLockWrite */,
7140 false /* fMediumLockWriteAll */,
7141 aParent,
7142 *pTargetMediumLockList);
7143 alock.acquire();
7144 if (FAILED(hrc))
7145 {
7146 delete pSourceMediumLockList;
7147 delete pTargetMediumLockList;
7148 throw hrc;
7149 }
7150
7151 alock.release();
7152 hrc = pSourceMediumLockList->Lock();
7153 alock.acquire();
7154 if (FAILED(hrc))
7155 {
7156 delete pSourceMediumLockList;
7157 delete pTargetMediumLockList;
7158 throw setError(hrc,
7159 tr("Failed to lock source media '%s'"),
7160 i_getLocationFull().c_str());
7161 }
7162 alock.release();
7163 hrc = pTargetMediumLockList->Lock();
7164 alock.acquire();
7165 if (FAILED(hrc))
7166 {
7167 delete pSourceMediumLockList;
7168 delete pTargetMediumLockList;
7169 throw setError(hrc,
7170 tr("Failed to lock target media '%s'"),
7171 aTarget->i_getLocationFull().c_str());
7172 }
7173
7174 pProgress.createObject();
7175 hrc = pProgress->init(m->pVirtualBox,
7176 static_cast <IMedium *>(this),
7177 BstrFmt(tr("Creating clone medium '%s'"), aTarget->m->strLocationFull.c_str()).raw(),
7178 TRUE /* aCancelable */);
7179 if (FAILED(hrc))
7180 {
7181 delete pSourceMediumLockList;
7182 delete pTargetMediumLockList;
7183 throw hrc;
7184 }
7185
7186 /* setup task object to carry out the operation asynchronously */
7187 pTask = new Medium::CloneTask(this, pProgress, aTarget, aVariant,
7188 aParent, idxSrcImageSame,
7189 idxDstImageSame, pSourceMediumLockList,
7190 pTargetMediumLockList, false, false, aNotify);
7191 hrc = pTask->hrc();
7192 AssertComRC(hrc);
7193 if (FAILED(hrc))
7194 throw hrc;
7195
7196 if (aTarget->m->state == MediumState_NotCreated)
7197 aTarget->m->state = MediumState_Creating;
7198 }
7199 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
7200
7201 if (SUCCEEDED(hrc))
7202 {
7203 hrc = pTask->createThread();
7204 pTask = NULL;
7205 if (SUCCEEDED(hrc))
7206 pProgress.queryInterfaceTo(aProgress);
7207 }
7208 else if (pTask != NULL)
7209 delete pTask;
7210
7211 return hrc;
7212}
7213
7214/**
7215 * Returns the key identifier for this medium if encryption is configured.
7216 *
7217 * @returns Key identifier or empty string if no encryption is configured.
7218 */
7219const Utf8Str& Medium::i_getKeyId()
7220{
7221 ComObjPtr<Medium> pBase = i_getBase();
7222
7223 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7224
7225 settings::StringsMap::const_iterator it = pBase->m->mapProperties.find("CRYPT/KeyId");
7226 if (it == pBase->m->mapProperties.end())
7227 return Utf8Str::Empty;
7228
7229 return it->second;
7230}
7231
7232
7233/**
7234 * Returns all filter related properties.
7235 *
7236 * @returns COM status code.
7237 * @param aReturnNames Where to store the properties names on success.
7238 * @param aReturnValues Where to store the properties values on success.
7239 */
7240HRESULT Medium::i_getFilterProperties(std::vector<com::Utf8Str> &aReturnNames,
7241 std::vector<com::Utf8Str> &aReturnValues)
7242{
7243 std::vector<com::Utf8Str> aPropNames;
7244 std::vector<com::Utf8Str> aPropValues;
7245 HRESULT hrc = getProperties(Utf8Str(""), aPropNames, aPropValues);
7246
7247 if (SUCCEEDED(hrc))
7248 {
7249 unsigned cReturnSize = 0;
7250 aReturnNames.resize(0);
7251 aReturnValues.resize(0);
7252 for (unsigned idx = 0; idx < aPropNames.size(); idx++)
7253 {
7254 if (i_isPropertyForFilter(aPropNames[idx]))
7255 {
7256 aReturnNames.resize(cReturnSize + 1);
7257 aReturnValues.resize(cReturnSize + 1);
7258 aReturnNames[cReturnSize] = aPropNames[idx];
7259 aReturnValues[cReturnSize] = aPropValues[idx];
7260 cReturnSize++;
7261 }
7262 }
7263 }
7264
7265 return hrc;
7266}
7267
7268/**
7269 * Preparation to move this medium to a new location
7270 *
7271 * @param aLocation Location of the storage unit. If the location is a FS-path,
7272 * then it can be relative to the VirtualBox home directory.
7273 *
7274 * @note Must be called from under this object's write lock.
7275 */
7276HRESULT Medium::i_preparationForMoving(const Utf8Str &aLocation)
7277{
7278 HRESULT hrc = E_FAIL;
7279
7280 if (i_getLocationFull() != aLocation)
7281 {
7282 m->strNewLocationFull = aLocation;
7283 m->fMoveThisMedium = true;
7284 hrc = S_OK;
7285 }
7286
7287 return hrc;
7288}
7289
7290/**
7291 * Checking whether current operation "moving" or not
7292 */
7293bool Medium::i_isMoveOperation(const ComObjPtr<Medium> &aTarget) const
7294{
7295 RT_NOREF(aTarget);
7296 return m->fMoveThisMedium;
7297}
7298
7299bool Medium::i_resetMoveOperationData()
7300{
7301 m->strNewLocationFull.setNull();
7302 m->fMoveThisMedium = false;
7303 return true;
7304}
7305
7306Utf8Str Medium::i_getNewLocationForMoving() const
7307{
7308 if (m->fMoveThisMedium == true)
7309 return m->strNewLocationFull;
7310 else
7311 return Utf8Str();
7312}
7313////////////////////////////////////////////////////////////////////////////////
7314//
7315// Private methods
7316//
7317////////////////////////////////////////////////////////////////////////////////
7318
7319/**
7320 * Queries information from the medium.
7321 *
7322 * As a result of this call, the accessibility state and data members such as
7323 * size and description will be updated with the current information.
7324 *
7325 * @note This method may block during a system I/O call that checks storage
7326 * accessibility.
7327 *
7328 * @note Caller MUST NOT hold the media tree or medium lock.
7329 *
7330 * @note Locks m->pParent for reading. Locks this object for writing.
7331 *
7332 * @param fSetImageId Whether to reset the UUID contained in the image file
7333 * to the UUID in the medium instance data (see SetIDs())
7334 * @param fSetParentId Whether to reset the parent UUID contained in the image
7335 * file to the parent UUID in the medium instance data (see
7336 * SetIDs())
7337 * @param autoCaller
7338 * @return
7339 */
7340HRESULT Medium::i_queryInfo(bool fSetImageId, bool fSetParentId, AutoCaller &autoCaller)
7341{
7342 Assert(!isWriteLockOnCurrentThread());
7343 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7344
7345 if ( ( m->state != MediumState_Created
7346 && m->state != MediumState_Inaccessible
7347 && m->state != MediumState_LockedRead)
7348 || m->fClosing)
7349 return E_FAIL;
7350
7351 HRESULT hrc = S_OK;
7352
7353 int vrc = VINF_SUCCESS;
7354
7355 /* check if a blocking i_queryInfo() call is in progress on some other thread,
7356 * and wait for it to finish if so instead of querying data ourselves */
7357 if (m->queryInfoRunning)
7358 {
7359 Assert( m->state == MediumState_LockedRead
7360 || m->state == MediumState_LockedWrite);
7361
7362 while (m->queryInfoRunning)
7363 {
7364 alock.release();
7365 /* must not hold the object lock now */
7366 Assert(!isWriteLockOnCurrentThread());
7367 {
7368 AutoReadLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
7369 }
7370 alock.acquire();
7371 }
7372
7373 return S_OK;
7374 }
7375
7376 bool success = false;
7377 Utf8Str lastAccessError;
7378
7379 /* are we dealing with a new medium constructed using the existing
7380 * location? */
7381 bool isImport = m->id.isZero();
7382 unsigned uOpenFlags = VD_OPEN_FLAGS_INFO;
7383
7384 /* Note that we don't use VD_OPEN_FLAGS_READONLY when opening new
7385 * media because that would prevent necessary modifications
7386 * when opening media of some third-party formats for the first
7387 * time in VirtualBox (such as VMDK for which VDOpen() needs to
7388 * generate an UUID if it is missing) */
7389 if ( m->hddOpenMode == OpenReadOnly
7390 || m->type == MediumType_Readonly
7391 || (!isImport && !fSetImageId && !fSetParentId)
7392 )
7393 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
7394
7395 /* Open shareable medium with the appropriate flags */
7396 if (m->type == MediumType_Shareable)
7397 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
7398
7399 /* Lock the medium, which makes the behavior much more consistent, must be
7400 * done before dropping the object lock and setting queryInfoRunning. */
7401 ComPtr<IToken> pToken;
7402 if (uOpenFlags & (VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_SHAREABLE))
7403 hrc = LockRead(pToken.asOutParam());
7404 else
7405 hrc = LockWrite(pToken.asOutParam());
7406 if (FAILED(hrc)) return hrc;
7407
7408 /* Copies of the input state fields which are not read-only,
7409 * as we're dropping the lock. CAUTION: be extremely careful what
7410 * you do with the contents of this medium object, as you will
7411 * create races if there are concurrent changes. */
7412 Utf8Str format(m->strFormat);
7413 Utf8Str location(m->strLocationFull);
7414 ComObjPtr<MediumFormat> formatObj = m->formatObj;
7415
7416 /* "Output" values which can't be set because the lock isn't held
7417 * at the time the values are determined. */
7418 Guid mediumId = m->id;
7419 uint64_t mediumSize = 0;
7420 uint64_t mediumLogicalSize = 0;
7421
7422 /* Flag whether a base image has a non-zero parent UUID and thus
7423 * need repairing after it was closed again. */
7424 bool fRepairImageZeroParentUuid = false;
7425
7426 ComObjPtr<VirtualBox> pVirtualBox = m->pVirtualBox;
7427
7428 /* must be set before leaving the object lock the first time */
7429 m->queryInfoRunning = true;
7430
7431 /* must leave object lock now, because a lock from a higher lock class
7432 * is needed and also a lengthy operation is coming */
7433 alock.release();
7434 autoCaller.release();
7435
7436 /* Note that taking the queryInfoSem after leaving the object lock above
7437 * can lead to short spinning of the loops waiting for i_queryInfo() to
7438 * complete. This is unavoidable since the other order causes a lock order
7439 * violation: here it would be requesting the object lock (at the beginning
7440 * of the method), then queryInfoSem, and below the other way round. */
7441 AutoWriteLock qlock(m->queryInfoSem COMMA_LOCKVAL_SRC_POS);
7442
7443 /* take the opportunity to have a media tree lock, released initially */
7444 Assert(!isWriteLockOnCurrentThread());
7445 Assert(!pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
7446 AutoWriteLock treeLock(pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
7447 treeLock.release();
7448
7449 /* re-take the caller, but not the object lock, to keep uninit away */
7450 autoCaller.add();
7451 if (FAILED(autoCaller.hrc()))
7452 {
7453 m->queryInfoRunning = false;
7454 return autoCaller.hrc();
7455 }
7456
7457 try
7458 {
7459 /* skip accessibility checks for host drives */
7460 if (m->hostDrive)
7461 {
7462 success = true;
7463 throw S_OK;
7464 }
7465
7466 PVDISK hdd;
7467 vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
7468 ComAssertRCThrow(vrc, E_FAIL);
7469
7470 try
7471 {
7472 /** @todo This kind of opening of media is assuming that diff
7473 * media can be opened as base media. Should be documented that
7474 * it must work for all medium format backends. */
7475 vrc = VDOpen(hdd,
7476 format.c_str(),
7477 location.c_str(),
7478 uOpenFlags | m->uOpenFlagsDef,
7479 m->vdImageIfaces);
7480 if (RT_FAILURE(vrc))
7481 {
7482 lastAccessError = Utf8StrFmt(tr("Could not open the medium '%s'%s"),
7483 location.c_str(), i_vdError(vrc).c_str());
7484 throw S_OK;
7485 }
7486
7487 if (formatObj->i_getCapabilities() & MediumFormatCapabilities_Uuid)
7488 {
7489 /* Modify the UUIDs if necessary. The associated fields are
7490 * not modified by other code, so no need to copy. */
7491 if (fSetImageId)
7492 {
7493 alock.acquire();
7494 vrc = VDSetUuid(hdd, 0, m->uuidImage.raw());
7495 alock.release();
7496 if (RT_FAILURE(vrc))
7497 {
7498 lastAccessError = Utf8StrFmt(tr("Could not update the UUID of medium '%s'%s"),
7499 location.c_str(), i_vdError(vrc).c_str());
7500 throw S_OK;
7501 }
7502 mediumId = m->uuidImage;
7503 }
7504 if (fSetParentId)
7505 {
7506 alock.acquire();
7507 vrc = VDSetParentUuid(hdd, 0, m->uuidParentImage.raw());
7508 alock.release();
7509 if (RT_FAILURE(vrc))
7510 {
7511 lastAccessError = Utf8StrFmt(tr("Could not update the parent UUID of medium '%s'%s"),
7512 location.c_str(), i_vdError(vrc).c_str());
7513 throw S_OK;
7514 }
7515 }
7516 /* zap the information, these are no long-term members */
7517 alock.acquire();
7518 unconst(m->uuidImage).clear();
7519 unconst(m->uuidParentImage).clear();
7520 alock.release();
7521
7522 /* check the UUID */
7523 RTUUID uuid;
7524 vrc = VDGetUuid(hdd, 0, &uuid);
7525 ComAssertRCThrow(vrc, E_FAIL);
7526
7527 if (isImport)
7528 {
7529 mediumId = uuid;
7530
7531 if (mediumId.isZero() && (m->hddOpenMode == OpenReadOnly))
7532 // only when importing a VDMK that has no UUID, create one in memory
7533 mediumId.create();
7534 }
7535 else
7536 {
7537 Assert(!mediumId.isZero());
7538
7539 if (mediumId != uuid)
7540 {
7541 /** @todo r=klaus this always refers to VirtualBox.xml as the medium registry, even for new VMs */
7542 lastAccessError = Utf8StrFmt(
7543 tr("UUID {%RTuuid} of the medium '%s' does not match the value {%RTuuid} stored in the media registry ('%s')"),
7544 &uuid,
7545 location.c_str(),
7546 mediumId.raw(),
7547 pVirtualBox->i_settingsFilePath().c_str());
7548 throw S_OK;
7549 }
7550 }
7551 }
7552 else
7553 {
7554 /* the backend does not support storing UUIDs within the
7555 * underlying storage so use what we store in XML */
7556
7557 if (fSetImageId)
7558 {
7559 /* set the UUID if an API client wants to change it */
7560 alock.acquire();
7561 mediumId = m->uuidImage;
7562 alock.release();
7563 }
7564 else if (isImport)
7565 {
7566 /* generate an UUID for an imported UUID-less medium */
7567 mediumId.create();
7568 }
7569 }
7570
7571 /* set the image uuid before the below parent uuid handling code
7572 * might place it somewhere in the media tree, so that the medium
7573 * UUID is valid at this point */
7574 alock.acquire();
7575 if (isImport || fSetImageId)
7576 unconst(m->id) = mediumId;
7577 alock.release();
7578
7579 /* get the medium variant */
7580 unsigned uImageFlags;
7581 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
7582 ComAssertRCThrow(vrc, E_FAIL);
7583 alock.acquire();
7584 m->variant = (MediumVariant_T)uImageFlags;
7585 alock.release();
7586
7587 /* check/get the parent uuid and update corresponding state */
7588 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
7589 {
7590 RTUUID parentId;
7591 vrc = VDGetParentUuid(hdd, 0, &parentId);
7592 ComAssertRCThrow(vrc, E_FAIL);
7593
7594 /* streamOptimized VMDK images are only accepted as base
7595 * images, as this allows automatic repair of OVF appliances.
7596 * Since such images don't support random writes they will not
7597 * be created for diff images. Only an overly smart user might
7598 * manually create this case. Too bad for him. */
7599 if ( (isImport || fSetParentId)
7600 && !(uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED))
7601 {
7602 /* the parent must be known to us. Note that we freely
7603 * call locking methods of mVirtualBox and parent, as all
7604 * relevant locks must be already held. There may be no
7605 * concurrent access to the just opened medium on other
7606 * threads yet (and init() will fail if this method reports
7607 * MediumState_Inaccessible) */
7608
7609 ComObjPtr<Medium> pParent;
7610 if (RTUuidIsNull(&parentId))
7611 hrc = VBOX_E_OBJECT_NOT_FOUND;
7612 else
7613 hrc = pVirtualBox->i_findHardDiskById(Guid(parentId), false /* aSetError */, &pParent);
7614 if (FAILED(hrc))
7615 {
7616 if (fSetImageId && !fSetParentId)
7617 {
7618 /* If the image UUID gets changed for an existing
7619 * image then the parent UUID can be stale. In such
7620 * cases clear the parent information. The parent
7621 * information may/will be re-set later if the
7622 * API client wants to adjust a complete medium
7623 * hierarchy one by one. */
7624 hrc = S_OK;
7625 alock.acquire();
7626 RTUuidClear(&parentId);
7627 vrc = VDSetParentUuid(hdd, 0, &parentId);
7628 alock.release();
7629 ComAssertRCThrow(vrc, E_FAIL);
7630 }
7631 else
7632 {
7633 lastAccessError = Utf8StrFmt(tr("Parent medium with UUID {%RTuuid} of the medium '%s' is not found in the media registry ('%s')"),
7634 &parentId, location.c_str(),
7635 pVirtualBox->i_settingsFilePath().c_str());
7636 throw S_OK;
7637 }
7638 }
7639
7640 /* must drop the caller before taking the tree lock */
7641 autoCaller.release();
7642 /* we set m->pParent & children() */
7643 treeLock.acquire();
7644 autoCaller.add();
7645 if (FAILED(autoCaller.hrc()))
7646 throw autoCaller.hrc();
7647
7648 if (m->pParent)
7649 i_deparent();
7650
7651 if (!pParent.isNull())
7652 if (pParent->i_getDepth() >= SETTINGS_MEDIUM_DEPTH_MAX)
7653 {
7654 AutoReadLock plock(pParent COMMA_LOCKVAL_SRC_POS);
7655 throw setError(VBOX_E_INVALID_OBJECT_STATE,
7656 tr("Cannot open differencing image for medium '%s', because it exceeds the medium tree depth limit. Please merge some images which you no longer need"),
7657 pParent->m->strLocationFull.c_str());
7658 }
7659 i_setParent(pParent);
7660
7661 treeLock.release();
7662 }
7663 else
7664 {
7665 /* must drop the caller before taking the tree lock */
7666 autoCaller.release();
7667 /* we access m->pParent */
7668 treeLock.acquire();
7669 autoCaller.add();
7670 if (FAILED(autoCaller.hrc()))
7671 throw autoCaller.hrc();
7672
7673 /* check that parent UUIDs match. Note that there's no need
7674 * for the parent's AutoCaller (our lifetime is bound to
7675 * it) */
7676
7677 if (m->pParent.isNull())
7678 {
7679 /* Due to a bug in VDCopy() in VirtualBox 3.0.0-3.0.14
7680 * and 3.1.0-3.1.8 there are base images out there
7681 * which have a non-zero parent UUID. No point in
7682 * complaining about them, instead automatically
7683 * repair the problem. Later we can bring back the
7684 * error message, but we should wait until really
7685 * most users have repaired their images, either with
7686 * VBoxFixHdd or this way. */
7687#if 1
7688 fRepairImageZeroParentUuid = true;
7689#else /* 0 */
7690 lastAccessError = Utf8StrFmt(
7691 tr("Medium type of '%s' is differencing but it is not associated with any parent medium in the media registry ('%s')"),
7692 location.c_str(),
7693 pVirtualBox->settingsFilePath().c_str());
7694 treeLock.release();
7695 throw S_OK;
7696#endif /* 0 */
7697 }
7698
7699 {
7700 autoCaller.release();
7701 AutoReadLock parentLock(m->pParent COMMA_LOCKVAL_SRC_POS);
7702 autoCaller.add();
7703 if (FAILED(autoCaller.hrc()))
7704 throw autoCaller.hrc();
7705
7706 if ( !fRepairImageZeroParentUuid
7707 && m->pParent->i_getState() != MediumState_Inaccessible
7708 && m->pParent->i_getId() != parentId)
7709 {
7710 /** @todo r=klaus this always refers to VirtualBox.xml as the medium registry, even for new VMs */
7711 lastAccessError = Utf8StrFmt(
7712 tr("Parent UUID {%RTuuid} of the medium '%s' does not match UUID {%RTuuid} of its parent medium stored in the media registry ('%s')"),
7713 &parentId, location.c_str(),
7714 m->pParent->i_getId().raw(),
7715 pVirtualBox->i_settingsFilePath().c_str());
7716 parentLock.release();
7717 treeLock.release();
7718 throw S_OK;
7719 }
7720 }
7721
7722 /// @todo NEWMEDIA what to do if the parent is not
7723 /// accessible while the diff is? Probably nothing. The
7724 /// real code will detect the mismatch anyway.
7725
7726 treeLock.release();
7727 }
7728 }
7729
7730 mediumSize = VDGetFileSize(hdd, 0);
7731 mediumLogicalSize = VDGetSize(hdd, 0);
7732
7733 success = true;
7734 }
7735 catch (HRESULT hrcXcpt)
7736 {
7737 hrc = hrcXcpt;
7738 }
7739
7740 vrc = VDDestroy(hdd);
7741 if (RT_FAILURE(vrc))
7742 {
7743 lastAccessError.printf(tr("Could not update and close the medium '%s'%s"),
7744 location.c_str(), i_vdError(vrc).c_str());
7745 success = false;
7746 throw S_OK;
7747 }
7748 }
7749 catch (HRESULT hrcXcpt)
7750 {
7751 hrc = hrcXcpt;
7752 }
7753
7754 autoCaller.release();
7755 treeLock.acquire();
7756 autoCaller.add();
7757 if (FAILED(autoCaller.hrc()))
7758 {
7759 m->queryInfoRunning = false;
7760 return autoCaller.hrc();
7761 }
7762 alock.acquire();
7763
7764 if (success)
7765 {
7766 m->size = mediumSize;
7767 m->logicalSize = mediumLogicalSize;
7768 m->strLastAccessError.setNull();
7769 }
7770 else
7771 {
7772 m->strLastAccessError = lastAccessError;
7773 Log1WarningFunc(("'%s' is not accessible (error='%s', hrc=%Rhrc, vrc=%Rrc)\n",
7774 location.c_str(), m->strLastAccessError.c_str(), hrc, vrc));
7775 }
7776
7777 /* Set the proper state according to the result of the check */
7778 if (success)
7779 m->preLockState = MediumState_Created;
7780 else
7781 m->preLockState = MediumState_Inaccessible;
7782
7783 /* unblock anyone waiting for the i_queryInfo results */
7784 qlock.release();
7785 m->queryInfoRunning = false;
7786
7787 pToken->Abandon();
7788 pToken.setNull();
7789
7790 if (FAILED(hrc))
7791 return hrc;
7792
7793 /* If this is a base image which incorrectly has a parent UUID set,
7794 * repair the image now by zeroing the parent UUID. This is only done
7795 * when we have structural information from a config file, on import
7796 * this is not possible. If someone would accidentally call openMedium
7797 * with a diff image before the base is registered this would destroy
7798 * the diff. Not acceptable. */
7799 do
7800 {
7801 if (fRepairImageZeroParentUuid)
7802 {
7803 hrc = LockWrite(pToken.asOutParam());
7804 if (FAILED(hrc))
7805 break;
7806
7807 alock.release();
7808
7809 try
7810 {
7811 PVDISK hdd;
7812 vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
7813 ComAssertRCThrow(vrc, E_FAIL);
7814
7815 try
7816 {
7817 vrc = VDOpen(hdd,
7818 format.c_str(),
7819 location.c_str(),
7820 (uOpenFlags & ~VD_OPEN_FLAGS_READONLY) | m->uOpenFlagsDef,
7821 m->vdImageIfaces);
7822 if (RT_FAILURE(vrc))
7823 throw S_OK;
7824
7825 RTUUID zeroParentUuid;
7826 RTUuidClear(&zeroParentUuid);
7827 vrc = VDSetParentUuid(hdd, 0, &zeroParentUuid);
7828 ComAssertRCThrow(vrc, E_FAIL);
7829 }
7830 catch (HRESULT hrcXcpt)
7831 {
7832 hrc = hrcXcpt;
7833 }
7834
7835 VDDestroy(hdd);
7836 }
7837 catch (HRESULT hrcXcpt)
7838 {
7839 hrc = hrcXcpt;
7840 }
7841
7842 pToken->Abandon();
7843 pToken.setNull();
7844 if (FAILED(hrc))
7845 break;
7846 }
7847 } while(0);
7848
7849 return hrc;
7850}
7851
7852/**
7853 * Performs extra checks if the medium can be closed and returns S_OK in
7854 * this case. Otherwise, returns a respective error message. Called by
7855 * Close() under the medium tree lock and the medium lock.
7856 *
7857 * @note Also reused by Medium::Reset().
7858 *
7859 * @note Caller must hold the media tree write lock!
7860 */
7861HRESULT Medium::i_canClose()
7862{
7863 Assert(m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
7864
7865 if (i_getChildren().size() != 0)
7866 return setError(VBOX_E_OBJECT_IN_USE,
7867 tr("Cannot close medium '%s' because it has %d child media", "", i_getChildren().size()),
7868 m->strLocationFull.c_str(), i_getChildren().size());
7869
7870 return S_OK;
7871}
7872
7873/**
7874 * Unregisters this medium with mVirtualBox. Called by close() under the medium tree lock.
7875 *
7876 * @note Caller must have locked the media tree lock for writing!
7877 */
7878HRESULT Medium::i_unregisterWithVirtualBox()
7879{
7880 /* Note that we need to de-associate ourselves from the parent to let
7881 * VirtualBox::i_unregisterMedium() properly save the registry */
7882
7883 /* we modify m->pParent and access children */
7884 Assert(m->pVirtualBox->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
7885
7886 Medium *pParentBackup = m->pParent;
7887 AssertReturn(i_getChildren().size() == 0, E_FAIL);
7888 if (m->pParent)
7889 i_deparent();
7890
7891 HRESULT hrc = m->pVirtualBox->i_unregisterMedium(this);
7892 if (FAILED(hrc))
7893 {
7894 if (pParentBackup)
7895 {
7896 // re-associate with the parent as we are still relatives in the registry
7897 i_setParent(pParentBackup);
7898 }
7899 }
7900
7901 return hrc;
7902}
7903
7904/**
7905 * Like SetProperty but do not trigger a settings store. Only for internal use!
7906 */
7907HRESULT Medium::i_setPropertyDirect(const Utf8Str &aName, const Utf8Str &aValue)
7908{
7909 AutoCaller autoCaller(this);
7910 if (FAILED(autoCaller.hrc())) return autoCaller.hrc();
7911
7912 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
7913
7914 switch (m->state)
7915 {
7916 case MediumState_Created:
7917 case MediumState_Inaccessible:
7918 break;
7919 default:
7920 return i_setStateError();
7921 }
7922
7923 m->mapProperties[aName] = aValue;
7924
7925 return S_OK;
7926}
7927
7928/**
7929 * Sets the extended error info according to the current media state.
7930 *
7931 * @note Must be called from under this object's write or read lock.
7932 */
7933HRESULT Medium::i_setStateError()
7934{
7935 HRESULT hrc;
7936
7937 switch (m->state)
7938 {
7939 case MediumState_NotCreated:
7940 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7941 tr("Storage for the medium '%s' is not created"),
7942 m->strLocationFull.c_str());
7943 break;
7944 case MediumState_Created:
7945 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7946 tr("Storage for the medium '%s' is already created"),
7947 m->strLocationFull.c_str());
7948 break;
7949 case MediumState_LockedRead:
7950 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7951 tr("Medium '%s' is locked for reading by another task"),
7952 m->strLocationFull.c_str());
7953 break;
7954 case MediumState_LockedWrite:
7955 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7956 tr("Medium '%s' is locked for writing by another task"),
7957 m->strLocationFull.c_str());
7958 break;
7959 case MediumState_Inaccessible:
7960 /* be in sync with Console::powerUpThread() */
7961 if (!m->strLastAccessError.isEmpty())
7962 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7963 tr("Medium '%s' is not accessible. %s"),
7964 m->strLocationFull.c_str(), m->strLastAccessError.c_str());
7965 else
7966 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7967 tr("Medium '%s' is not accessible"),
7968 m->strLocationFull.c_str());
7969 break;
7970 case MediumState_Creating:
7971 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7972 tr("Storage for the medium '%s' is being created"),
7973 m->strLocationFull.c_str());
7974 break;
7975 case MediumState_Deleting:
7976 hrc = setError(VBOX_E_INVALID_OBJECT_STATE,
7977 tr("Storage for the medium '%s' is being deleted"),
7978 m->strLocationFull.c_str());
7979 break;
7980 default:
7981 AssertFailed();
7982 hrc = E_FAIL;
7983 break;
7984 }
7985
7986 return hrc;
7987}
7988
7989/**
7990 * Sets the value of m->strLocationFull. The given location must be a fully
7991 * qualified path; relative paths are not supported here.
7992 *
7993 * As a special exception, if the specified location is a file path that ends with '/'
7994 * then the file name part will be generated by this method automatically in the format
7995 * '{\<uuid\>}.\<ext\>' where \<uuid\> is a fresh UUID that this method will generate
7996 * and assign to this medium, and \<ext\> is the default extension for this
7997 * medium's storage format. Note that this procedure requires the media state to
7998 * be NotCreated and will return a failure otherwise.
7999 *
8000 * @param aLocation Location of the storage unit. If the location is a FS-path,
8001 * then it can be relative to the VirtualBox home directory.
8002 * @param aFormat Optional fallback format if it is an import and the format
8003 * cannot be determined.
8004 *
8005 * @note Must be called from under this object's write lock.
8006 */
8007HRESULT Medium::i_setLocation(const Utf8Str &aLocation,
8008 const Utf8Str &aFormat /* = Utf8Str::Empty */)
8009{
8010 AssertReturn(!aLocation.isEmpty(), E_FAIL);
8011
8012 AutoCaller autoCaller(this);
8013 AssertComRCReturnRC(autoCaller.hrc());
8014
8015 /* formatObj may be null only when initializing from an existing path and
8016 * no format is known yet */
8017 AssertReturn( (!m->strFormat.isEmpty() && !m->formatObj.isNull())
8018 || ( getObjectState().getState() == ObjectState::InInit
8019 && m->state != MediumState_NotCreated
8020 && m->id.isZero()
8021 && m->strFormat.isEmpty()
8022 && m->formatObj.isNull()),
8023 E_FAIL);
8024
8025 /* are we dealing with a new medium constructed using the existing
8026 * location? */
8027 bool isImport = m->strFormat.isEmpty();
8028
8029 if ( isImport
8030 || ( (m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File)
8031 && !m->hostDrive))
8032 {
8033 Guid id;
8034
8035 Utf8Str locationFull(aLocation);
8036
8037 if (m->state == MediumState_NotCreated)
8038 {
8039 /* must be a file (formatObj must be already known) */
8040 Assert(m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File);
8041
8042 if (RTPathFilename(aLocation.c_str()) == NULL)
8043 {
8044 /* no file name is given (either an empty string or ends with a
8045 * slash), generate a new UUID + file name if the state allows
8046 * this */
8047
8048 ComAssertMsgRet(!m->formatObj->i_getFileExtensions().empty(),
8049 (tr("Must be at least one extension if it is MediumFormatCapabilities_File\n")),
8050 E_FAIL);
8051
8052 Utf8Str strExt = m->formatObj->i_getFileExtensions().front();
8053 ComAssertMsgRet(!strExt.isEmpty(),
8054 (tr("Default extension must not be empty\n")),
8055 E_FAIL);
8056
8057 id.create();
8058
8059 locationFull = Utf8StrFmt("%s{%RTuuid}.%s",
8060 aLocation.c_str(), id.raw(), strExt.c_str());
8061 }
8062 }
8063
8064 // we must always have full paths now (if it refers to a file)
8065 if ( ( m->formatObj.isNull()
8066 || m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File)
8067 && !RTPathStartsWithRoot(locationFull.c_str()))
8068 return setError(VBOX_E_FILE_ERROR,
8069 tr("The given path '%s' is not fully qualified"),
8070 locationFull.c_str());
8071
8072 /* detect the backend from the storage unit if importing */
8073 if (isImport)
8074 {
8075 VDTYPE const enmDesiredType = i_convertDeviceType();
8076 VDTYPE enmType = VDTYPE_INVALID;
8077 char *backendName = NULL;
8078
8079 /* is it a file? */
8080 RTFILE hFile;
8081 int vrc = RTFileOpen(&hFile, locationFull.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
8082 if (RT_SUCCESS(vrc))
8083 {
8084 RTFileClose(hFile);
8085 vrc = VDGetFormat(NULL /* pVDIfsDisk */, NULL /* pVDIfsImage */,
8086 locationFull.c_str(), enmDesiredType, &backendName, &enmType);
8087 }
8088 else if ( vrc != VERR_FILE_NOT_FOUND
8089 && vrc != VERR_PATH_NOT_FOUND
8090 && vrc != VERR_ACCESS_DENIED
8091 && locationFull != aLocation)
8092 {
8093 /* assume it's not a file, restore the original location */
8094 locationFull = aLocation;
8095 vrc = VDGetFormat(NULL /* pVDIfsDisk */, NULL /* pVDIfsImage */,
8096 locationFull.c_str(), enmDesiredType, &backendName, &enmType);
8097 }
8098
8099 if (RT_FAILURE(vrc))
8100 {
8101 if (vrc == VERR_ACCESS_DENIED)
8102 return setErrorBoth(VBOX_E_FILE_ERROR, vrc,
8103 tr("Permission problem accessing the file for the medium '%s' (%Rrc)"),
8104 locationFull.c_str(), vrc);
8105 if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
8106 return setErrorBoth(VBOX_E_FILE_ERROR, vrc,
8107 tr("Could not find file for the medium '%s' (%Rrc)"),
8108 locationFull.c_str(), vrc);
8109 if (aFormat.isEmpty())
8110 return setErrorBoth(VBOX_E_IPRT_ERROR, vrc,
8111 tr("Could not get the storage format of the medium '%s' (%Rrc)"),
8112 locationFull.c_str(), vrc);
8113 HRESULT hrc = i_setFormat(aFormat);
8114 /* setFormat() must not fail since we've just used the backend so
8115 * the format object must be there */
8116 AssertComRCReturnRC(hrc);
8117 }
8118 else if ( enmType == VDTYPE_INVALID
8119 || m->devType != i_convertToDeviceType(enmType))
8120 {
8121 /*
8122 * The user tried to use a image as a device which is not supported
8123 * by the backend.
8124 */
8125 RTStrFree(backendName);
8126 return setError(E_FAIL,
8127 tr("The medium '%s' can't be used as the requested device type (%s, detected %s)"),
8128 locationFull.c_str(), getDeviceTypeName(m->devType), getVDTypeName(enmType));
8129 }
8130 else
8131 {
8132 ComAssertRet(backendName != NULL && *backendName != '\0', E_FAIL);
8133
8134 HRESULT hrc = i_setFormat(backendName);
8135 RTStrFree(backendName);
8136
8137 /* setFormat() must not fail since we've just used the backend so
8138 * the format object must be there */
8139 AssertComRCReturnRC(hrc);
8140 }
8141 }
8142
8143 m->strLocationFull = locationFull;
8144
8145 /* is it still a file? */
8146 if ( (m->formatObj->i_getCapabilities() & MediumFormatCapabilities_File)
8147 && (m->state == MediumState_NotCreated)
8148 )
8149 /* assign a new UUID (this UUID will be used when calling
8150 * VDCreateBase/VDCreateDiff as a wanted UUID). Note that we
8151 * also do that if we didn't generate it to make sure it is
8152 * either generated by us or reset to null */
8153 unconst(m->id) = id;
8154 }
8155 else
8156 m->strLocationFull = aLocation;
8157
8158 return S_OK;
8159}
8160
8161/**
8162 * Checks that the format ID is valid and sets it on success.
8163 *
8164 * Note that this method will caller-reference the format object on success!
8165 * This reference must be released somewhere to let the MediumFormat object be
8166 * uninitialized.
8167 *
8168 * @note Must be called from under this object's write lock.
8169 */
8170HRESULT Medium::i_setFormat(const Utf8Str &aFormat)
8171{
8172 /* get the format object first */
8173 {
8174 SystemProperties *pSysProps = m->pVirtualBox->i_getSystemProperties();
8175 AutoReadLock propsLock(pSysProps COMMA_LOCKVAL_SRC_POS);
8176
8177 unconst(m->formatObj) = pSysProps->i_mediumFormat(aFormat);
8178 if (m->formatObj.isNull())
8179 return setError(E_INVALIDARG,
8180 tr("Invalid medium storage format '%s'"),
8181 aFormat.c_str());
8182
8183 /* get properties (preinsert them as keys in the map). Note that the
8184 * map doesn't grow over the object life time since the set of
8185 * properties is meant to be constant. */
8186
8187 Assert(m->mapProperties.empty());
8188
8189 for (MediumFormat::PropertyArray::const_iterator it = m->formatObj->i_getProperties().begin();
8190 it != m->formatObj->i_getProperties().end();
8191 ++it)
8192 {
8193 m->mapProperties.insert(std::make_pair(it->strName, Utf8Str::Empty));
8194 }
8195 }
8196
8197 unconst(m->strFormat) = aFormat;
8198
8199 return S_OK;
8200}
8201
8202/**
8203 * Converts the Medium device type to the VD type.
8204 */
8205VDTYPE Medium::i_convertDeviceType()
8206{
8207 VDTYPE enmType;
8208
8209 switch (m->devType)
8210 {
8211 case DeviceType_HardDisk:
8212 enmType = VDTYPE_HDD;
8213 break;
8214 case DeviceType_DVD:
8215 enmType = VDTYPE_OPTICAL_DISC;
8216 break;
8217 case DeviceType_Floppy:
8218 enmType = VDTYPE_FLOPPY;
8219 break;
8220 default:
8221 ComAssertFailedRet(VDTYPE_INVALID);
8222 }
8223
8224 return enmType;
8225}
8226
8227/**
8228 * Converts from the VD type to the medium type.
8229 */
8230DeviceType_T Medium::i_convertToDeviceType(VDTYPE enmType)
8231{
8232 DeviceType_T devType;
8233
8234 switch (enmType)
8235 {
8236 case VDTYPE_HDD:
8237 devType = DeviceType_HardDisk;
8238 break;
8239 case VDTYPE_OPTICAL_DISC:
8240 devType = DeviceType_DVD;
8241 break;
8242 case VDTYPE_FLOPPY:
8243 devType = DeviceType_Floppy;
8244 break;
8245 default:
8246 ComAssertFailedRet(DeviceType_Null);
8247 }
8248
8249 return devType;
8250}
8251
8252/**
8253 * Internal method which checks whether a property name is for a filter plugin.
8254 */
8255bool Medium::i_isPropertyForFilter(const com::Utf8Str &aName)
8256{
8257 /* If the name contains "/" use the part before as a filter name and lookup the filter. */
8258 size_t offSlash;
8259 if ((offSlash = aName.find("/", 0)) != aName.npos)
8260 {
8261 com::Utf8Str strFilter;
8262 com::Utf8Str strKey;
8263
8264 HRESULT hrc = strFilter.assignEx(aName, 0, offSlash);
8265 if (FAILED(hrc))
8266 return false;
8267
8268 hrc = strKey.assignEx(aName, offSlash + 1, aName.length() - offSlash - 1); /* Skip slash */
8269 if (FAILED(hrc))
8270 return false;
8271
8272 VDFILTERINFO FilterInfo;
8273 int vrc = VDFilterInfoOne(strFilter.c_str(), &FilterInfo);
8274 if (RT_SUCCESS(vrc))
8275 {
8276 /* Check that the property exists. */
8277 PCVDCONFIGINFO paConfig = FilterInfo.paConfigInfo;
8278 while (paConfig->pszKey)
8279 {
8280 if (strKey.equals(paConfig->pszKey))
8281 return true;
8282 paConfig++;
8283 }
8284 }
8285 }
8286
8287 return false;
8288}
8289
8290/**
8291 * Returns the last error message collected by the i_vdErrorCall callback and
8292 * resets it.
8293 *
8294 * The error message is returned prepended with a dot and a space, like this:
8295 * <code>
8296 * ". <error_text> (%Rrc)"
8297 * </code>
8298 * to make it easily appendable to a more general error message. The @c %Rrc
8299 * format string is given @a aVRC as an argument.
8300 *
8301 * If there is no last error message collected by i_vdErrorCall or if it is a
8302 * null or empty string, then this function returns the following text:
8303 * <code>
8304 * " (%Rrc)"
8305 * </code>
8306 *
8307 * @note Doesn't do any object locking; it is assumed that the caller makes sure
8308 * the callback isn't called by more than one thread at a time.
8309 *
8310 * @param aVRC VBox error code to use when no error message is provided.
8311 */
8312Utf8Str Medium::i_vdError(int aVRC)
8313{
8314 Utf8Str error;
8315
8316 if (m->vdError.isEmpty())
8317 error.printf(" (%Rrc)", aVRC);
8318 else
8319 error.printf(".\n%s", m->vdError.c_str());
8320
8321 m->vdError.setNull();
8322
8323 return error;
8324}
8325
8326/**
8327 * Error message callback.
8328 *
8329 * Puts the reported error message to the m->vdError field.
8330 *
8331 * @note Doesn't do any object locking; it is assumed that the caller makes sure
8332 * the callback isn't called by more than one thread at a time.
8333 *
8334 * @param pvUser The opaque data passed on container creation.
8335 * @param vrc The VBox error code.
8336 * @param SRC_POS Use RT_SRC_POS.
8337 * @param pszFormat Error message format string.
8338 * @param va Error message arguments.
8339 */
8340/*static*/
8341DECLCALLBACK(void) Medium::i_vdErrorCall(void *pvUser, int vrc, RT_SRC_POS_DECL,
8342 const char *pszFormat, va_list va)
8343{
8344 NOREF(pszFile); NOREF(iLine); NOREF(pszFunction); /* RT_SRC_POS_DECL */
8345
8346 Medium *that = static_cast<Medium*>(pvUser);
8347 AssertReturnVoid(that != NULL);
8348
8349 va_list vaCopy; /* For gcc */
8350 va_copy(vaCopy, va);
8351 if (that->m->vdError.isEmpty())
8352 that->m->vdError.printf("%N (%Rrc)", pszFormat, &vaCopy, vrc);
8353 else
8354 that->m->vdError.appendPrintf(".\n%N (%Rrc)", pszFormat, &vaCopy, vrc);
8355 va_end(vaCopy);
8356}
8357
8358/* static */
8359DECLCALLBACK(bool) Medium::i_vdConfigAreKeysValid(void *pvUser,
8360 const char * /* pszzValid */)
8361{
8362 Medium *that = static_cast<Medium*>(pvUser);
8363 AssertReturn(that != NULL, false);
8364
8365 /* we always return true since the only keys we have are those found in
8366 * VDBACKENDINFO */
8367 return true;
8368}
8369
8370/* static */
8371DECLCALLBACK(int) Medium::i_vdConfigQuerySize(void *pvUser,
8372 const char *pszName,
8373 size_t *pcbValue)
8374{
8375 AssertPtrReturn(pcbValue, VERR_INVALID_POINTER);
8376
8377 Medium *that = static_cast<Medium*>(pvUser);
8378 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
8379
8380 settings::StringsMap::const_iterator it = that->m->mapProperties.find(Utf8Str(pszName));
8381 if (it == that->m->mapProperties.end())
8382 return VERR_CFGM_VALUE_NOT_FOUND;
8383
8384 /* we interpret null values as "no value" in Medium */
8385 if (it->second.isEmpty())
8386 return VERR_CFGM_VALUE_NOT_FOUND;
8387
8388 *pcbValue = it->second.length() + 1 /* include terminator */;
8389
8390 return VINF_SUCCESS;
8391}
8392
8393/* static */
8394DECLCALLBACK(int) Medium::i_vdConfigQuery(void *pvUser,
8395 const char *pszName,
8396 char *pszValue,
8397 size_t cchValue)
8398{
8399 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
8400
8401 Medium *that = static_cast<Medium*>(pvUser);
8402 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
8403
8404 settings::StringsMap::const_iterator it = that->m->mapProperties.find(Utf8Str(pszName));
8405 if (it == that->m->mapProperties.end())
8406 return VERR_CFGM_VALUE_NOT_FOUND;
8407
8408 /* we interpret null values as "no value" in Medium */
8409 if (it->second.isEmpty())
8410 return VERR_CFGM_VALUE_NOT_FOUND;
8411
8412 const Utf8Str &value = it->second;
8413 if (value.length() >= cchValue)
8414 return VERR_CFGM_NOT_ENOUGH_SPACE;
8415
8416 memcpy(pszValue, value.c_str(), value.length() + 1);
8417
8418 return VINF_SUCCESS;
8419}
8420
8421DECLCALLBACK(bool) Medium::i_vdCryptoConfigAreKeysValid(void *pvUser, const char *pszzValid)
8422{
8423 /* Just return always true here. */
8424 NOREF(pvUser);
8425 NOREF(pszzValid);
8426 return true;
8427}
8428
8429DECLCALLBACK(int) Medium::i_vdCryptoConfigQuerySize(void *pvUser, const char *pszName, size_t *pcbValue)
8430{
8431 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8432 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8433 AssertPtrReturn(pcbValue, VERR_INVALID_POINTER);
8434
8435 size_t cbValue = 0;
8436 if (!strcmp(pszName, "Algorithm"))
8437 cbValue = strlen(pSettings->pszCipher) + 1;
8438 else if (!strcmp(pszName, "KeyId"))
8439 cbValue = sizeof("irrelevant");
8440 else if (!strcmp(pszName, "KeyStore"))
8441 {
8442 if (!pSettings->pszKeyStoreLoad)
8443 return VERR_CFGM_VALUE_NOT_FOUND;
8444 cbValue = strlen(pSettings->pszKeyStoreLoad) + 1;
8445 }
8446 else if (!strcmp(pszName, "CreateKeyStore"))
8447 cbValue = 2; /* Single digit + terminator. */
8448 else
8449 return VERR_CFGM_VALUE_NOT_FOUND;
8450
8451 *pcbValue = cbValue + 1 /* include terminator */;
8452
8453 return VINF_SUCCESS;
8454}
8455
8456DECLCALLBACK(int) Medium::i_vdCryptoConfigQuery(void *pvUser, const char *pszName,
8457 char *pszValue, size_t cchValue)
8458{
8459 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8460 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8461 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
8462
8463 const char *psz = NULL;
8464 if (!strcmp(pszName, "Algorithm"))
8465 psz = pSettings->pszCipher;
8466 else if (!strcmp(pszName, "KeyId"))
8467 psz = "irrelevant";
8468 else if (!strcmp(pszName, "KeyStore"))
8469 psz = pSettings->pszKeyStoreLoad;
8470 else if (!strcmp(pszName, "CreateKeyStore"))
8471 {
8472 if (pSettings->fCreateKeyStore)
8473 psz = "1";
8474 else
8475 psz = "0";
8476 }
8477 else
8478 return VERR_CFGM_VALUE_NOT_FOUND;
8479
8480 size_t cch = strlen(psz);
8481 if (cch >= cchValue)
8482 return VERR_CFGM_NOT_ENOUGH_SPACE;
8483
8484 memcpy(pszValue, psz, cch + 1);
8485 return VINF_SUCCESS;
8486}
8487
8488DECLCALLBACK(int) Medium::i_vdConfigUpdate(void *pvUser,
8489 bool fCreate,
8490 const char *pszName,
8491 const char *pszValue)
8492{
8493 Medium *that = (Medium *)pvUser;
8494
8495 // Detect if this runs inside i_queryInfo() on the current thread.
8496 // Skip if not. Check does not need synchronization.
8497 if (!that->m || !that->m->queryInfoRunning || !that->m->queryInfoSem.isWriteLockOnCurrentThread())
8498 return VINF_SUCCESS;
8499
8500 // It's guaranteed that this code is executing inside Medium::i_queryInfo,
8501 // can assume it took care of synchronization.
8502 int rv = VINF_SUCCESS;
8503 Utf8Str strName(pszName);
8504 settings::StringsMap::const_iterator it = that->m->mapProperties.find(strName);
8505 if (it == that->m->mapProperties.end() && !fCreate)
8506 rv = VERR_CFGM_VALUE_NOT_FOUND;
8507 else
8508 that->m->mapProperties[strName] = Utf8Str(pszValue);
8509 return rv;
8510}
8511
8512DECLCALLBACK(int) Medium::i_vdCryptoKeyRetain(void *pvUser, const char *pszId,
8513 const uint8_t **ppbKey, size_t *pcbKey)
8514{
8515 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8516 NOREF(pszId);
8517 NOREF(ppbKey);
8518 NOREF(pcbKey);
8519 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8520 AssertMsgFailedReturn(("This method should not be called here!\n"), VERR_INVALID_STATE);
8521}
8522
8523DECLCALLBACK(int) Medium::i_vdCryptoKeyRelease(void *pvUser, const char *pszId)
8524{
8525 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8526 NOREF(pszId);
8527 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8528 AssertMsgFailedReturn(("This method should not be called here!\n"), VERR_INVALID_STATE);
8529}
8530
8531DECLCALLBACK(int) Medium::i_vdCryptoKeyStorePasswordRetain(void *pvUser, const char *pszId, const char **ppszPassword)
8532{
8533 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8534 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8535
8536 NOREF(pszId);
8537 *ppszPassword = pSettings->pszPassword;
8538 return VINF_SUCCESS;
8539}
8540
8541DECLCALLBACK(int) Medium::i_vdCryptoKeyStorePasswordRelease(void *pvUser, const char *pszId)
8542{
8543 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8544 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8545 NOREF(pszId);
8546 return VINF_SUCCESS;
8547}
8548
8549DECLCALLBACK(int) Medium::i_vdCryptoKeyStoreSave(void *pvUser, const void *pvKeyStore, size_t cbKeyStore)
8550{
8551 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8552 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8553
8554 pSettings->pszKeyStore = (char *)RTMemAllocZ(cbKeyStore);
8555 if (!pSettings->pszKeyStore)
8556 return VERR_NO_MEMORY;
8557
8558 memcpy(pSettings->pszKeyStore, pvKeyStore, cbKeyStore);
8559 return VINF_SUCCESS;
8560}
8561
8562DECLCALLBACK(int) Medium::i_vdCryptoKeyStoreReturnParameters(void *pvUser, const char *pszCipher,
8563 const uint8_t *pbDek, size_t cbDek)
8564{
8565 MediumCryptoFilterSettings *pSettings = (MediumCryptoFilterSettings *)pvUser;
8566 AssertPtrReturn(pSettings, VERR_GENERAL_FAILURE);
8567
8568 pSettings->pszCipherReturned = RTStrDup(pszCipher);
8569 pSettings->pbDek = pbDek;
8570 pSettings->cbDek = cbDek;
8571
8572 return pSettings->pszCipherReturned ? VINF_SUCCESS : VERR_NO_MEMORY;
8573}
8574
8575/**
8576 * Creates a VDISK instance for this medium.
8577 *
8578 * @note Caller should not hold any medium related locks as this method will
8579 * acquire the medium lock for writing and others (VirtualBox).
8580 *
8581 * @returns COM status code.
8582 * @param fWritable Whether to return a writable VDISK instance
8583 * (true) or a read-only one (false).
8584 * @param pKeyStore The key store.
8585 * @param ppHdd Where to return the pointer to the VDISK on
8586 * success.
8587 * @param pMediumLockList The lock list to populate and lock. Caller
8588 * is responsible for calling the destructor or
8589 * MediumLockList::Clear() after destroying
8590 * @a *ppHdd
8591 * @param pCryptoSettings The crypto settings to use for setting up
8592 * decryption/encryption of the VDISK. This object
8593 * must be alive until the VDISK is destroyed!
8594 */
8595HRESULT Medium::i_openForIO(bool fWritable, SecretKeyStore *pKeyStore, PVDISK *ppHdd, MediumLockList *pMediumLockList,
8596 MediumCryptoFilterSettings *pCryptoSettings)
8597{
8598 /*
8599 * Create the media lock list and lock the media.
8600 */
8601 HRESULT hrc = i_createMediumLockList(true /* fFailIfInaccessible */,
8602 fWritable ? this : NULL /* pToLockWrite */,
8603 false /* fMediumLockWriteAll */,
8604 NULL,
8605 *pMediumLockList);
8606 if (SUCCEEDED(hrc))
8607 hrc = pMediumLockList->Lock();
8608 if (FAILED(hrc))
8609 return hrc;
8610
8611 /*
8612 * Get the base medium before write locking this medium.
8613 */
8614 ComObjPtr<Medium> pBase = i_getBase();
8615 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
8616
8617 /*
8618 * Create the VDISK instance.
8619 */
8620 PVDISK pHdd;
8621 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &pHdd);
8622 AssertRCReturn(vrc, E_FAIL);
8623
8624 /*
8625 * Goto avoidance using try/catch/throw(HRESULT).
8626 */
8627 try
8628 {
8629 settings::StringsMap::iterator itKeyStore = pBase->m->mapProperties.find("CRYPT/KeyStore");
8630 if (itKeyStore != pBase->m->mapProperties.end())
8631 {
8632#ifdef VBOX_WITH_EXTPACK
8633 settings::StringsMap::iterator itKeyId = pBase->m->mapProperties.find("CRYPT/KeyId");
8634
8635 ExtPackManager *pExtPackManager = m->pVirtualBox->i_getExtPackManager();
8636 if (pExtPackManager->i_isExtPackUsable(ORACLE_PUEL_EXTPACK_NAME))
8637 {
8638 /* Load the plugin */
8639 Utf8Str strPlugin;
8640 hrc = pExtPackManager->i_getLibraryPathForExtPack(g_szVDPlugin, ORACLE_PUEL_EXTPACK_NAME, &strPlugin);
8641 if (SUCCEEDED(hrc))
8642 {
8643 vrc = VDPluginLoadFromFilename(strPlugin.c_str());
8644 if (RT_FAILURE(vrc))
8645 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
8646 tr("Retrieving encryption settings of the image failed because the encryption plugin could not be loaded (%s)"),
8647 i_vdError(vrc).c_str());
8648 }
8649 else
8650 throw setError(VBOX_E_NOT_SUPPORTED,
8651 tr("Encryption is not supported because the extension pack '%s' is missing the encryption plugin (old extension pack installed?)"),
8652 ORACLE_PUEL_EXTPACK_NAME);
8653 }
8654 else
8655 throw setError(VBOX_E_NOT_SUPPORTED,
8656 tr("Encryption is not supported because the extension pack '%s' is missing"),
8657 ORACLE_PUEL_EXTPACK_NAME);
8658
8659 if (itKeyId == pBase->m->mapProperties.end())
8660 throw setError(VBOX_E_INVALID_OBJECT_STATE,
8661 tr("Image '%s' is configured for encryption but doesn't has a key identifier set"),
8662 pBase->m->strLocationFull.c_str());
8663
8664 /* Find the proper secret key in the key store. */
8665 if (!pKeyStore)
8666 throw setError(VBOX_E_INVALID_OBJECT_STATE,
8667 tr("Image '%s' is configured for encryption but there is no key store to retrieve the password from"),
8668 pBase->m->strLocationFull.c_str());
8669
8670 SecretKey *pKey = NULL;
8671 vrc = pKeyStore->retainSecretKey(itKeyId->second, &pKey);
8672 if (RT_FAILURE(vrc))
8673 throw setErrorBoth(VBOX_E_INVALID_OBJECT_STATE, vrc,
8674 tr("Failed to retrieve the secret key with ID \"%s\" from the store (%Rrc)"),
8675 itKeyId->second.c_str(), vrc);
8676
8677 i_taskEncryptSettingsSetup(pCryptoSettings, NULL, itKeyStore->second.c_str(), (const char *)pKey->getKeyBuffer(),
8678 false /* fCreateKeyStore */);
8679 vrc = VDFilterAdd(pHdd, "CRYPT", VD_FILTER_FLAGS_DEFAULT, pCryptoSettings->vdFilterIfaces);
8680 pKeyStore->releaseSecretKey(itKeyId->second);
8681 if (vrc == VERR_VD_PASSWORD_INCORRECT)
8682 throw setErrorBoth(VBOX_E_PASSWORD_INCORRECT, vrc, tr("The password to decrypt the image is incorrect"));
8683 if (RT_FAILURE(vrc))
8684 throw setErrorBoth(VBOX_E_INVALID_OBJECT_STATE, vrc, tr("Failed to load the decryption filter: %s"),
8685 i_vdError(vrc).c_str());
8686#else
8687 RT_NOREF(pKeyStore, pCryptoSettings);
8688 throw setError(VBOX_E_NOT_SUPPORTED,
8689 tr("Encryption is not supported because extension pack support is not built in"));
8690#endif /* VBOX_WITH_EXTPACK */
8691 }
8692
8693 /*
8694 * Open all media in the source chain.
8695 */
8696 MediumLockList::Base::const_iterator sourceListBegin = pMediumLockList->GetBegin();
8697 MediumLockList::Base::const_iterator sourceListEnd = pMediumLockList->GetEnd();
8698 MediumLockList::Base::const_iterator mediumListLast = sourceListEnd;
8699 --mediumListLast;
8700
8701 for (MediumLockList::Base::const_iterator it = sourceListBegin; it != sourceListEnd; ++it)
8702 {
8703 const MediumLock &mediumLock = *it;
8704 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
8705 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
8706
8707 /* sanity check */
8708 Assert(pMedium->m->state == (fWritable && it == mediumListLast ? MediumState_LockedWrite : MediumState_LockedRead));
8709
8710 /* Open all media in read-only mode. */
8711 vrc = VDOpen(pHdd,
8712 pMedium->m->strFormat.c_str(),
8713 pMedium->m->strLocationFull.c_str(),
8714 m->uOpenFlagsDef | (fWritable && it == mediumListLast ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY),
8715 pMedium->m->vdImageIfaces);
8716 if (RT_FAILURE(vrc))
8717 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
8718 tr("Could not open the medium storage unit '%s'%s"),
8719 pMedium->m->strLocationFull.c_str(),
8720 i_vdError(vrc).c_str());
8721 }
8722
8723 Assert(m->state == (fWritable ? MediumState_LockedWrite : MediumState_LockedRead));
8724
8725 /*
8726 * Done!
8727 */
8728 *ppHdd = pHdd;
8729 return S_OK;
8730 }
8731 catch (HRESULT hrc2)
8732 {
8733 hrc = hrc2;
8734 }
8735
8736 VDDestroy(pHdd);
8737 return hrc;
8738
8739}
8740
8741/**
8742 * Implementation code for the "create base" task.
8743 *
8744 * This only gets started from Medium::CreateBaseStorage() and always runs
8745 * asynchronously. As a result, we always save the VirtualBox.xml file when
8746 * we're done here.
8747 *
8748 * @param task
8749 * @return
8750 */
8751HRESULT Medium::i_taskCreateBaseHandler(Medium::CreateBaseTask &task)
8752{
8753 /** @todo r=klaus The code below needs to be double checked with regard
8754 * to lock order violations, it probably causes lock order issues related
8755 * to the AutoCaller usage. */
8756 HRESULT hrc = S_OK;
8757
8758 /* these parameters we need after creation */
8759 uint64_t size = 0, logicalSize = 0;
8760 MediumVariant_T variant = MediumVariant_Standard;
8761 bool fGenerateUuid = false;
8762
8763 try
8764 {
8765 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
8766
8767 /* The object may request a specific UUID (through a special form of
8768 * the moveTo() argument). Otherwise we have to generate it */
8769 Guid id = m->id;
8770
8771 fGenerateUuid = id.isZero();
8772 if (fGenerateUuid)
8773 {
8774 id.create();
8775 /* VirtualBox::i_registerMedium() will need UUID */
8776 unconst(m->id) = id;
8777 }
8778
8779 Utf8Str format(m->strFormat);
8780 Utf8Str location(m->strLocationFull);
8781 uint64_t capabilities = m->formatObj->i_getCapabilities();
8782 ComAssertThrow(capabilities & ( MediumFormatCapabilities_CreateFixed
8783 | MediumFormatCapabilities_CreateDynamic), E_FAIL);
8784 Assert(m->state == MediumState_Creating);
8785
8786 PVDISK hdd;
8787 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
8788 ComAssertRCThrow(vrc, E_FAIL);
8789
8790 /* unlock before the potentially lengthy operation */
8791 thisLock.release();
8792
8793 try
8794 {
8795 /* ensure the directory exists */
8796 if (capabilities & MediumFormatCapabilities_File)
8797 {
8798 hrc = VirtualBox::i_ensureFilePathExists(location, !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
8799 if (FAILED(hrc))
8800 throw hrc;
8801 }
8802
8803 VDGEOMETRY geo = { 0, 0, 0 }; /* auto-detect */
8804
8805 vrc = VDCreateBase(hdd,
8806 format.c_str(),
8807 location.c_str(),
8808 task.mSize,
8809 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted),
8810 NULL,
8811 &geo,
8812 &geo,
8813 id.raw(),
8814 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
8815 m->vdImageIfaces,
8816 task.mVDOperationIfaces);
8817 if (RT_FAILURE(vrc))
8818 {
8819 if (vrc == VERR_VD_INVALID_TYPE)
8820 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
8821 tr("Parameters for creating the medium storage unit '%s' are invalid%s"),
8822 location.c_str(), i_vdError(vrc).c_str());
8823 else
8824 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
8825 tr("Could not create the medium storage unit '%s'%s"),
8826 location.c_str(), i_vdError(vrc).c_str());
8827 }
8828
8829 if (task.mVariant & MediumVariant_Formatted)
8830 {
8831 RTVFSFILE hVfsFile;
8832 vrc = VDCreateVfsFileFromDisk(hdd, 0 /*fFlags*/, &hVfsFile);
8833 if (RT_FAILURE(vrc))
8834 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Opening medium storage unit '%s' failed%s"),
8835 location.c_str(), i_vdError(vrc).c_str());
8836 RTERRINFOSTATIC ErrInfo;
8837 vrc = RTFsFatVolFormat(hVfsFile, 0 /* offVol */, 0 /* cbVol */, RTFSFATVOL_FMT_F_FULL,
8838 0 /* cbSector */, 0 /* cbSectorPerCluster */, RTFSFATTYPE_INVALID,
8839 0 /* cHeads */, 0 /* cSectorsPerTrack*/, 0 /* bMedia */,
8840 0 /* cRootDirEntries */, 0 /* cHiddenSectors */,
8841 RTErrInfoInitStatic(&ErrInfo));
8842 RTVfsFileRelease(hVfsFile);
8843 if (RT_FAILURE(vrc) && RTErrInfoIsSet(&ErrInfo.Core))
8844 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Formatting medium storage unit '%s' failed: %s"),
8845 location.c_str(), ErrInfo.Core.pszMsg);
8846 if (RT_FAILURE(vrc))
8847 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc, tr("Formatting medium storage unit '%s' failed%s"),
8848 location.c_str(), i_vdError(vrc).c_str());
8849 }
8850
8851 size = VDGetFileSize(hdd, 0);
8852 logicalSize = VDGetSize(hdd, 0);
8853 unsigned uImageFlags;
8854 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
8855 if (RT_SUCCESS(vrc))
8856 variant = (MediumVariant_T)uImageFlags;
8857 }
8858 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
8859
8860 VDDestroy(hdd);
8861 }
8862 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
8863
8864 if (SUCCEEDED(hrc))
8865 {
8866 /* register with mVirtualBox as the last step and move to
8867 * Created state only on success (leaving an orphan file is
8868 * better than breaking media registry consistency) */
8869 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
8870 ComObjPtr<Medium> pMedium;
8871 hrc = m->pVirtualBox->i_registerMedium(this, &pMedium, treeLock);
8872 Assert(pMedium == NULL || this == pMedium);
8873 }
8874
8875 // re-acquire the lock before changing state
8876 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
8877
8878 if (SUCCEEDED(hrc))
8879 {
8880 m->state = MediumState_Created;
8881
8882 m->size = size;
8883 m->logicalSize = logicalSize;
8884 m->variant = variant;
8885
8886 thisLock.release();
8887 i_markRegistriesModified();
8888 if (task.isAsync())
8889 {
8890 // in asynchronous mode, save settings now
8891 m->pVirtualBox->i_saveModifiedRegistries();
8892 }
8893 }
8894 else
8895 {
8896 /* back to NotCreated on failure */
8897 m->state = MediumState_NotCreated;
8898
8899 /* reset UUID to prevent it from being reused next time */
8900 if (fGenerateUuid)
8901 unconst(m->id).clear();
8902 }
8903
8904 if (task.NotifyAboutChanges() && SUCCEEDED(hrc))
8905 {
8906 m->pVirtualBox->i_onMediumConfigChanged(this);
8907 m->pVirtualBox->i_onMediumRegistered(m->id, m->devType, TRUE);
8908 }
8909
8910 return hrc;
8911}
8912
8913/**
8914 * Implementation code for the "create diff" task.
8915 *
8916 * This task always gets started from Medium::createDiffStorage() and can run
8917 * synchronously or asynchronously depending on the "wait" parameter passed to
8918 * that function. If we run synchronously, the caller expects the medium
8919 * registry modification to be set before returning; otherwise (in asynchronous
8920 * mode), we save the settings ourselves.
8921 *
8922 * @param task
8923 * @return
8924 */
8925HRESULT Medium::i_taskCreateDiffHandler(Medium::CreateDiffTask &task)
8926{
8927 /** @todo r=klaus The code below needs to be double checked with regard
8928 * to lock order violations, it probably causes lock order issues related
8929 * to the AutoCaller usage. */
8930 HRESULT hrcTmp = S_OK;
8931
8932 const ComObjPtr<Medium> &pTarget = task.mTarget;
8933
8934 uint64_t size = 0, logicalSize = 0;
8935 MediumVariant_T variant = MediumVariant_Standard;
8936 bool fGenerateUuid = false;
8937
8938 try
8939 {
8940 if (i_getDepth() >= SETTINGS_MEDIUM_DEPTH_MAX)
8941 {
8942 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
8943 throw setError(VBOX_E_INVALID_OBJECT_STATE,
8944 tr("Cannot create differencing image for medium '%s', because it exceeds the medium tree depth limit. Please merge some images which you no longer need"),
8945 m->strLocationFull.c_str());
8946 }
8947
8948 /* Lock both in {parent,child} order. */
8949 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
8950
8951 /* The object may request a specific UUID (through a special form of
8952 * the moveTo() argument). Otherwise we have to generate it */
8953 Guid targetId = pTarget->m->id;
8954
8955 fGenerateUuid = targetId.isZero();
8956 if (fGenerateUuid)
8957 {
8958 targetId.create();
8959 /* VirtualBox::i_registerMedium() will need UUID */
8960 unconst(pTarget->m->id) = targetId;
8961 }
8962
8963 Guid id = m->id;
8964
8965 Utf8Str targetFormat(pTarget->m->strFormat);
8966 Utf8Str targetLocation(pTarget->m->strLocationFull);
8967 uint64_t capabilities = pTarget->m->formatObj->i_getCapabilities();
8968 ComAssertThrow(capabilities & MediumFormatCapabilities_CreateDynamic, E_FAIL);
8969
8970 Assert(pTarget->m->state == MediumState_Creating);
8971 Assert(m->state == MediumState_LockedRead);
8972
8973 PVDISK hdd;
8974 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
8975 ComAssertRCThrow(vrc, E_FAIL);
8976
8977 /* the two media are now protected by their non-default states;
8978 * unlock the media before the potentially lengthy operation */
8979 mediaLock.release();
8980
8981 try
8982 {
8983 /* Open all media in the target chain but the last. */
8984 MediumLockList::Base::const_iterator targetListBegin =
8985 task.mpMediumLockList->GetBegin();
8986 MediumLockList::Base::const_iterator targetListEnd =
8987 task.mpMediumLockList->GetEnd();
8988 for (MediumLockList::Base::const_iterator it = targetListBegin;
8989 it != targetListEnd;
8990 ++it)
8991 {
8992 const MediumLock &mediumLock = *it;
8993 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
8994
8995 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
8996
8997 /* Skip over the target diff medium */
8998 if (pMedium->m->state == MediumState_Creating)
8999 continue;
9000
9001 /* sanity check */
9002 Assert(pMedium->m->state == MediumState_LockedRead);
9003
9004 /* Open all media in appropriate mode. */
9005 vrc = VDOpen(hdd,
9006 pMedium->m->strFormat.c_str(),
9007 pMedium->m->strLocationFull.c_str(),
9008 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
9009 pMedium->m->vdImageIfaces);
9010 if (RT_FAILURE(vrc))
9011 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9012 tr("Could not open the medium storage unit '%s'%s"),
9013 pMedium->m->strLocationFull.c_str(),
9014 i_vdError(vrc).c_str());
9015 }
9016
9017 /* ensure the target directory exists */
9018 if (capabilities & MediumFormatCapabilities_File)
9019 {
9020 HRESULT hrc = VirtualBox::i_ensureFilePathExists(targetLocation,
9021 !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
9022 if (FAILED(hrc))
9023 throw hrc;
9024 }
9025
9026 vrc = VDCreateDiff(hdd,
9027 targetFormat.c_str(),
9028 targetLocation.c_str(),
9029 (task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk))
9030 | VD_IMAGE_FLAGS_DIFF,
9031 NULL,
9032 targetId.raw(),
9033 id.raw(),
9034 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
9035 pTarget->m->vdImageIfaces,
9036 task.mVDOperationIfaces);
9037 if (RT_FAILURE(vrc))
9038 {
9039 if (vrc == VERR_VD_INVALID_TYPE)
9040 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9041 tr("Parameters for creating the differencing medium storage unit '%s' are invalid%s"),
9042 targetLocation.c_str(), i_vdError(vrc).c_str());
9043 else
9044 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9045 tr("Could not create the differencing medium storage unit '%s'%s"),
9046 targetLocation.c_str(), i_vdError(vrc).c_str());
9047 }
9048
9049 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
9050 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
9051 unsigned uImageFlags;
9052 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
9053 if (RT_SUCCESS(vrc))
9054 variant = (MediumVariant_T)uImageFlags;
9055 }
9056 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9057
9058 VDDestroy(hdd);
9059 }
9060 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9061
9062 MultiResult mrc(hrcTmp);
9063
9064 if (SUCCEEDED(mrc))
9065 {
9066 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
9067
9068 Assert(pTarget->m->pParent.isNull());
9069
9070 /* associate child with the parent, maximum depth was checked above */
9071 pTarget->i_setParent(this);
9072
9073 /* diffs for immutable media are auto-reset by default */
9074 bool fAutoReset;
9075 {
9076 ComObjPtr<Medium> pBase = i_getBase();
9077 AutoReadLock block(pBase COMMA_LOCKVAL_SRC_POS);
9078 fAutoReset = (pBase->m->type == MediumType_Immutable);
9079 }
9080 {
9081 AutoWriteLock tlock(pTarget COMMA_LOCKVAL_SRC_POS);
9082 pTarget->m->autoReset = fAutoReset;
9083 }
9084
9085 /* register with mVirtualBox as the last step and move to
9086 * Created state only on success (leaving an orphan file is
9087 * better than breaking media registry consistency) */
9088 ComObjPtr<Medium> pMedium;
9089 mrc = m->pVirtualBox->i_registerMedium(pTarget, &pMedium, treeLock);
9090 Assert(pTarget == pMedium);
9091
9092 if (FAILED(mrc))
9093 /* break the parent association on failure to register */
9094 i_deparent();
9095 }
9096
9097 AutoMultiWriteLock2 mediaLock(this, pTarget COMMA_LOCKVAL_SRC_POS);
9098
9099 if (SUCCEEDED(mrc))
9100 {
9101 pTarget->m->state = MediumState_Created;
9102
9103 pTarget->m->size = size;
9104 pTarget->m->logicalSize = logicalSize;
9105 pTarget->m->variant = variant;
9106 }
9107 else
9108 {
9109 /* back to NotCreated on failure */
9110 pTarget->m->state = MediumState_NotCreated;
9111
9112 pTarget->m->autoReset = false;
9113
9114 /* reset UUID to prevent it from being reused next time */
9115 if (fGenerateUuid)
9116 unconst(pTarget->m->id).clear();
9117 }
9118
9119 // deregister the task registered in createDiffStorage()
9120 Assert(m->numCreateDiffTasks != 0);
9121 --m->numCreateDiffTasks;
9122
9123 mediaLock.release();
9124 i_markRegistriesModified();
9125 if (task.isAsync())
9126 {
9127 // in asynchronous mode, save settings now
9128 m->pVirtualBox->i_saveModifiedRegistries();
9129 }
9130
9131 /* Note that in sync mode, it's the caller's responsibility to
9132 * unlock the medium. */
9133
9134 if (task.NotifyAboutChanges() && SUCCEEDED(mrc))
9135 {
9136 m->pVirtualBox->i_onMediumConfigChanged(this);
9137 m->pVirtualBox->i_onMediumRegistered(m->id, m->devType, TRUE);
9138 }
9139
9140 return mrc;
9141}
9142
9143/**
9144 * Implementation code for the "merge" task.
9145 *
9146 * This task always gets started from Medium::mergeTo() and can run
9147 * synchronously or asynchronously depending on the "wait" parameter passed to
9148 * that function. If we run synchronously, the caller expects the medium
9149 * registry modification to be set before returning; otherwise (in asynchronous
9150 * mode), we save the settings ourselves.
9151 *
9152 * @param task
9153 * @return
9154 */
9155HRESULT Medium::i_taskMergeHandler(Medium::MergeTask &task)
9156{
9157 /** @todo r=klaus The code below needs to be double checked with regard
9158 * to lock order violations, it probably causes lock order issues related
9159 * to the AutoCaller usage. */
9160 HRESULT hrcTmp = S_OK;
9161
9162 const ComObjPtr<Medium> &pTarget = task.mTarget;
9163
9164 try
9165 {
9166 if (!task.mParentForTarget.isNull())
9167 if (task.mParentForTarget->i_getDepth() >= SETTINGS_MEDIUM_DEPTH_MAX)
9168 {
9169 AutoReadLock plock(task.mParentForTarget COMMA_LOCKVAL_SRC_POS);
9170 throw setError(VBOX_E_INVALID_OBJECT_STATE,
9171 tr("Cannot merge image for medium '%s', because it exceeds the medium tree depth limit. Please merge some images which you no longer need"),
9172 task.mParentForTarget->m->strLocationFull.c_str());
9173 }
9174
9175 // Resize target to source size, if possible. Otherwise throw an error.
9176 // It's offline resizing. Online resizing will be called in the
9177 // SessionMachine::onlineMergeMedium.
9178
9179 uint64_t sourceSize = 0;
9180 Utf8Str sourceName;
9181 {
9182 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9183 sourceSize = i_getLogicalSize();
9184 sourceName = i_getName();
9185 }
9186 uint64_t targetSize = 0;
9187 Utf8Str targetName;
9188 {
9189 AutoReadLock alock(pTarget COMMA_LOCKVAL_SRC_POS);
9190 targetSize = pTarget->i_getLogicalSize();
9191 targetName = pTarget->i_getName();
9192 }
9193
9194 //reducing vm disks are not implemented yet
9195 if (sourceSize > targetSize)
9196 {
9197 if (i_isMediumFormatFile())
9198 {
9199 /// @todo r=klaus Can this use the standard code for creating a medium lock list?
9200 // Have to make own lock list, because "resize" method resizes the last image
9201 // in the lock chain only. The lock chain is already in the task.mpMediumLockList,
9202 // so just make new lock list based on it, with the right last medium. The own
9203 // lock list skips double locking and therefore does not affect the general lock
9204 // state after the "resize" method.
9205 MediumLockList* pMediumLockListForResize = new MediumLockList();
9206
9207 for (MediumLockList::Base::iterator it = task.mpMediumLockList->GetBegin();
9208 it != task.mpMediumLockList->GetEnd();
9209 ++it)
9210 {
9211 ComObjPtr<Medium> pMedium = it->GetMedium();
9212 pMediumLockListForResize->Append(pMedium, pMedium->m->state == MediumState_LockedWrite);
9213 if (pMedium == pTarget)
9214 break;
9215 }
9216
9217 // just to switch internal state of the lock list to avoid errors during list deletion,
9218 // because all media in the list already locked by task.mpMediumLockList
9219 HRESULT hrc = pMediumLockListForResize->Lock(true /* fSkipOverLockedMedia */);
9220 if (FAILED(hrc))
9221 {
9222 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9223 delete pMediumLockListForResize;
9224 throw setError(hrc,
9225 tr("Failed to lock the medium '%s' to resize before merge"),
9226 targetName.c_str());
9227 }
9228
9229 ComObjPtr<Progress> pProgress(task.GetProgressObject());
9230 hrc = pTarget->i_resize(sourceSize, pMediumLockListForResize, &pProgress, true, false);
9231 if (FAILED(hrc))
9232 {
9233 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9234 delete pMediumLockListForResize;
9235 throw setError(hrc,
9236 tr("Failed to set size of '%s' to size of '%s'"),
9237 targetName.c_str(), sourceName.c_str());
9238 }
9239 delete pMediumLockListForResize;
9240 }
9241 else
9242 {
9243 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9244 throw setError(VBOX_E_NOT_SUPPORTED,
9245 tr("Sizes of '%s' and '%s' are different and medium format does not support resing"),
9246 sourceName.c_str(), targetName.c_str());
9247 }
9248 }
9249
9250 task.GetProgressObject()->SetNextOperation(BstrFmt(tr("Merging medium '%s' to '%s'"),
9251 i_getName().c_str(),
9252 targetName.c_str()).raw(),
9253 1);
9254
9255 PVDISK hdd;
9256 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
9257 ComAssertRCThrow(vrc, E_FAIL);
9258
9259 try
9260 {
9261 // Similar code appears in SessionMachine::onlineMergeMedium, so
9262 // if you make any changes below check whether they are applicable
9263 // in that context as well.
9264
9265 unsigned uTargetIdx = VD_LAST_IMAGE;
9266 unsigned uSourceIdx = VD_LAST_IMAGE;
9267 /* Open all media in the chain. */
9268 MediumLockList::Base::iterator lockListBegin =
9269 task.mpMediumLockList->GetBegin();
9270 MediumLockList::Base::iterator lockListEnd =
9271 task.mpMediumLockList->GetEnd();
9272 unsigned i = 0;
9273 for (MediumLockList::Base::iterator it = lockListBegin;
9274 it != lockListEnd;
9275 ++it)
9276 {
9277 MediumLock &mediumLock = *it;
9278 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
9279
9280 if (pMedium == this)
9281 uSourceIdx = i;
9282 else if (pMedium == pTarget)
9283 uTargetIdx = i;
9284
9285 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
9286
9287 /*
9288 * complex sanity (sane complexity)
9289 *
9290 * The current medium must be in the Deleting (medium is merged)
9291 * or LockedRead (parent medium) state if it is not the target.
9292 * If it is the target it must be in the LockedWrite state.
9293 */
9294 Assert( ( pMedium != pTarget
9295 && ( pMedium->m->state == MediumState_Deleting
9296 || pMedium->m->state == MediumState_LockedRead))
9297 || ( pMedium == pTarget
9298 && pMedium->m->state == MediumState_LockedWrite));
9299 /*
9300 * Medium must be the target, in the LockedRead state
9301 * or Deleting state where it is not allowed to be attached
9302 * to a virtual machine.
9303 */
9304 Assert( pMedium == pTarget
9305 || pMedium->m->state == MediumState_LockedRead
9306 || ( pMedium->m->backRefs.size() == 0
9307 && pMedium->m->state == MediumState_Deleting));
9308 /* The source medium must be in Deleting state. */
9309 Assert( pMedium != this
9310 || pMedium->m->state == MediumState_Deleting);
9311
9312 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
9313
9314 if ( pMedium->m->state == MediumState_LockedRead
9315 || pMedium->m->state == MediumState_Deleting)
9316 uOpenFlags = VD_OPEN_FLAGS_READONLY;
9317 if (pMedium->m->type == MediumType_Shareable)
9318 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
9319
9320 /* Open the medium */
9321 vrc = VDOpen(hdd,
9322 pMedium->m->strFormat.c_str(),
9323 pMedium->m->strLocationFull.c_str(),
9324 uOpenFlags | m->uOpenFlagsDef,
9325 pMedium->m->vdImageIfaces);
9326 if (RT_FAILURE(vrc))
9327 throw vrc;
9328
9329 i++;
9330 }
9331
9332 ComAssertThrow( uSourceIdx != VD_LAST_IMAGE
9333 && uTargetIdx != VD_LAST_IMAGE, E_FAIL);
9334
9335 vrc = VDMerge(hdd, uSourceIdx, uTargetIdx,
9336 task.mVDOperationIfaces);
9337 if (RT_FAILURE(vrc))
9338 throw vrc;
9339
9340 /* update parent UUIDs */
9341 if (!task.mfMergeForward)
9342 {
9343 /* we need to update UUIDs of all source's children
9344 * which cannot be part of the container at once so
9345 * add each one in there individually */
9346 if (task.mpChildrenToReparent)
9347 {
9348 MediumLockList::Base::iterator childrenBegin = task.mpChildrenToReparent->GetBegin();
9349 MediumLockList::Base::iterator childrenEnd = task.mpChildrenToReparent->GetEnd();
9350 for (MediumLockList::Base::iterator it = childrenBegin;
9351 it != childrenEnd;
9352 ++it)
9353 {
9354 Medium *pMedium = it->GetMedium();
9355 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
9356 vrc = VDOpen(hdd,
9357 pMedium->m->strFormat.c_str(),
9358 pMedium->m->strLocationFull.c_str(),
9359 VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
9360 pMedium->m->vdImageIfaces);
9361 if (RT_FAILURE(vrc))
9362 throw vrc;
9363
9364 vrc = VDSetParentUuid(hdd, VD_LAST_IMAGE,
9365 pTarget->m->id.raw());
9366 if (RT_FAILURE(vrc))
9367 throw vrc;
9368
9369 vrc = VDClose(hdd, false /* fDelete */);
9370 if (RT_FAILURE(vrc))
9371 throw vrc;
9372 }
9373 }
9374 }
9375 }
9376 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9377 catch (int aVRC)
9378 {
9379 hrcTmp = setErrorBoth(VBOX_E_FILE_ERROR, aVRC,
9380 tr("Could not merge the medium '%s' to '%s'%s"),
9381 m->strLocationFull.c_str(),
9382 pTarget->m->strLocationFull.c_str(),
9383 i_vdError(aVRC).c_str());
9384 }
9385
9386 VDDestroy(hdd);
9387 }
9388 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9389
9390 ErrorInfoKeeper eik;
9391 MultiResult mrc(hrcTmp);
9392 HRESULT hrc2;
9393
9394 std::set<ComObjPtr<Medium> > pMediaForNotify;
9395 std::map<Guid, DeviceType_T> uIdsForNotify;
9396
9397 if (SUCCEEDED(mrc))
9398 {
9399 /* all media but the target were successfully deleted by
9400 * VDMerge; reparent the last one and uninitialize deleted media. */
9401
9402 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
9403
9404 if (task.mfMergeForward)
9405 {
9406 /* first, unregister the target since it may become a base
9407 * medium which needs re-registration */
9408 hrc2 = m->pVirtualBox->i_unregisterMedium(pTarget);
9409 AssertComRC(hrc2);
9410
9411 /* then, reparent it and disconnect the deleted branch at both ends
9412 * (chain->parent() is source's parent). Depth check above. */
9413 pTarget->i_deparent();
9414 pTarget->i_setParent(task.mParentForTarget);
9415 if (task.mParentForTarget)
9416 {
9417 i_deparent();
9418 if (task.NotifyAboutChanges())
9419 pMediaForNotify.insert(task.mParentForTarget);
9420 }
9421
9422 /* then, register again */
9423 ComObjPtr<Medium> pMedium;
9424 hrc2 = m->pVirtualBox->i_registerMedium(pTarget, &pMedium, treeLock);
9425 AssertComRC(hrc2);
9426 }
9427 else
9428 {
9429 Assert(pTarget->i_getChildren().size() == 1);
9430 Medium *targetChild = pTarget->i_getChildren().front();
9431
9432 /* disconnect the deleted branch at the elder end */
9433 targetChild->i_deparent();
9434
9435 /* reparent source's children and disconnect the deleted
9436 * branch at the younger end */
9437 if (task.mpChildrenToReparent)
9438 {
9439 /* obey {parent,child} lock order */
9440 AutoWriteLock sourceLock(this COMMA_LOCKVAL_SRC_POS);
9441
9442 MediumLockList::Base::iterator childrenBegin = task.mpChildrenToReparent->GetBegin();
9443 MediumLockList::Base::iterator childrenEnd = task.mpChildrenToReparent->GetEnd();
9444 for (MediumLockList::Base::iterator it = childrenBegin;
9445 it != childrenEnd;
9446 ++it)
9447 {
9448 Medium *pMedium = it->GetMedium();
9449 AutoWriteLock childLock(pMedium COMMA_LOCKVAL_SRC_POS);
9450
9451 pMedium->i_deparent(); // removes pMedium from source
9452 // no depth check, reduces depth
9453 pMedium->i_setParent(pTarget);
9454
9455 if (task.NotifyAboutChanges())
9456 pMediaForNotify.insert(pMedium);
9457 }
9458 }
9459 pMediaForNotify.insert(pTarget);
9460 }
9461
9462 /* unregister and uninitialize all media removed by the merge */
9463 MediumLockList::Base::iterator lockListBegin =
9464 task.mpMediumLockList->GetBegin();
9465 MediumLockList::Base::iterator lockListEnd =
9466 task.mpMediumLockList->GetEnd();
9467 for (MediumLockList::Base::iterator it = lockListBegin;
9468 it != lockListEnd;
9469 )
9470 {
9471 MediumLock &mediumLock = *it;
9472 /* Create a real copy of the medium pointer, as the medium
9473 * lock deletion below would invalidate the referenced object. */
9474 const ComObjPtr<Medium> pMedium = mediumLock.GetMedium();
9475
9476 /* The target and all media not merged (readonly) are skipped */
9477 if ( pMedium == pTarget
9478 || pMedium->m->state == MediumState_LockedRead)
9479 {
9480 ++it;
9481 continue;
9482 }
9483
9484 uIdsForNotify[pMedium->i_getId()] = pMedium->i_getDeviceType();
9485 hrc2 = pMedium->m->pVirtualBox->i_unregisterMedium(pMedium);
9486 AssertComRC(hrc2);
9487
9488 /* now, uninitialize the deleted medium (note that
9489 * due to the Deleting state, uninit() will not touch
9490 * the parent-child relationship so we need to
9491 * uninitialize each disk individually) */
9492
9493 /* note that the operation initiator medium (which is
9494 * normally also the source medium) is a special case
9495 * -- there is one more caller added by Task to it which
9496 * we must release. Also, if we are in sync mode, the
9497 * caller may still hold an AutoCaller instance for it
9498 * and therefore we cannot uninit() it (it's therefore
9499 * the caller's responsibility) */
9500 if (pMedium == this)
9501 {
9502 Assert(i_getChildren().size() == 0);
9503 Assert(m->backRefs.size() == 0);
9504 task.mMediumCaller.release();
9505 }
9506
9507 /* Delete the medium lock list entry, which also releases the
9508 * caller added by MergeChain before uninit() and updates the
9509 * iterator to point to the right place. */
9510 hrc2 = task.mpMediumLockList->RemoveByIterator(it);
9511 AssertComRC(hrc2);
9512
9513 if (task.isAsync() || pMedium != this)
9514 {
9515 treeLock.release();
9516 pMedium->uninit();
9517 treeLock.acquire();
9518 }
9519 }
9520 }
9521
9522 i_markRegistriesModified();
9523 if (task.isAsync())
9524 {
9525 // in asynchronous mode, save settings now
9526 eik.restore();
9527 m->pVirtualBox->i_saveModifiedRegistries();
9528 eik.fetch();
9529 }
9530
9531 if (FAILED(mrc))
9532 {
9533 /* Here we come if either VDMerge() failed (in which case we
9534 * assume that it tried to do everything to make a further
9535 * retry possible -- e.g. not deleted intermediate media
9536 * and so on) or VirtualBox::saveRegistries() failed (where we
9537 * should have the original tree but with intermediate storage
9538 * units deleted by VDMerge()). We have to only restore states
9539 * (through the MergeChain dtor) unless we are run synchronously
9540 * in which case it's the responsibility of the caller as stated
9541 * in the mergeTo() docs. The latter also implies that we
9542 * don't own the merge chain, so release it in this case. */
9543 if (task.isAsync())
9544 i_cancelMergeTo(task.mpChildrenToReparent, task.mpMediumLockList);
9545 }
9546 else if (task.NotifyAboutChanges())
9547 {
9548 for (std::set<ComObjPtr<Medium> >::const_iterator it = pMediaForNotify.begin();
9549 it != pMediaForNotify.end();
9550 ++it)
9551 {
9552 if (it->isNotNull())
9553 m->pVirtualBox->i_onMediumConfigChanged(*it);
9554 }
9555 for (std::map<Guid, DeviceType_T>::const_iterator it = uIdsForNotify.begin();
9556 it != uIdsForNotify.end();
9557 ++it)
9558 {
9559 m->pVirtualBox->i_onMediumRegistered(it->first, it->second, FALSE);
9560 }
9561 }
9562
9563 return mrc;
9564}
9565
9566/**
9567 * Implementation code for the "clone" task.
9568 *
9569 * This only gets started from Medium::CloneTo() and always runs asynchronously.
9570 * As a result, we always save the VirtualBox.xml file when we're done here.
9571 *
9572 * @param task
9573 * @return
9574 */
9575HRESULT Medium::i_taskCloneHandler(Medium::CloneTask &task)
9576{
9577 /** @todo r=klaus The code below needs to be double checked with regard
9578 * to lock order violations, it probably causes lock order issues related
9579 * to the AutoCaller usage. */
9580 HRESULT hrcTmp = S_OK;
9581
9582 const ComObjPtr<Medium> &pTarget = task.mTarget;
9583 const ComObjPtr<Medium> &pParent = task.mParent;
9584
9585 bool fCreatingTarget = false;
9586 bool cloningWithinList = false;
9587 unsigned uTargetIdx = VD_LAST_IMAGE;
9588
9589 uint64_t size = 0, logicalSize = 0;
9590 MediumVariant_T variant = MediumVariant_Standard;
9591 bool fGenerateUuid = false;
9592
9593 try
9594 {
9595 if (!pParent.isNull())
9596 {
9597
9598 if (pParent->i_getDepth() >= SETTINGS_MEDIUM_DEPTH_MAX)
9599 {
9600 AutoReadLock plock(pParent COMMA_LOCKVAL_SRC_POS);
9601 throw setError(VBOX_E_INVALID_OBJECT_STATE,
9602 tr("Cannot clone image for medium '%s', because it exceeds the medium tree depth limit. Please merge some images which you no longer need"),
9603 pParent->m->strLocationFull.c_str());
9604 }
9605 }
9606
9607 /* Lock all in {parent,child} order. The lock is also used as a
9608 * signal from the task initiator (which releases it only after
9609 * RTThreadCreate()) that we can start the job. */
9610 AutoMultiWriteLock3 thisLock(this, pTarget, pParent COMMA_LOCKVAL_SRC_POS);
9611
9612 fCreatingTarget = pTarget->m->state == MediumState_Creating;
9613
9614 /* The object may request a specific UUID (through a special form of
9615 * the moveTo() argument). Otherwise we have to generate it */
9616 Guid targetId = pTarget->m->id;
9617
9618 fGenerateUuid = targetId.isZero();
9619 if (fGenerateUuid)
9620 {
9621 targetId.create();
9622 /* VirtualBox::registerMedium() will need UUID */
9623 unconst(pTarget->m->id) = targetId;
9624 }
9625
9626 PVDISK hdd;
9627 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
9628 ComAssertRCThrow(vrc, E_FAIL);
9629
9630 try
9631 {
9632 /* Open all media in the source chain. */
9633 MediumLockList::Base::const_iterator sourceListBegin =
9634 task.mpSourceMediumLockList->GetBegin();
9635 MediumLockList::Base::const_iterator sourceListEnd =
9636 task.mpSourceMediumLockList->GetEnd();
9637 unsigned i=0;
9638 for (MediumLockList::Base::const_iterator it = sourceListBegin;
9639 it != sourceListEnd;
9640 ++it)
9641 {
9642 const MediumLock &mediumLock = *it;
9643 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
9644 if (pMedium == pTarget)
9645 {
9646 cloningWithinList = true;
9647 uTargetIdx = i;
9648 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
9649 Assert(pMedium->m->state == MediumState_LockedWrite);
9650 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
9651 if (pMedium->m->type == MediumType_Shareable)
9652 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
9653 /* Open target in appropriate mode. */
9654 vrc = VDOpen(hdd,
9655 pMedium->m->strFormat.c_str(),
9656 pMedium->m->strLocationFull.c_str(),
9657 uOpenFlags | m->uOpenFlagsDef,
9658 pMedium->m->vdImageIfaces);
9659 if (RT_FAILURE(vrc))
9660 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9661 tr("Could not open the medium storage unit '%s'%s"),
9662 pMedium->m->strLocationFull.c_str(),
9663 i_vdError(vrc).c_str());
9664 }
9665 i++;
9666 if (!cloningWithinList || pMedium != pTarget)
9667 {
9668 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
9669 /* sanity check */
9670 Assert(pMedium->m->state == MediumState_LockedRead);
9671 /** Open all media in read-only mode. */
9672 vrc = VDOpen(hdd,
9673 pMedium->m->strFormat.c_str(),
9674 pMedium->m->strLocationFull.c_str(),
9675 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
9676 pMedium->m->vdImageIfaces);
9677 if (RT_FAILURE(vrc))
9678 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9679 tr("Could not open the medium storage unit '%s'%s"),
9680 pMedium->m->strLocationFull.c_str(),
9681 i_vdError(vrc).c_str());
9682 }
9683 }
9684
9685 Utf8Str targetFormat(pTarget->m->strFormat);
9686 Utf8Str targetLocation(pTarget->m->strLocationFull);
9687 uint64_t capabilities = pTarget->m->formatObj->i_getCapabilities();
9688
9689 Assert( pTarget->m->state == MediumState_Creating
9690 || pTarget->m->state == MediumState_LockedWrite);
9691 Assert(m->state == MediumState_LockedRead);
9692 Assert( pParent.isNull()
9693 || pParent->m->state == MediumState_LockedRead);
9694
9695 /* unlock before the potentially lengthy operation */
9696 thisLock.release();
9697
9698 /* ensure the target directory exists */
9699 if (capabilities & MediumFormatCapabilities_File)
9700 {
9701 HRESULT hrc = VirtualBox::i_ensureFilePathExists(targetLocation,
9702 !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
9703 if (FAILED(hrc))
9704 throw hrc;
9705 }
9706
9707 PVDISK targetHdd;
9708 vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &targetHdd);
9709 ComAssertRCThrow(vrc, E_FAIL);
9710
9711 try
9712 {
9713 if (!cloningWithinList)
9714 {
9715 /* Open all media in the target chain. */
9716 MediumLockList::Base::const_iterator targetListBegin =
9717 task.mpTargetMediumLockList->GetBegin();
9718 MediumLockList::Base::const_iterator targetListEnd =
9719 task.mpTargetMediumLockList->GetEnd();
9720 for (MediumLockList::Base::const_iterator it = targetListBegin;
9721 it != targetListEnd;
9722 ++it)
9723 {
9724 const MediumLock &mediumLock = *it;
9725 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
9726 /* If the target medium is not created yet there's no
9727 * reason to open it. */
9728 if (pMedium == pTarget && fCreatingTarget)
9729 continue;
9730 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
9731 /* sanity check */
9732 Assert( pMedium->m->state == MediumState_LockedRead
9733 || pMedium->m->state == MediumState_LockedWrite);
9734 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
9735 if (pMedium->m->state != MediumState_LockedWrite)
9736 uOpenFlags = VD_OPEN_FLAGS_READONLY;
9737 if (pMedium->m->type == MediumType_Shareable)
9738 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
9739 /* Open all media in appropriate mode. */
9740 vrc = VDOpen(targetHdd,
9741 pMedium->m->strFormat.c_str(),
9742 pMedium->m->strLocationFull.c_str(),
9743 uOpenFlags | m->uOpenFlagsDef,
9744 pMedium->m->vdImageIfaces);
9745 if (RT_FAILURE(vrc))
9746 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9747 tr("Could not open the medium storage unit '%s'%s"),
9748 pMedium->m->strLocationFull.c_str(),
9749 i_vdError(vrc).c_str());
9750 }
9751 /* target isn't locked, but no changing data is accessed */
9752 if (task.midxSrcImageSame == UINT32_MAX)
9753 {
9754 vrc = VDCopy(hdd,
9755 VD_LAST_IMAGE,
9756 targetHdd,
9757 targetFormat.c_str(),
9758 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
9759 false /* fMoveByRename */,
9760 task.mTargetLogicalSize /* cbSize */,
9761 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
9762 targetId.raw(),
9763 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
9764 NULL /* pVDIfsOperation */,
9765 pTarget->m->vdImageIfaces,
9766 task.mVDOperationIfaces);
9767 }
9768 else
9769 {
9770 vrc = VDCopyEx(hdd,
9771 VD_LAST_IMAGE,
9772 targetHdd,
9773 VD_LAST_IMAGE,
9774 targetFormat.c_str(),
9775 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
9776 false /* fMoveByRename */,
9777 task.mTargetLogicalSize /* cbSize */,
9778 task.midxSrcImageSame,
9779 task.midxDstImageSame,
9780 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
9781 targetId.raw(),
9782 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
9783 NULL /* pVDIfsOperation */,
9784 pTarget->m->vdImageIfaces,
9785 task.mVDOperationIfaces);
9786 }
9787 if (RT_FAILURE(vrc))
9788 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9789 tr("Could not create the clone medium '%s'%s"),
9790 targetLocation.c_str(), i_vdError(vrc).c_str());
9791
9792 size = VDGetFileSize(targetHdd, VD_LAST_IMAGE);
9793 logicalSize = VDGetSize(targetHdd, VD_LAST_IMAGE);
9794 unsigned uImageFlags;
9795 vrc = VDGetImageFlags(targetHdd, 0, &uImageFlags);
9796 if (RT_SUCCESS(vrc))
9797 variant = (MediumVariant_T)uImageFlags;
9798 }
9799 else //cloningWithinList - only use source chain
9800 {
9801 if (task.midxSrcImageSame == UINT32_MAX)
9802 {
9803 vrc = VDCopyEx(hdd,
9804 VD_LAST_IMAGE,
9805 hdd,
9806 uTargetIdx,
9807 targetFormat.c_str(),
9808 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
9809 false /* fMoveByRename */,
9810 task.mTargetLogicalSize /* cbSize */,
9811 VD_IMAGE_CONTENT_UNKNOWN,
9812 VD_IMAGE_CONTENT_UNKNOWN,
9813 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
9814 targetId.raw(),
9815 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
9816 NULL /* pVDIfsOperation */,
9817 pTarget->m->vdImageIfaces,
9818 task.mVDOperationIfaces);
9819 }
9820 else
9821 {
9822 vrc = VDCopyEx(hdd,
9823 VD_LAST_IMAGE,
9824 hdd,
9825 uTargetIdx,
9826 targetFormat.c_str(),
9827 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
9828 false /* fMoveByRename */,
9829 task.mTargetLogicalSize /* cbSize */,
9830 task.midxSrcImageSame,
9831 task.midxDstImageSame,
9832 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
9833 targetId.raw(),
9834 VD_OPEN_FLAGS_NORMAL | m->uOpenFlagsDef,
9835 NULL /* pVDIfsOperation */,
9836 pTarget->m->vdImageIfaces,
9837 task.mVDOperationIfaces);
9838 }
9839 if (RT_FAILURE(vrc))
9840 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
9841 tr("Could not create the clone medium '%s'%s"),
9842 targetLocation.c_str(), i_vdError(vrc).c_str());
9843
9844 size = VDGetFileSize(hdd, uTargetIdx);
9845 logicalSize = VDGetSize(hdd, uTargetIdx);
9846 unsigned uImageFlags;
9847 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
9848 if (RT_SUCCESS(vrc))
9849 variant = (MediumVariant_T)uImageFlags;
9850
9851 }
9852 }
9853 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9854
9855 VDDestroy(targetHdd);
9856 }
9857 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9858
9859 VDDestroy(hdd);
9860 }
9861 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
9862
9863 ErrorInfoKeeper eik;
9864 MultiResult mrc(hrcTmp);
9865
9866 /* Only do the parent changes for newly created media. */
9867 if (SUCCEEDED(mrc) && fCreatingTarget)
9868 {
9869 /* we set m->pParent & children() */
9870 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
9871
9872 Assert(pTarget->m->pParent.isNull());
9873
9874 if (pParent)
9875 {
9876 /* Associate the clone with the parent and deassociate
9877 * from VirtualBox. Depth check above. */
9878 pTarget->i_setParent(pParent);
9879
9880 /* register with mVirtualBox as the last step and move to
9881 * Created state only on success (leaving an orphan file is
9882 * better than breaking media registry consistency) */
9883 eik.restore();
9884 ComObjPtr<Medium> pMedium;
9885 mrc = pParent->m->pVirtualBox->i_registerMedium(pTarget, &pMedium,
9886 treeLock);
9887 Assert( FAILED(mrc)
9888 || pTarget == pMedium);
9889 eik.fetch();
9890
9891 if (FAILED(mrc))
9892 /* break parent association on failure to register */
9893 pTarget->i_deparent(); // removes target from parent
9894 }
9895 else
9896 {
9897 /* just register */
9898 eik.restore();
9899 ComObjPtr<Medium> pMedium;
9900 mrc = m->pVirtualBox->i_registerMedium(pTarget, &pMedium,
9901 treeLock);
9902 Assert( FAILED(mrc)
9903 || pTarget == pMedium);
9904 eik.fetch();
9905 }
9906 }
9907
9908 if (fCreatingTarget)
9909 {
9910 AutoWriteLock mLock(pTarget COMMA_LOCKVAL_SRC_POS);
9911
9912 if (SUCCEEDED(mrc))
9913 {
9914 pTarget->m->state = MediumState_Created;
9915
9916 pTarget->m->size = size;
9917 pTarget->m->logicalSize = logicalSize;
9918 pTarget->m->variant = variant;
9919 }
9920 else
9921 {
9922 /* back to NotCreated on failure */
9923 pTarget->m->state = MediumState_NotCreated;
9924
9925 /* reset UUID to prevent it from being reused next time */
9926 if (fGenerateUuid)
9927 unconst(pTarget->m->id).clear();
9928 }
9929 }
9930
9931 /* Copy any filter related settings over to the target. */
9932 if (SUCCEEDED(mrc))
9933 {
9934 /* Copy any filter related settings over. */
9935 ComObjPtr<Medium> pBase = i_getBase();
9936 ComObjPtr<Medium> pTargetBase = pTarget->i_getBase();
9937 std::vector<com::Utf8Str> aFilterPropNames;
9938 std::vector<com::Utf8Str> aFilterPropValues;
9939 mrc = pBase->i_getFilterProperties(aFilterPropNames, aFilterPropValues);
9940 if (SUCCEEDED(mrc))
9941 {
9942 /* Go through the properties and add them to the target medium. */
9943 for (unsigned idx = 0; idx < aFilterPropNames.size(); idx++)
9944 {
9945 mrc = pTargetBase->i_setPropertyDirect(aFilterPropNames[idx], aFilterPropValues[idx]);
9946 if (FAILED(mrc)) break;
9947 }
9948
9949 // now, at the end of this task (always asynchronous), save the settings
9950 if (SUCCEEDED(mrc))
9951 {
9952 // save the settings
9953 i_markRegistriesModified();
9954 /* collect multiple errors */
9955 eik.restore();
9956 m->pVirtualBox->i_saveModifiedRegistries();
9957 eik.fetch();
9958
9959 if (task.NotifyAboutChanges())
9960 {
9961 if (!fCreatingTarget)
9962 {
9963 if (!aFilterPropNames.empty())
9964 m->pVirtualBox->i_onMediumConfigChanged(pTargetBase);
9965 if (pParent)
9966 m->pVirtualBox->i_onMediumConfigChanged(pParent);
9967 }
9968 else
9969 {
9970 m->pVirtualBox->i_onMediumRegistered(pTarget->i_getId(), pTarget->i_getDeviceType(), TRUE);
9971 }
9972 }
9973 }
9974 }
9975 }
9976
9977 /* Everything is explicitly unlocked when the task exits,
9978 * as the task destruction also destroys the source chain. */
9979
9980 /* Make sure the source chain is released early. It could happen
9981 * that we get a deadlock in Appliance::Import when Medium::Close
9982 * is called & the source chain is released at the same time. */
9983 task.mpSourceMediumLockList->Clear();
9984
9985 return mrc;
9986}
9987
9988/**
9989 * Implementation code for the "move" task.
9990 *
9991 * This only gets started from Medium::MoveTo() and always
9992 * runs asynchronously.
9993 *
9994 * @param task
9995 * @return
9996 */
9997HRESULT Medium::i_taskMoveHandler(Medium::MoveTask &task)
9998{
9999 LogFlowFuncEnter();
10000 HRESULT hrcOut = S_OK;
10001
10002 /* pTarget is equal "this" in our case */
10003 const ComObjPtr<Medium> &pTarget = task.mMedium;
10004
10005 uint64_t size = 0; NOREF(size);
10006 uint64_t logicalSize = 0; NOREF(logicalSize);
10007 MediumVariant_T variant = MediumVariant_Standard; NOREF(variant);
10008
10009 /*
10010 * it's exactly moving, not cloning
10011 */
10012 if (!i_isMoveOperation(pTarget))
10013 {
10014 LogFlowFunc(("LEAVE: hrc=VBOX_E_FILE_ERROR (early)\n"));
10015 return setError(VBOX_E_FILE_ERROR,
10016 tr("Wrong preconditions for moving the medium %s"),
10017 pTarget->m->strLocationFull.c_str());
10018 }
10019
10020 try
10021 {
10022 /* Lock all in {parent,child} order. The lock is also used as a
10023 * signal from the task initiator (which releases it only after
10024 * RTThreadCreate()) that we can start the job. */
10025
10026 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10027
10028 PVDISK hdd;
10029 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10030 ComAssertRCThrow(vrc, E_FAIL);
10031
10032 try
10033 {
10034 /* Open all media in the source chain. */
10035 MediumLockList::Base::const_iterator sourceListBegin =
10036 task.mpMediumLockList->GetBegin();
10037 MediumLockList::Base::const_iterator sourceListEnd =
10038 task.mpMediumLockList->GetEnd();
10039 for (MediumLockList::Base::const_iterator it = sourceListBegin;
10040 it != sourceListEnd;
10041 ++it)
10042 {
10043 const MediumLock &mediumLock = *it;
10044 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
10045 AutoWriteLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
10046
10047 /* sanity check */
10048 Assert(pMedium->m->state == MediumState_LockedWrite);
10049
10050 vrc = VDOpen(hdd,
10051 pMedium->m->strFormat.c_str(),
10052 pMedium->m->strLocationFull.c_str(),
10053 VD_OPEN_FLAGS_NORMAL,
10054 pMedium->m->vdImageIfaces);
10055 if (RT_FAILURE(vrc))
10056 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10057 tr("Could not open the medium storage unit '%s'%s"),
10058 pMedium->m->strLocationFull.c_str(),
10059 i_vdError(vrc).c_str());
10060 }
10061
10062 /* we can directly use pTarget->m->"variables" but for better reading we use local copies */
10063 Guid targetId = pTarget->m->id;
10064 Utf8Str targetFormat(pTarget->m->strFormat);
10065 uint64_t targetCapabilities = pTarget->m->formatObj->i_getCapabilities();
10066
10067 /*
10068 * change target location
10069 * m->strNewLocationFull has been set already together with m->fMoveThisMedium in
10070 * i_preparationForMoving()
10071 */
10072 Utf8Str targetLocation = i_getNewLocationForMoving();
10073
10074 /* unlock before the potentially lengthy operation */
10075 thisLock.release();
10076
10077 /* ensure the target directory exists */
10078 if (targetCapabilities & MediumFormatCapabilities_File)
10079 {
10080 HRESULT hrc = VirtualBox::i_ensureFilePathExists(targetLocation,
10081 !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
10082 if (FAILED(hrc))
10083 throw hrc;
10084 }
10085
10086 try
10087 {
10088 vrc = VDCopy(hdd,
10089 VD_LAST_IMAGE,
10090 hdd,
10091 targetFormat.c_str(),
10092 targetLocation.c_str(),
10093 true /* fMoveByRename */,
10094 0 /* cbSize */,
10095 VD_IMAGE_FLAGS_NONE,
10096 targetId.raw(),
10097 VD_OPEN_FLAGS_NORMAL,
10098 NULL /* pVDIfsOperation */,
10099 pTarget->m->vdImageIfaces,
10100 task.mVDOperationIfaces);
10101 if (RT_FAILURE(vrc))
10102 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10103 tr("Could not move medium '%s'%s"),
10104 targetLocation.c_str(), i_vdError(vrc).c_str());
10105 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
10106 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
10107 unsigned uImageFlags;
10108 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
10109 if (RT_SUCCESS(vrc))
10110 variant = (MediumVariant_T)uImageFlags;
10111
10112 /*
10113 * set current location, because VDCopy\VDCopyEx doesn't do it.
10114 * also reset moving flag
10115 */
10116 i_resetMoveOperationData();
10117 m->strLocationFull = targetLocation;
10118
10119 }
10120 catch (HRESULT hrcXcpt) { hrcOut = hrcXcpt; }
10121
10122 }
10123 catch (HRESULT hrcXcpt) { hrcOut = hrcXcpt; }
10124
10125 VDDestroy(hdd);
10126 }
10127 catch (HRESULT hrcXcpt) { hrcOut = hrcXcpt; }
10128
10129 ErrorInfoKeeper eik;
10130 MultiResult mrc(hrcOut);
10131
10132 // now, at the end of this task (always asynchronous), save the settings
10133 if (SUCCEEDED(mrc))
10134 {
10135 // save the settings
10136 i_markRegistriesModified();
10137 /* collect multiple errors */
10138 eik.restore();
10139 m->pVirtualBox->i_saveModifiedRegistries();
10140 eik.fetch();
10141 }
10142
10143 /* Everything is explicitly unlocked when the task exits,
10144 * as the task destruction also destroys the source chain. */
10145
10146 task.mpMediumLockList->Clear();
10147
10148 if (task.NotifyAboutChanges() && SUCCEEDED(mrc))
10149 m->pVirtualBox->i_onMediumConfigChanged(this);
10150
10151 LogFlowFunc(("LEAVE: mrc=%Rhrc\n", (HRESULT)mrc));
10152 return mrc;
10153}
10154
10155/**
10156 * Implementation code for the "delete" task.
10157 *
10158 * This task always gets started from Medium::deleteStorage() and can run
10159 * synchronously or asynchronously depending on the "wait" parameter passed to
10160 * that function.
10161 *
10162 * @param task
10163 * @return
10164 */
10165HRESULT Medium::i_taskDeleteHandler(Medium::DeleteTask &task)
10166{
10167 NOREF(task);
10168 HRESULT hrc = S_OK;
10169
10170 try
10171 {
10172 /* The lock is also used as a signal from the task initiator (which
10173 * releases it only after RTThreadCreate()) that we can start the job */
10174 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10175
10176 PVDISK hdd;
10177 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10178 ComAssertRCThrow(vrc, E_FAIL);
10179
10180 Utf8Str format(m->strFormat);
10181 Utf8Str location(m->strLocationFull);
10182
10183 /* unlock before the potentially lengthy operation */
10184 Assert(m->state == MediumState_Deleting);
10185 thisLock.release();
10186
10187 try
10188 {
10189 vrc = VDOpen(hdd,
10190 format.c_str(),
10191 location.c_str(),
10192 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
10193 m->vdImageIfaces);
10194 if (RT_SUCCESS(vrc))
10195 vrc = VDClose(hdd, true /* fDelete */);
10196
10197 if (RT_FAILURE(vrc) && vrc != VERR_FILE_NOT_FOUND)
10198 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10199 tr("Could not delete the medium storage unit '%s'%s"),
10200 location.c_str(), i_vdError(vrc).c_str());
10201
10202 }
10203 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10204
10205 VDDestroy(hdd);
10206 }
10207 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10208
10209 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10210
10211 /* go to the NotCreated state even on failure since the storage
10212 * may have been already partially deleted and cannot be used any
10213 * more. One will be able to manually re-open the storage if really
10214 * needed to re-register it. */
10215 m->state = MediumState_NotCreated;
10216
10217 /* Reset UUID to prevent Create* from reusing it again */
10218 com::Guid uOldId = m->id;
10219 unconst(m->id).clear();
10220
10221 if (task.NotifyAboutChanges() && SUCCEEDED(hrc))
10222 {
10223 if (m->pParent.isNotNull())
10224 m->pVirtualBox->i_onMediumConfigChanged(m->pParent);
10225 m->pVirtualBox->i_onMediumRegistered(uOldId, m->devType, FALSE);
10226 }
10227
10228 return hrc;
10229}
10230
10231/**
10232 * Implementation code for the "reset" task.
10233 *
10234 * This always gets started asynchronously from Medium::Reset().
10235 *
10236 * @param task
10237 * @return
10238 */
10239HRESULT Medium::i_taskResetHandler(Medium::ResetTask &task)
10240{
10241 HRESULT hrc = S_OK;
10242
10243 uint64_t size = 0, logicalSize = 0;
10244 MediumVariant_T variant = MediumVariant_Standard;
10245
10246 try
10247 {
10248 /* The lock is also used as a signal from the task initiator (which
10249 * releases it only after RTThreadCreate()) that we can start the job */
10250 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10251
10252 /// @todo Below we use a pair of delete/create operations to reset
10253 /// the diff contents but the most efficient way will of course be
10254 /// to add a VDResetDiff() API call
10255
10256 PVDISK hdd;
10257 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10258 ComAssertRCThrow(vrc, E_FAIL);
10259
10260 Guid id = m->id;
10261 Utf8Str format(m->strFormat);
10262 Utf8Str location(m->strLocationFull);
10263
10264 Medium *pParent = m->pParent;
10265 Guid parentId = pParent->m->id;
10266 Utf8Str parentFormat(pParent->m->strFormat);
10267 Utf8Str parentLocation(pParent->m->strLocationFull);
10268
10269 Assert(m->state == MediumState_LockedWrite);
10270
10271 /* unlock before the potentially lengthy operation */
10272 thisLock.release();
10273
10274 try
10275 {
10276 /* Open all media in the target chain but the last. */
10277 MediumLockList::Base::const_iterator targetListBegin =
10278 task.mpMediumLockList->GetBegin();
10279 MediumLockList::Base::const_iterator targetListEnd =
10280 task.mpMediumLockList->GetEnd();
10281 for (MediumLockList::Base::const_iterator it = targetListBegin;
10282 it != targetListEnd;
10283 ++it)
10284 {
10285 const MediumLock &mediumLock = *it;
10286 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
10287
10288 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
10289
10290 /* sanity check, "this" is checked above */
10291 Assert( pMedium == this
10292 || pMedium->m->state == MediumState_LockedRead);
10293
10294 /* Open all media in appropriate mode. */
10295 vrc = VDOpen(hdd,
10296 pMedium->m->strFormat.c_str(),
10297 pMedium->m->strLocationFull.c_str(),
10298 VD_OPEN_FLAGS_READONLY | m->uOpenFlagsDef,
10299 pMedium->m->vdImageIfaces);
10300 if (RT_FAILURE(vrc))
10301 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10302 tr("Could not open the medium storage unit '%s'%s"),
10303 pMedium->m->strLocationFull.c_str(),
10304 i_vdError(vrc).c_str());
10305
10306 /* Done when we hit the media which should be reset */
10307 if (pMedium == this)
10308 break;
10309 }
10310
10311 /* first, delete the storage unit */
10312 vrc = VDClose(hdd, true /* fDelete */);
10313 if (RT_FAILURE(vrc))
10314 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10315 tr("Could not delete the medium storage unit '%s'%s"),
10316 location.c_str(), i_vdError(vrc).c_str());
10317
10318 /* next, create it again */
10319 vrc = VDOpen(hdd,
10320 parentFormat.c_str(),
10321 parentLocation.c_str(),
10322 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | m->uOpenFlagsDef,
10323 m->vdImageIfaces);
10324 if (RT_FAILURE(vrc))
10325 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10326 tr("Could not open the medium storage unit '%s'%s"),
10327 parentLocation.c_str(), i_vdError(vrc).c_str());
10328
10329 vrc = VDCreateDiff(hdd,
10330 format.c_str(),
10331 location.c_str(),
10332 /// @todo use the same medium variant as before
10333 VD_IMAGE_FLAGS_NONE,
10334 NULL,
10335 id.raw(),
10336 parentId.raw(),
10337 VD_OPEN_FLAGS_NORMAL,
10338 m->vdImageIfaces,
10339 task.mVDOperationIfaces);
10340 if (RT_FAILURE(vrc))
10341 {
10342 if (vrc == VERR_VD_INVALID_TYPE)
10343 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10344 tr("Parameters for creating the differencing medium storage unit '%s' are invalid%s"),
10345 location.c_str(), i_vdError(vrc).c_str());
10346 else
10347 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10348 tr("Could not create the differencing medium storage unit '%s'%s"),
10349 location.c_str(), i_vdError(vrc).c_str());
10350 }
10351
10352 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
10353 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
10354 unsigned uImageFlags;
10355 vrc = VDGetImageFlags(hdd, 0, &uImageFlags);
10356 if (RT_SUCCESS(vrc))
10357 variant = (MediumVariant_T)uImageFlags;
10358 }
10359 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10360
10361 VDDestroy(hdd);
10362 }
10363 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10364
10365 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10366
10367 m->size = size;
10368 m->logicalSize = logicalSize;
10369 m->variant = variant;
10370
10371 if (task.NotifyAboutChanges() && SUCCEEDED(hrc))
10372 m->pVirtualBox->i_onMediumConfigChanged(this);
10373
10374 /* Everything is explicitly unlocked when the task exits,
10375 * as the task destruction also destroys the media chain. */
10376
10377 return hrc;
10378}
10379
10380/**
10381 * Implementation code for the "compact" task.
10382 *
10383 * @param task
10384 * @return
10385 */
10386HRESULT Medium::i_taskCompactHandler(Medium::CompactTask &task)
10387{
10388 HRESULT hrc = S_OK;
10389
10390 /* Lock all in {parent,child} order. The lock is also used as a
10391 * signal from the task initiator (which releases it only after
10392 * RTThreadCreate()) that we can start the job. */
10393 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10394
10395 try
10396 {
10397 PVDISK hdd;
10398 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10399 ComAssertRCThrow(vrc, E_FAIL);
10400
10401 try
10402 {
10403 /* Open all media in the chain. */
10404 MediumLockList::Base::const_iterator mediumListBegin =
10405 task.mpMediumLockList->GetBegin();
10406 MediumLockList::Base::const_iterator mediumListEnd =
10407 task.mpMediumLockList->GetEnd();
10408 MediumLockList::Base::const_iterator mediumListLast =
10409 mediumListEnd;
10410 --mediumListLast;
10411 for (MediumLockList::Base::const_iterator it = mediumListBegin;
10412 it != mediumListEnd;
10413 ++it)
10414 {
10415 const MediumLock &mediumLock = *it;
10416 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
10417 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
10418
10419 /* sanity check */
10420 if (it == mediumListLast)
10421 Assert(pMedium->m->state == MediumState_LockedWrite);
10422 else
10423 Assert(pMedium->m->state == MediumState_LockedRead);
10424
10425 /* Open all media but last in read-only mode. Do not handle
10426 * shareable media, as compaction and sharing are mutually
10427 * exclusive. */
10428 vrc = VDOpen(hdd,
10429 pMedium->m->strFormat.c_str(),
10430 pMedium->m->strLocationFull.c_str(),
10431 m->uOpenFlagsDef | (it == mediumListLast ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY),
10432 pMedium->m->vdImageIfaces);
10433 if (RT_FAILURE(vrc))
10434 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10435 tr("Could not open the medium storage unit '%s'%s"),
10436 pMedium->m->strLocationFull.c_str(),
10437 i_vdError(vrc).c_str());
10438 }
10439
10440 Assert(m->state == MediumState_LockedWrite);
10441
10442 Utf8Str location(m->strLocationFull);
10443
10444 /* unlock before the potentially lengthy operation */
10445 thisLock.release();
10446
10447 vrc = VDCompact(hdd, VD_LAST_IMAGE, task.mVDOperationIfaces);
10448 if (RT_FAILURE(vrc))
10449 {
10450 if (vrc == VERR_NOT_SUPPORTED)
10451 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
10452 tr("Compacting is not yet supported for medium '%s'"),
10453 location.c_str());
10454 else if (vrc == VERR_NOT_IMPLEMENTED)
10455 throw setErrorBoth(E_NOTIMPL, vrc,
10456 tr("Compacting is not implemented, medium '%s'"),
10457 location.c_str());
10458 else
10459 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10460 tr("Could not compact medium '%s'%s"),
10461 location.c_str(),
10462 i_vdError(vrc).c_str());
10463 }
10464 }
10465 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10466
10467 VDDestroy(hdd);
10468 }
10469 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10470
10471 if (task.NotifyAboutChanges() && SUCCEEDED(hrc))
10472 m->pVirtualBox->i_onMediumConfigChanged(this);
10473
10474 /* Everything is explicitly unlocked when the task exits,
10475 * as the task destruction also destroys the media chain. */
10476
10477 return hrc;
10478}
10479
10480/**
10481 * Implementation code for the "resize" task.
10482 *
10483 * @param task
10484 * @return
10485 */
10486HRESULT Medium::i_taskResizeHandler(Medium::ResizeTask &task)
10487{
10488 HRESULT hrc = S_OK;
10489
10490 uint64_t size = 0, logicalSize = 0;
10491
10492 try
10493 {
10494 /* The lock is also used as a signal from the task initiator (which
10495 * releases it only after RTThreadCreate()) that we can start the job */
10496 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10497
10498 PVDISK hdd;
10499 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10500 ComAssertRCThrow(vrc, E_FAIL);
10501
10502 try
10503 {
10504 /* Open all media in the chain. */
10505 MediumLockList::Base::const_iterator mediumListBegin =
10506 task.mpMediumLockList->GetBegin();
10507 MediumLockList::Base::const_iterator mediumListEnd =
10508 task.mpMediumLockList->GetEnd();
10509 MediumLockList::Base::const_iterator mediumListLast =
10510 mediumListEnd;
10511 --mediumListLast;
10512 for (MediumLockList::Base::const_iterator it = mediumListBegin;
10513 it != mediumListEnd;
10514 ++it)
10515 {
10516 const MediumLock &mediumLock = *it;
10517 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
10518 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
10519
10520 /* sanity check */
10521 if (it == mediumListLast)
10522 Assert(pMedium->m->state == MediumState_LockedWrite);
10523 else
10524 Assert(pMedium->m->state == MediumState_LockedRead ||
10525 // Allow resize the target image during mergeTo in case
10526 // of direction from parent to child because all intermediate
10527 // images are marked to MediumState_Deleting and will be
10528 // destroyed after successful merge
10529 pMedium->m->state == MediumState_Deleting);
10530
10531 /* Open all media but last in read-only mode. Do not handle
10532 * shareable media, as compaction and sharing are mutually
10533 * exclusive. */
10534 vrc = VDOpen(hdd,
10535 pMedium->m->strFormat.c_str(),
10536 pMedium->m->strLocationFull.c_str(),
10537 m->uOpenFlagsDef | (it == mediumListLast ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY),
10538 pMedium->m->vdImageIfaces);
10539 if (RT_FAILURE(vrc))
10540 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10541 tr("Could not open the medium storage unit '%s'%s"),
10542 pMedium->m->strLocationFull.c_str(),
10543 i_vdError(vrc).c_str());
10544 }
10545
10546 Assert(m->state == MediumState_LockedWrite);
10547
10548 Utf8Str location(m->strLocationFull);
10549
10550 /* unlock before the potentially lengthy operation */
10551 thisLock.release();
10552
10553 VDGEOMETRY geo = {0, 0, 0}; /* auto */
10554 vrc = VDResize(hdd, task.mSize, &geo, &geo, task.mVDOperationIfaces);
10555 if (RT_FAILURE(vrc))
10556 {
10557 if (vrc == VERR_VD_SHRINK_NOT_SUPPORTED)
10558 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
10559 tr("Shrinking is not yet supported for medium '%s'"),
10560 location.c_str());
10561 if (vrc == VERR_NOT_SUPPORTED)
10562 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
10563 tr("Resizing to new size %llu is not yet supported for medium '%s'"),
10564 task.mSize, location.c_str());
10565 else if (vrc == VERR_NOT_IMPLEMENTED)
10566 throw setErrorBoth(E_NOTIMPL, vrc,
10567 tr("Resizing is not implemented, medium '%s'"),
10568 location.c_str());
10569 else
10570 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10571 tr("Could not resize medium '%s'%s"),
10572 location.c_str(),
10573 i_vdError(vrc).c_str());
10574 }
10575 size = VDGetFileSize(hdd, VD_LAST_IMAGE);
10576 logicalSize = VDGetSize(hdd, VD_LAST_IMAGE);
10577 }
10578 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10579
10580 VDDestroy(hdd);
10581 }
10582 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
10583
10584 if (SUCCEEDED(hrc))
10585 {
10586 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10587 m->size = size;
10588 m->logicalSize = logicalSize;
10589
10590 if (task.NotifyAboutChanges())
10591 m->pVirtualBox->i_onMediumConfigChanged(this);
10592 }
10593
10594 /* Everything is explicitly unlocked when the task exits,
10595 * as the task destruction also destroys the media chain. */
10596
10597 return hrc;
10598}
10599
10600/**
10601 * Implementation code for the "import" task.
10602 *
10603 * This only gets started from Medium::importFile() and always runs
10604 * asynchronously. It potentially touches the media registry, so we
10605 * always save the VirtualBox.xml file when we're done here.
10606 *
10607 * @param task
10608 * @return
10609 */
10610HRESULT Medium::i_taskImportHandler(Medium::ImportTask &task)
10611{
10612 /** @todo r=klaus The code below needs to be double checked with regard
10613 * to lock order violations, it probably causes lock order issues related
10614 * to the AutoCaller usage. */
10615 HRESULT hrcTmp = S_OK;
10616
10617 const ComObjPtr<Medium> &pParent = task.mParent;
10618
10619 bool fCreatingTarget = false;
10620
10621 uint64_t size = 0, logicalSize = 0;
10622 MediumVariant_T variant = MediumVariant_Standard;
10623 bool fGenerateUuid = false;
10624
10625 try
10626 {
10627 if (!pParent.isNull())
10628 if (pParent->i_getDepth() >= SETTINGS_MEDIUM_DEPTH_MAX)
10629 {
10630 AutoReadLock plock(pParent COMMA_LOCKVAL_SRC_POS);
10631 throw setError(VBOX_E_INVALID_OBJECT_STATE,
10632 tr("Cannot import image for medium '%s', because it exceeds the medium tree depth limit. Please merge some images which you no longer need"),
10633 pParent->m->strLocationFull.c_str());
10634 }
10635
10636 /* Lock all in {parent,child} order. The lock is also used as a
10637 * signal from the task initiator (which releases it only after
10638 * RTThreadCreate()) that we can start the job. */
10639 AutoMultiWriteLock2 thisLock(this, pParent COMMA_LOCKVAL_SRC_POS);
10640
10641 fCreatingTarget = m->state == MediumState_Creating;
10642
10643 /* The object may request a specific UUID (through a special form of
10644 * the moveTo() argument). Otherwise we have to generate it */
10645 Guid targetId = m->id;
10646
10647 fGenerateUuid = targetId.isZero();
10648 if (fGenerateUuid)
10649 {
10650 targetId.create();
10651 /* VirtualBox::i_registerMedium() will need UUID */
10652 unconst(m->id) = targetId;
10653 }
10654
10655
10656 PVDISK hdd;
10657 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &hdd);
10658 ComAssertRCThrow(vrc, E_FAIL);
10659
10660 try
10661 {
10662 /* Open source medium. */
10663 vrc = VDOpen(hdd,
10664 task.mFormat->i_getId().c_str(),
10665 task.mFilename.c_str(),
10666 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_SEQUENTIAL | m->uOpenFlagsDef,
10667 task.mVDImageIfaces);
10668 if (RT_FAILURE(vrc))
10669 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10670 tr("Could not open the medium storage unit '%s'%s"),
10671 task.mFilename.c_str(),
10672 i_vdError(vrc).c_str());
10673
10674 Utf8Str targetFormat(m->strFormat);
10675 Utf8Str targetLocation(m->strLocationFull);
10676 uint64_t capabilities = task.mFormat->i_getCapabilities();
10677
10678 Assert( m->state == MediumState_Creating
10679 || m->state == MediumState_LockedWrite);
10680 Assert( pParent.isNull()
10681 || pParent->m->state == MediumState_LockedRead);
10682
10683 /* unlock before the potentially lengthy operation */
10684 thisLock.release();
10685
10686 /* ensure the target directory exists */
10687 if (capabilities & MediumFormatCapabilities_File)
10688 {
10689 HRESULT hrc = VirtualBox::i_ensureFilePathExists(targetLocation,
10690 !(task.mVariant & MediumVariant_NoCreateDir) /* fCreate */);
10691 if (FAILED(hrc))
10692 throw hrc;
10693 }
10694
10695 PVDISK targetHdd;
10696 vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &targetHdd);
10697 ComAssertRCThrow(vrc, E_FAIL);
10698
10699 try
10700 {
10701 /* Open all media in the target chain. */
10702 MediumLockList::Base::const_iterator targetListBegin =
10703 task.mpTargetMediumLockList->GetBegin();
10704 MediumLockList::Base::const_iterator targetListEnd =
10705 task.mpTargetMediumLockList->GetEnd();
10706 for (MediumLockList::Base::const_iterator it = targetListBegin;
10707 it != targetListEnd;
10708 ++it)
10709 {
10710 const MediumLock &mediumLock = *it;
10711 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
10712
10713 /* If the target medium is not created yet there's no
10714 * reason to open it. */
10715 if (pMedium == this && fCreatingTarget)
10716 continue;
10717
10718 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
10719
10720 /* sanity check */
10721 Assert( pMedium->m->state == MediumState_LockedRead
10722 || pMedium->m->state == MediumState_LockedWrite);
10723
10724 unsigned uOpenFlags = VD_OPEN_FLAGS_NORMAL;
10725 if (pMedium->m->state != MediumState_LockedWrite)
10726 uOpenFlags = VD_OPEN_FLAGS_READONLY;
10727 if (pMedium->m->type == MediumType_Shareable)
10728 uOpenFlags |= VD_OPEN_FLAGS_SHAREABLE;
10729
10730 /* Open all media in appropriate mode. */
10731 vrc = VDOpen(targetHdd,
10732 pMedium->m->strFormat.c_str(),
10733 pMedium->m->strLocationFull.c_str(),
10734 uOpenFlags | m->uOpenFlagsDef,
10735 pMedium->m->vdImageIfaces);
10736 if (RT_FAILURE(vrc))
10737 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10738 tr("Could not open the medium storage unit '%s'%s"),
10739 pMedium->m->strLocationFull.c_str(),
10740 i_vdError(vrc).c_str());
10741 }
10742
10743 vrc = VDCopy(hdd,
10744 VD_LAST_IMAGE,
10745 targetHdd,
10746 targetFormat.c_str(),
10747 (fCreatingTarget) ? targetLocation.c_str() : (char *)NULL,
10748 false /* fMoveByRename */,
10749 0 /* cbSize */,
10750 task.mVariant & ~(MediumVariant_NoCreateDir | MediumVariant_Formatted | MediumVariant_VmdkESX | MediumVariant_VmdkRawDisk),
10751 targetId.raw(),
10752 VD_OPEN_FLAGS_NORMAL,
10753 NULL /* pVDIfsOperation */,
10754 m->vdImageIfaces,
10755 task.mVDOperationIfaces);
10756 if (RT_FAILURE(vrc))
10757 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
10758 tr("Could not create the imported medium '%s'%s"),
10759 targetLocation.c_str(), i_vdError(vrc).c_str());
10760
10761 size = VDGetFileSize(targetHdd, VD_LAST_IMAGE);
10762 logicalSize = VDGetSize(targetHdd, VD_LAST_IMAGE);
10763 unsigned uImageFlags;
10764 vrc = VDGetImageFlags(targetHdd, 0, &uImageFlags);
10765 if (RT_SUCCESS(vrc))
10766 variant = (MediumVariant_T)uImageFlags;
10767 }
10768 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
10769
10770 VDDestroy(targetHdd);
10771 }
10772 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
10773
10774 VDDestroy(hdd);
10775 }
10776 catch (HRESULT hrcXcpt) { hrcTmp = hrcXcpt; }
10777
10778 ErrorInfoKeeper eik;
10779 MultiResult mrc(hrcTmp);
10780
10781 /* Only do the parent changes for newly created media. */
10782 if (SUCCEEDED(mrc) && fCreatingTarget)
10783 {
10784 /* we set m->pParent & children() */
10785 AutoWriteLock treeLock(m->pVirtualBox->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
10786
10787 Assert(m->pParent.isNull());
10788
10789 if (pParent)
10790 {
10791 /* Associate the imported medium with the parent and deassociate
10792 * from VirtualBox. Depth check above. */
10793 i_setParent(pParent);
10794
10795 /* register with mVirtualBox as the last step and move to
10796 * Created state only on success (leaving an orphan file is
10797 * better than breaking media registry consistency) */
10798 eik.restore();
10799 ComObjPtr<Medium> pMedium;
10800 mrc = pParent->m->pVirtualBox->i_registerMedium(this, &pMedium,
10801 treeLock);
10802 Assert(this == pMedium);
10803 eik.fetch();
10804
10805 if (FAILED(mrc))
10806 /* break parent association on failure to register */
10807 this->i_deparent(); // removes target from parent
10808 }
10809 else
10810 {
10811 /* just register */
10812 eik.restore();
10813 ComObjPtr<Medium> pMedium;
10814 mrc = m->pVirtualBox->i_registerMedium(this, &pMedium, treeLock);
10815 Assert(this == pMedium);
10816 eik.fetch();
10817 }
10818 }
10819
10820 if (fCreatingTarget)
10821 {
10822 AutoWriteLock mLock(this COMMA_LOCKVAL_SRC_POS);
10823
10824 if (SUCCEEDED(mrc))
10825 {
10826 m->state = MediumState_Created;
10827
10828 m->size = size;
10829 m->logicalSize = logicalSize;
10830 m->variant = variant;
10831 }
10832 else
10833 {
10834 /* back to NotCreated on failure */
10835 m->state = MediumState_NotCreated;
10836
10837 /* reset UUID to prevent it from being reused next time */
10838 if (fGenerateUuid)
10839 unconst(m->id).clear();
10840 }
10841 }
10842
10843 // now, at the end of this task (always asynchronous), save the settings
10844 {
10845 // save the settings
10846 i_markRegistriesModified();
10847 /* collect multiple errors */
10848 eik.restore();
10849 m->pVirtualBox->i_saveModifiedRegistries();
10850 eik.fetch();
10851 }
10852
10853 /* Everything is explicitly unlocked when the task exits,
10854 * as the task destruction also destroys the target chain. */
10855
10856 /* Make sure the target chain is released early, otherwise it can
10857 * lead to deadlocks with concurrent IAppliance activities. */
10858 task.mpTargetMediumLockList->Clear();
10859
10860 if (task.NotifyAboutChanges() && SUCCEEDED(mrc))
10861 {
10862 if (pParent)
10863 m->pVirtualBox->i_onMediumConfigChanged(pParent);
10864 if (fCreatingTarget)
10865 m->pVirtualBox->i_onMediumConfigChanged(this);
10866 else
10867 m->pVirtualBox->i_onMediumRegistered(m->id, m->devType, TRUE);
10868 }
10869
10870 return mrc;
10871}
10872
10873/**
10874 * Sets up the encryption settings for a filter.
10875 */
10876void Medium::i_taskEncryptSettingsSetup(MediumCryptoFilterSettings *pSettings, const char *pszCipher,
10877 const char *pszKeyStore, const char *pszPassword,
10878 bool fCreateKeyStore)
10879{
10880 pSettings->pszCipher = pszCipher;
10881 pSettings->pszPassword = pszPassword;
10882 pSettings->pszKeyStoreLoad = pszKeyStore;
10883 pSettings->fCreateKeyStore = fCreateKeyStore;
10884 pSettings->pbDek = NULL;
10885 pSettings->cbDek = 0;
10886 pSettings->vdFilterIfaces = NULL;
10887
10888 pSettings->vdIfCfg.pfnAreKeysValid = i_vdCryptoConfigAreKeysValid;
10889 pSettings->vdIfCfg.pfnQuerySize = i_vdCryptoConfigQuerySize;
10890 pSettings->vdIfCfg.pfnQuery = i_vdCryptoConfigQuery;
10891 pSettings->vdIfCfg.pfnQueryBytes = NULL;
10892
10893 pSettings->vdIfCrypto.pfnKeyRetain = i_vdCryptoKeyRetain;
10894 pSettings->vdIfCrypto.pfnKeyRelease = i_vdCryptoKeyRelease;
10895 pSettings->vdIfCrypto.pfnKeyStorePasswordRetain = i_vdCryptoKeyStorePasswordRetain;
10896 pSettings->vdIfCrypto.pfnKeyStorePasswordRelease = i_vdCryptoKeyStorePasswordRelease;
10897 pSettings->vdIfCrypto.pfnKeyStoreSave = i_vdCryptoKeyStoreSave;
10898 pSettings->vdIfCrypto.pfnKeyStoreReturnParameters = i_vdCryptoKeyStoreReturnParameters;
10899
10900 int vrc = VDInterfaceAdd(&pSettings->vdIfCfg.Core,
10901 "Medium::vdInterfaceCfgCrypto",
10902 VDINTERFACETYPE_CONFIG, pSettings,
10903 sizeof(VDINTERFACECONFIG), &pSettings->vdFilterIfaces);
10904 AssertRC(vrc);
10905
10906 vrc = VDInterfaceAdd(&pSettings->vdIfCrypto.Core,
10907 "Medium::vdInterfaceCrypto",
10908 VDINTERFACETYPE_CRYPTO, pSettings,
10909 sizeof(VDINTERFACECRYPTO), &pSettings->vdFilterIfaces);
10910 AssertRC(vrc);
10911}
10912
10913/**
10914 * Implementation code for the "encrypt" task.
10915 *
10916 * @param task
10917 * @return
10918 */
10919HRESULT Medium::i_taskEncryptHandler(Medium::EncryptTask &task)
10920{
10921# ifndef VBOX_WITH_EXTPACK
10922 RT_NOREF(task);
10923# endif
10924 HRESULT hrc = S_OK;
10925
10926 /* Lock all in {parent,child} order. The lock is also used as a
10927 * signal from the task initiator (which releases it only after
10928 * RTThreadCreate()) that we can start the job. */
10929 ComObjPtr<Medium> pBase = i_getBase();
10930 AutoWriteLock thisLock(this COMMA_LOCKVAL_SRC_POS);
10931
10932 try
10933 {
10934# ifdef VBOX_WITH_EXTPACK
10935 ExtPackManager *pExtPackManager = m->pVirtualBox->i_getExtPackManager();
10936 if (pExtPackManager->i_isExtPackUsable(ORACLE_PUEL_EXTPACK_NAME))
10937 {
10938 /* Load the plugin */
10939 Utf8Str strPlugin;
10940 hrc = pExtPackManager->i_getLibraryPathForExtPack(g_szVDPlugin, ORACLE_PUEL_EXTPACK_NAME, &strPlugin);
10941 if (SUCCEEDED(hrc))
10942 {
10943 int vrc = VDPluginLoadFromFilename(strPlugin.c_str());
10944 if (RT_FAILURE(vrc))
10945 throw setErrorBoth(VBOX_E_NOT_SUPPORTED, vrc,
10946 tr("Encrypting the image failed because the encryption plugin could not be loaded (%s)"),
10947 i_vdError(vrc).c_str());
10948 }
10949 else
10950 throw setError(VBOX_E_NOT_SUPPORTED,
10951 tr("Encryption is not supported because the extension pack '%s' is missing the encryption plugin (old extension pack installed?)"),
10952 ORACLE_PUEL_EXTPACK_NAME);
10953 }
10954 else
10955 throw setError(VBOX_E_NOT_SUPPORTED,
10956 tr("Encryption is not supported because the extension pack '%s' is missing"),
10957 ORACLE_PUEL_EXTPACK_NAME);
10958
10959 PVDISK pDisk = NULL;
10960 int vrc = VDCreate(m->vdDiskIfaces, i_convertDeviceType(), &pDisk);
10961 ComAssertRCThrow(vrc, E_FAIL);
10962
10963 MediumCryptoFilterSettings CryptoSettingsRead;
10964 MediumCryptoFilterSettings CryptoSettingsWrite;
10965
10966 void *pvBuf = NULL;
10967 const char *pszPasswordNew = NULL;
10968 try
10969 {
10970 /* Set up disk encryption filters. */
10971 if (task.mstrCurrentPassword.isEmpty())
10972 {
10973 /*
10974 * Query whether the medium property indicating that encryption is
10975 * configured is existing.
10976 */
10977 settings::StringsMap::iterator it = pBase->m->mapProperties.find("CRYPT/KeyStore");
10978 if (it != pBase->m->mapProperties.end())
10979 throw setError(VBOX_E_PASSWORD_INCORRECT,
10980 tr("The password given for the encrypted image is incorrect"));
10981 }
10982 else
10983 {
10984 settings::StringsMap::iterator it = pBase->m->mapProperties.find("CRYPT/KeyStore");
10985 if (it == pBase->m->mapProperties.end())
10986 throw setError(VBOX_E_INVALID_OBJECT_STATE,
10987 tr("The image is not configured for encryption"));
10988
10989 i_taskEncryptSettingsSetup(&CryptoSettingsRead, NULL, it->second.c_str(), task.mstrCurrentPassword.c_str(),
10990 false /* fCreateKeyStore */);
10991 vrc = VDFilterAdd(pDisk, "CRYPT", VD_FILTER_FLAGS_READ, CryptoSettingsRead.vdFilterIfaces);
10992 if (vrc == VERR_VD_PASSWORD_INCORRECT)
10993 throw setError(VBOX_E_PASSWORD_INCORRECT,
10994 tr("The password to decrypt the image is incorrect"));
10995 else if (RT_FAILURE(vrc))
10996 throw setError(VBOX_E_INVALID_OBJECT_STATE,
10997 tr("Failed to load the decryption filter: %s"),
10998 i_vdError(vrc).c_str());
10999 }
11000
11001 if (task.mstrCipher.isNotEmpty())
11002 {
11003 if ( task.mstrNewPassword.isEmpty()
11004 && task.mstrNewPasswordId.isEmpty()
11005 && task.mstrCurrentPassword.isNotEmpty())
11006 {
11007 /* An empty password and password ID will default to the current password. */
11008 pszPasswordNew = task.mstrCurrentPassword.c_str();
11009 }
11010 else if (task.mstrNewPassword.isEmpty())
11011 throw setError(VBOX_E_OBJECT_NOT_FOUND,
11012 tr("A password must be given for the image encryption"));
11013 else if (task.mstrNewPasswordId.isEmpty())
11014 throw setError(VBOX_E_INVALID_OBJECT_STATE,
11015 tr("A valid identifier for the password must be given"));
11016 else
11017 pszPasswordNew = task.mstrNewPassword.c_str();
11018
11019 i_taskEncryptSettingsSetup(&CryptoSettingsWrite, task.mstrCipher.c_str(), NULL,
11020 pszPasswordNew, true /* fCreateKeyStore */);
11021 vrc = VDFilterAdd(pDisk, "CRYPT", VD_FILTER_FLAGS_WRITE, CryptoSettingsWrite.vdFilterIfaces);
11022 if (RT_FAILURE(vrc))
11023 throw setErrorBoth(VBOX_E_INVALID_OBJECT_STATE, vrc,
11024 tr("Failed to load the encryption filter: %s"),
11025 i_vdError(vrc).c_str());
11026 }
11027 else if (task.mstrNewPasswordId.isNotEmpty() || task.mstrNewPassword.isNotEmpty())
11028 throw setError(VBOX_E_INVALID_OBJECT_STATE,
11029 tr("The password and password identifier must be empty if the output should be unencrypted"));
11030
11031 /* Open all media in the chain. */
11032 MediumLockList::Base::const_iterator mediumListBegin =
11033 task.mpMediumLockList->GetBegin();
11034 MediumLockList::Base::const_iterator mediumListEnd =
11035 task.mpMediumLockList->GetEnd();
11036 MediumLockList::Base::const_iterator mediumListLast =
11037 mediumListEnd;
11038 --mediumListLast;
11039 for (MediumLockList::Base::const_iterator it = mediumListBegin;
11040 it != mediumListEnd;
11041 ++it)
11042 {
11043 const MediumLock &mediumLock = *it;
11044 const ComObjPtr<Medium> &pMedium = mediumLock.GetMedium();
11045 AutoReadLock alock(pMedium COMMA_LOCKVAL_SRC_POS);
11046
11047 Assert(pMedium->m->state == MediumState_LockedWrite);
11048
11049 /* Open all media but last in read-only mode. Do not handle
11050 * shareable media, as compaction and sharing are mutually
11051 * exclusive. */
11052 vrc = VDOpen(pDisk,
11053 pMedium->m->strFormat.c_str(),
11054 pMedium->m->strLocationFull.c_str(),
11055 m->uOpenFlagsDef | (it == mediumListLast ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY),
11056 pMedium->m->vdImageIfaces);
11057 if (RT_FAILURE(vrc))
11058 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
11059 tr("Could not open the medium storage unit '%s'%s"),
11060 pMedium->m->strLocationFull.c_str(),
11061 i_vdError(vrc).c_str());
11062 }
11063
11064 Assert(m->state == MediumState_LockedWrite);
11065
11066 Utf8Str location(m->strLocationFull);
11067
11068 /* unlock before the potentially lengthy operation */
11069 thisLock.release();
11070
11071 vrc = VDPrepareWithFilters(pDisk, task.mVDOperationIfaces);
11072 if (RT_FAILURE(vrc))
11073 throw setErrorBoth(VBOX_E_FILE_ERROR, vrc,
11074 tr("Could not prepare disk images for encryption (%Rrc): %s"),
11075 vrc, i_vdError(vrc).c_str());
11076
11077 thisLock.acquire();
11078 /* If everything went well set the new key store. */
11079 settings::StringsMap::iterator it = pBase->m->mapProperties.find("CRYPT/KeyStore");
11080 if (it != pBase->m->mapProperties.end())
11081 pBase->m->mapProperties.erase(it);
11082
11083 /* Delete KeyId if encryption is removed or the password did change. */
11084 if ( task.mstrNewPasswordId.isNotEmpty()
11085 || task.mstrCipher.isEmpty())
11086 {
11087 it = pBase->m->mapProperties.find("CRYPT/KeyId");
11088 if (it != pBase->m->mapProperties.end())
11089 pBase->m->mapProperties.erase(it);
11090 }
11091
11092 if (CryptoSettingsWrite.pszKeyStore)
11093 {
11094 pBase->m->mapProperties["CRYPT/KeyStore"] = Utf8Str(CryptoSettingsWrite.pszKeyStore);
11095 if (task.mstrNewPasswordId.isNotEmpty())
11096 pBase->m->mapProperties["CRYPT/KeyId"] = task.mstrNewPasswordId;
11097 }
11098
11099 if (CryptoSettingsRead.pszCipherReturned)
11100 RTStrFree(CryptoSettingsRead.pszCipherReturned);
11101
11102 if (CryptoSettingsWrite.pszCipherReturned)
11103 RTStrFree(CryptoSettingsWrite.pszCipherReturned);
11104
11105 thisLock.release();
11106 pBase->i_markRegistriesModified();
11107 m->pVirtualBox->i_saveModifiedRegistries();
11108 }
11109 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
11110
11111 if (pvBuf)
11112 RTMemFree(pvBuf);
11113
11114 VDDestroy(pDisk);
11115# else
11116 throw setError(VBOX_E_NOT_SUPPORTED,
11117 tr("Encryption is not supported because extension pack support is not built in"));
11118# endif
11119 }
11120 catch (HRESULT hrcXcpt) { hrc = hrcXcpt; }
11121
11122 /* Everything is explicitly unlocked when the task exits,
11123 * as the task destruction also destroys the media chain. */
11124
11125 return hrc;
11126}
11127
11128/* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note: See TracBrowser for help on using the repository browser.

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