VirtualBox

source: vbox/trunk/src/VBox/Main/HardDiskImpl.cpp@ 1804

Last change on this file since 1804 was 1553, checked in by vboxsync, 18 years ago

Main: More meaningful assertions.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 113.5 KB
Line 
1/** @file
2 *
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22#include "HardDiskImpl.h"
23#include "ProgressImpl.h"
24#include "VirtualBoxImpl.h"
25#include "SystemPropertiesImpl.h"
26#include "Logging.h"
27
28#include <iprt/string.h>
29#include <iprt/thread.h>
30#include <iprt/file.h>
31#include <iprt/path.h>
32#include <iprt/dir.h>
33#include <VBox/VBoxHDD.h>
34#include <VBox/err.h>
35
36#include <algorithm>
37
38#define CHECK_BUSY() \
39 do { \
40 if (isBusy()) \
41 return setError (E_UNEXPECTED, \
42 tr ("Hard disk '%ls' is being used by another task"), \
43 toString().raw()); \
44 } while (0)
45
46#define CHECK_BUSY_AND_READERS() \
47do { \
48 if (readers() > 0 || isBusy()) \
49 return setError (E_UNEXPECTED, \
50 tr ("Hard disk '%ls' is being used by another task"), \
51 toString().raw()); \
52} while (0)
53
54/** Task structure for asynchronous VDI operations */
55struct VDITask
56{
57 enum Op { CreateDynamic, CreateStatic, CloneToImage };
58
59 VDITask (Op op, HVirtualDiskImage *i, Progress *p)
60 : operation (op)
61 , vdi (i)
62 , progress (p)
63 {}
64
65 Op operation;
66 ComObjPtr <HVirtualDiskImage> vdi;
67 ComObjPtr <Progress> progress;
68
69 /* for CreateDynamic, CreateStatic */
70 uint64_t size;
71
72 /* for CloneToImage */
73 ComObjPtr <HardDisk> source;
74};
75
76/**
77 * Progress callback handler for VDI operations.
78 *
79 * @param uPercent Completetion precentage (0-100).
80 * @param pvUser Pointer to the Progress instance.
81 */
82static DECLCALLBACK(int) progressCallback (PVM /* pVM */, unsigned uPercent, void *pvUser)
83{
84 Progress *progress = static_cast <Progress *> (pvUser);
85
86 /* update the progress object */
87 if (progress)
88 progress->notifyProgress (uPercent);
89
90 return VINF_SUCCESS;
91}
92
93////////////////////////////////////////////////////////////////////////////////
94// HardDisk class
95////////////////////////////////////////////////////////////////////////////////
96
97// constructor / destructor
98////////////////////////////////////////////////////////////////////////////////
99
100/** Shold be called by subclasses from #FinalConstruct() */
101HRESULT HardDisk::FinalConstruct()
102{
103 mRegistered = FALSE;
104
105 mStorageType = HardDiskStorageType_VirtualDiskImage;
106 mType = HardDiskType_NormalHardDisk;
107
108 mBusy = false;
109 mReaders = 0;
110
111 return S_OK;
112}
113
114/**
115 * Shold be called by subclasses from #FinalRelease().
116 * Uninitializes this object by calling #uninit() if it's not yet done.
117 */
118void HardDisk::FinalRelease()
119{
120 uninit();
121}
122
123// protected initializer/uninitializer for internal purposes only
124////////////////////////////////////////////////////////////////////////////////
125
126/**
127 * Initializes the hard disk object.
128 *
129 * Subclasses should call this or any other #init() method from their
130 * init() implementations.
131 *
132 * @note
133 * This method doesn't do |isReady()| check and doesn't call
134 * |setReady (true)| on success!
135 * @note
136 * This method must be called from under the object's lock!
137 */
138HRESULT HardDisk::protectedInit (VirtualBox *aVirtualBox, HardDisk *aParent)
139{
140 LogFlowMember (("HardDisk::protectedInit (aParent=%p)\n", aParent));
141
142 ComAssertRet (aVirtualBox, E_INVALIDARG);
143
144 mVirtualBox = aVirtualBox;
145 mParent = aParent;
146
147 if (!aParent)
148 aVirtualBox->addDependentChild (this);
149 else
150 aParent->addDependentChild (this);
151
152 return S_OK;
153}
154
155/**
156 * Uninitializes the instance.
157 * Subclasses should call this from their uninit() implementations.
158 * The readiness flag must be true on input and will be set to false
159 * on output.
160 *
161 * @param alock this object's autolock
162 *
163 * @note
164 * Using mParent and mVirtualBox members after this method returns
165 * is forbidden.
166 */
167void HardDisk::protectedUninit (AutoLock &alock)
168{
169 LogFlowMember (("HardDisk::protectedUninit()\n"));
170
171 Assert (alock.belongsTo (this));
172 Assert (isReady());
173
174 /* uninit all children */
175 uninitDependentChildren();
176
177 setReady (false);
178
179 if (mParent)
180 mParent->removeDependentChild (this);
181 else
182 {
183 alock.leave();
184 mVirtualBox->removeDependentChild (this);
185 alock.enter();
186 }
187
188 mParent.setNull();
189 mVirtualBox.setNull();
190}
191
192// IHardDisk properties
193/////////////////////////////////////////////////////////////////////////////
194
195STDMETHODIMP HardDisk::COMGETTER(Id) (GUIDPARAMOUT aId)
196{
197 if (!aId)
198 return E_POINTER;
199
200 AutoLock alock (this);
201 CHECK_READY();
202
203 mId.cloneTo (aId);
204 return S_OK;
205}
206
207STDMETHODIMP HardDisk::COMGETTER(StorageType) (HardDiskStorageType_T *aStorageType)
208{
209 if (!aStorageType)
210 return E_POINTER;
211
212 AutoLock alock (this);
213 CHECK_READY();
214
215 *aStorageType = mStorageType;
216 return S_OK;
217}
218
219STDMETHODIMP HardDisk::COMGETTER(Location) (BSTR *aLocation)
220{
221 if (!aLocation)
222 return E_POINTER;
223
224 AutoLock alock (this);
225 CHECK_READY();
226
227 toString (false /* aShort */).cloneTo (aLocation);
228 return S_OK;
229}
230
231STDMETHODIMP HardDisk::COMGETTER(Type) (HardDiskType_T *aType)
232{
233 if (!aType)
234 return E_POINTER;
235
236 AutoLock alock (this);
237 CHECK_READY();
238
239 *aType = mType;
240 return S_OK;
241}
242
243STDMETHODIMP HardDisk::COMSETTER(Type) (HardDiskType_T aType)
244{
245 AutoLock alock (this);
246 CHECK_READY();
247
248 if (mRegistered)
249 return setError (E_FAIL,
250 tr ("You cannot change the type of the registered hard disk '%ls'"),
251 toString().raw());
252
253 if (mStorageType == HardDiskStorageType_ISCSIHardDisk)
254 return setError (E_FAIL,
255 tr ("Currently, changing the type of iSCSI hard disks is not allowed"));
256
257 /// @todo (dmik) later: allow to change the type on any registered hard disk
258 // depending on whether it is attached or not, has children etc.
259 // Don't forget to save hdd configuration afterwards.
260
261 mType = aType;
262 return S_OK;
263}
264
265STDMETHODIMP HardDisk::COMGETTER(Parent) (IHardDisk **aParent)
266{
267 if (!aParent)
268 return E_POINTER;
269
270 AutoLock alock (this);
271 CHECK_READY();
272
273 mParent.queryInterfaceTo (aParent);
274 return S_OK;
275}
276
277STDMETHODIMP HardDisk::COMGETTER(Children) (IHardDiskCollection **aChildren)
278{
279 if (!aChildren)
280 return E_POINTER;
281
282 AutoLock lock(this);
283 CHECK_READY();
284
285 AutoLock chLock (childrenLock());
286
287 ComObjPtr <HardDiskCollection> collection;
288 collection.createObject();
289 collection->init (children());
290 collection.queryInterfaceTo (aChildren);
291 return S_OK;
292}
293
294STDMETHODIMP HardDisk::COMGETTER(Root) (IHardDisk **aRoot)
295{
296 if (!aRoot)
297 return E_POINTER;
298
299 AutoLock lock(this);
300 CHECK_READY();
301
302 root().queryInterfaceTo (aRoot);
303 return S_OK;
304}
305
306STDMETHODIMP HardDisk::COMGETTER(Accessible) (BOOL *aAccessible)
307{
308 if (!aAccessible)
309 return E_POINTER;
310
311 AutoLock alock (this);
312 CHECK_READY();
313
314 HRESULT rc = getAccessible (mLastAccessError);
315 if (FAILED (rc))
316 return rc;
317
318 *aAccessible = mLastAccessError.isNull();
319 return S_OK;
320}
321
322STDMETHODIMP HardDisk::COMGETTER(AllAccessible) (BOOL *aAllAccessible)
323{
324 if (!aAllAccessible)
325 return E_POINTER;
326
327 AutoLock alock (this);
328 CHECK_READY();
329
330 if (mParent)
331 {
332 HRESULT rc = S_OK;
333
334 /* check the accessibility state of all ancestors */
335 ComObjPtr <HardDisk> parent = (HardDisk *) mParent;
336 while (parent)
337 {
338 AutoLock parentLock (parent);
339 HRESULT rc = parent->getAccessible (mLastAccessError);
340 if (FAILED (rc))
341 break;
342 *aAllAccessible = mLastAccessError.isNull();
343 if (!*aAllAccessible)
344 break;
345 parent = parent->mParent;
346 }
347
348 return rc;
349 }
350
351 HRESULT rc = getAccessible (mLastAccessError);
352 if (FAILED (rc))
353 return rc;
354
355 *aAllAccessible = mLastAccessError.isNull();
356 return S_OK;
357}
358
359STDMETHODIMP HardDisk::COMGETTER(LastAccessError) (BSTR *aLastAccessError)
360{
361 if (!aLastAccessError)
362 return E_POINTER;
363
364 AutoLock alock (this);
365 CHECK_READY();
366
367 mLastAccessError.cloneTo (aLastAccessError);
368 return S_OK;
369}
370
371STDMETHODIMP HardDisk::COMGETTER(MachineId) (GUIDPARAMOUT aMachineId)
372{
373 if (!aMachineId)
374 return E_POINTER;
375
376 AutoLock alock (this);
377 CHECK_READY();
378
379 mMachineId.cloneTo (aMachineId);
380 return S_OK;
381}
382
383STDMETHODIMP HardDisk::COMGETTER(SnapshotId) (GUIDPARAMOUT aSnapshotId)
384{
385 if (!aSnapshotId)
386 return E_POINTER;
387
388 AutoLock alock (this);
389 CHECK_READY();
390
391 mSnapshotId.cloneTo (aSnapshotId);
392 return S_OK;
393}
394
395STDMETHODIMP HardDisk::CloneToImage (INPTR BSTR aFilePath,
396 IVirtualDiskImage **aImage,
397 IProgress **aProgress)
398{
399 if (!aFilePath || !(*aFilePath))
400 return E_INVALIDARG;
401 if (!aImage || !aProgress)
402 return E_POINTER;
403
404 AutoLock alock (this);
405 CHECK_READY();
406 CHECK_BUSY();
407
408 if (!mParent.isNull())
409 return setError (E_FAIL,
410 tr ("Cloning differencing VDI images is not yet supported ('%ls')"),
411 toString().raw());
412
413 HRESULT rc = S_OK;
414
415 /* create a project object */
416 ComObjPtr <Progress> progress;
417 progress.createObject();
418 rc = progress->init (mVirtualBox, (IVirtualDiskImage *) this,
419 Bstr (tr ("Creating a hard disk clone")),
420 FALSE /* aCancelable */);
421 CheckComRCReturnRC (rc);
422
423 /* create an imageless resulting object */
424 ComObjPtr <HVirtualDiskImage> image;
425 image.createObject();
426 rc = image->init (mVirtualBox, NULL, NULL);
427 CheckComRCReturnRC (rc);
428
429 /* append the default path if only a name is given */
430 Bstr path = aFilePath;
431 {
432 Utf8Str fp = aFilePath;
433 if (!RTPathHavePath (fp))
434 {
435 AutoReaderLock propsLock (mVirtualBox->systemProperties());
436 path = Utf8StrFmt ("%ls%c%s",
437 mVirtualBox->systemProperties()->defaultVDIFolder().raw(),
438 RTPATH_DELIMITER,
439 fp.raw());
440 }
441 }
442
443 /* set the desired path */
444 rc = image->setFilePath (path);
445 CheckComRCReturnRC (rc);
446
447 /* ensure the directory exists */
448 {
449 Utf8Str imageDir = image->filePath();
450 RTPathStripFilename (imageDir.mutableRaw());
451 if (!RTDirExists (imageDir))
452 {
453 int vrc = RTDirCreateFullPath (imageDir, 0777);
454 if (VBOX_FAILURE (vrc))
455 {
456 return setError (E_FAIL,
457 tr ("Could not create a directory '%s' "
458 "to store the image file (%Vrc)"),
459 imageDir.raw(), vrc);
460 }
461 }
462 }
463
464 /* mark as busy (being created)
465 * (VDI task thread will unmark it) */
466 image->setBusy();
467
468 /* fill in a VDI task data */
469 VDITask *task = new VDITask (VDITask::CloneToImage, image, progress);
470 task->source = this;
471
472 /* increase readers until finished
473 * (VDI task thread will decrease them) */
474 addReader();
475
476 /* create the hard disk creation thread, pass operation data */
477 int vrc = RTThreadCreate (NULL, HVirtualDiskImage::vdiTaskThread,
478 (void *) task, 0, RTTHREADTYPE_MAIN_HEAVY_WORKER,
479 0, "VDITask");
480 ComAssertMsgRC (vrc, ("Could not create a thread (%Vrc)", vrc));
481 if (VBOX_FAILURE (vrc))
482 {
483 releaseReader();
484 image->clearBusy();
485 delete task;
486 return E_FAIL;
487 }
488
489 /* return interfaces to the caller */
490 image.queryInterfaceTo (aImage);
491 progress.queryInterfaceTo (aProgress);
492
493 return S_OK;
494}
495
496// public methods for internal purposes only
497/////////////////////////////////////////////////////////////////////////////
498
499/**
500 * Returns the very first (grand-) parent of this hard disk or the hard
501 * disk itself, if it doesn't have a parent.
502 *
503 * @note
504 * Must be called from under the object's lock
505 */
506ComObjPtr <HardDisk> HardDisk::root() const
507{
508 ComObjPtr <HardDisk> root = const_cast <HardDisk *> (this);
509 ComObjPtr <HardDisk> parent;
510 while ((parent = root->parent()))
511 root = parent;
512
513 return root;
514}
515
516/**
517 * Attempts to mark the hard disk as registered.
518 * Must be always called by every reimplementation.
519 * Only VirtualBox can call this method.
520 *
521 * @param aRegistered true to set registered and false to set unregistered
522 */
523HRESULT HardDisk::trySetRegistered (BOOL aRegistered)
524{
525 AutoLock alock (this);
526 CHECK_READY();
527
528 if (aRegistered)
529 {
530 ComAssertRet (mMachineId.isEmpty(), E_FAIL);
531 ComAssertRet (mId && children().size() == 0, E_FAIL);
532
533 if (mRegistered)
534 return setError (E_FAIL,
535 tr ("Hard disk '%ls' is already registered"),
536 toString().raw());
537
538 CHECK_BUSY();
539 }
540 else
541 {
542 if (!mRegistered)
543 return setError (E_FAIL,
544 tr ("Hard disk '%ls' is already unregistered"),
545 toString().raw());
546
547 if (!mMachineId.isEmpty())
548 return setError (E_FAIL,
549 tr ("Hard disk '%ls' is attached to a virtual machine with UUID {%s}"),
550 toString().raw(), mMachineId.toString().raw());
551
552 if (children().size() > 0)
553 return setError (E_FAIL,
554 tr ("Hard disk '%ls' has %d differencing hard disks based on it"),
555 toString().raw(), children().size());
556
557 CHECK_BUSY_AND_READERS();
558 }
559
560 mRegistered = aRegistered;
561 return S_OK;
562}
563
564/**
565 * Checks basic accessibility of this hard disk only (w/o parents).
566 * Must be always called by every HardDisk::getAccessible() reimplementation
567 * in the first place.
568 *
569 * When @a aCheckBusy is true, this method checks that mBusy = false and
570 * mReaders = 0 (and returns an appropriate error if not). This lets
571 * reimplementations successfully call setBusy() to lock the disk for the
572 * duration of the check if they need. Note that in this case, the
573 * reimplementation must enter the object lock before calling this method
574 * and not leave it before calling setBusy() to avoid race condition.
575 *
576 * @param aAccessError On output, a null string indicates the hard disk is
577 * accessible, otherwise contains a message describing
578 * the reason of inaccessibility.
579 * @param aCheckBusy Whether to do the busy check or not.
580 */
581HRESULT HardDisk::getBaseAccessible (Bstr &aAccessError,
582 bool aCheckBusy /* = false */)
583{
584 AutoLock alock (this);
585 CHECK_READY();
586
587 aAccessError.setNull();
588
589 if (aCheckBusy)
590 {
591 if (mBusy)
592 {
593 aAccessError = Utf8StrFmt (
594 tr ("Hard disk '%ls' is being exclusively used by another task"),
595 toString().raw());
596 }
597 else if (mReaders > 0)
598 {
599 aAccessError = Utf8StrFmt (
600 tr ("Hard disk '%ls' is being used by another task (%d readers)"),
601 toString().raw(), mReaders);
602 }
603 }
604
605 return S_OK;
606}
607
608/**
609 * Returns true if the set of properties that makes this object unique
610 * is equal to the same set of properties in the given object.
611 */
612bool HardDisk::sameAs (HardDisk *that)
613{
614 AutoLock alock (this);
615 if (!isReady())
616 return false;
617
618 /// @todo (dmik) besides UUID, we temporarily use toString() to uniquely
619 // identify objects. This is ok for VDIs but may be not good for iSCSI,
620 // so it will need a reimp of this method.
621
622 return that->mId == mId ||
623 toString (false /* aShort */) == that->toString (false /* aShort */);
624}
625
626/**
627 * Marks this hard disk as busy.
628 * A busy hard disk cannot have readers and its properties (UUID, description)
629 * cannot be externally modified.
630 */
631void HardDisk::setBusy()
632{
633 AutoLock alock (this);
634 AssertReturnVoid (isReady());
635
636 AssertMsgReturnVoid (mBusy == false, ("%ls", toString().raw()));
637 AssertMsgReturnVoid (mReaders == 0, ("%ls", toString().raw()));
638
639 mBusy = true;
640}
641
642/**
643 * Clears the busy flag previously set by #setBusy().
644 */
645void HardDisk::clearBusy()
646{
647 AutoLock alock (this);
648 AssertReturnVoid (isReady());
649
650 AssertMsgReturnVoid (mBusy == true, ("%ls", toString().raw()));
651
652 mBusy = false;
653}
654
655/**
656 * Increases the number of readers of this hard disk.
657 * A hard disk that have readers cannot be marked as busy (and vice versa)
658 * and its properties (UUID, description) cannot be externally modified.
659 */
660void HardDisk::addReader()
661{
662 AutoLock alock (this);
663 AssertReturnVoid (isReady());
664
665 AssertMsgReturnVoid (mBusy == false, ("%ls", toString().raw()));
666
667 ++ mReaders;
668}
669
670/**
671 * Decreases the number of readers of this hard disk.
672 */
673void HardDisk::releaseReader()
674{
675 AutoLock alock (this);
676 AssertReturnVoid (isReady());
677
678 AssertMsgReturnVoid (mBusy == false, ("%ls", toString().raw()));
679 AssertMsgReturnVoid (mReaders > 0, ("%ls", toString().raw()));
680
681 -- mReaders;
682}
683
684/**
685 * Increases the number of readers on all ancestors of this hard disk.
686 */
687void HardDisk::addReaderOnAncestors()
688{
689 AutoLock alock (this);
690 AssertReturnVoid (isReady());
691
692 if (mParent)
693 {
694 AutoLock alock (mParent);
695 mParent->addReader();
696 mParent->addReaderOnAncestors();
697 }
698}
699
700/**
701 * Decreases the number of readers on all ancestors of this hard disk.
702 */
703void HardDisk::releaseReaderOnAncestors()
704{
705 AutoLock alock (this);
706 AssertReturnVoid (isReady());
707
708 if (mParent)
709 {
710 AutoLock alock (mParent);
711 mParent->releaseReaderOnAncestors();
712 mParent->releaseReader();
713 }
714}
715
716/**
717 * Returns true if this hard disk has children not belonging to the same
718 * machine.
719 */
720bool HardDisk::hasForeignChildren()
721{
722 AutoLock alock (this);
723 AssertReturn (isReady(), false);
724
725 AssertReturn (!mMachineId.isEmpty(), false);
726
727 /* check all children */
728 AutoLock chLock (childrenLock());
729 for (HardDiskList::const_iterator it = children().begin();
730 it != children().end();
731 ++ it)
732 {
733 ComObjPtr <HardDisk> child = *it;
734 AutoLock childLock (child);
735 if (child->mMachineId != mMachineId)
736 return true;
737 }
738
739 return false;
740}
741
742/**
743 * Marks this hard disk and all its children as busy.
744 * Used for merge operations.
745 * Returns a meaningful error info on failure.
746 */
747HRESULT HardDisk::setBusyWithChildren()
748{
749 AutoLock alock (this);
750 AssertReturn (isReady(), E_FAIL);
751
752 const char *errMsg = tr ("Hard disk '%ls' is being used by another task");
753
754 if (mReaders > 0 || mBusy)
755 return setError (E_FAIL, errMsg, toString().raw());
756
757 AutoLock chLock (childrenLock());
758
759 for (HardDiskList::const_iterator it = children().begin();
760 it != children().end();
761 ++ it)
762 {
763 ComObjPtr <HardDisk> child = *it;
764 AutoLock childLock (child);
765 if (child->mReaders > 0 || child->mBusy)
766 {
767 /* reset the busy flag of all previous children */
768 while (it != children().begin())
769 (*(-- it))->clearBusy();
770 return setError (E_FAIL, errMsg, child->toString().raw());
771 }
772 else
773 child->mBusy = true;
774 }
775
776 mBusy = true;
777
778 return S_OK;
779}
780
781/**
782 * Clears the busy flag of this hard disk and all its children.
783 * An opposite to #setBusyWithChildren.
784 */
785void HardDisk::clearBusyWithChildren()
786{
787 AutoLock alock (this);
788 AssertReturn (isReady(), (void) 0);
789
790 AssertReturn (mBusy == true, (void) 0);
791
792 AutoLock chLock (childrenLock());
793
794 for (HardDiskList::const_iterator it = children().begin();
795 it != children().end();
796 ++ it)
797 {
798 ComObjPtr <HardDisk> child = *it;
799 AutoLock childLock (child);
800 Assert (child->mBusy == true);
801 child->mBusy = false;
802 }
803
804 mBusy = false;
805}
806
807/**
808 * Checks that this hard disk and all its direct children are accessible.
809 */
810HRESULT HardDisk::getAccessibleWithChildren (Bstr &aAccessError)
811{
812 AutoLock alock (this);
813 AssertReturn (isReady(), E_FAIL);
814
815 HRESULT rc = getAccessible (aAccessError);
816 if (FAILED (rc) || !aAccessError.isNull())
817 return rc;
818
819 AutoLock chLock (childrenLock());
820
821 for (HardDiskList::const_iterator it = children().begin();
822 it != children().end();
823 ++ it)
824 {
825 ComObjPtr <HardDisk> child = *it;
826 rc = child->getAccessible (aAccessError);
827 if (FAILED (rc) || !aAccessError.isNull())
828 return rc;
829 }
830
831 return rc;
832}
833
834/**
835 * Checks that this hard disk and all its descendants are consistent.
836 * For now, the consistency means that:
837 *
838 * 1) every differencing image is associated with a registered machine
839 * 2) every root image that has differencing children is associated with
840 * a registered machine.
841 *
842 * This method is used by the VirtualBox constructor after loading all hard
843 * disks and all machines.
844 */
845HRESULT HardDisk::checkConsistency()
846{
847 AutoLock alock (this);
848 AssertReturn (isReady(), E_FAIL);
849
850 if (isDifferencing())
851 {
852 Assert (mVirtualBox->isMachineIdValid (mMachineId) ||
853 mMachineId.isEmpty());
854
855 if (mMachineId.isEmpty())
856 return setError (E_FAIL,
857 tr ("Differencing hard disk '%ls' is not associated with "
858 "any registered virtual machine or snapshot"),
859 toString().raw());
860 }
861
862 HRESULT rc = S_OK;
863
864 AutoLock chLock (childrenLock());
865
866 if (mParent.isNull() && mType == HardDiskType_NormalHardDisk &&
867 children().size() != 0)
868 {
869 if (mMachineId.isEmpty())
870 return setError (E_FAIL,
871 tr ("Hard disk '%ls' is not associated with any registered "
872 "virtual machine or snapshot, but has differencing child "
873 "hard disks based on it"),
874 toString().raw());
875 }
876
877 for (HardDiskList::const_iterator it = children().begin();
878 it != children().end() && SUCCEEDED (rc);
879 ++ it)
880 {
881 rc = (*it)->checkConsistency();
882 }
883
884 return rc;
885}
886
887/**
888 * Creates a differencing hard disk for this hard disk and returns the
889 * created hard disk object to the caller.
890 *
891 * The created differencing hard disk is automatically added to the list of
892 * children of this hard disk object and registered within VirtualBox.
893
894 * The specified progress object (if not NULL) receives the percentage
895 * of the operation completion. However, it is responsibility of the caller to
896 * call Progress::notifyComplete() after this method returns.
897 *
898 * @param aFolder folder where to create the differencing disk
899 * (must be a full path)
900 * @param aMachineId machine ID the new hard disk will belong to
901 * @param aHardDisk resulting hard disk object
902 * @param aProgress progress object to run during copy operation
903 * (may be NULL)
904 *
905 * @note
906 * Must be NOT called from under locks of other objects that need external
907 * access dirung this method execurion!
908 */
909HRESULT HardDisk::createDiffHardDisk (const Bstr &aFolder, const Guid &aMachineId,
910 ComObjPtr <HVirtualDiskImage> &aHardDisk,
911 Progress *aProgress)
912{
913 AssertReturn (!aFolder.isEmpty() && !aMachineId.isEmpty(),
914 E_FAIL);
915
916 AutoLock alock (this);
917 CHECK_READY();
918
919 ComAssertRet (isBusy() == false, E_FAIL);
920
921 Guid id;
922 id.create();
923
924 Utf8Str filePathTo = Utf8StrFmt ("%ls%c{%Vuuid}.vdi",
925 aFolder.raw(), RTPATH_DELIMITER, id.ptr());
926
927 /* try to make the path relative to the vbox home dir */
928 const char *filePathToRel = filePathTo;
929 {
930 const Utf8Str &homeDir = mVirtualBox->homeDir();
931 if (!strncmp (filePathTo, homeDir, homeDir.length()))
932 filePathToRel = (filePathToRel + homeDir.length() + 1);
933 }
934
935 /* first ensure the directory exists */
936 {
937 Utf8Str dir = aFolder;
938 if (!RTDirExists (dir))
939 {
940 int vrc = RTDirCreateFullPath (dir, 0777);
941 if (VBOX_FAILURE (vrc))
942 {
943 return setError (E_FAIL,
944 tr ("Could not create a directory '%s' "
945 "to store the image file (%Vrc)"),
946 dir.raw(), vrc);
947 }
948 }
949 }
950
951 alock.leave();
952
953 /* call storage type specific diff creation method */
954 HRESULT rc = createDiffImage (id, filePathTo, aProgress);
955
956 alock.enter();
957
958 CheckComRCReturnRC (rc);
959
960 ComObjPtr <HVirtualDiskImage> vdi;
961 vdi.createObject();
962 rc = vdi->init (mVirtualBox, this, Bstr (filePathToRel),
963 TRUE /* aRegistered */);
964 CheckComRCReturnRC (rc);
965
966 /* associate the created hard disk with the given machine */
967 vdi->setMachineId (aMachineId);
968
969 rc = mVirtualBox->registerHardDisk (vdi, VirtualBox::RHD_Internal);
970 CheckComRCReturnRC (rc);
971
972 aHardDisk = vdi;
973
974 return S_OK;
975}
976
977/**
978 * Checks if the given change of \a aOldPath to \a aNewPath affects the path
979 * of this hard disk or any of its children and updates it if necessary (by
980 * calling #updatePath()). Intended to be called only by
981 * VirtualBox::updateSettings() if a machine's name change causes directory
982 * renaming that affects this image.
983 *
984 * @param aOldPath old path (full)
985 * @param aNewPath new path (full)
986 *
987 * @note Locks this object and all children for writing.
988 */
989void HardDisk::updatePaths (const char *aOldPath, const char *aNewPath)
990{
991 AssertReturnVoid (aOldPath);
992 AssertReturnVoid (aNewPath);
993
994 AutoLock alock (this);
995 AssertReturnVoid (isReady());
996
997 updatePath (aOldPath, aNewPath);
998
999 /* update paths of all children */
1000 AutoLock chLock (childrenLock());
1001 for (HardDiskList::const_iterator it = children().begin();
1002 it != children().end();
1003 ++ it)
1004 {
1005 (*it)->updatePaths (aOldPath, aNewPath);
1006 }
1007}
1008
1009/**
1010 * Helper method that deduces a hard disk object type to create from
1011 * the location string format and from the contents of the resource
1012 * pointed to by the location string.
1013 *
1014 * Currently, the location string must be a file path which is
1015 * passed to the HVirtualDiskImage or HVMDKImage initializer in
1016 * attempt to create a hard disk object.
1017 *
1018 * @param aVirtualBox
1019 * @param aLocation
1020 * @param hardDisk
1021 *
1022 * @return
1023 */
1024/* static */
1025HRESULT HardDisk::openHardDisk (VirtualBox *aVirtualBox, INPTR BSTR aLocation,
1026 ComObjPtr <HardDisk> &hardDisk)
1027{
1028 LogFlowFunc (("aLocation=\"%ls\"\n", aLocation));
1029
1030 AssertReturn (aVirtualBox, E_POINTER);
1031
1032 /* null and empty strings are not allowed locations */
1033 AssertReturn (aLocation, E_INVALIDARG);
1034 AssertReturn (*aLocation, E_INVALIDARG);
1035
1036 HRESULT rc = S_OK;
1037
1038 /* try to guess the probe order by extension */
1039 Utf8Str loc = aLocation;
1040 char *ext = RTPathExt (loc);
1041
1042 HardDiskStorageType_T order [2];
1043
1044 if (RTPathCompare (ext, ".vmdk") == 0)
1045 {
1046 order [0] = HardDiskStorageType_VMDKImage;
1047 order [1] = HardDiskStorageType_VirtualDiskImage;
1048 }
1049 else
1050 {
1051 order [0] = HardDiskStorageType_VirtualDiskImage;
1052 order [1] = HardDiskStorageType_VMDKImage;
1053 }
1054
1055 for (size_t i = 0; i < ELEMENTS (order); ++ i)
1056 {
1057 switch (order [i])
1058 {
1059 case HardDiskStorageType_VirtualDiskImage:
1060 {
1061 ComObjPtr <HVirtualDiskImage> obj;
1062 obj.createObject();
1063 rc = obj->init (aVirtualBox, NULL, aLocation,
1064 FALSE /* aRegistered */);
1065 if (SUCCEEDED (rc))
1066 {
1067 hardDisk = obj;
1068 return rc;
1069 }
1070 break;
1071 }
1072 case HardDiskStorageType_VMDKImage:
1073 {
1074 ComObjPtr <HVMDKImage> obj;
1075 obj.createObject();
1076 rc = obj->init (aVirtualBox, NULL, aLocation,
1077 FALSE /* aRegistered */);
1078 if (SUCCEEDED (rc))
1079 {
1080 hardDisk = obj;
1081 return rc;
1082 }
1083 break;
1084 }
1085 default:
1086 {
1087 ComAssertComRCRetRC (E_FAIL);
1088 }
1089 }
1090 }
1091
1092 return rc;
1093}
1094
1095// protected methods
1096/////////////////////////////////////////////////////////////////////////////
1097
1098/**
1099 * Loads the base settings of the hard disk from the given node, registers
1100 * it and loads and registers all child hard disks as HVirtualDiskImage
1101 * instances.
1102 *
1103 * Subclasses must call this method in their init() or loadSettings() methods
1104 * *after* they load specific parts of data (at least, necessary to let
1105 * toString() function correctly), in order to be properly loaded from the
1106 * settings file and registered.
1107 *
1108 * @param aHDNode <HardDisk> node when #isDifferencing() = false, or
1109 * <DiffHardDisk> node otherwise
1110 *
1111 * @note
1112 * Must be called from under the object's lock
1113 */
1114HRESULT HardDisk::loadSettings (CFGNODE aHDNode)
1115{
1116 AssertReturn (aHDNode, E_FAIL);
1117
1118 Guid uuid; /* uuid (required) */
1119 CFGLDRQueryUUID (aHDNode, "uuid", uuid.ptr());
1120 mId = uuid;
1121
1122 if (!isDifferencing())
1123 {
1124 Bstr type; /* type (required for <HardDisk> nodes only) */
1125 CFGLDRQueryBSTR (aHDNode, "type", type.asOutParam());
1126 if (type == L"normal")
1127 mType = HardDiskType_NormalHardDisk;
1128 else if (type == L"immutable")
1129 mType = HardDiskType_ImmutableHardDisk;
1130 else if (type == L"writethrough")
1131 mType = HardDiskType_WritethroughHardDisk;
1132 else
1133 ComAssertMsgFailedRet (("Invalid hard disk type '%ls'\n", type.raw()),
1134 E_FAIL);
1135 }
1136 else
1137 mType = HardDiskType_NormalHardDisk;
1138
1139 HRESULT rc = mVirtualBox->registerHardDisk (this, VirtualBox::RHD_OnStartUp);
1140 if (FAILED (rc))
1141 return rc;
1142
1143 /* load all children */
1144 unsigned count = 0;
1145 CFGLDRCountChildren (aHDNode, "DiffHardDisk", &count);
1146 for (unsigned i = 0; i < count && SUCCEEDED (rc); ++ i)
1147 {
1148 CFGNODE hdNode = 0;
1149
1150 CFGLDRGetChildNode (aHDNode, "DiffHardDisk", i, &hdNode);
1151 ComAssertBreak (hdNode, rc = E_FAIL);
1152
1153 do
1154 {
1155 CFGNODE vdiNode = 0;
1156 CFGLDRGetChildNode (hdNode, "VirtualDiskImage", 0, &vdiNode);
1157 ComAssertBreak (vdiNode, rc = E_FAIL);
1158
1159 ComObjPtr <HVirtualDiskImage> vdi;
1160 vdi.createObject();
1161 rc = vdi->init (mVirtualBox, this, hdNode, vdiNode);
1162
1163 CFGLDRReleaseNode (vdiNode);
1164 }
1165 while (0);
1166
1167 CFGLDRReleaseNode (hdNode);
1168 }
1169
1170 return rc;
1171}
1172
1173/**
1174 * Saves the base settings of the hard disk to the given node
1175 * and saves all child hard disks as <DiffHardDisk> nodes.
1176 *
1177 * Subclasses must call this method in their saveSettings() methods
1178 * in order to be properly saved to the settings file.
1179 *
1180 * @param aHDNode <HardDisk> node when #isDifferencing() = false, or
1181 * <DiffHardDisk> node otherwise
1182 *
1183 * @note
1184 * Must be called from under the object's lock
1185 */
1186HRESULT HardDisk::saveSettings (CFGNODE aHDNode)
1187{
1188 AssertReturn (aHDNode, E_FAIL);
1189
1190 /* uuid (required) */
1191 CFGLDRSetUUID (aHDNode, "uuid", mId.ptr());
1192
1193 if (!isDifferencing())
1194 {
1195 /* type (required) */
1196 const char *type = NULL;
1197 switch (mType)
1198 {
1199 case HardDiskType_NormalHardDisk:
1200 type = "normal";
1201 break;
1202 case HardDiskType_ImmutableHardDisk:
1203 type = "immutable";
1204 break;
1205 case HardDiskType_WritethroughHardDisk:
1206 type = "writethrough";
1207 break;
1208 }
1209 CFGLDRSetString (aHDNode, "type", type);
1210 }
1211
1212 HRESULT rc = S_OK;
1213
1214 /* save all children */
1215 AutoLock chLock (childrenLock());
1216 for (HardDiskList::const_iterator it = children().begin();
1217 it != children().end() && SUCCEEDED (rc);
1218 ++ it)
1219 {
1220 ComObjPtr <HardDisk> child = *it;
1221 AutoLock childLock (child);
1222
1223 CFGNODE hdNode = 0;
1224 CFGLDRAppendChildNode (aHDNode, "DiffHardDisk", &hdNode);
1225 ComAssertBreak (hdNode, rc = E_FAIL);
1226
1227 do
1228 {
1229 CFGNODE vdiNode = 0;
1230 CFGLDRAppendChildNode (hdNode, "VirtualDiskImage", &vdiNode);
1231 ComAssertBreak (vdiNode, rc = E_FAIL);
1232
1233 rc = child->saveSettings (hdNode, vdiNode);
1234
1235 CFGLDRReleaseNode (vdiNode);
1236 }
1237 while (0);
1238
1239 CFGLDRReleaseNode (hdNode);
1240 }
1241
1242 return rc;
1243}
1244
1245////////////////////////////////////////////////////////////////////////////////
1246// HVirtualDiskImage class
1247////////////////////////////////////////////////////////////////////////////////
1248
1249// constructor / destructor
1250////////////////////////////////////////////////////////////////////////////////
1251
1252HRESULT HVirtualDiskImage::FinalConstruct()
1253{
1254 HRESULT rc = HardDisk::FinalConstruct();
1255 if (FAILED (rc))
1256 return rc;
1257
1258 mState = NotCreated;
1259
1260 mStateCheckSem = NIL_RTSEMEVENTMULTI;
1261 mStateCheckWaiters = 0;
1262
1263 mSize = 0;
1264 mActualSize = 0;
1265
1266 return S_OK;
1267}
1268
1269void HVirtualDiskImage::FinalRelease()
1270{
1271 HardDisk::FinalRelease();
1272}
1273
1274// public initializer/uninitializer for internal purposes only
1275////////////////////////////////////////////////////////////////////////////////
1276
1277// public methods for internal purposes only
1278/////////////////////////////////////////////////////////////////////////////
1279
1280/**
1281 * Initializes the VDI hard disk object by reading its properties from
1282 * the given configuration node. The created hard disk will be marked as
1283 * registered on success.
1284 *
1285 * @param aHDNode <HardDisk> node
1286 * @param aVDINode <VirtualDiskImage> node
1287 */
1288HRESULT HVirtualDiskImage::init (VirtualBox *aVirtualBox, HardDisk *aParent,
1289 CFGNODE aHDNode, CFGNODE aVDINode)
1290{
1291 LogFlowMember (("HVirtualDiskImage::init (load)\n"));
1292
1293 AssertReturn (aHDNode && aVDINode, E_FAIL);
1294
1295 AutoLock alock (this);
1296 ComAssertRet (!isReady(), E_UNEXPECTED);
1297
1298 mStorageType = HardDiskStorageType_VirtualDiskImage;
1299
1300 HRESULT rc = S_OK;
1301
1302 do
1303 {
1304 rc = protectedInit (aVirtualBox, aParent);
1305 CheckComRCBreakRC (rc);
1306
1307 /* set ready to let protectedUninit() be called on failure */
1308 setReady (true);
1309
1310 /* filePath (required) */
1311 Bstr filePath;
1312 CFGLDRQueryBSTR (aVDINode, "filePath", filePath.asOutParam());
1313
1314 rc = setFilePath (filePath);
1315 CheckComRCBreakRC (rc);
1316
1317 LogFlowMember ((" '%ls'\n", mFilePathFull.raw()));
1318
1319 /* load basic settings and children */
1320 rc = loadSettings (aHDNode);
1321 CheckComRCBreakRC (rc);
1322
1323 mState = Created;
1324 mRegistered = TRUE;
1325
1326 /* Don't call queryInformation() for registered hard disks to
1327 * prevent the calling thread (i.e. the VirtualBox server startup
1328 * thread) from an unexpected freeze. The vital mId property (UUID)
1329 * is read from the registry file in loadSettings(). To get the rest,
1330 * the user will have to call COMGETTER(Accessible) manually. */
1331 }
1332 while (0);
1333
1334 if (FAILED (rc))
1335 uninit();
1336
1337 return rc;
1338}
1339
1340/**
1341 * Initializes the VDI hard disk object using the given image file name.
1342 *
1343 * @param aFilePath path to the image file (can be NULL to create an
1344 * imageless object)
1345 * @param aRegistered whether to mark this disk as registered or not
1346 * (ignored when \a aFilePath is NULL, assuming FALSE)
1347 */
1348HRESULT HVirtualDiskImage::init (VirtualBox *aVirtualBox, HardDisk *aParent,
1349 const BSTR aFilePath, BOOL aRegistered /* = FALSE */)
1350{
1351 LogFlowMember (("HVirtualDiskImage::init (aFilePath='%ls', aRegistered=%d)\n",
1352 aFilePath, aRegistered));
1353
1354 AutoLock alock (this);
1355 ComAssertRet (!isReady(), E_UNEXPECTED);
1356
1357 mStorageType = HardDiskStorageType_VirtualDiskImage;
1358
1359 HRESULT rc = S_OK;
1360
1361 do
1362 {
1363 rc = protectedInit (aVirtualBox, aParent);
1364 CheckComRCBreakRC (rc);
1365
1366 /* set ready to let protectedUninit() be called on failure */
1367 setReady (true);
1368
1369 rc = setFilePath (aFilePath);
1370 CheckComRCBreakRC (rc);
1371
1372 Assert (mId.isEmpty());
1373
1374 if (aFilePath && *aFilePath)
1375 {
1376 mRegistered = aRegistered;
1377 mState = Created;
1378
1379 /* Call queryInformation() anyway (even if it will block), because
1380 * it is the only way to get the UUID of the existing VDI and
1381 * initialize the vital mId property. */
1382 Bstr errMsg;
1383 rc = queryInformation (&errMsg);
1384 if (SUCCEEDED (rc))
1385 {
1386 /* We are constructing a new HVirtualDiskImage object. If there
1387 * is a fatal accessibility error (we cannot read image UUID),
1388 * we have to fail. We do so even on non-fatal errors as well,
1389 * because it's not worth to keep going with the inaccessible
1390 * image from the very beginning (when nothing else depends on
1391 * it yet). */
1392 if (!errMsg.isNull())
1393 rc = setErrorBstr (E_FAIL, errMsg);
1394 }
1395 }
1396 else
1397 {
1398 mRegistered = FALSE;
1399 mState = NotCreated;
1400 mId.create();
1401 }
1402 }
1403 while (0);
1404
1405 if (FAILED (rc))
1406 uninit();
1407
1408 return rc;
1409}
1410
1411/**
1412 * Uninitializes the instance and sets the ready flag to FALSE.
1413 * Called either from FinalRelease(), by the parent when it gets destroyed,
1414 * or by a third party when it decides this object is no more valid.
1415 */
1416void HVirtualDiskImage::uninit()
1417{
1418 LogFlowMember (("HVirtualDiskImage::uninit()\n"));
1419
1420 AutoLock alock (this);
1421 if (!isReady())
1422 return;
1423
1424 HardDisk::protectedUninit (alock);
1425}
1426
1427// IHardDisk properties
1428////////////////////////////////////////////////////////////////////////////////
1429
1430STDMETHODIMP HVirtualDiskImage::COMGETTER(Description) (BSTR *aDescription)
1431{
1432 if (!aDescription)
1433 return E_POINTER;
1434
1435 AutoLock alock (this);
1436 CHECK_READY();
1437
1438 mDescription.cloneTo (aDescription);
1439 return S_OK;
1440}
1441
1442STDMETHODIMP HVirtualDiskImage::COMSETTER(Description) (INPTR BSTR aDescription)
1443{
1444 AutoLock alock (this);
1445 CHECK_READY();
1446
1447 CHECK_BUSY_AND_READERS();
1448
1449 if (mState >= Created)
1450 {
1451 int vrc = VDISetImageComment (Utf8Str (mFilePathFull), Utf8Str (aDescription));
1452 if (VBOX_FAILURE (vrc))
1453 return setError (E_FAIL,
1454 tr ("Could not change the description of the VDI hard disk '%ls' "
1455 "(%Vrc)"),
1456 toString().raw(), vrc);
1457 }
1458
1459 mDescription = aDescription;
1460 return S_OK;
1461}
1462
1463STDMETHODIMP HVirtualDiskImage::COMGETTER(Size) (ULONG64 *aSize)
1464{
1465 if (!aSize)
1466 return E_POINTER;
1467
1468 AutoLock alock (this);
1469 CHECK_READY();
1470
1471 /* only a non-differencing image knows the logical size */
1472 if (isDifferencing())
1473 return root()->COMGETTER(Size) (aSize);
1474
1475 *aSize = mSize;
1476 return S_OK;
1477}
1478
1479STDMETHODIMP HVirtualDiskImage::COMGETTER(ActualSize) (ULONG64 *aActualSize)
1480{
1481 if (!aActualSize)
1482 return E_POINTER;
1483
1484 AutoLock alock (this);
1485 CHECK_READY();
1486
1487 *aActualSize = mActualSize;
1488 return S_OK;
1489}
1490
1491// IVirtualDiskImage properties
1492////////////////////////////////////////////////////////////////////////////////
1493
1494STDMETHODIMP HVirtualDiskImage::COMGETTER(FilePath) (BSTR *aFilePath)
1495{
1496 if (!aFilePath)
1497 return E_POINTER;
1498
1499 AutoLock alock (this);
1500 CHECK_READY();
1501
1502 mFilePathFull.cloneTo (aFilePath);
1503 return S_OK;
1504}
1505
1506STDMETHODIMP HVirtualDiskImage::COMSETTER(FilePath) (INPTR BSTR aFilePath)
1507{
1508 AutoLock alock (this);
1509 CHECK_READY();
1510
1511 if (mState != NotCreated)
1512 return setError (E_ACCESSDENIED,
1513 tr ("Cannot change the file path of the existing hard disk '%ls'"),
1514 toString().raw());
1515
1516 CHECK_BUSY_AND_READERS();
1517
1518 // append the default path if only a name is given
1519 Bstr path = aFilePath;
1520 if (aFilePath && *aFilePath)
1521 {
1522 Utf8Str fp = aFilePath;
1523 if (!RTPathHavePath (fp))
1524 {
1525 AutoReaderLock propsLock (mVirtualBox->systemProperties());
1526 path = Utf8StrFmt ("%ls%c%s",
1527 mVirtualBox->systemProperties()->defaultVDIFolder().raw(),
1528 RTPATH_DELIMITER,
1529 fp.raw());
1530 }
1531 }
1532
1533 return setFilePath (path);
1534}
1535
1536STDMETHODIMP HVirtualDiskImage::COMGETTER(Created) (BOOL *aCreated)
1537{
1538 if (!aCreated)
1539 return E_POINTER;
1540
1541 AutoLock alock (this);
1542 CHECK_READY();
1543
1544 *aCreated = mState >= Created;
1545 return S_OK;
1546}
1547
1548// IVirtualDiskImage methods
1549/////////////////////////////////////////////////////////////////////////////
1550
1551STDMETHODIMP HVirtualDiskImage::CreateDynamicImage (ULONG64 aSize, IProgress **aProgress)
1552{
1553 if (!aProgress)
1554 return E_POINTER;
1555
1556 AutoLock alock (this);
1557 CHECK_READY();
1558
1559 return createImage (aSize, TRUE /* aDynamic */, aProgress);
1560}
1561
1562STDMETHODIMP HVirtualDiskImage::CreateFixedImage (ULONG64 aSize, IProgress **aProgress)
1563{
1564 if (!aProgress)
1565 return E_POINTER;
1566
1567 AutoLock alock (this);
1568 CHECK_READY();
1569
1570 return createImage (aSize, FALSE /* aDynamic */, aProgress);
1571}
1572
1573STDMETHODIMP HVirtualDiskImage::DeleteImage()
1574{
1575 AutoLock alock (this);
1576 CHECK_READY();
1577 CHECK_BUSY_AND_READERS();
1578
1579 if (mRegistered)
1580 return setError (E_ACCESSDENIED,
1581 tr ("Cannot delete an image of the registered hard disk image '%ls"),
1582 mFilePathFull.raw());
1583 if (mState == NotCreated)
1584 return setError (E_FAIL,
1585 tr ("Hard disk image has been already deleted or never created"));
1586
1587 int vrc = RTFileDelete (Utf8Str (mFilePathFull));
1588 if (VBOX_FAILURE (vrc))
1589 return setError (E_FAIL, tr ("Could not delete the image file '%ls' (%Vrc)"),
1590 mFilePathFull.raw(), vrc);
1591
1592 mState = NotCreated;
1593
1594 // reset the fields
1595 mSize = 0;
1596 mActualSize = 0;
1597
1598 return S_OK;
1599}
1600
1601// public/protected methods for internal purposes only
1602/////////////////////////////////////////////////////////////////////////////
1603
1604/**
1605 * Attempts to mark the hard disk as registered.
1606 * Only VirtualBox can call this method.
1607 */
1608HRESULT HVirtualDiskImage::trySetRegistered (BOOL aRegistered)
1609{
1610 AutoLock alock (this);
1611 CHECK_READY();
1612
1613 if (aRegistered)
1614 {
1615 if (mState == NotCreated)
1616 return setError (E_FAIL,
1617 tr ("Image file '%ls' is not yet created for this hard disk"),
1618 mFilePathFull.raw());
1619 if (isDifferencing())
1620 return setError (E_FAIL,
1621 tr ("Hard disk '%ls' is differencing and cannot be unregistered "
1622 "explicitly"),
1623 mFilePathFull.raw());
1624 }
1625 else
1626 {
1627 ComAssertRet (mState >= Created, E_FAIL);
1628 }
1629
1630 return HardDisk::trySetRegistered (aRegistered);
1631}
1632
1633/**
1634 * Checks accessibility of this hard disk image only (w/o parents).
1635 *
1636 * @param aAccessError on output, a null string indicates the hard disk is
1637 * accessible, otherwise contains a message describing
1638 * the reason of inaccessibility.
1639 */
1640HRESULT HVirtualDiskImage::getAccessible (Bstr &aAccessError)
1641{
1642 AutoLock alock (this);
1643 CHECK_READY();
1644
1645 if (mStateCheckSem != NIL_RTSEMEVENTMULTI)
1646 {
1647 /* An accessibility check in progress on some other thread,
1648 * wait for it to finish. */
1649
1650 ComAssertRet (mStateCheckWaiters != (ULONG) ~0, E_FAIL);
1651 ++ mStateCheckWaiters;
1652 alock.leave();
1653
1654 int vrc = RTSemEventMultiWait (mStateCheckSem, RT_INDEFINITE_WAIT);
1655
1656 alock.enter();
1657 AssertReturn (mStateCheckWaiters != 0, E_FAIL);
1658 -- mStateCheckWaiters;
1659 if (mStateCheckWaiters == 0)
1660 {
1661 RTSemEventMultiDestroy (mStateCheckSem);
1662 mStateCheckSem = NIL_RTSEMEVENTMULTI;
1663 }
1664
1665 AssertRCReturn (vrc, E_FAIL);
1666
1667 /* don't touch aAccessError, it has been already set */
1668 return S_OK;
1669 }
1670
1671 /* check the basic accessibility */
1672 HRESULT rc = getBaseAccessible (aAccessError, true /* aCheckBusy */);
1673 if (FAILED (rc) || !aAccessError.isNull())
1674 return rc;
1675
1676 if (mState >= Created)
1677 {
1678 return queryInformation (&aAccessError);
1679 }
1680
1681 aAccessError = Utf8StrFmt ("Hard disk image '%ls' is not yet created",
1682 mFilePathFull.raw());
1683 return S_OK;
1684}
1685
1686/**
1687 * Saves hard disk settings to the specified storage node and saves
1688 * all children to the specified hard disk node
1689 *
1690 * @param aHDNode <HardDisk> or <DiffHardDisk> node
1691 * @param aStorageNode <VirtualDiskImage> node
1692 */
1693HRESULT HVirtualDiskImage::saveSettings (CFGNODE aHDNode, CFGNODE aStorageNode)
1694{
1695 AssertReturn (aHDNode && aStorageNode, E_FAIL);
1696
1697 AutoLock alock (this);
1698 CHECK_READY();
1699
1700 // filePath (required)
1701 CFGLDRSetBSTR (aStorageNode, "filePath", mFilePath);
1702
1703 // save basic settings and children
1704 return HardDisk::saveSettings (aHDNode);
1705}
1706
1707/**
1708 * Checks if the given change of \a aOldPath to \a aNewPath affects the path
1709 * of this hard disk and updates it if necessary to reflect the new location.
1710 * Intended to be from HardDisk::updatePaths().
1711 *
1712 * @param aOldPath old path (full)
1713 * @param aNewPath new path (full)
1714 *
1715 * @note Locks this object for writing.
1716 */
1717void HVirtualDiskImage::updatePath (const char *aOldPath, const char *aNewPath)
1718{
1719 AssertReturnVoid (aOldPath);
1720 AssertReturnVoid (aNewPath);
1721
1722 AutoLock alock (this);
1723 AssertReturnVoid (isReady());
1724
1725 size_t oldPathLen = strlen (aOldPath);
1726
1727 Utf8Str path = mFilePathFull;
1728 LogFlowThisFunc (("VDI.fullPath={%s}\n", path.raw()));
1729
1730 if (RTPathStartsWith (path, aOldPath))
1731 {
1732 Utf8Str newPath = Utf8StrFmt ("%s%s", aNewPath,
1733 path.raw() + oldPathLen);
1734 path = newPath;
1735
1736 mVirtualBox->calculateRelativePath (path, path);
1737
1738 unconst (mFilePathFull) = newPath;
1739 unconst (mFilePath) = path;
1740
1741 LogFlowThisFunc (("-> updated: full={%s} short={%s}\n",
1742 newPath.raw(), path.raw()));
1743 }
1744}
1745
1746/**
1747 * Returns the string representation of this hard disk.
1748 * When \a aShort is false, returns the full image file path.
1749 * Otherwise, returns the image file name only.
1750 *
1751 * @param aShort if true, a short representation is returned
1752 */
1753Bstr HVirtualDiskImage::toString (bool aShort /* = false */)
1754{
1755 AutoLock alock (this);
1756
1757 if (!aShort)
1758 return mFilePathFull;
1759 else
1760 {
1761 Utf8Str fname = mFilePathFull;
1762 return RTPathFilename (fname.mutableRaw());
1763 }
1764}
1765
1766/**
1767 * Creates a clone of this hard disk by storing hard disk data in the given
1768 * VDI file name.
1769 *
1770 * @param aId UUID to assign to the created image
1771 * @param aTargetPath VDI file where the cloned image is to be to stored
1772 * @param aProgress progress object to run during operation
1773 */
1774HRESULT
1775HVirtualDiskImage::cloneToImage (const Guid &aId, const Utf8Str &aTargetPath,
1776 Progress *aProgress)
1777{
1778 AssertReturn (!aId.isEmpty(), E_FAIL);
1779 AssertReturn (!aTargetPath.isNull(), E_FAIL);
1780 AssertReturn (aProgress, E_FAIL);
1781
1782 AutoLock alock (this);
1783 AssertReturn (isReady(), E_FAIL);
1784
1785 AssertReturn (isBusy() == false, E_FAIL);
1786
1787 /// @todo (dmik) cloning of differencing images is not yet supported
1788 AssertReturn (mParent.isNull(), E_FAIL);
1789
1790 Utf8Str filePathFull = mFilePathFull;
1791
1792 if (mState == NotCreated)
1793 return setError (E_FAIL,
1794 tr ("Source hard disk image '%s' is not yet created"),
1795 filePathFull.raw());
1796
1797 addReader();
1798 alock.leave();
1799
1800 int vrc = VDICopyImage (aTargetPath, filePathFull, NULL,
1801 progressCallback,
1802 static_cast <Progress *> (aProgress));
1803
1804 alock.enter();
1805 releaseReader();
1806
1807 if (VBOX_SUCCESS (vrc))
1808 vrc = VDISetImageUUIDs (aTargetPath, aId, NULL, NULL, NULL);
1809
1810 if (VBOX_FAILURE (vrc))
1811 return setError (E_FAIL,
1812 tr ("Could not copy the hard disk image '%s' to '%s' (%Vrc)"),
1813 filePathFull.raw(), aTargetPath.raw(), vrc);
1814
1815 return S_OK;
1816}
1817
1818/**
1819 * Creates a new differencing image for this hard disk with the given
1820 * VDI file name.
1821 *
1822 * @param aId UUID to assign to the created image
1823 * @param aTargetPath VDI file where to store the created differencing image
1824 * @param aProgress progress object to run during operation
1825 * (can be NULL)
1826 */
1827HRESULT
1828HVirtualDiskImage::createDiffImage (const Guid &aId, const Utf8Str &aTargetPath,
1829 Progress *aProgress)
1830{
1831 AssertReturn (!aId.isEmpty(), E_FAIL);
1832 AssertReturn (!aTargetPath.isNull(), E_FAIL);
1833
1834 AutoLock alock (this);
1835 AssertReturn (isReady(), E_FAIL);
1836
1837 AssertReturn (isBusy() == false, E_FAIL);
1838 AssertReturn (mState >= Created, E_FAIL);
1839
1840 addReader();
1841 alock.leave();
1842
1843 int vrc = VDICreateDifferenceImage (aTargetPath, Utf8Str (mFilePathFull),
1844 NULL, aProgress ? progressCallback : NULL,
1845 static_cast <Progress *> (aProgress));
1846 alock.enter();
1847 releaseReader();
1848
1849 /* update the UUID to correspond to the file name */
1850 if (VBOX_SUCCESS (vrc))
1851 vrc = VDISetImageUUIDs (aTargetPath, aId, NULL, NULL, NULL);
1852
1853 if (VBOX_FAILURE (vrc))
1854 return setError (E_FAIL,
1855 tr ("Could not create a differencing hard disk '%s' (%Vrc)"),
1856 aTargetPath.raw(), vrc);
1857
1858 return S_OK;
1859}
1860
1861/**
1862 * Copies the image file of this hard disk to a separate VDI file (with an
1863 * unique creation UUID) and creates a new hard disk object for the copied
1864 * image. The copy will be created as a child of this hard disk's parent
1865 * (so that this hard disk must be a differencing one).
1866 *
1867 * The specified progress object (if not NULL) receives the percentage
1868 * of the operation completion. However, it is responsibility of the caller to
1869 * call Progress::notifyComplete() after this method returns.
1870 *
1871 * @param aFolder folder where to create a copy (must be a full path)
1872 * @param aMachineId machine ID the new hard disk will belong to
1873 * @param aHardDisk resulting hard disk object
1874 * @param aProgress progress object to run during copy operation
1875 * (may be NULL)
1876 *
1877 * @note
1878 * Must be NOT called from under locks of other objects that need external
1879 * access dirung this method execurion!
1880 */
1881HRESULT
1882HVirtualDiskImage::cloneDiffImage (const Bstr &aFolder, const Guid &aMachineId,
1883 ComObjPtr <HVirtualDiskImage> &aHardDisk,
1884 Progress *aProgress)
1885{
1886 AssertReturn (!aFolder.isEmpty() && !aMachineId.isEmpty(),
1887 E_FAIL);
1888
1889 AutoLock alock (this);
1890 CHECK_READY();
1891
1892 AssertReturn (!mParent.isNull(), E_FAIL);
1893
1894 ComAssertRet (isBusy() == false, E_FAIL);
1895 ComAssertRet (mState >= Created, E_FAIL);
1896
1897 Guid id;
1898 id.create();
1899
1900 Utf8Str filePathTo = Utf8StrFmt ("%ls%c{%Vuuid}.vdi",
1901 aFolder.raw(), RTPATH_DELIMITER, id.ptr());
1902
1903 /* try to make the path relative to the vbox home dir */
1904 const char *filePathToRel = filePathTo;
1905 {
1906 const Utf8Str &homeDir = mVirtualBox->homeDir();
1907 if (!strncmp (filePathTo, homeDir, homeDir.length()))
1908 filePathToRel = (filePathToRel + homeDir.length() + 1);
1909 }
1910
1911 /* first ensure the directory exists */
1912 {
1913 Utf8Str dir = aFolder;
1914 if (!RTDirExists (dir))
1915 {
1916 int vrc = RTDirCreateFullPath (dir, 0777);
1917 if (VBOX_FAILURE (vrc))
1918 {
1919 return setError (E_FAIL,
1920 tr ("Could not create a directory '%s' "
1921 "to store the image file (%Vrc)"),
1922 dir.raw(), vrc);
1923 }
1924 }
1925 }
1926
1927 Utf8Str filePathFull = mFilePathFull;
1928
1929 alock.leave();
1930
1931 int vrc = VDICopyImage (filePathTo, filePathFull, NULL,
1932 progressCallback,
1933 static_cast <Progress *> (aProgress));
1934
1935 alock.enter();
1936
1937 /* get modification and parent UUIDs of this image */
1938 RTUUID modUuid, parentUuid, parentModUuid;
1939 if (VBOX_SUCCESS (vrc))
1940 vrc = VDIGetImageUUIDs (filePathFull, NULL, &modUuid,
1941 &parentUuid, &parentModUuid);
1942
1943 // update the UUID of the copy to correspond to the file name
1944 // and copy all other UUIDs from this image
1945 if (VBOX_SUCCESS (vrc))
1946 vrc = VDISetImageUUIDs (filePathTo, id, &modUuid,
1947 &parentUuid, &parentModUuid);
1948
1949 if (VBOX_FAILURE (vrc))
1950 return setError (E_FAIL,
1951 tr ("Could not copy the hard disk image '%s' to '%s' (%Vrc)"),
1952 filePathFull.raw(), filePathTo.raw(), vrc);
1953
1954 ComObjPtr <HVirtualDiskImage> vdi;
1955 vdi.createObject();
1956 HRESULT rc = vdi->init (mVirtualBox, mParent, Bstr (filePathToRel),
1957 TRUE /* aRegistered */);
1958 if (FAILED (rc))
1959 return rc;
1960
1961 /* associate the created hard disk with the given machine */
1962 vdi->setMachineId (aMachineId);
1963
1964 rc = mVirtualBox->registerHardDisk (vdi, VirtualBox::RHD_Internal);
1965 if (FAILED (rc))
1966 return rc;
1967
1968 aHardDisk = vdi;
1969
1970 return S_OK;
1971}
1972
1973/**
1974 * Merges this child image to its parent image and updates the parent UUID
1975 * of all children of this image (to point to this image's parent).
1976 * It's a responsibility of the caller to unregister and uninitialize
1977 * the merged image on success.
1978 *
1979 * This method is intended to be called on a worker thread (the operation
1980 * can be time consuming).
1981 *
1982 * The specified progress object (if not NULL) receives the percentage
1983 * of the operation completion. However, it is responsibility of the caller to
1984 * call Progress::notifyComplete() after this method returns.
1985 *
1986 * @param aProgress progress object to run during copy operation
1987 * (may be NULL)
1988 *
1989 * @note
1990 * This method expects that both this hard disk and the paret hard disk
1991 * are marked as busy using #setBusyWithChildren() prior to calling it!
1992 * Busy flags of both hard disks will be cleared by this method
1993 * on a successful return. In case of failure, #clearBusyWithChildren()
1994 * must be called on a parent.
1995 *
1996 * @note
1997 * Must be NOT called from under locks of other objects that need external
1998 * access dirung this method execurion!
1999 */
2000HRESULT HVirtualDiskImage::mergeImageToParent (Progress *aProgress)
2001{
2002 LogFlowMember (("HVirtualDiskImage::mergeImageToParent(): image='%ls'\n",
2003 mFilePathFull.raw()));
2004
2005 AutoLock alock (this);
2006 CHECK_READY();
2007
2008 AssertReturn (!mParent.isNull(), E_FAIL);
2009 AutoLock parentLock (mParent);
2010
2011 ComAssertRet (isBusy() == true, E_FAIL);
2012 ComAssertRet (mParent->isBusy() == true, E_FAIL);
2013
2014 ComAssertRet (mState >= Created && mParent->asVDI()->mState >= Created, E_FAIL);
2015
2016 ComAssertMsgRet (mParent->storageType() == HardDiskStorageType_VirtualDiskImage,
2017 ("non VDI storage types are not yet supported!"), E_FAIL);
2018
2019 parentLock.leave();
2020 alock.leave();
2021
2022 int vrc = VDIMergeImage (Utf8Str (mFilePathFull),
2023 Utf8Str (mParent->asVDI()->mFilePathFull),
2024 progressCallback,
2025 static_cast <Progress *> (aProgress));
2026 alock.enter();
2027 parentLock.enter();
2028
2029 if (VBOX_FAILURE (vrc))
2030 return setError (E_FAIL,
2031 tr ("Could not merge the hard disk image '%ls' to "
2032 "its parent image '%ls' (%Vrc)"),
2033 mFilePathFull.raw(), mParent->asVDI()->mFilePathFull.raw(), vrc);
2034
2035 {
2036 HRESULT rc = S_OK;
2037
2038 AutoLock chLock (childrenLock());
2039
2040 for (HardDiskList::const_iterator it = children().begin();
2041 it != children().end(); ++ it)
2042 {
2043 ComObjPtr <HVirtualDiskImage> child = (*it)->asVDI();
2044 AutoLock childLock (child);
2045
2046 /* reparent the child */
2047 child->mParent = mParent;
2048 if (mParent)
2049 mParent->addDependentChild (child);
2050
2051 /* change the parent UUID in the image as well */
2052 RTUUID parentUuid, parentModUuid;
2053 vrc = VDIGetImageUUIDs (Utf8Str (mParent->asVDI()->mFilePathFull),
2054 &parentUuid, &parentModUuid, NULL, NULL);
2055 if (VBOX_FAILURE (vrc))
2056 {
2057 rc = setError (E_FAIL,
2058 tr ("Could not access the hard disk image '%ls' (%Vrc)"),
2059 mParent->asVDI()->mFilePathFull.raw(), vrc);
2060 break;
2061 }
2062 ComAssertBreak (mParent->id() == Guid (parentUuid), rc = E_FAIL);
2063 vrc = VDISetImageUUIDs (Utf8Str (child->mFilePathFull),
2064 NULL, NULL, &parentUuid, &parentModUuid);
2065 if (VBOX_FAILURE (vrc))
2066 {
2067 rc = setError (E_FAIL,
2068 tr ("Could not update parent UUID of the hard disk image "
2069 "'%ls' (%Vrc)"),
2070 child->mFilePathFull.raw(), vrc);
2071 break;
2072 }
2073 }
2074
2075 if (FAILED (rc))
2076 return rc;
2077 }
2078
2079 /* detach all our children to avoid their uninit in #uninit() */
2080 removeDependentChildren();
2081
2082 mParent->clearBusy();
2083 clearBusy();
2084
2085 return S_OK;
2086}
2087
2088/**
2089 * Merges this image to all its child images, updates the parent UUID
2090 * of all children of this image (to point to this image's parent).
2091 * It's a responsibility of the caller to unregister and uninitialize
2092 * the merged image on success.
2093 *
2094 * This method is intended to be called on a worker thread (the operation
2095 * can be time consuming).
2096 *
2097 * The specified progress object (if not NULL) receives the percentage
2098 * of the operation completion. However, it is responsibility of the caller to
2099 * call Progress::notifyComplete() after this method returns.
2100 *
2101 * @param aProgress progress object to run during copy operation
2102 * (may be NULL)
2103 *
2104 * @note
2105 * This method expects that both this hard disk and all children
2106 * are marked as busy using setBusyWithChildren() prior to calling it!
2107 * Busy flags of all affected hard disks will be cleared by this method
2108 * on a successful return. In case of failure, #clearBusyWithChildren()
2109 * must be called for this hard disk.
2110 *
2111 * @note
2112 * Must be NOT called from under locks of other objects that need external
2113 * access dirung this method execurion!
2114 */
2115HRESULT HVirtualDiskImage::mergeImageToChildren (Progress *aProgress)
2116{
2117 LogFlowMember (("HVirtualDiskImage::mergeImageToChildren(): image='%ls'\n",
2118 mFilePathFull.raw()));
2119
2120 AutoLock alock (this);
2121 CHECK_READY();
2122
2123 /* this must be a diff image */
2124 AssertReturn (isDifferencing(), E_FAIL);
2125
2126 ComAssertRet (isBusy() == true, E_FAIL);
2127 ComAssertRet (mState >= Created, E_FAIL);
2128
2129 ComAssertMsgRet (mParent->storageType() == HardDiskStorageType_VirtualDiskImage,
2130 ("non VDI storage types are not yet supported!"), E_FAIL);
2131
2132 {
2133 HRESULT rc = S_OK;
2134
2135 AutoLock chLock (childrenLock());
2136
2137 /* iterate over a copy since we will modify the list */
2138 HardDiskList list = children();
2139
2140 for (HardDiskList::const_iterator it = list.begin();
2141 it != list.end(); ++ it)
2142 {
2143 ComObjPtr <HardDisk> hd = *it;
2144 ComObjPtr <HVirtualDiskImage> child = hd->asVDI();
2145 AutoLock childLock (child);
2146
2147 ComAssertRet (child->isBusy() == true, E_FAIL);
2148 ComAssertBreak (child->mState >= Created, rc = E_FAIL);
2149
2150 childLock.leave();
2151 alock.leave();
2152
2153 int vrc = VDIMergeImage (Utf8Str (mFilePathFull),
2154 Utf8Str (child->mFilePathFull),
2155 progressCallback,
2156 static_cast <Progress *> (aProgress));
2157 alock.enter();
2158 childLock.enter();
2159
2160 if (VBOX_FAILURE (vrc))
2161 {
2162 rc = setError (E_FAIL,
2163 tr ("Could not merge the hard disk image '%ls' to "
2164 "its parent image '%ls' (%Vrc)"),
2165 mFilePathFull.raw(), child->mFilePathFull.raw(), vrc);
2166 break;
2167 }
2168
2169 /* reparent the child */
2170 child->mParent = mParent;
2171 if (mParent)
2172 mParent->addDependentChild (child);
2173
2174 /* change the parent UUID in the image as well */
2175 RTUUID parentUuid, parentModUuid;
2176 vrc = VDIGetImageUUIDs (Utf8Str (mParent->asVDI()->mFilePathFull),
2177 &parentUuid, &parentModUuid, NULL, NULL);
2178 if (VBOX_FAILURE (vrc))
2179 {
2180 rc = setError (E_FAIL,
2181 tr ("Could not access the hard disk image '%ls' (%Vrc)"),
2182 mParent->asVDI()->mFilePathFull.raw(), vrc);
2183 break;
2184 }
2185 ComAssertBreak (mParent->id() == Guid (parentUuid), rc = E_FAIL);
2186 vrc = VDISetImageUUIDs (Utf8Str (child->mFilePathFull),
2187 NULL, NULL, &parentUuid, &parentModUuid);
2188 if (VBOX_FAILURE (vrc))
2189 {
2190 rc = setError (E_FAIL,
2191 tr ("Could not update parent UUID of the hard disk image "
2192 "'%ls' (%Vrc)"),
2193 child->mFilePathFull.raw(), vrc);
2194 break;
2195 }
2196
2197 /* detach child to avoid its uninit in #uninit() */
2198 removeDependentChild (child);
2199
2200 /* remove the busy flag */
2201 child->clearBusy();
2202 }
2203
2204 if (FAILED (rc))
2205 return rc;
2206 }
2207
2208 clearBusy();
2209
2210 return S_OK;
2211}
2212
2213/**
2214 * Deletes and recreates the differencing hard disk image from scratch.
2215 * The file name and UUID remain the same.
2216 */
2217HRESULT HVirtualDiskImage::wipeOutImage()
2218{
2219 AutoLock alock (this);
2220 CHECK_READY();
2221
2222 AssertReturn (isDifferencing(), E_FAIL);
2223 AssertReturn (mRegistered, E_FAIL);
2224 AssertReturn (mState >= Created, E_FAIL);
2225
2226 ComAssertMsgRet (mParent->storageType() == HardDiskStorageType_VirtualDiskImage,
2227 ("non-VDI storage types are not yet supported!"), E_FAIL);
2228
2229 Utf8Str filePathFull = mFilePathFull;
2230
2231 int vrc = RTFileDelete (filePathFull);
2232 if (VBOX_FAILURE (vrc))
2233 return setError (E_FAIL,
2234 tr ("Could not delete the image file '%s' (%Vrc)"),
2235 filePathFull.raw(), vrc);
2236
2237 vrc = VDICreateDifferenceImage (filePathFull,
2238 Utf8Str (mParent->asVDI()->mFilePathFull),
2239 NULL, NULL, NULL);
2240 /* update the UUID to correspond to the file name */
2241 if (VBOX_SUCCESS (vrc))
2242 vrc = VDISetImageUUIDs (filePathFull, mId, NULL, NULL, NULL);
2243
2244 if (VBOX_FAILURE (vrc))
2245 return setError (E_FAIL,
2246 tr ("Could not create a differencing hard disk '%s' (%Vrc)"),
2247 filePathFull.raw(), vrc);
2248
2249 return S_OK;
2250}
2251
2252// private methods
2253/////////////////////////////////////////////////////////////////////////////
2254
2255/**
2256 * Helper to set a new file path.
2257 * Resolves a path relatively to the Virtual Box home directory.
2258 *
2259 * @note
2260 * Must be called from under the object's lock!
2261 */
2262HRESULT HVirtualDiskImage::setFilePath (const BSTR aFilePath)
2263{
2264 if (aFilePath && *aFilePath)
2265 {
2266 /* get the full file name */
2267 char filePathFull [RTPATH_MAX];
2268 int vrc = RTPathAbsEx (mVirtualBox->homeDir(), Utf8Str (aFilePath),
2269 filePathFull, sizeof (filePathFull));
2270 if (VBOX_FAILURE (vrc))
2271 return setError (E_FAIL,
2272 tr ("Invalid image file path '%ls' (%Vrc)"), aFilePath, vrc);
2273
2274 mFilePath = aFilePath;
2275 mFilePathFull = filePathFull;
2276 }
2277 else
2278 {
2279 mFilePath.setNull();
2280 mFilePathFull.setNull();
2281 }
2282
2283 return S_OK;
2284}
2285
2286/**
2287 * Helper to query information about the VDI hard disk.
2288 *
2289 * @param aAccessError not used when NULL, otherwise see #getAccessible()
2290 *
2291 * @note Must be called from under the object's lock, only after
2292 * CHECK_BUSY_AND_READERS() succeeds.
2293 */
2294HRESULT HVirtualDiskImage::queryInformation (Bstr *aAccessError)
2295{
2296 AssertReturn (isLockedOnCurrentThread(), E_FAIL);
2297
2298 /* create a lock object to completely release it later */
2299 AutoLock alock (this);
2300
2301 AssertReturn (mStateCheckWaiters == 0, E_FAIL);
2302
2303 ComAssertRet (mState >= Created, E_FAIL);
2304
2305 HRESULT rc = S_OK;
2306 int vrc = VINF_SUCCESS;
2307
2308 /* lazily create a semaphore */
2309 vrc = RTSemEventMultiCreate (&mStateCheckSem);
2310 ComAssertRCRet (vrc, E_FAIL);
2311
2312 /* go to Busy state to prevent any concurrent modifications
2313 * after releasing the lock below (to unblock getters before
2314 * a lengthy operation) */
2315 setBusy();
2316
2317 alock.leave();
2318
2319 /* VBoxVHDD management interface needs to be optimized: we're opening a
2320 * file three times in a raw to get three bits of information. */
2321
2322 Utf8Str filePath = mFilePathFull;
2323 Bstr errMsg;
2324
2325 do
2326 {
2327 /* check the image file */
2328 Guid id, parentId;
2329 vrc = VDICheckImage (filePath, NULL, NULL, NULL,
2330 id.ptr(), parentId.ptr(), NULL, 0);
2331
2332 if (VBOX_FAILURE (vrc))
2333 break;
2334
2335 if (!mId.isEmpty())
2336 {
2337 /* check that the actual UUID of the image matches the stored UUID */
2338 if (mId != id)
2339 {
2340 errMsg = Utf8StrFmt (
2341 tr ("Actual UUID {%Vuuid} of the hard disk image '%s' doesn't "
2342 "match UUID {%Vuuid} stored in the registry"),
2343 id.ptr(), filePath.raw(), mId.ptr());
2344 break;
2345 }
2346 }
2347 else
2348 {
2349 /* assgn an UUID read from the image file */
2350 mId = id;
2351 }
2352
2353 if (mParent)
2354 {
2355 /* check parent UUID */
2356 AutoLock parentLock (mParent);
2357 if (mParent->id() != parentId)
2358 {
2359 errMsg = Utf8StrFmt (
2360 tr ("UUID {%Vuuid} of the parent image '%ls' stored in "
2361 "the hard disk image file '%s' doesn't match "
2362 "UUID {%Vuuid} stored in the registry"),
2363 parentId.raw(), mParent->toString().raw(),
2364 filePath.raw(), mParent->id().raw());
2365 break;
2366 }
2367 }
2368 else if (!parentId.isEmpty())
2369 {
2370 errMsg = Utf8StrFmt (
2371 tr ("Hard disk image '%s' is a differencing image that is linked "
2372 "to a hard disk with UUID {%Vuuid} and cannot be used "
2373 "directly as a base hard disk"),
2374 filePath.raw(), parentId.raw());
2375 break;
2376 }
2377
2378 {
2379 RTFILE file = NIL_RTFILE;
2380 vrc = RTFileOpen (&file, filePath, RTFILE_O_READ);
2381 if (VBOX_SUCCESS (vrc))
2382 {
2383 uint64_t size = 0;
2384 vrc = RTFileGetSize (file, &size);
2385 if (VBOX_SUCCESS (vrc))
2386 mActualSize = size;
2387 RTFileClose (file);
2388 }
2389 if (VBOX_FAILURE (vrc))
2390 break;
2391 }
2392
2393 if (!mParent)
2394 {
2395 /* query logical size only for non-differencing images */
2396
2397 PVDIDISK disk = VDIDiskCreate();
2398 vrc = VDIDiskOpenImage (disk, Utf8Str (mFilePathFull),
2399 VDI_OPEN_FLAGS_READONLY);
2400 if (VBOX_SUCCESS (vrc))
2401 {
2402 uint64_t size = VDIDiskGetSize (disk);
2403 /* convert to MBytes */
2404 mSize = size / 1024 / 1024;
2405 }
2406
2407 VDIDiskDestroy (disk);
2408 if (VBOX_FAILURE (vrc))
2409 break;
2410 }
2411 }
2412 while (0);
2413
2414 /* enter the lock again */
2415 alock.enter();
2416
2417 /* remove the busy flag */
2418 clearBusy();
2419
2420 if (FAILED (rc) || VBOX_FAILURE (vrc) || !errMsg.isNull())
2421 {
2422 LogWarningFunc (("'%ls' is not accessible "
2423 "(rc=%08X, vrc=%Vrc, errMsg='%ls')\n",
2424 mFilePathFull.raw(), rc, vrc, errMsg.raw()));
2425
2426 if (aAccessError)
2427 {
2428 if (!errMsg.isNull())
2429 *aAccessError = errMsg;
2430 else if (VBOX_FAILURE (vrc))
2431 *aAccessError = Utf8StrFmt (
2432 tr ("Could not access hard disk image '%ls' (%Vrc)"),
2433 mFilePathFull.raw(), vrc);
2434 }
2435
2436 /* downgrade to not accessible */
2437 mState = Created;
2438 }
2439 else
2440 {
2441 if (aAccessError)
2442 aAccessError->setNull();
2443
2444 mState = Accessible;
2445 }
2446
2447 /* inform waiters if there are any */
2448 if (mStateCheckWaiters > 0)
2449 {
2450 RTSemEventMultiSignal (mStateCheckSem);
2451 }
2452 else
2453 {
2454 /* delete the semaphore ourselves */
2455 RTSemEventMultiDestroy (mStateCheckSem);
2456 mStateCheckSem = NIL_RTSEMEVENTMULTI;
2457 }
2458
2459 return rc;
2460}
2461
2462/**
2463 * Helper to create hard disk images.
2464 *
2465 * @param aSize size in MB
2466 * @param aDynamic dynamic or fixed image
2467 * @param aProgress address of IProgress pointer to return
2468 */
2469HRESULT HVirtualDiskImage::createImage (ULONG64 aSize, BOOL aDynamic,
2470 IProgress **aProgress)
2471{
2472 AutoLock alock (this);
2473
2474 CHECK_BUSY_AND_READERS();
2475
2476 if (mState != NotCreated)
2477 return setError (E_ACCESSDENIED,
2478 tr ("Hard disk image '%ls' is already created"),
2479 mFilePathFull.raw());
2480
2481 if (!mFilePathFull)
2482 return setError (E_ACCESSDENIED,
2483 tr ("Cannot create a hard disk image using an empty (null) file path"),
2484 mFilePathFull.raw());
2485
2486 /* first ensure the directory exists */
2487 {
2488 Utf8Str imageDir = mFilePathFull;
2489 RTPathStripFilename (imageDir.mutableRaw());
2490 if (!RTDirExists (imageDir))
2491 {
2492 int vrc = RTDirCreateFullPath (imageDir, 0777);
2493 if (VBOX_FAILURE (vrc))
2494 {
2495 return setError (E_FAIL,
2496 tr ("Could not create a directory '%s' "
2497 "to store the image file (%Vrc)"),
2498 imageDir.raw(), vrc);
2499 }
2500 }
2501 }
2502
2503 /* check whether the given file exists or not */
2504 RTFILE file;
2505 int vrc = RTFileOpen (&file, Utf8Str (mFilePathFull),
2506 RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
2507 if (vrc != VERR_FILE_NOT_FOUND)
2508 {
2509 if (VBOX_SUCCESS (vrc))
2510 RTFileClose (file);
2511 switch(vrc)
2512 {
2513 case VINF_SUCCESS:
2514 return setError (E_FAIL,
2515 tr ("Image file '%ls' already exists"),
2516 mFilePathFull.raw());
2517
2518 default:
2519 return setError(E_FAIL,
2520 tr ("Invalid image file path '%ls' (%Vrc)"),
2521 mFilePathFull.raw(), vrc);
2522 }
2523 }
2524
2525 /* check VDI size limits */
2526 {
2527 HRESULT rc;
2528 ULONG64 maxVDISize;
2529 ComPtr <ISystemProperties> props;
2530 rc = mVirtualBox->COMGETTER(SystemProperties) (props.asOutParam());
2531 ComAssertComRCRet (rc, E_FAIL);
2532 rc = props->COMGETTER(MaxVDISize) (&maxVDISize);
2533 ComAssertComRCRet (rc, E_FAIL);
2534
2535 if (aSize < 1 || aSize > maxVDISize)
2536 return setError (E_INVALIDARG,
2537 tr ("Invalid VDI size: %llu MB (must be in range [1, %llu] MB)"),
2538 aSize, maxVDISize);
2539 }
2540
2541 HRESULT rc;
2542
2543 /* create a project object */
2544 ComObjPtr <Progress> progress;
2545 progress.createObject();
2546 {
2547 Bstr desc = aDynamic ? tr ("Creating a dynamically expanding hard disk")
2548 : tr ("Creating a fixed-size hard disk");
2549 rc = progress->init (mVirtualBox, (IVirtualDiskImage *) this, desc,
2550 FALSE /* aCancelable */);
2551 CheckComRCReturnRC (rc);
2552 }
2553
2554 /* mark as busy (being created)
2555 * (VDI task thread will unmark it) */
2556 setBusy();
2557
2558 /* fill in VDI task data */
2559 VDITask *task = new VDITask (aDynamic ? VDITask::CreateDynamic
2560 : VDITask::CreateStatic,
2561 this, progress);
2562 task->size = aSize;
2563 task->size *= 1024 * 1024; /* convert to bytes */
2564
2565 /* create the hard disk creation thread, pass operation data */
2566 vrc = RTThreadCreate (NULL, vdiTaskThread, (void *) task, 0,
2567 RTTHREADTYPE_MAIN_HEAVY_WORKER, 0, "VDITask");
2568 ComAssertMsgRC (vrc, ("Could not create a thread (%Vrc)", vrc));
2569 if (VBOX_FAILURE (vrc))
2570 {
2571 clearBusy();
2572 delete task;
2573 rc = E_FAIL;
2574 }
2575 else
2576 {
2577 /* get one interface for the caller */
2578 progress.queryInterfaceTo (aProgress);
2579 }
2580
2581 return rc;
2582}
2583
2584/* static */
2585DECLCALLBACK(int) HVirtualDiskImage::vdiTaskThread (RTTHREAD thread, void *pvUser)
2586{
2587 VDITask *task = static_cast <VDITask *> (pvUser);
2588 AssertReturn (task, VERR_GENERAL_FAILURE);
2589
2590 LogFlow (("vdiTaskThread(): operation=%d, size=%llu\n",
2591 task->operation, task->size));
2592
2593 VDIIMAGETYPE type = (VDIIMAGETYPE) 0;
2594
2595 switch (task->operation)
2596 {
2597 case VDITask::CreateDynamic: type = VDI_IMAGE_TYPE_NORMAL; break;
2598 case VDITask::CreateStatic: type = VDI_IMAGE_TYPE_FIXED; break;
2599 case VDITask::CloneToImage: break;
2600 default: AssertFailedReturn (VERR_GENERAL_FAILURE); break;
2601 }
2602
2603 HRESULT rc = S_OK;
2604 Utf8Str errorMsg;
2605
2606 if (task->operation == VDITask::CloneToImage)
2607 {
2608 Assert (!task->vdi->id().isEmpty());
2609 /// @todo (dmik) check locks
2610 AutoLock sourceLock (task->source);
2611 rc = task->source->cloneToImage (task->vdi->id(),
2612 Utf8Str (task->vdi->filePathFull()),
2613 task->progress);
2614
2615 /* release reader added in HardDisk::CloneToImage() */
2616 task->source->releaseReader();
2617 }
2618 else
2619 {
2620 int vrc = VDICreateBaseImage (Utf8Str (task->vdi->filePathFull()),
2621 type, task->size,
2622 Utf8Str (task->vdi->mDescription),
2623 progressCallback,
2624 static_cast <Progress *> (task->progress));
2625
2626 if (VBOX_SUCCESS (vrc) && task->vdi->id())
2627 {
2628 /* we have a non-null UUID, update the created image */
2629 vrc = VDISetImageUUIDs (Utf8Str (task->vdi->filePathFull()),
2630 task->vdi->id().raw(), NULL, NULL, NULL);
2631 }
2632
2633 if (VBOX_FAILURE (vrc))
2634 {
2635 errorMsg = Utf8StrFmt (
2636 tr ("Falied to create a hard disk image '%ls' (%Vrc)"),
2637 task->vdi->filePathFull().raw(), vrc);
2638 rc = E_FAIL;
2639 }
2640 }
2641
2642 LogFlow (("vdiTaskThread(): rc=%08X\n", rc));
2643
2644 AutoLock alock (task->vdi);
2645
2646 /* clear busy set in in HardDisk::CloneToImage() or
2647 * in HVirtualDiskImage::createImage() */
2648 task->vdi->clearBusy();
2649
2650 if (SUCCEEDED (rc))
2651 {
2652 task->vdi->mState = HVirtualDiskImage::Created;
2653 /* update VDI data fields */
2654 Bstr errMsg;
2655 rc = task->vdi->queryInformation (&errMsg);
2656 /* we want to deliver the access check result to the caller
2657 * immediately, before he calls HardDisk::GetAccssible() himself. */
2658 if (SUCCEEDED (rc) && !errMsg.isNull())
2659 task->progress->notifyCompleteBstr (
2660 E_FAIL, COM_IIDOF (IVirtualDiskImage), getComponentName(),
2661 errMsg);
2662 else
2663 task->progress->notifyComplete (rc);
2664 }
2665 else
2666 {
2667 /* delete the target file so we don't have orphaned files */
2668 RTFileDelete(Utf8Str (task->vdi->filePathFull()));
2669
2670 task->vdi->mState = HVirtualDiskImage::NotCreated;
2671 /* complete the progress object */
2672 if (errorMsg)
2673 task->progress->notifyComplete (
2674 E_FAIL, COM_IIDOF (IVirtualDiskImage), getComponentName(),
2675 errorMsg);
2676 else
2677 task->progress->notifyComplete (rc);
2678 }
2679
2680 delete task;
2681
2682 return VINF_SUCCESS;
2683}
2684
2685////////////////////////////////////////////////////////////////////////////////
2686// HISCSIHardDisk class
2687////////////////////////////////////////////////////////////////////////////////
2688
2689// constructor / destructor
2690////////////////////////////////////////////////////////////////////////////////
2691
2692HRESULT HISCSIHardDisk::FinalConstruct()
2693{
2694 HRESULT rc = HardDisk::FinalConstruct();
2695 if (FAILED (rc))
2696 return rc;
2697
2698 mSize = 0;
2699 mActualSize = 0;
2700
2701 mPort = 0;
2702 mLun = 0;
2703
2704 return S_OK;
2705}
2706
2707void HISCSIHardDisk::FinalRelease()
2708{
2709 HardDisk::FinalRelease();
2710}
2711
2712// public initializer/uninitializer for internal purposes only
2713////////////////////////////////////////////////////////////////////////////////
2714
2715// public methods for internal purposes only
2716/////////////////////////////////////////////////////////////////////////////
2717
2718/**
2719 * Initializes the iSCSI hard disk object by reading its properties from
2720 * the given configuration node. The created hard disk will be marked as
2721 * registered on success.
2722 *
2723 * @param aHDNode <HardDisk> node
2724 * @param aVDINod <ISCSIHardDisk> node
2725 */
2726HRESULT HISCSIHardDisk::init (VirtualBox *aVirtualBox,
2727 CFGNODE aHDNode, CFGNODE aISCSINode)
2728{
2729 LogFlowMember (("HISCSIHardDisk::init (load)\n"));
2730
2731 AssertReturn (aHDNode && aISCSINode, E_FAIL);
2732
2733 AutoLock alock (this);
2734 ComAssertRet (!isReady(), E_UNEXPECTED);
2735
2736 mStorageType = HardDiskStorageType_ISCSIHardDisk;
2737
2738 HRESULT rc = S_OK;
2739
2740 do
2741 {
2742 rc = protectedInit (aVirtualBox, NULL);
2743 CheckComRCBreakRC (rc);
2744
2745 /* set ready to let protectedUninit() be called on failure */
2746 setReady (true);
2747
2748 /* server (required) */
2749 CFGLDRQueryBSTR (aISCSINode, "server", mServer.asOutParam());
2750 /* target (required) */
2751 CFGLDRQueryBSTR (aISCSINode, "target", mTarget.asOutParam());
2752
2753 /* port (optional) */
2754 CFGLDRQueryUInt16 (aISCSINode, "port", &mPort);
2755 /* lun (optional) */
2756 CFGLDRQueryUInt64 (aISCSINode, "lun", &mLun);
2757 /* userName (optional) */
2758 CFGLDRQueryBSTR (aISCSINode, "userName", mUserName.asOutParam());
2759 /* password (optional) */
2760 CFGLDRQueryBSTR (aISCSINode, "password", mPassword.asOutParam());
2761
2762 LogFlowMember ((" 'iscsi:%ls:%hu@%ls/%ls:%llu'\n",
2763 mServer.raw(), mPort, mUserName.raw(), mTarget.raw(),
2764 mLun));
2765
2766 /* load basic settings and children */
2767 rc = loadSettings (aHDNode);
2768 CheckComRCBreakRC (rc);
2769
2770 if (mType != HardDiskType_WritethroughHardDisk)
2771 {
2772 rc = setError (E_FAIL,
2773 tr ("Currently, non-Writethrough iSCSI hard disks are not "
2774 "allowed ('%ls')"),
2775 toString().raw());
2776 break;
2777 }
2778
2779 mRegistered = TRUE;
2780 }
2781 while (0);
2782
2783 if (FAILED (rc))
2784 uninit();
2785
2786 return rc;
2787}
2788
2789/**
2790 * Initializes the iSCSI hard disk object using default values for all
2791 * properties. The created hard disk will NOT be marked as registered.
2792 */
2793HRESULT HISCSIHardDisk::init (VirtualBox *aVirtualBox)
2794{
2795 LogFlowMember (("HISCSIHardDisk::init()\n"));
2796
2797 AutoLock alock (this);
2798 ComAssertRet (!isReady(), E_UNEXPECTED);
2799
2800 mStorageType = HardDiskStorageType_ISCSIHardDisk;
2801
2802 HRESULT rc = S_OK;
2803
2804 do
2805 {
2806 rc = protectedInit (aVirtualBox, NULL);
2807 CheckComRCBreakRC (rc);
2808
2809 /* set ready to let protectedUninit() be called on failure */
2810 setReady (true);
2811
2812 /* we have to generate a new UUID */
2813 mId.create();
2814 /* currently, all iSCSI hard disks are writethrough */
2815 mType = HardDiskType_WritethroughHardDisk;
2816 mRegistered = FALSE;
2817 }
2818 while (0);
2819
2820 if (FAILED (rc))
2821 uninit();
2822
2823 return rc;
2824}
2825
2826/**
2827 * Uninitializes the instance and sets the ready flag to FALSE.
2828 * Called either from FinalRelease(), by the parent when it gets destroyed,
2829 * or by a third party when it decides this object is no more valid.
2830 */
2831void HISCSIHardDisk::uninit()
2832{
2833 LogFlowMember (("HISCSIHardDisk::uninit()\n"));
2834
2835 AutoLock alock (this);
2836 if (!isReady())
2837 return;
2838
2839 HardDisk::protectedUninit (alock);
2840}
2841
2842// IHardDisk properties
2843////////////////////////////////////////////////////////////////////////////////
2844
2845STDMETHODIMP HISCSIHardDisk::COMGETTER(Description) (BSTR *aDescription)
2846{
2847 if (!aDescription)
2848 return E_POINTER;
2849
2850 AutoLock alock (this);
2851 CHECK_READY();
2852
2853 mDescription.cloneTo (aDescription);
2854 return S_OK;
2855}
2856
2857STDMETHODIMP HISCSIHardDisk::COMSETTER(Description) (INPTR BSTR aDescription)
2858{
2859 AutoLock alock (this);
2860 CHECK_READY();
2861
2862 CHECK_BUSY_AND_READERS();
2863
2864 if (mDescription != aDescription)
2865 {
2866 mDescription = aDescription;
2867 if (mRegistered)
2868 return mVirtualBox->saveSettings();
2869 }
2870
2871 return S_OK;
2872}
2873
2874STDMETHODIMP HISCSIHardDisk::COMGETTER(Size) (ULONG64 *aSize)
2875{
2876 if (!aSize)
2877 return E_POINTER;
2878
2879 AutoLock alock (this);
2880 CHECK_READY();
2881
2882 *aSize = mSize;
2883 return S_OK;
2884}
2885
2886STDMETHODIMP HISCSIHardDisk::COMGETTER(ActualSize) (ULONG64 *aActualSize)
2887{
2888 if (!aActualSize)
2889 return E_POINTER;
2890
2891 AutoLock alock (this);
2892 CHECK_READY();
2893
2894 *aActualSize = mActualSize;
2895 return S_OK;
2896}
2897
2898// IISCSIHardDisk properties
2899////////////////////////////////////////////////////////////////////////////////
2900
2901STDMETHODIMP HISCSIHardDisk::COMGETTER(Server) (BSTR *aServer)
2902{
2903 if (!aServer)
2904 return E_POINTER;
2905
2906 AutoLock alock (this);
2907 CHECK_READY();
2908
2909 mServer.cloneTo (aServer);
2910 return S_OK;
2911}
2912
2913STDMETHODIMP HISCSIHardDisk::COMSETTER(Server) (INPTR BSTR aServer)
2914{
2915 if (!aServer || !*aServer)
2916 return E_INVALIDARG;
2917
2918 AutoLock alock (this);
2919 CHECK_READY();
2920
2921 CHECK_BUSY_AND_READERS();
2922
2923 if (mServer != aServer)
2924 {
2925 mServer = aServer;
2926 if (mRegistered)
2927 return mVirtualBox->saveSettings();
2928 }
2929
2930 return S_OK;
2931}
2932
2933STDMETHODIMP HISCSIHardDisk::COMGETTER(Port) (USHORT *aPort)
2934{
2935 if (!aPort)
2936 return E_POINTER;
2937
2938 AutoLock alock (this);
2939 CHECK_READY();
2940
2941 *aPort = mPort;
2942 return S_OK;
2943}
2944
2945STDMETHODIMP HISCSIHardDisk::COMSETTER(Port) (USHORT aPort)
2946{
2947 AutoLock alock (this);
2948 CHECK_READY();
2949
2950 CHECK_BUSY_AND_READERS();
2951
2952 if (mPort != aPort)
2953 {
2954 mPort = aPort;
2955 if (mRegistered)
2956 return mVirtualBox->saveSettings();
2957 }
2958
2959 return S_OK;
2960}
2961
2962STDMETHODIMP HISCSIHardDisk::COMGETTER(Target) (BSTR *aTarget)
2963{
2964 if (!aTarget)
2965 return E_POINTER;
2966
2967 AutoLock alock (this);
2968 CHECK_READY();
2969
2970 mTarget.cloneTo (aTarget);
2971 return S_OK;
2972}
2973
2974STDMETHODIMP HISCSIHardDisk::COMSETTER(Target) (INPTR BSTR aTarget)
2975{
2976 if (!aTarget || !*aTarget)
2977 return E_INVALIDARG;
2978
2979 AutoLock alock (this);
2980 CHECK_READY();
2981
2982 CHECK_BUSY_AND_READERS();
2983
2984 if (mTarget != aTarget)
2985 {
2986 mTarget = aTarget;
2987 if (mRegistered)
2988 return mVirtualBox->saveSettings();
2989 }
2990
2991 return S_OK;
2992}
2993
2994STDMETHODIMP HISCSIHardDisk::COMGETTER(Lun) (ULONG64 *aLun)
2995{
2996 if (!aLun)
2997 return E_POINTER;
2998
2999 AutoLock alock (this);
3000 CHECK_READY();
3001
3002 *aLun = mLun;
3003 return S_OK;
3004}
3005
3006STDMETHODIMP HISCSIHardDisk::COMSETTER(Lun) (ULONG64 aLun)
3007{
3008 AutoLock alock (this);
3009 CHECK_READY();
3010
3011 CHECK_BUSY_AND_READERS();
3012
3013 if (mLun != aLun)
3014 {
3015 mLun = aLun;
3016 if (mRegistered)
3017 return mVirtualBox->saveSettings();
3018 }
3019
3020 return S_OK;
3021}
3022
3023STDMETHODIMP HISCSIHardDisk::COMGETTER(UserName) (BSTR *aUserName)
3024{
3025 if (!aUserName)
3026 return E_POINTER;
3027
3028 AutoLock alock (this);
3029 CHECK_READY();
3030
3031 mUserName.cloneTo (aUserName);
3032 return S_OK;
3033}
3034
3035STDMETHODIMP HISCSIHardDisk::COMSETTER(UserName) (INPTR BSTR aUserName)
3036{
3037 AutoLock alock (this);
3038 CHECK_READY();
3039
3040 CHECK_BUSY_AND_READERS();
3041
3042 if (mUserName != aUserName)
3043 {
3044 mUserName = aUserName;
3045 if (mRegistered)
3046 return mVirtualBox->saveSettings();
3047 }
3048
3049 return S_OK;
3050}
3051
3052STDMETHODIMP HISCSIHardDisk::COMGETTER(Password) (BSTR *aPassword)
3053{
3054 if (!aPassword)
3055 return E_POINTER;
3056
3057 AutoLock alock (this);
3058 CHECK_READY();
3059
3060 mPassword.cloneTo (aPassword);
3061 return S_OK;
3062}
3063
3064STDMETHODIMP HISCSIHardDisk::COMSETTER(Password) (INPTR BSTR aPassword)
3065{
3066 AutoLock alock (this);
3067 CHECK_READY();
3068
3069 CHECK_BUSY_AND_READERS();
3070
3071 if (mPassword != aPassword)
3072 {
3073 mPassword = aPassword;
3074 if (mRegistered)
3075 return mVirtualBox->saveSettings();
3076 }
3077
3078 return S_OK;
3079}
3080
3081// public/protected methods for internal purposes only
3082/////////////////////////////////////////////////////////////////////////////
3083
3084/**
3085 * Attempts to mark the hard disk as registered.
3086 * Only VirtualBox can call this method.
3087 */
3088HRESULT HISCSIHardDisk::trySetRegistered (BOOL aRegistered)
3089{
3090 AutoLock alock (this);
3091 CHECK_READY();
3092
3093 if (aRegistered)
3094 {
3095 if (mServer.isEmpty() || mTarget.isEmpty())
3096 return setError (E_FAIL,
3097 tr ("iSCSI Hard disk has no server or target defined"));
3098 }
3099 else
3100 {
3101 }
3102
3103 return HardDisk::trySetRegistered (aRegistered);
3104}
3105
3106/**
3107 * Checks accessibility of this iSCSI hard disk.
3108 */
3109HRESULT HISCSIHardDisk::getAccessible (Bstr &aAccessError)
3110{
3111 AutoLock alock (this);
3112 CHECK_READY();
3113
3114 /* check the basic accessibility */
3115 HRESULT rc = getBaseAccessible (aAccessError);
3116 if (FAILED (rc) || !aAccessError.isNull())
3117 return rc;
3118
3119 return queryInformation (aAccessError);
3120}
3121
3122/**
3123 * Saves hard disk settings to the specified storage node and saves
3124 * all children to the specified hard disk node
3125 *
3126 * @param aHDNode <HardDisk>
3127 * @param aStorageNode <ISCSIHardDisk> node
3128 */
3129HRESULT HISCSIHardDisk::saveSettings (CFGNODE aHDNode, CFGNODE aStorageNode)
3130{
3131 AssertReturn (aHDNode && aStorageNode, E_FAIL);
3132
3133 AutoLock alock (this);
3134 CHECK_READY();
3135
3136 /* server (required) */
3137 CFGLDRSetBSTR (aStorageNode, "server", mServer);
3138 /* target (required) */
3139 CFGLDRSetBSTR (aStorageNode, "target", mTarget);
3140
3141 /* port (optional) */
3142 if (mPort != 0)
3143 CFGLDRSetUInt16 (aStorageNode, "port", mPort);
3144 else
3145 CFGLDRDeleteAttribute (aStorageNode, "port");
3146 /* lun (optional) */
3147 if (mLun != 0)
3148 CFGLDRSetUInt64 (aStorageNode, "lun", mLun);
3149 else
3150 CFGLDRDeleteAttribute (aStorageNode, "lun");
3151 /* userName (optional) */
3152 if (!mUserName.isNull())
3153 CFGLDRSetBSTR (aStorageNode, "userName", mUserName);
3154 else
3155 CFGLDRDeleteAttribute (aStorageNode, "userName");
3156 /* password (optional) */
3157 if (!mPassword.isNull())
3158 CFGLDRSetBSTR (aStorageNode, "password", mPassword);
3159 else
3160 CFGLDRDeleteAttribute (aStorageNode, "password");
3161
3162 /* save basic settings and children */
3163 return HardDisk::saveSettings (aHDNode);
3164}
3165
3166/**
3167 * Returns the string representation of this hard disk.
3168 * When \a aShort is false, returns the full image file path.
3169 * Otherwise, returns the image file name only.
3170 *
3171 * @param aShort if true, a short representation is returned
3172 */
3173Bstr HISCSIHardDisk::toString (bool aShort /* = false */)
3174{
3175 AutoLock alock (this);
3176
3177 Bstr str;
3178 if (!aShort)
3179 {
3180 /* the format is iscsi:[<user@>]<server>[:<port>]/<target>[:<lun>] */
3181 str = Utf8StrFmt ("iscsi:%s%ls%s/%ls%s",
3182 !mUserName.isNull() ? Utf8StrFmt ("%ls@", mUserName.raw()).raw() : "",
3183 mServer.raw(),
3184 mPort != 0 ? Utf8StrFmt (":%hu", mPort).raw() : "",
3185 mTarget.raw(),
3186 mLun != 0 ? Utf8StrFmt (":%llu", mLun).raw() : "");
3187 }
3188 else
3189 {
3190 str = Utf8StrFmt ("%ls%s",
3191 mTarget.raw(),
3192 mLun != 0 ? Utf8StrFmt (":%llu", mLun).raw() : "");
3193 }
3194
3195 return str;
3196}
3197
3198/**
3199 * Creates a clone of this hard disk by storing hard disk data in the given
3200 * VDI file name.
3201 *
3202 * @param aId UUID to assign to the created image
3203 * @param aTargetPath VDI file where the cloned image is to be to stored
3204 * @param aProgress progress object to run during operation
3205 */
3206HRESULT
3207HISCSIHardDisk::cloneToImage (const Guid &aId, const Utf8Str &aTargetPath,
3208 Progress *aProgress)
3209{
3210 ComAssertMsgFailed (("Not implemented"));
3211 return E_NOTIMPL;
3212
3213// AssertReturn (isBusy() == false, E_FAIL);
3214// addReader();
3215// releaseReader();
3216}
3217
3218/**
3219 * Creates a new differencing image for this hard disk with the given
3220 * VDI file name.
3221 *
3222 * @param aId UUID to assign to the created image
3223 * @param aTargetPath VDI file where to store the created differencing image
3224 * @param aProgress progress object to run during operation
3225 * (can be NULL)
3226 */
3227HRESULT
3228HISCSIHardDisk::createDiffImage (const Guid &aId, const Utf8Str &aTargetPath,
3229 Progress *aProgress)
3230{
3231 ComAssertMsgFailed (("Not implemented"));
3232 return E_NOTIMPL;
3233
3234// AssertReturn (isBusy() == false, E_FAIL);
3235// addReader();
3236// releaseReader();
3237}
3238
3239// private methods
3240/////////////////////////////////////////////////////////////////////////////
3241
3242/**
3243 * Helper to query information about the iSCSI hard disk.
3244 *
3245 * @param aAccessError see #getAccessible()
3246 * @note
3247 * Must be called from under the object's lock!
3248 */
3249HRESULT HISCSIHardDisk::queryInformation (Bstr &aAccessError)
3250{
3251 /// @todo (dmik) query info about this iSCSI disk,
3252 // set mSize and mActualSize,
3253 // or set aAccessError in case of failure
3254
3255 aAccessError.setNull();
3256 return S_OK;
3257}
3258
3259////////////////////////////////////////////////////////////////////////////////
3260// HVMDKImage class
3261////////////////////////////////////////////////////////////////////////////////
3262
3263// constructor / destructor
3264////////////////////////////////////////////////////////////////////////////////
3265
3266HRESULT HVMDKImage::FinalConstruct()
3267{
3268 HRESULT rc = HardDisk::FinalConstruct();
3269 if (FAILED (rc))
3270 return rc;
3271
3272 mState = NotCreated;
3273
3274 mStateCheckSem = NIL_RTSEMEVENTMULTI;
3275 mStateCheckWaiters = 0;
3276
3277 mSize = 0;
3278 mActualSize = 0;
3279
3280 return S_OK;
3281}
3282
3283void HVMDKImage::FinalRelease()
3284{
3285 HardDisk::FinalRelease();
3286}
3287
3288// public initializer/uninitializer for internal purposes only
3289////////////////////////////////////////////////////////////////////////////////
3290
3291// public methods for internal purposes only
3292/////////////////////////////////////////////////////////////////////////////
3293
3294/**
3295 * Initializes the VMDK hard disk object by reading its properties from
3296 * the given configuration node. The created hard disk will be marked as
3297 * registered on success.
3298 *
3299 * @param aHDNode <HardDisk> node
3300 * @param aVMDKNode <VirtualDiskImage> node
3301 */
3302HRESULT HVMDKImage::init (VirtualBox *aVirtualBox, HardDisk *aParent,
3303 CFGNODE aHDNode, CFGNODE aVMDKNode)
3304{
3305 LogFlowMember (("HVMDKImage::init (load)\n"));
3306
3307 AssertReturn (aHDNode && aVMDKNode, E_FAIL);
3308
3309 AutoLock alock (this);
3310 ComAssertRet (!isReady(), E_UNEXPECTED);
3311
3312 mStorageType = HardDiskStorageType_VMDKImage;
3313
3314 HRESULT rc = S_OK;
3315
3316 do
3317 {
3318 rc = protectedInit (aVirtualBox, aParent);
3319 CheckComRCBreakRC (rc);
3320
3321 /* set ready to let protectedUninit() be called on failure */
3322 setReady (true);
3323
3324 /* filePath (required) */
3325 Bstr filePath;
3326 CFGLDRQueryBSTR (aVMDKNode, "filePath", filePath.asOutParam());
3327
3328 rc = setFilePath (filePath);
3329 CheckComRCBreakRC (rc);
3330
3331 LogFlowMember ((" '%ls'\n", mFilePathFull.raw()));
3332
3333 /* load basic settings and children */
3334 rc = loadSettings (aHDNode);
3335 CheckComRCBreakRC (rc);
3336
3337 if (mType != HardDiskType_WritethroughHardDisk)
3338 {
3339 rc = setError (E_FAIL,
3340 tr ("Currently, non-Writethrough VMDK images are not "
3341 "allowed ('%ls')"),
3342 toString().raw());
3343 break;
3344 }
3345
3346 mState = Created;
3347 mRegistered = TRUE;
3348
3349 /* Don't call queryInformation() for registered hard disks to
3350 * prevent the calling thread (i.e. the VirtualBox server startup
3351 * thread) from an unexpected freeze. The vital mId property (UUID)
3352 * is read from the registry file in loadSettings(). To get the rest,
3353 * the user will have to call COMGETTER(Accessible) manually. */
3354 }
3355 while (0);
3356
3357 if (FAILED (rc))
3358 uninit();
3359
3360 return rc;
3361}
3362
3363/**
3364 * Initializes the VMDK hard disk object using the given image file name.
3365 *
3366 * @param aVirtualBox VirtualBox parent.
3367 * @param aParent Currently, must always be NULL.
3368 * @param aFilePath path to the image file (can be NULL to create an
3369 * imageless object)
3370 * @param aRegistered whether to mark this disk as registered or not
3371 * (ignored when \a aFilePath is NULL, assuming FALSE)
3372 */
3373HRESULT HVMDKImage::init (VirtualBox *aVirtualBox, HardDisk *aParent,
3374 const BSTR aFilePath, BOOL aRegistered /* = FALSE */)
3375{
3376 LogFlowMember (("HVMDKImage::init (aFilePath='%ls', aRegistered=%d)\n",
3377 aFilePath, aRegistered));
3378
3379 AssertReturn (aParent == NULL, E_FAIL);
3380
3381 AutoLock alock (this);
3382 ComAssertRet (!isReady(), E_UNEXPECTED);
3383
3384 mStorageType = HardDiskStorageType_VMDKImage;
3385
3386 HRESULT rc = S_OK;
3387
3388 do
3389 {
3390 rc = protectedInit (aVirtualBox, aParent);
3391 CheckComRCBreakRC (rc);
3392
3393 /* set ready to let protectedUninit() be called on failure */
3394 setReady (true);
3395
3396 rc = setFilePath (aFilePath);
3397 CheckComRCBreakRC (rc);
3398
3399 /* currently, all VMDK hard disks are writethrough */
3400 mType = HardDiskType_WritethroughHardDisk;
3401
3402 Assert (mId.isEmpty());
3403
3404 if (aFilePath && *aFilePath)
3405 {
3406 mRegistered = aRegistered;
3407 mState = Created;
3408
3409 /* Call queryInformation() anyway (even if it will block), because
3410 * it is the only way to get the UUID of the existing VDI and
3411 * initialize the vital mId property. */
3412 Bstr errMsg;
3413 rc = queryInformation (&errMsg);
3414 if (SUCCEEDED (rc))
3415 {
3416 /* We are constructing a new HVirtualDiskImage object. If there
3417 * is a fatal accessibility error (we cannot read image UUID),
3418 * we have to fail. We do so even on non-fatal errors as well,
3419 * because it's not worth to keep going with the inaccessible
3420 * image from the very beginning (when nothing else depends on
3421 * it yet). */
3422 if (!errMsg.isNull())
3423 rc = setErrorBstr (E_FAIL, errMsg);
3424 }
3425 }
3426 else
3427 {
3428 mRegistered = FALSE;
3429 mState = NotCreated;
3430 mId.create();
3431 }
3432 }
3433 while (0);
3434
3435 if (FAILED (rc))
3436 uninit();
3437
3438 return rc;
3439}
3440
3441/**
3442 * Uninitializes the instance and sets the ready flag to FALSE.
3443 * Called either from FinalRelease(), by the parent when it gets destroyed,
3444 * or by a third party when it decides this object is no more valid.
3445 */
3446void HVMDKImage::uninit()
3447{
3448 LogFlowMember (("HVMDKImage::uninit()\n"));
3449
3450 AutoLock alock (this);
3451 if (!isReady())
3452 return;
3453
3454 HardDisk::protectedUninit (alock);
3455}
3456
3457// IHardDisk properties
3458////////////////////////////////////////////////////////////////////////////////
3459
3460STDMETHODIMP HVMDKImage::COMGETTER(Description) (BSTR *aDescription)
3461{
3462 if (!aDescription)
3463 return E_POINTER;
3464
3465 AutoLock alock (this);
3466 CHECK_READY();
3467
3468 mDescription.cloneTo (aDescription);
3469 return S_OK;
3470}
3471
3472STDMETHODIMP HVMDKImage::COMSETTER(Description) (INPTR BSTR aDescription)
3473{
3474 AutoLock alock (this);
3475 CHECK_READY();
3476
3477 CHECK_BUSY_AND_READERS();
3478
3479 return E_NOTIMPL;
3480
3481/// @todo (r=dmik) implement
3482//
3483// if (mState >= Created)
3484// {
3485// int vrc = VDISetImageComment (Utf8Str (mFilePathFull), Utf8Str (aDescription));
3486// if (VBOX_FAILURE (vrc))
3487// return setError (E_FAIL,
3488// tr ("Could not change the description of the VDI hard disk '%ls' "
3489// "(%Vrc)"),
3490// toString().raw(), vrc);
3491// }
3492//
3493// mDescription = aDescription;
3494// return S_OK;
3495}
3496
3497STDMETHODIMP HVMDKImage::COMGETTER(Size) (ULONG64 *aSize)
3498{
3499 if (!aSize)
3500 return E_POINTER;
3501
3502 AutoLock alock (this);
3503 CHECK_READY();
3504
3505/// @todo (r=dmik) will need this if we add suppord for differencing VMDKs
3506//
3507// /* only a non-differencing image knows the logical size */
3508// if (isDifferencing())
3509// return root()->COMGETTER(Size) (aSize);
3510
3511 *aSize = mSize;
3512 return S_OK;
3513}
3514
3515STDMETHODIMP HVMDKImage::COMGETTER(ActualSize) (ULONG64 *aActualSize)
3516{
3517 if (!aActualSize)
3518 return E_POINTER;
3519
3520 AutoLock alock (this);
3521 CHECK_READY();
3522
3523 *aActualSize = mActualSize;
3524 return S_OK;
3525}
3526
3527// IVirtualDiskImage properties
3528////////////////////////////////////////////////////////////////////////////////
3529
3530STDMETHODIMP HVMDKImage::COMGETTER(FilePath) (BSTR *aFilePath)
3531{
3532 if (!aFilePath)
3533 return E_POINTER;
3534
3535 AutoLock alock (this);
3536 CHECK_READY();
3537
3538 mFilePathFull.cloneTo (aFilePath);
3539 return S_OK;
3540}
3541
3542STDMETHODIMP HVMDKImage::COMSETTER(FilePath) (INPTR BSTR aFilePath)
3543{
3544 AutoLock alock (this);
3545 CHECK_READY();
3546
3547 if (mState != NotCreated)
3548 return setError (E_ACCESSDENIED,
3549 tr ("Cannot change the file path of the existing hard disk '%ls'"),
3550 toString().raw());
3551
3552 CHECK_BUSY_AND_READERS();
3553
3554 /* append the default path if only a name is given */
3555 Bstr path = aFilePath;
3556 if (aFilePath && *aFilePath)
3557 {
3558 Utf8Str fp = aFilePath;
3559 if (!RTPathHavePath (fp))
3560 {
3561 AutoReaderLock propsLock (mVirtualBox->systemProperties());
3562 path = Utf8StrFmt ("%ls%c%s",
3563 mVirtualBox->systemProperties()->defaultVDIFolder().raw(),
3564 RTPATH_DELIMITER,
3565 fp.raw());
3566 }
3567 }
3568
3569 return setFilePath (path);
3570}
3571
3572STDMETHODIMP HVMDKImage::COMGETTER(Created) (BOOL *aCreated)
3573{
3574 if (!aCreated)
3575 return E_POINTER;
3576
3577 AutoLock alock (this);
3578 CHECK_READY();
3579
3580 *aCreated = mState >= Created;
3581 return S_OK;
3582}
3583
3584// IVMDKImage methods
3585/////////////////////////////////////////////////////////////////////////////
3586
3587STDMETHODIMP HVMDKImage::CreateDynamicImage (ULONG64 aSize, IProgress **aProgress)
3588{
3589 if (!aProgress)
3590 return E_POINTER;
3591
3592 AutoLock alock (this);
3593 CHECK_READY();
3594
3595 return createImage (aSize, TRUE /* aDynamic */, aProgress);
3596}
3597
3598STDMETHODIMP HVMDKImage::CreateFixedImage (ULONG64 aSize, IProgress **aProgress)
3599{
3600 if (!aProgress)
3601 return E_POINTER;
3602
3603 AutoLock alock (this);
3604 CHECK_READY();
3605
3606 return createImage (aSize, FALSE /* aDynamic */, aProgress);
3607}
3608
3609STDMETHODIMP HVMDKImage::DeleteImage()
3610{
3611 AutoLock alock (this);
3612 CHECK_READY();
3613 CHECK_BUSY_AND_READERS();
3614
3615 return E_NOTIMPL;
3616
3617/// @todo (r=dmik) later
3618// We will need to parse the file in order to delete all related delta and
3619// sparse images etc. We may also want to obey the .vmdk.lck file
3620// which is (as far as I understood) created when the VMware VM is
3621// running or saved etc.
3622//
3623// if (mRegistered)
3624// return setError (E_ACCESSDENIED,
3625// tr ("Cannot delete an image of the registered hard disk image '%ls"),
3626// mFilePathFull.raw());
3627// if (mState == NotCreated)
3628// return setError (E_FAIL,
3629// tr ("Hard disk image has been already deleted or never created"));
3630//
3631// int vrc = RTFileDelete (Utf8Str (mFilePathFull));
3632// if (VBOX_FAILURE (vrc))
3633// return setError (E_FAIL, tr ("Could not delete the image file '%ls' (%Vrc)"),
3634// mFilePathFull.raw(), vrc);
3635//
3636// mState = NotCreated;
3637//
3638// /* reset the fields */
3639// mSize = 0;
3640// mActualSize = 0;
3641//
3642// return S_OK;
3643}
3644
3645// public/protected methods for internal purposes only
3646/////////////////////////////////////////////////////////////////////////////
3647
3648/**
3649 * Attempts to mark the hard disk as registered.
3650 * Only VirtualBox can call this method.
3651 */
3652HRESULT HVMDKImage::trySetRegistered (BOOL aRegistered)
3653{
3654 AutoLock alock (this);
3655 CHECK_READY();
3656
3657 if (aRegistered)
3658 {
3659 if (mState == NotCreated)
3660 return setError (E_FAIL,
3661 tr ("Image file '%ls' is not yet created for this hard disk"),
3662 mFilePathFull.raw());
3663
3664/// @todo (r=dmik) will need this if we add suppord for differencing VMDKs
3665// if (isDifferencing())
3666// return setError (E_FAIL,
3667// tr ("Hard disk '%ls' is differencing and cannot be unregistered "
3668// "explicitly"),
3669// mFilePathFull.raw());
3670 }
3671 else
3672 {
3673 ComAssertRet (mState >= Created, E_FAIL);
3674 }
3675
3676 return HardDisk::trySetRegistered (aRegistered);
3677}
3678
3679/**
3680 * Checks accessibility of this hard disk image only (w/o parents).
3681 *
3682 * @param aAccessError on output, a null string indicates the hard disk is
3683 * accessible, otherwise contains a message describing
3684 * the reason of inaccessibility.
3685 */
3686HRESULT HVMDKImage::getAccessible (Bstr &aAccessError)
3687{
3688 AutoLock alock (this);
3689 CHECK_READY();
3690
3691 if (mStateCheckSem != NIL_RTSEMEVENTMULTI)
3692 {
3693 /* An accessibility check in progress on some other thread,
3694 * wait for it to finish. */
3695
3696 ComAssertRet (mStateCheckWaiters != (ULONG) ~0, E_FAIL);
3697 ++ mStateCheckWaiters;
3698 alock.leave();
3699
3700 int vrc = RTSemEventMultiWait (mStateCheckSem, RT_INDEFINITE_WAIT);
3701
3702 alock.enter();
3703 AssertReturn (mStateCheckWaiters != 0, E_FAIL);
3704 -- mStateCheckWaiters;
3705 if (mStateCheckWaiters == 0)
3706 {
3707 RTSemEventMultiDestroy (mStateCheckSem);
3708 mStateCheckSem = NIL_RTSEMEVENTMULTI;
3709 }
3710
3711 AssertRCReturn (vrc, E_FAIL);
3712
3713 /* don't touch aAccessError, it has been already set */
3714 return S_OK;
3715 }
3716
3717 /* check the basic accessibility */
3718 HRESULT rc = getBaseAccessible (aAccessError, true /* aCheckBusy */);
3719 if (FAILED (rc) || !aAccessError.isNull())
3720 return rc;
3721
3722 if (mState >= Created)
3723 {
3724 return queryInformation (&aAccessError);
3725 }
3726
3727 aAccessError = Utf8StrFmt ("Hard disk image '%ls' is not yet created",
3728 mFilePathFull.raw());
3729 return S_OK;
3730}
3731
3732/**
3733 * Saves hard disk settings to the specified storage node and saves
3734 * all children to the specified hard disk node
3735 *
3736 * @param aHDNode <HardDisk> or <DiffHardDisk> node
3737 * @param aStorageNode <VirtualDiskImage> node
3738 */
3739HRESULT HVMDKImage::saveSettings (CFGNODE aHDNode, CFGNODE aStorageNode)
3740{
3741 AssertReturn (aHDNode && aStorageNode, E_FAIL);
3742
3743 AutoLock alock (this);
3744 CHECK_READY();
3745
3746 /* filePath (required) */
3747 CFGLDRSetBSTR (aStorageNode, "filePath", mFilePath);
3748
3749 /* save basic settings and children */
3750 return HardDisk::saveSettings (aHDNode);
3751}
3752
3753/**
3754 * Checks if the given change of \a aOldPath to \a aNewPath affects the path
3755 * of this hard disk and updates it if necessary to reflect the new location.
3756 * Intended to be from HardDisk::updatePaths().
3757 *
3758 * @param aOldPath old path (full)
3759 * @param aNewPath new path (full)
3760 *
3761 * @note Locks this object for writing.
3762 */
3763void HVMDKImage::updatePath (const char *aOldPath, const char *aNewPath)
3764{
3765 AssertReturnVoid (aOldPath);
3766 AssertReturnVoid (aNewPath);
3767
3768 AutoLock alock (this);
3769 AssertReturnVoid (isReady());
3770
3771 size_t oldPathLen = strlen (aOldPath);
3772
3773 Utf8Str path = mFilePathFull;
3774 LogFlowThisFunc (("VMDK.fullPath={%s}\n", path.raw()));
3775
3776 if (RTPathStartsWith (path, aOldPath))
3777 {
3778 Utf8Str newPath = Utf8StrFmt ("%s%s", aNewPath,
3779 path.raw() + oldPathLen);
3780 path = newPath;
3781
3782 mVirtualBox->calculateRelativePath (path, path);
3783
3784 unconst (mFilePathFull) = newPath;
3785 unconst (mFilePath) = path;
3786
3787 LogFlowThisFunc (("-> updated: full={%s} short={%s}\n",
3788 newPath.raw(), path.raw()));
3789 }
3790}
3791
3792/**
3793 * Returns the string representation of this hard disk.
3794 * When \a aShort is false, returns the full image file path.
3795 * Otherwise, returns the image file name only.
3796 *
3797 * @param aShort if true, a short representation is returned
3798 */
3799Bstr HVMDKImage::toString (bool aShort /* = false */)
3800{
3801 AutoLock alock (this);
3802
3803 if (!aShort)
3804 return mFilePathFull;
3805 else
3806 {
3807 Utf8Str fname = mFilePathFull;
3808 return RTPathFilename (fname.mutableRaw());
3809 }
3810}
3811
3812/**
3813 * Creates a clone of this hard disk by storing hard disk data in the given
3814 * VDI file name.
3815 *
3816 * @param aId UUID to assign to the created image
3817 * @param aTargetPath VDI file where the cloned image is to be to stored
3818 * @param aProgress progress object to run during operation
3819 */
3820HRESULT
3821HVMDKImage::cloneToImage (const Guid &aId, const Utf8Str &aTargetPath,
3822 Progress *aProgress)
3823{
3824 ComAssertMsgFailed (("Not implemented"));
3825 return E_NOTIMPL;
3826
3827/// @todo (r=dmik) will need this if we add suppord for differencing VMDKs
3828// Use code from HVirtualDiskImage::cloneToImage as an example.
3829}
3830
3831/**
3832 * Creates a new differencing image for this hard disk with the given
3833 * VDI file name.
3834 *
3835 * @param aId UUID to assign to the created image
3836 * @param aTargetPath VDI file where to store the created differencing image
3837 * @param aProgress progress object to run during operation
3838 * (can be NULL)
3839 */
3840HRESULT
3841HVMDKImage::createDiffImage (const Guid &aId, const Utf8Str &aTargetPath,
3842 Progress *aProgress)
3843{
3844 ComAssertMsgFailed (("Not implemented"));
3845 return E_NOTIMPL;
3846
3847/// @todo (r=dmik) will need this if we add suppord for differencing VMDKs
3848// Use code from HVirtualDiskImage::createDiffImage as an example.
3849}
3850
3851// private methods
3852/////////////////////////////////////////////////////////////////////////////
3853
3854/**
3855 * Helper to set a new file path.
3856 * Resolves a path relatively to the Virtual Box home directory.
3857 *
3858 * @note
3859 * Must be called from under the object's lock!
3860 */
3861HRESULT HVMDKImage::setFilePath (const BSTR aFilePath)
3862{
3863 if (aFilePath && *aFilePath)
3864 {
3865 /* get the full file name */
3866 char filePathFull [RTPATH_MAX];
3867 int vrc = RTPathAbsEx (mVirtualBox->homeDir(), Utf8Str (aFilePath),
3868 filePathFull, sizeof (filePathFull));
3869 if (VBOX_FAILURE (vrc))
3870 return setError (E_FAIL,
3871 tr ("Invalid image file path '%ls' (%Vrc)"), aFilePath, vrc);
3872
3873 mFilePath = aFilePath;
3874 mFilePathFull = filePathFull;
3875 }
3876 else
3877 {
3878 mFilePath.setNull();
3879 mFilePathFull.setNull();
3880 }
3881
3882 return S_OK;
3883}
3884
3885/**
3886 * Helper to query information about the VDI hard disk.
3887 *
3888 * @param aAccessError not used when NULL, otherwise see #getAccessible()
3889 *
3890 * @note Must be called from under the object's lock, only after
3891 * CHECK_BUSY_AND_READERS() succeeds.
3892 */
3893HRESULT HVMDKImage::queryInformation (Bstr *aAccessError)
3894{
3895 AssertReturn (isLockedOnCurrentThread(), E_FAIL);
3896
3897 /* create a lock object to completely release it later */
3898 AutoLock alock (this);
3899
3900 AssertReturn (mStateCheckWaiters == 0, E_FAIL);
3901
3902 ComAssertRet (mState >= Created, E_FAIL);
3903
3904 HRESULT rc = S_OK;
3905 int vrc = VINF_SUCCESS;
3906
3907 /* lazily create a semaphore */
3908 vrc = RTSemEventMultiCreate (&mStateCheckSem);
3909 ComAssertRCRet (vrc, E_FAIL);
3910
3911 /* go to Busy state to prevent any concurrent modifications
3912 * after releasing the lock below (to unblock getters before
3913 * a lengthy operation) */
3914 setBusy();
3915
3916 alock.leave();
3917
3918 /* VBoxVHDD management interface needs to be optimized: we're opening a
3919 * file three times in a raw to get three bits of information. */
3920
3921 Utf8Str filePath = mFilePathFull;
3922 Bstr errMsg;
3923
3924 do
3925 {
3926 /// @todo remove when the code below is implemented
3927 if (mId.isEmpty())
3928 mId.create();
3929
3930/// @todo implement
3931//
3932// /* check the image file */
3933// Guid id, parentId;
3934// vrc = VDICheckImage (filePath, NULL, NULL, NULL,
3935// id.ptr(), parentId.ptr(), NULL, 0);
3936//
3937// if (VBOX_FAILURE (vrc))
3938// break;
3939//
3940// if (!mId.isEmpty())
3941// {
3942// /* check that the actual UUID of the image matches the stored UUID */
3943// if (mId != id)
3944// {
3945// errMsg = Utf8StrFmt (
3946// tr ("Actual UUID {%Vuuid} of the hard disk image '%s' doesn't "
3947// "match UUID {%Vuuid} stored in the registry"),
3948// id.ptr(), filePath.raw(), mId.ptr());
3949// break;
3950// }
3951// }
3952// else
3953// {
3954// /* assgn an UUID read from the image file */
3955// mId = id;
3956// }
3957//
3958// if (mParent)
3959// {
3960// /* check parent UUID */
3961// AutoLock parentLock (mParent);
3962// if (mParent->id() != parentId)
3963// {
3964// errMsg = Utf8StrFmt (
3965// tr ("UUID {%Vuuid} of the parent image '%ls' stored in "
3966// "the hard disk image file '%s' doesn't match "
3967// "UUID {%Vuuid} stored in the registry"),
3968// parentId.raw(), mParent->toString().raw(),
3969// filePath.raw(), mParent->id().raw());
3970// break;
3971// }
3972// }
3973// else if (!parentId.isEmpty())
3974// {
3975// errMsg = Utf8StrFmt (
3976// tr ("Hard disk image '%s' is a differencing image that is linked "
3977// "to a hard disk with UUID {%Vuuid} and cannot be used "
3978// "directly as a base hard disk"),
3979// filePath.raw(), parentId.raw());
3980// break;
3981// }
3982//
3983// {
3984// RTFILE file = NIL_RTFILE;
3985// vrc = RTFileOpen (&file, filePath, RTFILE_O_READ);
3986// if (VBOX_SUCCESS (vrc))
3987// {
3988// uint64_t size = 0;
3989// vrc = RTFileGetSize (file, &size);
3990// if (VBOX_SUCCESS (vrc))
3991// mActualSize = size;
3992// RTFileClose (file);
3993// }
3994// if (VBOX_FAILURE (vrc))
3995// break;
3996// }
3997//
3998// if (!mParent)
3999// {
4000// /* query logical size only for non-differencing images */
4001//
4002// PVDIDISK disk = VDIDiskCreate();
4003// vrc = VDIDiskOpenImage (disk, Utf8Str (mFilePathFull),
4004// VDI_OPEN_FLAGS_READONLY);
4005// if (VBOX_SUCCESS (vrc))
4006// {
4007// uint64_t size = VDIDiskGetSize (disk);
4008// /* convert to MBytes */
4009// mSize = size / 1024 / 1024;
4010// }
4011//
4012// VDIDiskDestroy (disk);
4013// if (VBOX_FAILURE (vrc))
4014// break;
4015// }
4016 }
4017 while (0);
4018
4019 /* enter the lock again */
4020 alock.enter();
4021
4022 /* remove the busy flag */
4023 clearBusy();
4024
4025 if (FAILED (rc) || VBOX_FAILURE (vrc) || !errMsg.isNull())
4026 {
4027 LogWarningFunc (("'%ls' is not accessible "
4028 "(rc=%08X, vrc=%Vrc, errMsg='%ls')\n",
4029 mFilePathFull.raw(), rc, vrc, errMsg.raw()));
4030
4031 if (aAccessError)
4032 {
4033 if (!errMsg.isNull())
4034 *aAccessError = errMsg;
4035 else if (VBOX_FAILURE (vrc))
4036 *aAccessError = Utf8StrFmt (
4037 tr ("Could not access hard disk image '%ls' (%Vrc)"),
4038 mFilePathFull.raw(), vrc);
4039 }
4040
4041 /* downgrade to not accessible */
4042 mState = Created;
4043 }
4044 else
4045 {
4046 if (aAccessError)
4047 aAccessError->setNull();
4048
4049 mState = Accessible;
4050 }
4051
4052 /* inform waiters if there are any */
4053 if (mStateCheckWaiters > 0)
4054 {
4055 RTSemEventMultiSignal (mStateCheckSem);
4056 }
4057 else
4058 {
4059 /* delete the semaphore ourselves */
4060 RTSemEventMultiDestroy (mStateCheckSem);
4061 mStateCheckSem = NIL_RTSEMEVENTMULTI;
4062 }
4063
4064 return rc;
4065}
4066
4067/**
4068 * Helper to create hard disk images.
4069 *
4070 * @param aSize size in MB
4071 * @param aDynamic dynamic or fixed image
4072 * @param aProgress address of IProgress pointer to return
4073 */
4074HRESULT HVMDKImage::createImage (ULONG64 aSize, BOOL aDynamic,
4075 IProgress **aProgress)
4076{
4077 ComAssertMsgFailed (("Not implemented"));
4078 return E_NOTIMPL;
4079
4080/// @todo (r=dmik) later
4081// Use code from HVirtualDiskImage::createImage as an example.
4082}
4083
4084/* static */
4085DECLCALLBACK(int) HVMDKImage::vdiTaskThread (RTTHREAD thread, void *pvUser)
4086{
4087 AssertMsgFailed (("Not implemented"));
4088 return VERR_GENERAL_FAILURE;
4089
4090/// @todo (r=dmik) later
4091// Use code from HVirtualDiskImage::vdiTaskThread as an example.
4092}
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