VirtualBox

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

Last change on this file since 22663 was 22624, checked in by vboxsync, 15 years ago

Main: cosmetics

  • Property svn:eol-style set to native
  • Property svn:keywords set to Date Revision Author Id
File size: 159.1 KB
Line 
1/* $Id: HardDiskImpl.cpp 22624 2009-08-31 17:32:17Z vboxsync $ */
2
3/** @file
4 *
5 * VirtualBox COM class implementation
6 */
7
8/*
9 * Copyright (C) 2008 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24#include <iprt/param.h>
25#include <iprt/path.h>
26#include <iprt/file.h>
27#include <iprt/tcp.h>
28
29#include <VBox/com/array.h>
30#include <VBox/com/SupportErrorInfo.h>
31
32#include <VBox/err.h>
33#include <VBox/settings.h>
34
35#include <list>
36#include <memory>
37
38#include "HardDiskImpl.h"
39#include "SystemPropertiesImpl.h"
40#include "ProgressImpl.h"
41
42#include "Logging.h"
43
44////////////////////////////////////////////////////////////////////////////////
45// Globals
46////////////////////////////////////////////////////////////////////////////////
47
48/**
49 * Asynchronous task thread parameter bucket.
50 *
51 * Note that instances of this class must be created using new() because the
52 * task thread function will delete them when the task is complete!
53 *
54 * @note The constructor of this class adds a caller on the managed HardDisk
55 * object which is automatically released upon destruction.
56 */
57struct HardDisk::Task : public com::SupportErrorInfoBase
58{
59 enum Operation { CreateBase, CreateDiff,
60 Merge, Clone, Delete, Reset, Compact };
61
62 HardDisk *that;
63 VirtualBoxBaseProto::AutoCaller autoCaller;
64
65 ComObjPtr<Progress> progress;
66 Operation operation;
67
68 /** Where to save the result when executed using #runNow(). */
69 HRESULT rc;
70
71 Task (HardDisk *aThat, Progress *aProgress, Operation aOperation)
72 : that (aThat), autoCaller (aThat)
73 , progress (aProgress)
74 , operation (aOperation)
75 , rc (S_OK) {}
76
77 ~Task();
78
79 void setData (HardDisk *aTarget)
80 {
81 d.target = aTarget;
82 HRESULT rc = d.target->addCaller();
83 AssertComRC (rc);
84 }
85
86 void setData (HardDisk *aTarget, HardDisk *aParent)
87 {
88 d.target = aTarget;
89 HRESULT rc = d.target->addCaller();
90 AssertComRC (rc);
91 d.parentDisk = aParent;
92 if (aParent)
93 {
94 rc = d.parentDisk->addCaller();
95 AssertComRC (rc);
96 }
97 }
98
99 void setData (MergeChain *aChain)
100 {
101 AssertReturnVoid (aChain != NULL);
102 d.chain.reset (aChain);
103 }
104
105 void setData (ImageChain *aSrcChain, ImageChain *aParentChain)
106 {
107 AssertReturnVoid (aSrcChain != NULL);
108 AssertReturnVoid (aParentChain != NULL);
109 d.source.reset (aSrcChain);
110 d.parent.reset (aParentChain);
111 }
112
113 void setData (ImageChain *aImgChain)
114 {
115 AssertReturnVoid (aImgChain != NULL);
116 d.images.reset (aImgChain);
117 }
118
119 HRESULT startThread();
120 HRESULT runNow();
121
122 struct Data
123 {
124 Data() : size (0) {}
125
126 /* CreateBase */
127
128 uint64_t size;
129
130 /* CreateBase, CreateDiff, Clone */
131
132 HardDiskVariant_T variant;
133
134 /* CreateDiff, Clone */
135
136 ComObjPtr<HardDisk> target;
137
138 /* Clone */
139
140 /** Hard disks to open, in {parent,child} order */
141 std::auto_ptr <ImageChain> source;
142 /** Hard disks which are parent of target, in {parent,child} order */
143 std::auto_ptr <ImageChain> parent;
144 /** The to-be parent hard disk object */
145 ComObjPtr<HardDisk> parentDisk;
146
147 /* Merge */
148
149 /** Hard disks to merge, in {parent,child} order */
150 std::auto_ptr <MergeChain> chain;
151
152 /* Compact */
153
154 /** Hard disks to open, in {parent,child} order */
155 std::auto_ptr <ImageChain> images;
156 }
157 d;
158
159protected:
160
161 // SupportErrorInfoBase interface
162 const GUID &mainInterfaceID() const { return COM_IIDOF (IHardDisk); }
163 const char *componentName() const { return HardDisk::ComponentName(); }
164};
165
166HardDisk::Task::~Task()
167{
168 /* remove callers added by setData() */
169 if (!d.target.isNull())
170 d.target->releaseCaller();
171}
172
173/**
174 * Starts a new thread driven by the HardDisk::taskThread() function and passes
175 * this Task instance as an argument.
176 *
177 * Note that if this method returns success, this Task object becomes an ownee
178 * of the started thread and will be automatically deleted when the thread
179 * terminates.
180 *
181 * @note When the task is executed by this method, IProgress::notifyComplete()
182 * is automatically called for the progress object associated with this
183 * task when the task is finished to signal the operation completion for
184 * other threads asynchronously waiting for it.
185 */
186HRESULT HardDisk::Task::startThread()
187{
188 int vrc = RTThreadCreate (NULL, HardDisk::taskThread, this,
189 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
190 "HardDisk::Task");
191 ComAssertMsgRCRet (vrc,
192 ("Could not create HardDisk::Task thread (%Rrc)\n", vrc), E_FAIL);
193
194 return S_OK;
195}
196
197/**
198 * Runs HardDisk::taskThread() by passing it this Task instance as an argument
199 * on the current thread instead of creating a new one.
200 *
201 * This call implies that it is made on another temporary thread created for
202 * some asynchronous task. Avoid calling it from a normal thread since the task
203 * operatinos are potentially lengthy and will block the calling thread in this
204 * case.
205 *
206 * Note that this Task object will be deleted by taskThread() when this method
207 * returns!
208 *
209 * @note When the task is executed by this method, IProgress::notifyComplete()
210 * is not called for the progress object associated with this task when
211 * the task is finished. Instead, the result of the operation is returned
212 * by this method directly and it's the caller's responsibility to
213 * complete the progress object in this case.
214 */
215HRESULT HardDisk::Task::runNow()
216{
217 HardDisk::taskThread (NIL_RTTHREAD, this);
218
219 return rc;
220}
221
222////////////////////////////////////////////////////////////////////////////////
223
224/**
225 * Helper class for merge operations.
226 *
227 * @note It is assumed that when modifying methods of this class are called,
228 * HardDisk::treeLock() is held in read mode.
229 */
230class HardDisk::MergeChain : public HardDisk::List,
231 public com::SupportErrorInfoBase
232{
233public:
234
235 MergeChain (bool aForward, bool aIgnoreAttachments)
236 : mForward (aForward)
237 , mIgnoreAttachments (aIgnoreAttachments) {}
238
239 ~MergeChain()
240 {
241 for (iterator it = mChildren.begin(); it != mChildren.end(); ++ it)
242 {
243 HRESULT rc = (*it)->UnlockWrite (NULL);
244 AssertComRC (rc);
245
246 (*it)->releaseCaller();
247 }
248
249 for (iterator it = begin(); it != end(); ++ it)
250 {
251 AutoWriteLock alock(*it);
252 Assert ((*it)->m.state == MediaState_LockedWrite ||
253 (*it)->m.state == MediaState_Deleting);
254 if ((*it)->m.state == MediaState_LockedWrite)
255 (*it)->UnlockWrite (NULL);
256 else
257 (*it)->m.state = MediaState_Created;
258
259 (*it)->releaseCaller();
260 }
261
262 if (!mParent.isNull())
263 mParent->releaseCaller();
264 }
265
266 HRESULT addSource (HardDisk *aHardDisk)
267 {
268 HRESULT rc = aHardDisk->addCaller();
269 CheckComRCReturnRC(rc);
270
271 AutoWriteLock alock(aHardDisk);
272
273 if (mForward)
274 {
275 rc = checkChildrenAndAttachmentsAndImmutable (aHardDisk);
276 if (FAILED (rc))
277 {
278 aHardDisk->releaseCaller();
279 return rc;
280 }
281 }
282
283 /* We have to fetch the state with the COM method, cause it's possible
284 that the hard disk isn't fully initialized yet. See HRESULT
285 ImageMediumBase::protectedInit (VirtualBox *aVirtualBox, const
286 settings::Key &aImageNode) for an explanation why. */
287 MediaState_T m;
288 rc = aHardDisk->COMGETTER(State)(&m);
289 CheckComRCReturnRC(rc);
290 /* go to Deleting */
291 switch (m)
292 {
293 case MediaState_Created:
294 aHardDisk->m.state = MediaState_Deleting;
295 break;
296 default:
297 aHardDisk->releaseCaller();
298 return aHardDisk->setStateError();
299 }
300
301 push_front (aHardDisk);
302
303 if (mForward)
304 {
305 /* we will need parent to reparent target */
306 if (!aHardDisk->mParent.isNull())
307 {
308 rc = aHardDisk->mParent->addCaller();
309 CheckComRCReturnRC(rc);
310
311 mParent = aHardDisk->mParent;
312 }
313 }
314 else
315 {
316 /* we will need to reparent children */
317 for (List::const_iterator it = aHardDisk->children().begin();
318 it != aHardDisk->children().end(); ++ it)
319 {
320 rc = (*it)->addCaller();
321 CheckComRCReturnRC(rc);
322
323 rc = (*it)->LockWrite (NULL);
324 if (FAILED (rc))
325 {
326 (*it)->releaseCaller();
327 return rc;
328 }
329
330 mChildren.push_back (*it);
331 }
332 }
333
334 return S_OK;
335 }
336
337 HRESULT addTarget (HardDisk *aHardDisk)
338 {
339 HRESULT rc = aHardDisk->addCaller();
340 CheckComRCReturnRC(rc);
341
342 AutoWriteLock alock(aHardDisk);
343
344 if (!mForward)
345 {
346 rc = checkChildrenAndImmutable (aHardDisk);
347 if (FAILED (rc))
348 {
349 aHardDisk->releaseCaller();
350 return rc;
351 }
352 }
353
354 /* go to LockedWrite */
355 rc = aHardDisk->LockWrite (NULL);
356 if (FAILED (rc))
357 {
358 aHardDisk->releaseCaller();
359 return rc;
360 }
361
362 push_front (aHardDisk);
363
364 return S_OK;
365 }
366
367 HRESULT addIntermediate (HardDisk *aHardDisk)
368 {
369 HRESULT rc = aHardDisk->addCaller();
370 CheckComRCReturnRC(rc);
371
372 AutoWriteLock alock(aHardDisk);
373
374 rc = checkChildrenAndAttachments (aHardDisk);
375 if (FAILED (rc))
376 {
377 aHardDisk->releaseCaller();
378 return rc;
379 }
380
381 /* go to Deleting */
382 switch (aHardDisk->m.state)
383 {
384 case MediaState_Created:
385 aHardDisk->m.state = MediaState_Deleting;
386 break;
387 default:
388 aHardDisk->releaseCaller();
389 return aHardDisk->setStateError();
390 }
391
392 push_front (aHardDisk);
393
394 return S_OK;
395 }
396
397 bool isForward() const { return mForward; }
398 HardDisk *parent() const { return mParent; }
399 const List &children() const { return mChildren; }
400
401 HardDisk *source() const
402 { AssertReturn(size() > 0, NULL); return mForward ? front() : back(); }
403
404 HardDisk *target() const
405 { AssertReturn(size() > 0, NULL); return mForward ? back() : front(); }
406
407protected:
408
409 // SupportErrorInfoBase interface
410 const GUID &mainInterfaceID() const { return COM_IIDOF (IHardDisk); }
411 const char *componentName() const { return HardDisk::ComponentName(); }
412
413private:
414
415 HRESULT check (HardDisk *aHardDisk, bool aChildren, bool aAttachments,
416 bool aImmutable)
417 {
418 if (aChildren)
419 {
420 /* not going to multi-merge as it's too expensive */
421 if (aHardDisk->children().size() > 1)
422 {
423 return setError (E_FAIL,
424 tr ("Hard disk '%ls' involved in the merge operation has more than one child hard disk (%d)"),
425 aHardDisk->m.locationFull.raw(),
426 aHardDisk->children().size());
427 }
428 }
429
430 if (aAttachments && !mIgnoreAttachments)
431 {
432 if (aHardDisk->m.backRefs.size() != 0)
433 return setError (E_FAIL,
434 tr ("Hard disk '%ls' is attached to %d virtual machines"),
435 aHardDisk->m.locationFull.raw(),
436 aHardDisk->m.backRefs.size());
437 }
438
439 if (aImmutable)
440 {
441 if (aHardDisk->mm.type == HardDiskType_Immutable)
442 return setError (E_FAIL,
443 tr ("Hard disk '%ls' is immutable"),
444 aHardDisk->m.locationFull.raw());
445 }
446
447 return S_OK;
448 }
449
450 HRESULT checkChildren (HardDisk *aHardDisk)
451 { return check (aHardDisk, true, false, false); }
452
453 HRESULT checkChildrenAndImmutable (HardDisk *aHardDisk)
454 { return check (aHardDisk, true, false, true); }
455
456 HRESULT checkChildrenAndAttachments (HardDisk *aHardDisk)
457 { return check (aHardDisk, true, true, false); }
458
459 HRESULT checkChildrenAndAttachmentsAndImmutable (HardDisk *aHardDisk)
460 { return check (aHardDisk, true, true, true); }
461
462 /** true if forward merge, false if backward */
463 bool mForward : 1;
464 /** true to not perform attachment checks */
465 bool mIgnoreAttachments : 1;
466
467 /** Parent of the source when forward merge (if any) */
468 ComObjPtr<HardDisk> mParent;
469 /** Children of the source when backward merge (if any) */
470 List mChildren;
471};
472
473////////////////////////////////////////////////////////////////////////////////
474
475/**
476 * Helper class for image operations involving the entire parent chain.
477 *
478 * @note It is assumed that when modifying methods of this class are called,
479 * HardDisk::treeLock() is held in read mode.
480 */
481class HardDisk::ImageChain : public HardDisk::List,
482 public com::SupportErrorInfoBase
483{
484public:
485
486 ImageChain () {}
487
488 ~ImageChain()
489 {
490 /* empty? */
491 if (begin() != end())
492 {
493 List::const_iterator last = end();
494 last--;
495 for (List::const_iterator it = begin(); it != end(); ++ it)
496 {
497 AutoWriteLock alock(*it);
498 if (it == last)
499 {
500 Assert ( (*it)->m.state == MediaState_LockedRead
501 || (*it)->m.state == MediaState_LockedWrite);
502 if ((*it)->m.state == MediaState_LockedRead)
503 (*it)->UnlockRead (NULL);
504 else if ((*it)->m.state == MediaState_LockedWrite)
505 (*it)->UnlockWrite (NULL);
506 }
507 else
508 {
509 Assert ((*it)->m.state == MediaState_LockedRead);
510 if ((*it)->m.state == MediaState_LockedRead)
511 (*it)->UnlockRead (NULL);
512 }
513
514 (*it)->releaseCaller();
515 }
516 }
517 }
518
519 HRESULT addImage (HardDisk *aHardDisk)
520 {
521 HRESULT rc = aHardDisk->addCaller();
522 CheckComRCReturnRC(rc);
523
524 push_front (aHardDisk);
525
526 return S_OK;
527 }
528
529 HRESULT lockImagesRead ()
530 {
531 /* Lock all disks in the chain in {parent, child} order,
532 * and make sure they are accessible. */
533 /// @todo code duplication with SessionMachine::lockMedia, see below
534 ErrorInfoKeeper eik (true /* aIsNull */);
535 MultiResult mrc (S_OK);
536 for (List::const_iterator it = begin(); it != end(); ++ it)
537 {
538 HRESULT rc = S_OK;
539 MediaState_T mediaState;
540 rc = (*it)->LockRead(&mediaState);
541 CheckComRCReturnRC(rc);
542
543 if (mediaState == MediaState_Inaccessible)
544 {
545 rc = (*it)->COMGETTER(State) (&mediaState);
546 CheckComRCReturnRC(rc);
547 Assert (mediaState == MediaState_LockedRead);
548
549 /* Note that we locked the medium already, so use the error
550 * value to see if there was an accessibility failure */
551 Bstr error;
552 rc = (*it)->COMGETTER(LastAccessError) (error.asOutParam());
553 CheckComRCReturnRC(rc);
554
555 if (!error.isEmpty())
556 {
557 Bstr loc;
558 rc = (*it)->COMGETTER(Location) (loc.asOutParam());
559 CheckComRCThrowRC (rc);
560
561 /* collect multiple errors */
562 eik.restore();
563
564 /* be in sync with MediumBase::setStateError() */
565 Assert (!error.isEmpty());
566 mrc = setError (E_FAIL,
567 tr ("Medium '%ls' is not accessible. %ls"),
568 loc.raw(), error.raw());
569
570 eik.fetch();
571 }
572 }
573 }
574
575 eik.restore();
576 CheckComRCReturnRC((HRESULT) mrc);
577
578 return S_OK;
579 }
580
581 HRESULT lockImagesReadAndLastWrite ()
582 {
583 /* Lock all disks in the chain in {parent, child} order,
584 * and make sure they are accessible. */
585 /// @todo code duplication with SessionMachine::lockMedia, see below
586 ErrorInfoKeeper eik (true /* aIsNull */);
587 MultiResult mrc (S_OK);
588 List::const_iterator last = end();
589 last--;
590 for (List::const_iterator it = begin(); it != end(); ++ it)
591 {
592 HRESULT rc = S_OK;
593 MediaState_T mediaState;
594 if (it == last)
595 rc = (*it)->LockWrite(&mediaState);
596 else
597 rc = (*it)->LockRead(&mediaState);
598 CheckComRCReturnRC(rc);
599
600 if (mediaState == MediaState_Inaccessible)
601 {
602 rc = (*it)->COMGETTER(State) (&mediaState);
603 CheckComRCReturnRC(rc);
604 if (it == last)
605 Assert (mediaState == MediaState_LockedWrite);
606 else
607 Assert (mediaState == MediaState_LockedRead);
608
609 /* Note that we locked the medium already, so use the error
610 * value to see if there was an accessibility failure */
611 Bstr error;
612 rc = (*it)->COMGETTER(LastAccessError) (error.asOutParam());
613 CheckComRCReturnRC(rc);
614
615 if (!error.isEmpty())
616 {
617 Bstr loc;
618 rc = (*it)->COMGETTER(Location) (loc.asOutParam());
619 CheckComRCThrowRC (rc);
620
621 /* collect multiple errors */
622 eik.restore();
623
624 /* be in sync with MediumBase::setStateError() */
625 Assert (!error.isEmpty());
626 mrc = setError (E_FAIL,
627 tr ("Medium '%ls' is not accessible. %ls"),
628 loc.raw(), error.raw());
629
630 eik.fetch();
631 }
632 }
633 }
634
635 eik.restore();
636 CheckComRCReturnRC((HRESULT) mrc);
637
638 return S_OK;
639 }
640
641protected:
642
643 // SupportErrorInfoBase interface
644 const GUID &mainInterfaceID() const { return COM_IIDOF (IHardDisk); }
645 const char *componentName() const { return HardDisk::ComponentName(); }
646
647private:
648
649};
650
651////////////////////////////////////////////////////////////////////////////////
652// HardDisk class
653////////////////////////////////////////////////////////////////////////////////
654
655// constructor / destructor
656////////////////////////////////////////////////////////////////////////////////
657
658DEFINE_EMPTY_CTOR_DTOR (HardDisk)
659
660HRESULT HardDisk::FinalConstruct()
661{
662 /* Initialize the callbacks of the VD error interface */
663 mm.vdIfCallsError.cbSize = sizeof (VDINTERFACEERROR);
664 mm.vdIfCallsError.enmInterface = VDINTERFACETYPE_ERROR;
665 mm.vdIfCallsError.pfnError = vdErrorCall;
666 mm.vdIfCallsError.pfnMessage = NULL;
667
668 /* Initialize the callbacks of the VD progress interface */
669 mm.vdIfCallsProgress.cbSize = sizeof (VDINTERFACEPROGRESS);
670 mm.vdIfCallsProgress.enmInterface = VDINTERFACETYPE_PROGRESS;
671 mm.vdIfCallsProgress.pfnProgress = vdProgressCall;
672
673 /* Initialize the callbacks of the VD config interface */
674 mm.vdIfCallsConfig.cbSize = sizeof (VDINTERFACECONFIG);
675 mm.vdIfCallsConfig.enmInterface = VDINTERFACETYPE_CONFIG;
676 mm.vdIfCallsConfig.pfnAreKeysValid = vdConfigAreKeysValid;
677 mm.vdIfCallsConfig.pfnQuerySize = vdConfigQuerySize;
678 mm.vdIfCallsConfig.pfnQuery = vdConfigQuery;
679
680 /* Initialize the callbacks of the VD TCP interface (we always use the host
681 * IP stack for now) */
682 mm.vdIfCallsTcpNet.cbSize = sizeof (VDINTERFACETCPNET);
683 mm.vdIfCallsTcpNet.enmInterface = VDINTERFACETYPE_TCPNET;
684 mm.vdIfCallsTcpNet.pfnClientConnect = RTTcpClientConnect;
685 mm.vdIfCallsTcpNet.pfnClientClose = RTTcpClientClose;
686 mm.vdIfCallsTcpNet.pfnSelectOne = RTTcpSelectOne;
687 mm.vdIfCallsTcpNet.pfnRead = RTTcpRead;
688 mm.vdIfCallsTcpNet.pfnWrite = RTTcpWrite;
689 mm.vdIfCallsTcpNet.pfnFlush = RTTcpFlush;
690
691 /* Initialize the per-disk interface chain */
692 int vrc;
693 vrc = VDInterfaceAdd (&mm.vdIfError,
694 "HardDisk::vdInterfaceError",
695 VDINTERFACETYPE_ERROR,
696 &mm.vdIfCallsError, this, &mm.vdDiskIfaces);
697 AssertRCReturn (vrc, E_FAIL);
698
699 vrc = VDInterfaceAdd (&mm.vdIfProgress,
700 "HardDisk::vdInterfaceProgress",
701 VDINTERFACETYPE_PROGRESS,
702 &mm.vdIfCallsProgress, this, &mm.vdDiskIfaces);
703 AssertRCReturn (vrc, E_FAIL);
704
705 vrc = VDInterfaceAdd (&mm.vdIfConfig,
706 "HardDisk::vdInterfaceConfig",
707 VDINTERFACETYPE_CONFIG,
708 &mm.vdIfCallsConfig, this, &mm.vdDiskIfaces);
709 AssertRCReturn (vrc, E_FAIL);
710
711 vrc = VDInterfaceAdd (&mm.vdIfTcpNet,
712 "HardDisk::vdInterfaceTcpNet",
713 VDINTERFACETYPE_TCPNET,
714 &mm.vdIfCallsTcpNet, this, &mm.vdDiskIfaces);
715 AssertRCReturn (vrc, E_FAIL);
716
717 return S_OK;
718}
719
720void HardDisk::FinalRelease()
721{
722 uninit();
723}
724
725// public initializer/uninitializer for internal purposes only
726////////////////////////////////////////////////////////////////////////////////
727
728/**
729 * Initializes the hard disk object without creating or opening an associated
730 * storage unit.
731 *
732 * For hard disks that don't have the VD_CAP_CREATE_FIXED or
733 * VD_CAP_CREATE_DYNAMIC capability (and therefore cannot be created or deleted
734 * with the means of VirtualBox) the associated storage unit is assumed to be
735 * ready for use so the state of the hard disk object will be set to Created.
736 *
737 * @param aVirtualBox VirtualBox object.
738 * @param aLocation Storage unit location.
739 */
740HRESULT HardDisk::init (VirtualBox *aVirtualBox,
741 CBSTR aFormat,
742 CBSTR aLocation)
743{
744 AssertReturn(aVirtualBox != NULL, E_FAIL);
745 AssertReturn(aFormat != NULL && *aFormat != '\0', E_FAIL);
746
747 /* Enclose the state transition NotReady->InInit->Ready */
748 AutoInitSpan autoInitSpan(this);
749 AssertReturn(autoInitSpan.isOk(), E_FAIL);
750
751 HRESULT rc = S_OK;
752
753 /* share VirtualBox weakly (parent remains NULL so far) */
754 unconst(mVirtualBox) = aVirtualBox;
755
756 /* register with VirtualBox early, since uninit() will
757 * unconditionally unregister on failure */
758 aVirtualBox->addDependentChild (this);
759
760 /* no storage yet */
761 m.state = MediaState_NotCreated;
762
763 /* No storage unit is created yet, no need to queryInfo() */
764
765 rc = setFormat (aFormat);
766 CheckComRCReturnRC(rc);
767
768 if (mm.formatObj->capabilities() & HardDiskFormatCapabilities_File)
769 {
770 rc = setLocation (aLocation);
771 CheckComRCReturnRC(rc);
772 }
773 else
774 {
775 rc = setLocation (aLocation);
776 CheckComRCReturnRC(rc);
777
778 /// @todo later we may want to use a pfnComposeLocation backend info
779 /// callback to generate a well-formed location value (based on the hard
780 /// disk properties we have) rather than allowing each caller to invent
781 /// its own (pseudo-)location.
782 }
783
784 if (!(mm.formatObj->capabilities() &
785 (HardDiskFormatCapabilities_CreateFixed |
786 HardDiskFormatCapabilities_CreateDynamic)))
787 {
788 /* storage for hard disks of this format can neither be explicitly
789 * created by VirtualBox nor deleted, so we place the hard disk to
790 * Created state here and also add it to the registry */
791 m.state = MediaState_Created;
792 unconst(m.id).create();
793 rc = mVirtualBox->registerHardDisk (this);
794
795 /// @todo later we may want to use a pfnIsConfigSufficient backend info
796 /// callback that would tell us when we have enough properties to work
797 /// with the hard disk and this information could be used to actually
798 /// move such hard disks from NotCreated to Created state. Instead of
799 /// pfnIsConfigSufficient we can use HardDiskFormat property
800 /// descriptions to see which properties are mandatory
801 }
802
803 /* Confirm a successful initialization when it's the case */
804 if (SUCCEEDED(rc))
805 autoInitSpan.setSucceeded();
806
807 return rc;
808}
809
810/**
811 * Initializes the hard disk object by opening the storage unit at the specified
812 * location. The enOpenMode parameter defines whether the image will be opened
813 * read/write or read-only.
814 *
815 * Note that the UUID, format and the parent of this hard disk will be
816 * determined when reading the hard disk storage unit, unless new values are
817 * specified by the parameters. If the detected or set parent is
818 * not known to VirtualBox, then this method will fail.
819 *
820 * @param aVirtualBox VirtualBox object.
821 * @param aLocation Storage unit location.
822 * @param enOpenMode Whether to open the image read/write or read-only.
823 * @param aSetImageId Whether to set the image UUID or not.
824 * @param aImageId New image UUID if @aSetId is true. Empty string means
825 * create a new UUID, and a zero UUID is invalid.
826 * @param aSetParentId Whether to set the parent UUID or not.
827 * @param aParentId New parent UUID if @aSetParentId is true. Empty string
828 * means create a new UUID, and a zero UUID is valid.
829 */
830HRESULT HardDisk::init(VirtualBox *aVirtualBox,
831 CBSTR aLocation,
832 HDDOpenMode enOpenMode,
833 BOOL aSetImageId,
834 const Guid &aImageId,
835 BOOL aSetParentId,
836 const Guid &aParentId)
837{
838 AssertReturn(aVirtualBox, E_INVALIDARG);
839 AssertReturn(aLocation, E_INVALIDARG);
840
841 /* Enclose the state transition NotReady->InInit->Ready */
842 AutoInitSpan autoInitSpan(this);
843 AssertReturn(autoInitSpan.isOk(), E_FAIL);
844
845 HRESULT rc = S_OK;
846
847 /* share VirtualBox weakly (parent remains NULL so far) */
848 unconst(mVirtualBox) = aVirtualBox;
849
850 /* register with VirtualBox early, since uninit() will
851 * unconditionally unregister on failure */
852 aVirtualBox->addDependentChild (this);
853
854 /* there must be a storage unit */
855 m.state = MediaState_Created;
856
857 /* remember the open mode (defaults to ReadWrite) */
858 mm.hddOpenMode = enOpenMode;
859
860 rc = setLocation (aLocation);
861 CheckComRCReturnRC(rc);
862
863 /* save the new uuid values, will be used by queryInfo() */
864 mm.setImageId = aSetImageId;
865 unconst(mm.imageId) = aImageId;
866 mm.setParentId = aSetParentId;
867 unconst(mm.parentId) = aParentId;
868
869 /* get all the information about the medium from the storage unit */
870 rc = queryInfo();
871
872 if (SUCCEEDED(rc))
873 {
874 /* if the storage unit is not accessible, it's not acceptable for the
875 * newly opened media so convert this into an error */
876 if (m.state == MediaState_Inaccessible)
877 {
878 Assert (!m.lastAccessError.isEmpty());
879 rc = setError(E_FAIL, Utf8Str(m.lastAccessError).c_str());
880 }
881 else
882 {
883 AssertReturn(!m.id.isEmpty(), E_FAIL);
884
885 /* storage format must be detected by queryInfo() if the medium is accessible */
886 AssertReturn(!mm.format.isNull(), E_FAIL);
887 }
888 }
889
890 /* Confirm a successful initialization when it's the case */
891 if (SUCCEEDED(rc))
892 autoInitSpan.setSucceeded();
893
894 return rc;
895}
896
897/**
898 * Initializes the hard disk object by loading its data from the given settings
899 * node. In this mode, the image will always be opened read/write.
900 *
901 * @param aVirtualBox VirtualBox object.
902 * @param aParent Parent hard disk or NULL for a root (base) hard disk.
903 * @param aNode <HardDisk> settings node.
904 *
905 * @note Locks VirtualBox lock for writing, treeLock() for writing.
906 */
907HRESULT HardDisk::init(VirtualBox *aVirtualBox,
908 HardDisk *aParent,
909 const settings::Medium &data)
910{
911 AssertReturn(aVirtualBox, E_INVALIDARG);
912
913 /* Enclose the state transition NotReady->InInit->Ready */
914 AutoInitSpan autoInitSpan(this);
915 AssertReturn(autoInitSpan.isOk(), E_FAIL);
916
917 HRESULT rc = S_OK;
918
919 /* share VirtualBox and parent weakly */
920 unconst(mVirtualBox) = aVirtualBox;
921
922 /* register with VirtualBox/parent early, since uninit() will
923 * unconditionally unregister on failure */
924 if (aParent == NULL)
925 aVirtualBox->addDependentChild (this);
926 else
927 {
928 /* we set mParent */
929 AutoWriteLock treeLock(this->treeLock());
930
931 mParent = aParent;
932 aParent->addDependentChild (this);
933 }
934
935 /* see below why we don't call queryInfo() (and therefore treat the medium
936 * as inaccessible for now */
937 m.state = MediaState_Inaccessible;
938 m.lastAccessError = tr("Accessibility check was not yet performed");
939
940 /* required */
941 unconst(m.id) = data.uuid;
942
943 /* optional */
944 m.description = data.strDescription;
945
946 /* required */
947 AssertReturn(!data.strFormat.isEmpty(), E_FAIL);
948 rc = setFormat(Bstr(data.strFormat));
949 CheckComRCReturnRC(rc);
950
951 /* optional, only for diffs, default is false */
952 if (aParent != NULL)
953 mm.autoReset = data.fAutoReset;
954 else
955 mm.autoReset = false;
956
957 /* properties (after setting the format as it populates the map). Note that
958 * if some properties are not supported but preseint in the settings file,
959 * they will still be read and accessible (for possible backward
960 * compatibility; we can also clean them up from the XML upon next
961 * XML format version change if we wish) */
962 for (settings::PropertiesMap::const_iterator it = data.properties.begin();
963 it != data.properties.end();
964 ++it)
965 {
966 const Utf8Str &name = it->first;
967 const Utf8Str &value = it->second;
968 mm.properties[Bstr(name)] = Bstr(value);
969 }
970
971 /* required */
972 rc = setLocation(data.strLocation);
973 CheckComRCReturnRC(rc);
974
975 /* type is only for base hard disks */
976 if (mParent.isNull())
977 mm.type = data.hdType;
978
979 LogFlowThisFunc(("m.locationFull='%ls', mm.format=%ls, m.id={%RTuuid}\n",
980 m.locationFull.raw(), mm.format.raw(), m.id.raw()));
981
982 /* Don't call queryInfo() for registered media to prevent the calling
983 * thread (i.e. the VirtualBox server startup thread) from an unexpected
984 * freeze but mark it as initially inaccessible instead. The vital UUID,
985 * location and format properties are read from the registry file above; to
986 * get the actual state and the rest of the data, the user will have to call
987 * COMGETTER(State). */
988
989 /* load all children */
990 for (settings::MediaList::const_iterator it = data.llChildren.begin();
991 it != data.llChildren.end();
992 ++it)
993 {
994 const settings::Medium &m = *it;
995
996 ComObjPtr<HardDisk> pHD;
997 pHD.createObject();
998 rc = pHD->init(aVirtualBox,
999 this, // parent
1000 m); // child data
1001 CheckComRCBreakRC (rc);
1002
1003 rc = mVirtualBox->registerHardDisk(pHD, false /* aSaveRegistry */);
1004 CheckComRCBreakRC (rc);
1005 }
1006
1007 /* Confirm a successful initialization when it's the case */
1008 if (SUCCEEDED(rc))
1009 autoInitSpan.setSucceeded();
1010
1011 return rc;
1012}
1013
1014/**
1015 * Uninitializes the instance.
1016 *
1017 * Called either from FinalRelease() or by the parent when it gets destroyed.
1018 *
1019 * @note All children of this hard disk get uninitialized by calling their
1020 * uninit() methods.
1021 *
1022 * @note Locks treeLock() for writing, VirtualBox for writing.
1023 */
1024void HardDisk::uninit()
1025{
1026 /* Enclose the state transition Ready->InUninit->NotReady */
1027 AutoUninitSpan autoUninitSpan(this);
1028 if (autoUninitSpan.uninitDone())
1029 return;
1030
1031 if (!mm.formatObj.isNull())
1032 {
1033 /* remove the caller reference we added in setFormat() */
1034 mm.formatObj->releaseCaller();
1035 mm.formatObj.setNull();
1036 }
1037
1038 if (m.state == MediaState_Deleting)
1039 {
1040 /* we are being uninitialized after've been deleted by merge.
1041 * Reparenting has already been done so don't touch it here (we are
1042 * now orphans and remoeDependentChild() will assert) */
1043
1044 Assert (mParent.isNull());
1045 }
1046 else
1047 {
1048 /* we uninit children and reset mParent
1049 * and VirtualBox::removeDependentChild() needs a write lock */
1050 AutoMultiWriteLock2 alock (mVirtualBox->lockHandle(), this->treeLock());
1051
1052 uninitDependentChildren();
1053
1054 if (!mParent.isNull())
1055 {
1056 mParent->removeDependentChild (this);
1057 mParent.setNull();
1058 }
1059 else
1060 mVirtualBox->removeDependentChild (this);
1061 }
1062
1063 unconst(mVirtualBox).setNull();
1064}
1065
1066// IHardDisk properties
1067////////////////////////////////////////////////////////////////////////////////
1068
1069STDMETHODIMP HardDisk::COMGETTER(Format) (BSTR *aFormat)
1070{
1071 if (aFormat == NULL)
1072 return E_POINTER;
1073
1074 AutoCaller autoCaller(this);
1075 CheckComRCReturnRC(autoCaller.rc());
1076
1077 /* no need to lock, mm.format is const */
1078 mm.format.cloneTo(aFormat);
1079
1080 return S_OK;
1081}
1082
1083STDMETHODIMP HardDisk::COMGETTER(Type) (HardDiskType_T *aType)
1084{
1085 if (aType == NULL)
1086 return E_POINTER;
1087
1088 AutoCaller autoCaller(this);
1089 CheckComRCReturnRC(autoCaller.rc());
1090
1091 AutoReadLock alock(this);
1092
1093 *aType = mm.type;
1094
1095 return S_OK;
1096}
1097
1098STDMETHODIMP HardDisk::COMSETTER(Type) (HardDiskType_T aType)
1099{
1100 AutoCaller autoCaller(this);
1101 CheckComRCReturnRC(autoCaller.rc());
1102
1103 /* VirtualBox::saveSettings() needs a write lock */
1104 AutoMultiWriteLock2 alock (mVirtualBox, this);
1105
1106 switch (m.state)
1107 {
1108 case MediaState_Created:
1109 case MediaState_Inaccessible:
1110 break;
1111 default:
1112 return setStateError();
1113 }
1114
1115 if (mm.type == aType)
1116 {
1117 /* Nothing to do */
1118 return S_OK;
1119 }
1120
1121 /* we access mParent & children() */
1122 AutoReadLock treeLock (this->treeLock());
1123
1124 /* cannot change the type of a differencing hard disk */
1125 if (!mParent.isNull())
1126 return setError (E_FAIL,
1127 tr ("Hard disk '%ls' is a differencing hard disk"),
1128 m.locationFull.raw());
1129
1130 /* cannot change the type of a hard disk being in use */
1131 if (m.backRefs.size() != 0)
1132 return setError (E_FAIL,
1133 tr ("Hard disk '%ls' is attached to %d virtual machines"),
1134 m.locationFull.raw(), m.backRefs.size());
1135
1136 switch (aType)
1137 {
1138 case HardDiskType_Normal:
1139 case HardDiskType_Immutable:
1140 {
1141 /* normal can be easily converted to imutable and vice versa even
1142 * if they have children as long as they are not attached to any
1143 * machine themselves */
1144 break;
1145 }
1146 case HardDiskType_Writethrough:
1147 {
1148 /* cannot change to writethrough if there are children */
1149 if (children().size() != 0)
1150 return setError (E_FAIL,
1151 tr ("Hard disk '%ls' has %d child hard disks"),
1152 children().size());
1153 break;
1154 }
1155 default:
1156 AssertFailedReturn (E_FAIL);
1157 }
1158
1159 mm.type = aType;
1160
1161 HRESULT rc = mVirtualBox->saveSettings();
1162
1163 return rc;
1164}
1165
1166STDMETHODIMP HardDisk::COMGETTER(Parent) (IHardDisk **aParent)
1167{
1168 if (aParent == NULL)
1169 return E_POINTER;
1170
1171 AutoCaller autoCaller(this);
1172 CheckComRCReturnRC(autoCaller.rc());
1173
1174 /* we access mParent */
1175 AutoReadLock treeLock (this->treeLock());
1176
1177 mParent.queryInterfaceTo(aParent);
1178
1179 return S_OK;
1180}
1181
1182STDMETHODIMP HardDisk::COMGETTER(Children) (ComSafeArrayOut(IHardDisk *, aChildren))
1183{
1184 if (ComSafeArrayOutIsNull(aChildren))
1185 return E_POINTER;
1186
1187 AutoCaller autoCaller(this);
1188 CheckComRCReturnRC(autoCaller.rc());
1189
1190 /* we access children */
1191 AutoReadLock treeLock (this->treeLock());
1192
1193 SafeIfaceArray<IHardDisk> children (this->children());
1194 children.detachTo(ComSafeArrayOutArg(aChildren));
1195
1196 return S_OK;
1197}
1198
1199STDMETHODIMP HardDisk::COMGETTER(Root)(IHardDisk **aRoot)
1200{
1201 if (aRoot == NULL)
1202 return E_POINTER;
1203
1204 /* root() will do callers/locking */
1205
1206 root().queryInterfaceTo(aRoot);
1207
1208 return S_OK;
1209}
1210
1211STDMETHODIMP HardDisk::COMGETTER(ReadOnly) (BOOL *aReadOnly)
1212{
1213 if (aReadOnly == NULL)
1214 return E_POINTER;
1215
1216 AutoCaller autoCaller(this);
1217 CheckComRCReturnRC(autoCaller.rc());
1218
1219 /* isRadOnly() will do locking */
1220
1221 *aReadOnly = isReadOnly();
1222
1223 return S_OK;
1224}
1225
1226STDMETHODIMP HardDisk::COMGETTER(LogicalSize) (ULONG64 *aLogicalSize)
1227{
1228 CheckComArgOutPointerValid(aLogicalSize);
1229
1230 {
1231 AutoCaller autoCaller(this);
1232 CheckComRCReturnRC(autoCaller.rc());
1233
1234 AutoReadLock alock(this);
1235
1236 /* we access mParent */
1237 AutoReadLock treeLock (this->treeLock());
1238
1239 if (mParent.isNull())
1240 {
1241 *aLogicalSize = mm.logicalSize;
1242
1243 return S_OK;
1244 }
1245 }
1246
1247 /* We assume that some backend may decide to return a meaningless value in
1248 * response to VDGetSize() for differencing hard disks and therefore
1249 * always ask the base hard disk ourselves. */
1250
1251 /* root() will do callers/locking */
1252
1253 return root()->COMGETTER (LogicalSize) (aLogicalSize);
1254}
1255
1256STDMETHODIMP HardDisk::COMGETTER(AutoReset) (BOOL *aAutoReset)
1257{
1258 CheckComArgOutPointerValid(aAutoReset);
1259
1260 AutoCaller autoCaller(this);
1261 CheckComRCReturnRC(autoCaller.rc());
1262
1263 AutoReadLock alock(this);
1264
1265 if (mParent.isNull())
1266 *aAutoReset = FALSE;
1267
1268 *aAutoReset = mm.autoReset;
1269
1270 return S_OK;
1271}
1272
1273STDMETHODIMP HardDisk::COMSETTER(AutoReset) (BOOL aAutoReset)
1274{
1275 AutoCaller autoCaller(this);
1276 CheckComRCReturnRC(autoCaller.rc());
1277
1278 /* VirtualBox::saveSettings() needs a write lock */
1279 AutoMultiWriteLock2 alock (mVirtualBox, this);
1280
1281 if (mParent.isNull())
1282 return setError (VBOX_E_NOT_SUPPORTED,
1283 tr ("Hard disk '%ls' is not differencing"),
1284 m.locationFull.raw());
1285
1286 if (mm.autoReset != aAutoReset)
1287 {
1288 mm.autoReset = aAutoReset;
1289
1290 return mVirtualBox->saveSettings();
1291 }
1292
1293 return S_OK;
1294}
1295
1296// IHardDisk methods
1297////////////////////////////////////////////////////////////////////////////////
1298
1299STDMETHODIMP HardDisk::GetProperty (IN_BSTR aName, BSTR *aValue)
1300{
1301 CheckComArgStrNotEmptyOrNull (aName);
1302 CheckComArgOutPointerValid(aValue);
1303
1304 AutoCaller autoCaller(this);
1305 CheckComRCReturnRC(autoCaller.rc());
1306
1307 AutoReadLock alock(this);
1308
1309 Data::PropertyMap::const_iterator it = mm.properties.find (Bstr (aName));
1310 if (it == mm.properties.end())
1311 return setError (VBOX_E_OBJECT_NOT_FOUND,
1312 tr ("Property '%ls' does not exist"), aName);
1313
1314 if (it->second.isEmpty())
1315 Bstr("").cloneTo(aValue);
1316 else
1317 it->second.cloneTo(aValue);
1318
1319 return S_OK;
1320}
1321
1322STDMETHODIMP HardDisk::SetProperty (IN_BSTR aName, IN_BSTR aValue)
1323{
1324 CheckComArgStrNotEmptyOrNull (aName);
1325
1326 AutoCaller autoCaller(this);
1327 CheckComRCReturnRC(autoCaller.rc());
1328
1329 /* VirtualBox::saveSettings() needs a write lock */
1330 AutoMultiWriteLock2 alock (mVirtualBox, this);
1331
1332 switch (m.state)
1333 {
1334 case MediaState_Created:
1335 case MediaState_Inaccessible:
1336 break;
1337 default:
1338 return setStateError();
1339 }
1340
1341 Data::PropertyMap::iterator it = mm.properties.find (Bstr (aName));
1342 if (it == mm.properties.end())
1343 return setError (VBOX_E_OBJECT_NOT_FOUND,
1344 tr ("Property '%ls' does not exist"), aName);
1345
1346 if (aValue && !*aValue)
1347 it->second = (const char *)NULL;
1348 else
1349 it->second = aValue;
1350
1351 HRESULT rc = mVirtualBox->saveSettings();
1352
1353 return rc;
1354}
1355
1356STDMETHODIMP HardDisk::GetProperties(IN_BSTR aNames,
1357 ComSafeArrayOut(BSTR, aReturnNames),
1358 ComSafeArrayOut(BSTR, aReturnValues))
1359{
1360 CheckComArgOutSafeArrayPointerValid(aReturnNames);
1361 CheckComArgOutSafeArrayPointerValid(aReturnValues);
1362
1363 AutoCaller autoCaller(this);
1364 CheckComRCReturnRC(autoCaller.rc());
1365
1366 AutoReadLock alock(this);
1367
1368 /// @todo make use of aNames according to the documentation
1369 NOREF (aNames);
1370
1371 com::SafeArray<BSTR> names (mm.properties.size());
1372 com::SafeArray<BSTR> values (mm.properties.size());
1373 size_t i = 0;
1374
1375 for (Data::PropertyMap::const_iterator it = mm.properties.begin();
1376 it != mm.properties.end(); ++ it)
1377 {
1378 it->first.cloneTo(&names [i]);
1379 if (it->second.isEmpty())
1380 Bstr("").cloneTo(&values [i]);
1381 else
1382 it->second.cloneTo(&values [i]);
1383 ++ i;
1384 }
1385
1386 names.detachTo(ComSafeArrayOutArg(aReturnNames));
1387 values.detachTo(ComSafeArrayOutArg(aReturnValues));
1388
1389 return S_OK;
1390}
1391
1392STDMETHODIMP HardDisk::SetProperties(ComSafeArrayIn (IN_BSTR, aNames),
1393 ComSafeArrayIn (IN_BSTR, aValues))
1394{
1395 CheckComArgSafeArrayNotNull (aNames);
1396 CheckComArgSafeArrayNotNull (aValues);
1397
1398 AutoCaller autoCaller(this);
1399 CheckComRCReturnRC(autoCaller.rc());
1400
1401 /* VirtualBox::saveSettings() needs a write lock */
1402 AutoMultiWriteLock2 alock (mVirtualBox, this);
1403
1404 com::SafeArray<IN_BSTR> names (ComSafeArrayInArg (aNames));
1405 com::SafeArray<IN_BSTR> values (ComSafeArrayInArg (aValues));
1406
1407 /* first pass: validate names */
1408 for (size_t i = 0; i < names.size(); ++ i)
1409 {
1410 if (mm.properties.find (Bstr (names [i])) == mm.properties.end())
1411 return setError (VBOX_E_OBJECT_NOT_FOUND,
1412 tr ("Property '%ls' does not exist"), names [i]);
1413 }
1414
1415 /* second pass: assign */
1416 for (size_t i = 0; i < names.size(); ++ i)
1417 {
1418 Data::PropertyMap::iterator it = mm.properties.find (Bstr (names [i]));
1419 AssertReturn(it != mm.properties.end(), E_FAIL);
1420
1421 if (values[i] && !*values[i])
1422 it->second = (const char *)NULL;
1423 else
1424 it->second = values [i];
1425 }
1426
1427 HRESULT rc = mVirtualBox->saveSettings();
1428
1429 return rc;
1430}
1431
1432STDMETHODIMP HardDisk::CreateBaseStorage(ULONG64 aLogicalSize,
1433 HardDiskVariant_T aVariant,
1434 IProgress **aProgress)
1435{
1436 CheckComArgOutPointerValid(aProgress);
1437
1438 AutoCaller autoCaller(this);
1439 CheckComRCReturnRC(autoCaller.rc());
1440
1441 AutoWriteLock alock(this);
1442
1443 aVariant = (HardDiskVariant_T)((unsigned)aVariant & (unsigned)~HardDiskVariant_Diff);
1444 if ( !(aVariant & HardDiskVariant_Fixed)
1445 && !(mm.formatObj->capabilities() & HardDiskFormatCapabilities_CreateDynamic))
1446 return setError (VBOX_E_NOT_SUPPORTED,
1447 tr ("Hard disk format '%ls' does not support dynamic storage creation"),
1448 mm.format.raw());
1449 if ( (aVariant & HardDiskVariant_Fixed)
1450 && !(mm.formatObj->capabilities() & HardDiskFormatCapabilities_CreateDynamic))
1451 return setError (VBOX_E_NOT_SUPPORTED,
1452 tr ("Hard disk format '%ls' does not support fixed storage creation"),
1453 mm.format.raw());
1454
1455 switch (m.state)
1456 {
1457 case MediaState_NotCreated:
1458 break;
1459 default:
1460 return setStateError();
1461 }
1462
1463 ComObjPtr<Progress> progress;
1464 progress.createObject();
1465 /// @todo include fixed/dynamic
1466 HRESULT rc = progress->init (mVirtualBox, static_cast<IHardDisk*>(this),
1467 (aVariant & HardDiskVariant_Fixed)
1468 ? BstrFmt (tr ("Creating fixed hard disk storage unit '%ls'"), m.locationFull.raw())
1469 : BstrFmt (tr ("Creating dynamic hard disk storage unit '%ls'"), m.locationFull.raw()),
1470 TRUE /* aCancelable */);
1471 CheckComRCReturnRC(rc);
1472
1473 /* setup task object and thread to carry out the operation
1474 * asynchronously */
1475
1476 std::auto_ptr <Task> task (new Task (this, progress, Task::CreateBase));
1477 AssertComRCReturnRC(task->autoCaller.rc());
1478
1479 task->d.size = aLogicalSize;
1480 task->d.variant = aVariant;
1481
1482 rc = task->startThread();
1483 CheckComRCReturnRC(rc);
1484
1485 /* go to Creating state on success */
1486 m.state = MediaState_Creating;
1487
1488 /* task is now owned by taskThread() so release it */
1489 task.release();
1490
1491 /* return progress to the caller */
1492 progress.queryInterfaceTo(aProgress);
1493
1494 return S_OK;
1495}
1496
1497STDMETHODIMP HardDisk::DeleteStorage (IProgress **aProgress)
1498{
1499 CheckComArgOutPointerValid(aProgress);
1500
1501 AutoCaller autoCaller(this);
1502 CheckComRCReturnRC(autoCaller.rc());
1503
1504 ComObjPtr<Progress> progress;
1505
1506 HRESULT rc = deleteStorageNoWait (progress);
1507 if (SUCCEEDED(rc))
1508 {
1509 /* return progress to the caller */
1510 progress.queryInterfaceTo(aProgress);
1511 }
1512
1513 return rc;
1514}
1515
1516STDMETHODIMP HardDisk::CreateDiffStorage (IHardDisk *aTarget,
1517 HardDiskVariant_T aVariant,
1518 IProgress **aProgress)
1519{
1520 CheckComArgNotNull (aTarget);
1521 CheckComArgOutPointerValid(aProgress);
1522
1523 AutoCaller autoCaller(this);
1524 CheckComRCReturnRC(autoCaller.rc());
1525
1526 ComObjPtr<HardDisk> diff;
1527 HRESULT rc = mVirtualBox->cast (aTarget, diff);
1528 CheckComRCReturnRC(rc);
1529
1530 AutoWriteLock alock(this);
1531
1532 if (mm.type == HardDiskType_Writethrough)
1533 return setError (E_FAIL,
1534 tr ("Hard disk '%ls' is Writethrough"),
1535 m.locationFull.raw());
1536
1537 /* We want to be locked for reading as long as our diff child is being
1538 * created */
1539 rc = LockRead (NULL);
1540 CheckComRCReturnRC(rc);
1541
1542 ComObjPtr<Progress> progress;
1543
1544 rc = createDiffStorageNoWait (diff, aVariant, progress);
1545 if (FAILED (rc))
1546 {
1547 HRESULT rc2 = UnlockRead (NULL);
1548 AssertComRC (rc2);
1549 /* Note: on success, taskThread() will unlock this */
1550 }
1551 else
1552 {
1553 /* return progress to the caller */
1554 progress.queryInterfaceTo(aProgress);
1555 }
1556
1557 return rc;
1558}
1559
1560STDMETHODIMP HardDisk::MergeTo (IN_BSTR /* aTargetId */, IProgress ** /* aProgress */)
1561{
1562 AutoCaller autoCaller(this);
1563 CheckComRCReturnRC(autoCaller.rc());
1564
1565 ReturnComNotImplemented();
1566}
1567
1568STDMETHODIMP HardDisk::CloneTo (IHardDisk *aTarget,
1569 HardDiskVariant_T aVariant,
1570 IHardDisk *aParent,
1571 IProgress **aProgress)
1572{
1573 CheckComArgNotNull (aTarget);
1574 CheckComArgOutPointerValid(aProgress);
1575
1576 AutoCaller autoCaller(this);
1577 CheckComRCReturnRC(autoCaller.rc());
1578
1579 ComObjPtr<HardDisk> target;
1580 HRESULT rc = mVirtualBox->cast (aTarget, target);
1581 CheckComRCReturnRC(rc);
1582 ComObjPtr<HardDisk> parent;
1583 if (aParent)
1584 {
1585 rc = mVirtualBox->cast (aParent, parent);
1586 CheckComRCReturnRC(rc);
1587 }
1588
1589 AutoMultiWriteLock3 alock (this, target, parent);
1590
1591 ComObjPtr<Progress> progress;
1592
1593 try
1594 {
1595 if ( target->m.state != MediaState_NotCreated
1596 && target->m.state != MediaState_Created)
1597 throw target->setStateError();
1598
1599 /** @todo separate out creating/locking an image chain from
1600 * SessionMachine::lockMedia and use it from here too.
1601 * logically this belongs into HardDisk functionality. */
1602
1603 /* Build the source chain and lock images in the proper order. */
1604 std::auto_ptr <ImageChain> srcChain (new ImageChain ());
1605
1606 /* we walk the source tree */
1607 AutoReadLock srcTreeLock (this->treeLock());
1608 for (HardDisk *hd = this; hd; hd = hd->mParent)
1609 {
1610 rc = srcChain->addImage(hd);
1611 CheckComRCThrowRC (rc);
1612 }
1613 rc = srcChain->lockImagesRead();
1614 CheckComRCThrowRC (rc);
1615
1616 /* Build the parent chain and lock images in the proper order. */
1617 std::auto_ptr <ImageChain> parentChain (new ImageChain ());
1618
1619 /* we walk the future parent tree */
1620 AutoReadLock parentTreeLock;
1621 if (parent)
1622 parentTreeLock.attach(parent->treeLock());
1623 for (HardDisk *hd = parent; hd; hd = hd->mParent)
1624 {
1625 rc = parentChain->addImage(hd);
1626 CheckComRCThrowRC (rc);
1627 }
1628 if (target->m.state == MediaState_Created)
1629 {
1630 /* If we're cloning to an existing image the parent chain also
1631 * contains the target image, and it gets locked for writing. */
1632 rc = parentChain->addImage(target);
1633 CheckComRCThrowRC (rc);
1634 rc = parentChain->lockImagesReadAndLastWrite();
1635 CheckComRCThrowRC (rc);
1636 }
1637 else
1638 {
1639 rc = parentChain->lockImagesRead();
1640 CheckComRCThrowRC (rc);
1641 }
1642
1643 progress.createObject();
1644 rc = progress->init (mVirtualBox, static_cast <IHardDisk *> (this),
1645 BstrFmt (tr ("Creating clone hard disk '%ls'"),
1646 target->m.locationFull.raw()),
1647 TRUE /* aCancelable */);
1648 CheckComRCThrowRC (rc);
1649
1650 /* setup task object and thread to carry out the operation
1651 * asynchronously */
1652
1653 std::auto_ptr <Task> task (new Task (this, progress, Task::Clone));
1654 AssertComRCThrowRC (task->autoCaller.rc());
1655
1656 task->setData (target, parent);
1657 task->d.variant = aVariant;
1658 task->setData (srcChain.release(), parentChain.release());
1659
1660 rc = task->startThread();
1661 CheckComRCThrowRC (rc);
1662
1663 if (target->m.state == MediaState_NotCreated)
1664 {
1665 /* go to Creating state before leaving the lock */
1666 target->m.state = MediaState_Creating;
1667 }
1668
1669 /* task is now owned (or already deleted) by taskThread() so release it */
1670 task.release();
1671 }
1672 catch (HRESULT aRC)
1673 {
1674 rc = aRC;
1675 }
1676
1677 if (SUCCEEDED(rc))
1678 {
1679 /* return progress to the caller */
1680 progress.queryInterfaceTo(aProgress);
1681 }
1682
1683 return rc;
1684}
1685
1686STDMETHODIMP HardDisk::Compact (IProgress **aProgress)
1687{
1688 CheckComArgOutPointerValid(aProgress);
1689
1690 AutoCaller autoCaller(this);
1691 CheckComRCReturnRC(autoCaller.rc());
1692
1693 AutoWriteLock alock(this);
1694
1695 ComObjPtr<Progress> progress;
1696
1697 HRESULT rc = S_OK;
1698
1699 try
1700 {
1701 /** @todo separate out creating/locking an image chain from
1702 * SessionMachine::lockMedia and use it from here too.
1703 * logically this belongs into HardDisk functionality. */
1704
1705 /* Build the image chain and lock images in the proper order. */
1706 std::auto_ptr <ImageChain> imgChain (new ImageChain ());
1707
1708 /* we walk the image tree */
1709 AutoReadLock srcTreeLock (this->treeLock());
1710 for (HardDisk *hd = this; hd; hd = hd->mParent)
1711 {
1712 rc = imgChain->addImage(hd);
1713 CheckComRCThrowRC (rc);
1714 }
1715 rc = imgChain->lockImagesReadAndLastWrite();
1716 CheckComRCThrowRC (rc);
1717
1718 progress.createObject();
1719 rc = progress->init (mVirtualBox, static_cast <IHardDisk *> (this),
1720 BstrFmt (tr ("Compacting hard disk '%ls'"), m.locationFull.raw()),
1721 TRUE /* aCancelable */);
1722 CheckComRCThrowRC (rc);
1723
1724 /* setup task object and thread to carry out the operation
1725 * asynchronously */
1726
1727 std::auto_ptr <Task> task (new Task (this, progress, Task::Compact));
1728 AssertComRCThrowRC (task->autoCaller.rc());
1729
1730 task->setData (imgChain.release());
1731
1732 rc = task->startThread();
1733 CheckComRCThrowRC (rc);
1734
1735 /* task is now owned (or already deleted) by taskThread() so release it */
1736 task.release();
1737 }
1738 catch (HRESULT aRC)
1739 {
1740 rc = aRC;
1741 }
1742
1743 if (SUCCEEDED(rc))
1744 {
1745 /* return progress to the caller */
1746 progress.queryInterfaceTo(aProgress);
1747 }
1748
1749 return rc;
1750}
1751
1752STDMETHODIMP HardDisk::Reset (IProgress **aProgress)
1753{
1754 CheckComArgOutPointerValid(aProgress);
1755
1756 AutoCaller autoCaller(this);
1757 CheckComRCReturnRC(autoCaller.rc());
1758
1759 AutoWriteLock alock(this);
1760
1761 if (mParent.isNull())
1762 return setError (VBOX_E_NOT_SUPPORTED,
1763 tr ("Hard disk '%ls' is not differencing"),
1764 m.locationFull.raw());
1765
1766 HRESULT rc = canClose();
1767 CheckComRCReturnRC(rc);
1768
1769 rc = LockWrite (NULL);
1770 CheckComRCReturnRC(rc);
1771
1772 ComObjPtr<Progress> progress;
1773
1774 try
1775 {
1776 progress.createObject();
1777 rc = progress->init (mVirtualBox, static_cast <IHardDisk *> (this),
1778 BstrFmt (tr ("Resetting differencing hard disk '%ls'"),
1779 m.locationFull.raw()),
1780 FALSE /* aCancelable */);
1781 CheckComRCThrowRC (rc);
1782
1783 /* setup task object and thread to carry out the operation
1784 * asynchronously */
1785
1786 std::auto_ptr <Task> task (new Task (this, progress, Task::Reset));
1787 AssertComRCThrowRC (task->autoCaller.rc());
1788
1789 rc = task->startThread();
1790 CheckComRCThrowRC (rc);
1791
1792 /* task is now owned (or already deleted) by taskThread() so release it */
1793 task.release();
1794 }
1795 catch (HRESULT aRC)
1796 {
1797 rc = aRC;
1798 }
1799
1800 if (FAILED (rc))
1801 {
1802 HRESULT rc2 = UnlockWrite (NULL);
1803 AssertComRC (rc2);
1804 /* Note: on success, taskThread() will unlock this */
1805 }
1806 else
1807 {
1808 /* return progress to the caller */
1809 progress.queryInterfaceTo(aProgress);
1810 }
1811
1812 return rc;
1813}
1814
1815// public methods for internal purposes only
1816////////////////////////////////////////////////////////////////////////////////
1817
1818/**
1819 * Checks if the given change of \a aOldPath to \a aNewPath affects the location
1820 * of this hard disk or any its child and updates the paths if necessary to
1821 * reflect the new location.
1822 *
1823 * @param aOldPath Old path (full).
1824 * @param aNewPath New path (full).
1825 *
1826 * @note Locks treeLock() for reading, this object and all children for writing.
1827 */
1828void HardDisk::updatePaths (const char *aOldPath, const char *aNewPath)
1829{
1830 AssertReturnVoid (aOldPath);
1831 AssertReturnVoid (aNewPath);
1832
1833 AutoCaller autoCaller(this);
1834 AssertComRCReturnVoid (autoCaller.rc());
1835
1836 AutoWriteLock alock(this);
1837
1838 /* we access children() */
1839 AutoReadLock treeLock (this->treeLock());
1840
1841 updatePath (aOldPath, aNewPath);
1842
1843 /* update paths of all children */
1844 for (List::const_iterator it = children().begin();
1845 it != children().end();
1846 ++ it)
1847 {
1848 (*it)->updatePaths (aOldPath, aNewPath);
1849 }
1850}
1851
1852/**
1853 * Returns the base hard disk of the hard disk chain this hard disk is part of.
1854 *
1855 * The root hard disk is found by walking up the parent-child relationship axis.
1856 * If the hard disk doesn't have a parent (i.e. it's a base hard disk), it
1857 * returns itself in response to this method.
1858 *
1859 * @param aLevel Where to store the number of ancestors of this hard disk
1860 * (zero for the root), may be @c NULL.
1861 *
1862 * @note Locks treeLock() for reading.
1863 */
1864ComObjPtr<HardDisk> HardDisk::root (uint32_t *aLevel /*= NULL*/)
1865{
1866 ComObjPtr<HardDisk> root;
1867 uint32_t level;
1868
1869 AutoCaller autoCaller(this);
1870 AssertReturn(autoCaller.isOk(), root);
1871
1872 /* we access mParent */
1873 AutoReadLock treeLock (this->treeLock());
1874
1875 root = this;
1876 level = 0;
1877
1878 if (!mParent.isNull())
1879 {
1880 for (;;)
1881 {
1882 AutoCaller rootCaller (root);
1883 AssertReturn(rootCaller.isOk(), root);
1884
1885 if (root->mParent.isNull())
1886 break;
1887
1888 root = root->mParent;
1889 ++ level;
1890 }
1891 }
1892
1893 if (aLevel != NULL)
1894 *aLevel = level;
1895
1896 return root;
1897}
1898
1899/**
1900 * Returns @c true if this hard disk cannot be modified because it has
1901 * dependants (children) or is part of the snapshot. Related to the hard disk
1902 * type and posterity, not to the current media state.
1903 *
1904 * @note Locks this object and treeLock() for reading.
1905 */
1906bool HardDisk::isReadOnly()
1907{
1908 AutoCaller autoCaller(this);
1909 AssertComRCReturn (autoCaller.rc(), false);
1910
1911 AutoReadLock alock(this);
1912
1913 /* we access children */
1914 AutoReadLock treeLock (this->treeLock());
1915
1916 switch (mm.type)
1917 {
1918 case HardDiskType_Normal:
1919 {
1920 if (children().size() != 0)
1921 return true;
1922
1923 for (BackRefList::const_iterator it = m.backRefs.begin();
1924 it != m.backRefs.end(); ++ it)
1925 if (it->snapshotIds.size() != 0)
1926 return true;
1927
1928 return false;
1929 }
1930 case HardDiskType_Immutable:
1931 {
1932 return true;
1933 }
1934 case HardDiskType_Writethrough:
1935 {
1936 return false;
1937 }
1938 default:
1939 break;
1940 }
1941
1942 AssertFailedReturn (false);
1943}
1944
1945/**
1946 * Saves hard disk data by appending a new <HardDisk> child node to the given
1947 * parent node which can be either <HardDisks> or <HardDisk>.
1948 *
1949 * @param aaParentNode Parent <HardDisks> or <HardDisk> node.
1950 *
1951 * @note Locks this object, treeLock() and children for reading.
1952 */
1953HRESULT HardDisk::saveSettings(settings::Medium &data)
1954{
1955 AutoCaller autoCaller(this);
1956 CheckComRCReturnRC(autoCaller.rc());
1957
1958 AutoReadLock alock(this);
1959
1960 /* we access mParent */
1961 AutoReadLock treeLock (this->treeLock());
1962
1963 data.uuid = m.id;
1964 data.strLocation = m.location;
1965 data.strFormat = mm.format;
1966
1967 /* optional, only for diffs, default is false */
1968 if (!mParent.isNull())
1969 data.fAutoReset = !!mm.autoReset;
1970 else
1971 data.fAutoReset = false;
1972
1973 /* optional */
1974 data.strDescription = m.description;
1975
1976 /* optional properties */
1977 data.properties.clear();
1978 for (Data::PropertyMap::const_iterator it = mm.properties.begin();
1979 it != mm.properties.end();
1980 ++it)
1981 {
1982 /* only save properties that have non-default values */
1983 if (!it->second.isNull())
1984 {
1985 const Utf8Str &name = it->first;
1986 const Utf8Str &value = it->second;
1987 data.properties[name] = value;
1988 }
1989 }
1990
1991 /* only for base hard disks */
1992 if (mParent.isNull())
1993 data.hdType = mm.type;
1994
1995 /* save all children */
1996 for (List::const_iterator it = children().begin();
1997 it != children().end();
1998 ++it)
1999 {
2000 settings::Medium m;
2001 HRESULT rc = (*it)->saveSettings(m);
2002 AssertComRCReturnRC (rc);
2003 data.llChildren.push_back(m);
2004 }
2005
2006 return S_OK;
2007}
2008
2009/**
2010 * Compares the location of this hard disk to the given location.
2011 *
2012 * The comparison takes the location details into account. For example, if the
2013 * location is a file in the host's filesystem, a case insensitive comparison
2014 * will be performed for case insensitive filesystems.
2015 *
2016 * @param aLocation Location to compare to (as is).
2017 * @param aResult Where to store the result of comparison: 0 if locations
2018 * are equal, 1 if this object's location is greater than
2019 * the specified location, and -1 otherwise.
2020 */
2021HRESULT HardDisk::compareLocationTo(const Utf8Str &strLocation, int &aResult)
2022{
2023 AutoCaller autoCaller(this);
2024 AssertComRCReturnRC(autoCaller.rc());
2025
2026 AutoReadLock alock(this);
2027
2028 Utf8Str locationFull (m.locationFull);
2029
2030 /// @todo NEWMEDIA delegate the comparison to the backend?
2031
2032 if (mm.formatObj->capabilities() & HardDiskFormatCapabilities_File)
2033 {
2034 Utf8Str location(strLocation);
2035
2036 /* For locations represented by files, append the default path if
2037 * only the name is given, and then get the full path. */
2038 if (!RTPathHavePath(strLocation.c_str()))
2039 location = Utf8StrFmt("%s%c%s",
2040 mVirtualBox->getDefaultHardDiskFolder().raw(),
2041 RTPATH_DELIMITER,
2042 strLocation.c_str());
2043
2044 int vrc = mVirtualBox->calculateFullPath(location, location);
2045 if (RT_FAILURE(vrc))
2046 return setError(E_FAIL,
2047 tr("Invalid hard disk storage file location '%s' (%Rrc)"),
2048 location.raw(),
2049 vrc);
2050
2051 aResult = RTPathCompare(locationFull.c_str(), location.c_str());
2052 }
2053 else
2054 aResult = locationFull.compare(strLocation);
2055
2056 return S_OK;
2057}
2058
2059/**
2060 * Returns a short version of the location attribute.
2061 *
2062 * Reimplements MediumBase::name() to specially treat non-FS-path locations.
2063 *
2064 * @note Must be called from under this object's read or write lock.
2065 */
2066Utf8Str HardDisk::name()
2067{
2068 /// @todo NEWMEDIA treat non-FS-paths specially! (may require to requiest
2069 /// this information from the VD backend)
2070
2071 Utf8Str location (m.locationFull);
2072
2073 Utf8Str name = RTPathFilename(location.c_str());
2074 return name;
2075}
2076
2077/**
2078 * Checks that this hard disk may be discarded and performs necessary state
2079 * changes.
2080 *
2081 * This method is to be called prior to calling the #discard() to perform
2082 * necessary consistency checks and place involved hard disks to appropriate
2083 * states. If #discard() is not called or fails, the state modifications
2084 * performed by this method must be undone by #cancelDiscard().
2085 *
2086 * See #discard() for more info about discarding hard disks.
2087 *
2088 * @param aChain Where to store the created merge chain (may return NULL
2089 * if no real merge is necessary).
2090 *
2091 * @note Locks treeLock() for reading. Locks this object, aTarget and all
2092 * intermediate hard disks for writing.
2093 */
2094HRESULT HardDisk::prepareDiscard (MergeChain * &aChain)
2095{
2096 AutoCaller autoCaller(this);
2097 AssertComRCReturnRC(autoCaller.rc());
2098
2099 aChain = NULL;
2100
2101 AutoWriteLock alock(this);
2102
2103 /* we access mParent & children() */
2104 AutoReadLock treeLock (this->treeLock());
2105
2106 AssertReturn(mm.type == HardDiskType_Normal, E_FAIL);
2107
2108 if (children().size() == 0)
2109 {
2110 /* special treatment of the last hard disk in the chain: */
2111
2112 if (mParent.isNull())
2113 {
2114 /* lock only, to prevent any usage; discard() will unlock */
2115 return LockWrite (NULL);
2116 }
2117
2118 /* the differencing hard disk w/o children will be deleted, protect it
2119 * from attaching to other VMs (this is why Deleting) */
2120
2121 switch (m.state)
2122 {
2123 case MediaState_Created:
2124 m.state = MediaState_Deleting;
2125 break;
2126 default:
2127 return setStateError();
2128 }
2129
2130 /* aChain is intentionally NULL here */
2131
2132 return S_OK;
2133 }
2134
2135 /* not going multi-merge as it's too expensive */
2136 if (children().size() > 1)
2137 return setError (E_FAIL,
2138 tr ("Hard disk '%ls' has more than one child hard disk (%d)"),
2139 m.locationFull.raw(), children().size());
2140
2141 /* this is a read-only hard disk with children; it must be associated with
2142 * exactly one snapshot (when the snapshot is being taken, none of the
2143 * current VM's hard disks may be attached to other VMs). Note that by the
2144 * time when discard() is called, there must be no any attachments at all
2145 * (the code calling prepareDiscard() should detach). */
2146 AssertReturn(m.backRefs.size() == 1 &&
2147 !m.backRefs.front().inCurState &&
2148 m.backRefs.front().snapshotIds.size() == 1, E_FAIL);
2149
2150 ComObjPtr<HardDisk> child = children().front();
2151
2152 /* we keep this locked, so lock the affected child to make sure the lock
2153 * order is correct when calling prepareMergeTo() */
2154 AutoWriteLock childLock (child);
2155
2156 /* delegate the rest to the profi */
2157 if (mParent.isNull())
2158 {
2159 /* base hard disk, backward merge */
2160
2161 Assert (child->m.backRefs.size() == 1);
2162 if (child->m.backRefs.front().machineId != m.backRefs.front().machineId)
2163 {
2164 /* backward merge is too tricky, we'll just detach on discard, so
2165 * lock only, to prevent any usage; discard() will only unlock
2166 * (since we return NULL in aChain) */
2167 return LockWrite (NULL);
2168 }
2169
2170 return child->prepareMergeTo (this, aChain,
2171 true /* aIgnoreAttachments */);
2172 }
2173 else
2174 {
2175 /* forward merge */
2176 return prepareMergeTo (child, aChain,
2177 true /* aIgnoreAttachments */);
2178 }
2179}
2180
2181/**
2182 * Discards this hard disk.
2183 *
2184 * Discarding the hard disk is merging its contents to its differencing child
2185 * hard disk (forward merge) or contents of its child hard disk to itself
2186 * (backward merge) if this hard disk is a base hard disk. If this hard disk is
2187 * a differencing hard disk w/o children, then it will be simply deleted.
2188 * Calling this method on a base hard disk w/o children will do nothing and
2189 * silently succeed. If this hard disk has more than one child, the method will
2190 * currently return an error (since merging in this case would be too expensive
2191 * and result in data duplication).
2192 *
2193 * When the backward merge takes place (i.e. this hard disk is a target) then,
2194 * on success, this hard disk will automatically replace the differencing child
2195 * hard disk used as a source (which will then be deleted) in the attachment
2196 * this child hard disk is associated with. This will happen only if both hard
2197 * disks belong to the same machine because otherwise such a replace would be
2198 * too tricky and could be not expected by the other machine. Same relates to a
2199 * case when the child hard disk is not associated with any machine at all. When
2200 * the backward merge is not applied, the method behaves as if the base hard
2201 * disk were not attached at all -- i.e. simply detaches it from the machine but
2202 * leaves the hard disk chain intact.
2203 *
2204 * This method is basically a wrapper around #mergeTo() that selects the correct
2205 * merge direction and performs additional actions as described above and.
2206 *
2207 * Note that this method will not return until the merge operation is complete
2208 * (which may be quite time consuming depending on the size of the merged hard
2209 * disks).
2210 *
2211 * Note that #prepareDiscard() must be called before calling this method. If
2212 * this method returns a failure, the caller must call #cancelDiscard(). On
2213 * success, #cancelDiscard() must not be called (this method will perform all
2214 * necessary steps such as resetting states of all involved hard disks and
2215 * deleting @a aChain).
2216 *
2217 * @param aChain Merge chain created by #prepareDiscard() (may be NULL if
2218 * no real merge takes place).
2219 *
2220 * @note Locks the hard disks from the chain for writing. Locks the machine
2221 * object when the backward merge takes place. Locks treeLock() lock for
2222 * reading or writing.
2223 */
2224HRESULT HardDisk::discard (ComObjPtr<Progress> &aProgress, MergeChain *aChain)
2225{
2226 AssertReturn(!aProgress.isNull(), E_FAIL);
2227
2228 ComObjPtr<HardDisk> hdFrom;
2229
2230 HRESULT rc = S_OK;
2231
2232 {
2233 AutoCaller autoCaller(this);
2234 AssertComRCReturnRC(autoCaller.rc());
2235
2236 aProgress->setNextOperation(BstrFmt(tr("Discarding hard disk '%s'"), name().raw()),
2237 1); // weight
2238
2239 if (aChain == NULL)
2240 {
2241 AutoWriteLock alock(this);
2242
2243 /* we access mParent & children() */
2244 AutoReadLock treeLock (this->treeLock());
2245
2246 Assert (children().size() == 0);
2247
2248 /* special treatment of the last hard disk in the chain: */
2249
2250 if (mParent.isNull())
2251 {
2252 rc = UnlockWrite (NULL);
2253 AssertComRC (rc);
2254 return rc;
2255 }
2256
2257 /* delete the differencing hard disk w/o children */
2258
2259 Assert (m.state == MediaState_Deleting);
2260
2261 /* go back to Created since deleteStorage() expects this state */
2262 m.state = MediaState_Created;
2263
2264 hdFrom = this;
2265
2266 rc = deleteStorageAndWait (&aProgress);
2267 }
2268 else
2269 {
2270 hdFrom = aChain->source();
2271
2272 rc = hdFrom->mergeToAndWait (aChain, &aProgress);
2273 }
2274 }
2275
2276 if (SUCCEEDED(rc))
2277 {
2278 /* mergeToAndWait() cannot uninitialize the initiator because of
2279 * possible AutoCallers on the current thread, deleteStorageAndWait()
2280 * doesn't do it either; do it ourselves */
2281 hdFrom->uninit();
2282 }
2283
2284 return rc;
2285}
2286
2287/**
2288 * Undoes what #prepareDiscard() did. Must be called if #discard() is not called
2289 * or fails. Frees memory occupied by @a aChain.
2290 *
2291 * @param aChain Merge chain created by #prepareDiscard() (may be NULL if
2292 * no real merge takes place).
2293 *
2294 * @note Locks the hard disks from the chain for writing. Locks treeLock() for
2295 * reading.
2296 */
2297void HardDisk::cancelDiscard (MergeChain *aChain)
2298{
2299 AutoCaller autoCaller(this);
2300 AssertComRCReturnVoid (autoCaller.rc());
2301
2302 if (aChain == NULL)
2303 {
2304 AutoWriteLock alock(this);
2305
2306 /* we access mParent & children() */
2307 AutoReadLock treeLock (this->treeLock());
2308
2309 Assert (children().size() == 0);
2310
2311 /* special treatment of the last hard disk in the chain: */
2312
2313 if (mParent.isNull())
2314 {
2315 HRESULT rc = UnlockWrite (NULL);
2316 AssertComRC (rc);
2317 return;
2318 }
2319
2320 /* the differencing hard disk w/o children will be deleted, protect it
2321 * from attaching to other VMs (this is why Deleting) */
2322
2323 Assert (m.state == MediaState_Deleting);
2324 m.state = MediaState_Created;
2325
2326 return;
2327 }
2328
2329 /* delegate the rest to the profi */
2330 cancelMergeTo (aChain);
2331}
2332
2333/**
2334 * Returns a preferred format for differencing hard disks.
2335 */
2336Bstr HardDisk::preferredDiffFormat()
2337{
2338 AutoCaller autoCaller(this);
2339 AssertReturn(!autoCaller.rc(), "");
2340
2341 /* mm.format is const, no need to lock */
2342 Bstr bstrFormat = mm.format;
2343
2344 /* check that our own format supports diffs */
2345 if (!(mm.formatObj->capabilities() & HardDiskFormatCapabilities_Differencing))
2346 /* use the default format if not */
2347 bstrFormat = mVirtualBox->getDefaultHardDiskFormat();
2348
2349 return bstrFormat;
2350}
2351
2352// protected methods
2353////////////////////////////////////////////////////////////////////////////////
2354
2355/**
2356 * Deletes the hard disk storage unit.
2357 *
2358 * If @a aProgress is not NULL but the object it points to is @c null then a new
2359 * progress object will be created and assigned to @a *aProgress on success,
2360 * otherwise the existing progress object is used. If Progress is NULL, then no
2361 * progress object is created/used at all.
2362 *
2363 * When @a aWait is @c false, this method will create a thread to perform the
2364 * delete operation asynchronously and will return immediately. Otherwise, it
2365 * will perform the operation on the calling thread and will not return to the
2366 * caller until the operation is completed. Note that @a aProgress cannot be
2367 * NULL when @a aWait is @c false (this method will assert in this case).
2368 *
2369 * @param aProgress Where to find/store a Progress object to track operation
2370 * completion.
2371 * @param aWait @c true if this method should block instead of creating
2372 * an asynchronous thread.
2373 *
2374 * @note Locks mVirtualBox and this object for writing. Locks treeLock() for
2375 * writing.
2376 */
2377HRESULT HardDisk::deleteStorage (ComObjPtr<Progress> *aProgress, bool aWait)
2378{
2379 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
2380
2381 /* unregisterWithVirtualBox() needs a write lock. We want to unregister
2382 * ourselves atomically after detecting that deletion is possible to make
2383 * sure that we don't do that after another thread has done
2384 * VirtualBox::findHardDisk() but before it starts using us (provided that
2385 * it holds a mVirtualBox lock too of course). */
2386
2387 AutoWriteLock vboxLock (mVirtualBox);
2388
2389 AutoWriteLock alock(this);
2390
2391 if (!(mm.formatObj->capabilities() &
2392 (HardDiskFormatCapabilities_CreateDynamic |
2393 HardDiskFormatCapabilities_CreateFixed)))
2394 return setError (VBOX_E_NOT_SUPPORTED,
2395 tr ("Hard disk format '%ls' does not support storage deletion"),
2396 mm.format.raw());
2397
2398 /* Note that we are fine with Inaccessible state too: a) for symmetry with
2399 * create calls and b) because it doesn't really harm to try, if it is
2400 * really inaccessibke, the delete operation will fail anyway. Accepting
2401 * Inaccessible state is especially important because all registered hard
2402 * disks are initially Inaccessible upon VBoxSVC startup until
2403 * COMGETTER(State) is called. */
2404
2405 switch (m.state)
2406 {
2407 case MediaState_Created:
2408 case MediaState_Inaccessible:
2409 break;
2410 default:
2411 return setStateError();
2412 }
2413
2414 if (m.backRefs.size() != 0)
2415 return setError (VBOX_E_OBJECT_IN_USE,
2416 tr ("Hard disk '%ls' is attached to %d virtual machines"),
2417 m.locationFull.raw(), m.backRefs.size());
2418
2419 HRESULT rc = canClose();
2420 CheckComRCReturnRC(rc);
2421
2422 /* go to Deleting state before leaving the lock */
2423 m.state = MediaState_Deleting;
2424
2425 /* we need to leave this object's write lock now because of
2426 * unregisterWithVirtualBox() that locks treeLock() for writing */
2427 alock.leave();
2428
2429 /* try to remove from the list of known hard disks before performing actual
2430 * deletion (we favor the consistency of the media registry in the first
2431 * place which would have been broken if unregisterWithVirtualBox() failed
2432 * after we successfully deleted the storage) */
2433
2434 rc = unregisterWithVirtualBox();
2435
2436 alock.enter();
2437
2438 /* restore the state because we may fail below; we will set it later again*/
2439 m.state = MediaState_Created;
2440
2441 CheckComRCReturnRC(rc);
2442
2443 ComObjPtr<Progress> progress;
2444
2445 if (aProgress != NULL)
2446 {
2447 /* use the existing progress object... */
2448 progress = *aProgress;
2449
2450 /* ...but create a new one if it is null */
2451 if (progress.isNull())
2452 {
2453 progress.createObject();
2454 rc = progress->init (mVirtualBox, static_cast<IHardDisk*>(this),
2455 BstrFmt (tr ("Deleting hard disk storage unit '%ls'"),
2456 m.locationFull.raw()),
2457 FALSE /* aCancelable */);
2458 CheckComRCReturnRC(rc);
2459 }
2460 }
2461
2462 std::auto_ptr <Task> task (new Task (this, progress, Task::Delete));
2463 AssertComRCReturnRC(task->autoCaller.rc());
2464
2465 if (aWait)
2466 {
2467 /* go to Deleting state before starting the task */
2468 m.state = MediaState_Deleting;
2469
2470 rc = task->runNow();
2471 }
2472 else
2473 {
2474 rc = task->startThread();
2475 CheckComRCReturnRC(rc);
2476
2477 /* go to Deleting state before leaving the lock */
2478 m.state = MediaState_Deleting;
2479 }
2480
2481 /* task is now owned (or already deleted) by taskThread() so release it */
2482 task.release();
2483
2484 if (aProgress != NULL)
2485 {
2486 /* return progress to the caller */
2487 *aProgress = progress;
2488 }
2489
2490 return rc;
2491}
2492
2493/**
2494 * Creates a new differencing storage unit using the given target hard disk's
2495 * format and the location. Note that @c aTarget must be NotCreated.
2496 *
2497 * As opposed to the CreateDiffStorage() method, this method doesn't try to lock
2498 * this hard disk for reading assuming that the caller has already done so. This
2499 * is used when taking an online snaopshot (where all original hard disks are
2500 * locked for writing and must remain such). Note however that if @a aWait is
2501 * @c false and this method returns a success then the thread started by
2502 * this method will unlock the hard disk (unless it is in
2503 * MediaState_LockedWrite state) so make sure the hard disk is either in
2504 * MediaState_LockedWrite or call #LockRead() before calling this method! If @a
2505 * aWait is @c true then this method neither locks nor unlocks the hard disk, so
2506 * make sure you do it yourself as needed.
2507 *
2508 * If @a aProgress is not NULL but the object it points to is @c null then a new
2509 * progress object will be created and assigned to @a *aProgress on success,
2510 * otherwise the existing progress object is used. If @a aProgress is NULL, then no
2511 * progress object is created/used at all.
2512 *
2513 * When @a aWait is @c false, this method will create a thread to perform the
2514 * create operation asynchronously and will return immediately. Otherwise, it
2515 * will perform the operation on the calling thread and will not return to the
2516 * caller until the operation is completed. Note that @a aProgress cannot be
2517 * NULL when @a aWait is @c false (this method will assert in this case).
2518 *
2519 * @param aTarget Target hard disk.
2520 * @param aVariant Precise image variant to create.
2521 * @param aProgress Where to find/store a Progress object to track operation
2522 * completion.
2523 * @param aWait @c true if this method should block instead of creating
2524 * an asynchronous thread.
2525 *
2526 * @note Locks this object and @a aTarget for writing.
2527 */
2528HRESULT HardDisk::createDiffStorage(ComObjPtr<HardDisk> &aTarget,
2529 HardDiskVariant_T aVariant,
2530 ComObjPtr<Progress> *aProgress,
2531 bool aWait)
2532{
2533 AssertReturn(!aTarget.isNull(), E_FAIL);
2534 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
2535
2536 AutoCaller autoCaller(this);
2537 CheckComRCReturnRC(autoCaller.rc());
2538
2539 AutoCaller targetCaller (aTarget);
2540 CheckComRCReturnRC(targetCaller.rc());
2541
2542 AutoMultiWriteLock2 alock (this, aTarget);
2543
2544 AssertReturn(mm.type != HardDiskType_Writethrough, E_FAIL);
2545
2546 LogFlowThisFunc(("Hard disk '%s' has media status %d\n", name().raw(), m.state));
2547
2548 /* Note: MediaState_LockedWrite is ok when taking an online snapshot */
2549 AssertReturn(m.state == MediaState_LockedRead ||
2550 m.state == MediaState_LockedWrite, E_FAIL);
2551
2552 if (aTarget->m.state != MediaState_NotCreated)
2553 return aTarget->setStateError();
2554
2555 HRESULT rc = S_OK;
2556
2557 /* check that the hard disk is not attached to any VM in the current state*/
2558 for (BackRefList::const_iterator it = m.backRefs.begin();
2559 it != m.backRefs.end();
2560 ++it)
2561 {
2562 if (it->inCurState)
2563 {
2564 /* Note: when a VM snapshot is being taken, all normal hard disks
2565 * attached to the VM in the current state will be, as an exception,
2566 * also associated with the snapshot which is about to create (see
2567 * SnapshotMachine::init()) before deassociating them from the
2568 * current state (which takes place only on success in
2569 * Machine::fixupHardDisks()), so that the size of snapshotIds
2570 * will be 1 in this case. The given condition is used to filter out
2571 * this legal situatinon and do not report an error. */
2572
2573 if (it->snapshotIds.size() == 0)
2574 {
2575 return setError (VBOX_E_INVALID_OBJECT_STATE,
2576 tr ("Hard disk '%ls' is attached to a virtual machine with UUID {%RTuuid}. No differencing hard disks based on it may be created until it is detached"),
2577 m.locationFull.raw(), it->machineId.raw());
2578 }
2579
2580 Assert (it->snapshotIds.size() == 1);
2581 }
2582 }
2583
2584 ComObjPtr<Progress> progress;
2585
2586 if (aProgress != NULL)
2587 {
2588 /* use the existing progress object... */
2589 progress = *aProgress;
2590
2591 /* ...but create a new one if it is null */
2592 if (progress.isNull())
2593 {
2594 progress.createObject();
2595 rc = progress->init(mVirtualBox,
2596 static_cast<IHardDisk*>(this),
2597 BstrFmt(tr("Creating differencing hard disk storage unit '%ls'"),
2598 aTarget->m.locationFull.raw()),
2599 TRUE /* aCancelable */);
2600 CheckComRCReturnRC(rc);
2601 }
2602 }
2603
2604 /* setup task object and thread to carry out the operation
2605 * asynchronously */
2606
2607 std::auto_ptr<Task> task(new Task(this, progress, Task::CreateDiff));
2608 AssertComRCReturnRC(task->autoCaller.rc());
2609
2610 task->setData (aTarget);
2611 task->d.variant = aVariant;
2612
2613 /* register a task (it will deregister itself when done) */
2614 ++mm.numCreateDiffTasks;
2615 Assert (mm.numCreateDiffTasks != 0); /* overflow? */
2616
2617 if (aWait)
2618 {
2619 /* go to Creating state before starting the task */
2620 aTarget->m.state = MediaState_Creating;
2621
2622 rc = task->runNow();
2623 }
2624 else
2625 {
2626 rc = task->startThread();
2627 CheckComRCReturnRC(rc);
2628
2629 /* go to Creating state before leaving the lock */
2630 aTarget->m.state = MediaState_Creating;
2631 }
2632
2633 /* task is now owned (or already deleted) by taskThread() so release it */
2634 task.release();
2635
2636 if (aProgress != NULL)
2637 {
2638 /* return progress to the caller */
2639 *aProgress = progress;
2640 }
2641
2642 return rc;
2643}
2644
2645/**
2646 * Prepares this (source) hard disk, target hard disk and all intermediate hard
2647 * disks for the merge operation.
2648 *
2649 * This method is to be called prior to calling the #mergeTo() to perform
2650 * necessary consistency checks and place involved hard disks to appropriate
2651 * states. If #mergeTo() is not called or fails, the state modifications
2652 * performed by this method must be undone by #cancelMergeTo().
2653 *
2654 * Note that when @a aIgnoreAttachments is @c true then it's the caller's
2655 * responsibility to detach the source and all intermediate hard disks before
2656 * calling #mergeTo() (which will fail otherwise).
2657 *
2658 * See #mergeTo() for more information about merging.
2659 *
2660 * @param aTarget Target hard disk.
2661 * @param aChain Where to store the created merge chain.
2662 * @param aIgnoreAttachments Don't check if the source or any intermediate
2663 * hard disk is attached to any VM.
2664 *
2665 * @note Locks treeLock() for reading. Locks this object, aTarget and all
2666 * intermediate hard disks for writing.
2667 */
2668HRESULT HardDisk::prepareMergeTo(HardDisk *aTarget,
2669 MergeChain * &aChain,
2670 bool aIgnoreAttachments /*= false*/)
2671{
2672 AssertReturn(aTarget != NULL, E_FAIL);
2673
2674 AutoCaller autoCaller(this);
2675 AssertComRCReturnRC(autoCaller.rc());
2676
2677 AutoCaller targetCaller (aTarget);
2678 AssertComRCReturnRC(targetCaller.rc());
2679
2680 aChain = NULL;
2681
2682 /* we walk the tree */
2683 AutoReadLock treeLock (this->treeLock());
2684
2685 HRESULT rc = S_OK;
2686
2687 /* detect the merge direction */
2688 bool forward;
2689 {
2690 HardDisk *parent = mParent;
2691 while (parent != NULL && parent != aTarget)
2692 parent = parent->mParent;
2693 if (parent == aTarget)
2694 forward = false;
2695 else
2696 {
2697 parent = aTarget->mParent;
2698 while (parent != NULL && parent != this)
2699 parent = parent->mParent;
2700 if (parent == this)
2701 forward = true;
2702 else
2703 {
2704 Bstr tgtLoc;
2705 {
2706 AutoReadLock alock(this);
2707 tgtLoc = aTarget->locationFull();
2708 }
2709
2710 AutoReadLock alock(this);
2711 return setError (E_FAIL,
2712 tr ("Hard disks '%ls' and '%ls' are unrelated"),
2713 m.locationFull.raw(), tgtLoc.raw());
2714 }
2715 }
2716 }
2717
2718 /* build the chain (will do necessary checks and state changes) */
2719 std::auto_ptr <MergeChain> chain (new MergeChain (forward,
2720 aIgnoreAttachments));
2721 {
2722 HardDisk *last = forward ? aTarget : this;
2723 HardDisk *first = forward ? this : aTarget;
2724
2725 for (;;)
2726 {
2727 if (last == aTarget)
2728 rc = chain->addTarget (last);
2729 else if (last == this)
2730 rc = chain->addSource (last);
2731 else
2732 rc = chain->addIntermediate (last);
2733 CheckComRCReturnRC(rc);
2734
2735 if (last == first)
2736 break;
2737
2738 last = last->mParent;
2739 }
2740 }
2741
2742 aChain = chain.release();
2743
2744 return S_OK;
2745}
2746
2747/**
2748 * Merges this hard disk to the specified hard disk which must be either its
2749 * direct ancestor or descendant.
2750 *
2751 * Given this hard disk is SOURCE and the specified hard disk is TARGET, we will
2752 * get two varians of the merge operation:
2753 *
2754 * forward merge
2755 * ------------------------->
2756 * [Extra] <- SOURCE <- Intermediate <- TARGET
2757 * Any Del Del LockWr
2758 *
2759 *
2760 * backward merge
2761 * <-------------------------
2762 * TARGET <- Intermediate <- SOURCE <- [Extra]
2763 * LockWr Del Del LockWr
2764 *
2765 * Each scheme shows the involved hard disks on the hard disk chain where
2766 * SOURCE and TARGET belong. Under each hard disk there is a state value which
2767 * the hard disk must have at a time of the mergeTo() call.
2768 *
2769 * The hard disks in the square braces may be absent (e.g. when the forward
2770 * operation takes place and SOURCE is the base hard disk, or when the backward
2771 * merge operation takes place and TARGET is the last child in the chain) but if
2772 * they present they are involved too as shown.
2773 *
2774 * Nor the source hard disk neither intermediate hard disks may be attached to
2775 * any VM directly or in the snapshot, otherwise this method will assert.
2776 *
2777 * The #prepareMergeTo() method must be called prior to this method to place all
2778 * involved to necessary states and perform other consistency checks.
2779 *
2780 * If @a aWait is @c true then this method will perform the operation on the
2781 * calling thread and will not return to the caller until the operation is
2782 * completed. When this method succeeds, all intermediate hard disk objects in
2783 * the chain will be uninitialized, the state of the target hard disk (and all
2784 * involved extra hard disks) will be restored and @a aChain will be deleted.
2785 * Note that this (source) hard disk is not uninitialized because of possible
2786 * AutoCaller instances held by the caller of this method on the current thread.
2787 * It's therefore the responsibility of the caller to call HardDisk::uninit()
2788 * after releasing all callers in this case!
2789 *
2790 * If @a aWait is @c false then this method will crea,te a thread to perform the
2791 * create operation asynchronously and will return immediately. If the operation
2792 * succeeds, the thread will uninitialize the source hard disk object and all
2793 * intermediate hard disk objects in the chain, reset the state of the target
2794 * hard disk (and all involved extra hard disks) and delete @a aChain. If the
2795 * operation fails, the thread will only reset the states of all involved hard
2796 * disks and delete @a aChain.
2797 *
2798 * When this method fails (regardless of the @a aWait mode), it is a caller's
2799 * responsiblity to undo state changes and delete @a aChain using
2800 * #cancelMergeTo().
2801 *
2802 * If @a aProgress is not NULL but the object it points to is @c null then a new
2803 * progress object will be created and assigned to @a *aProgress on success,
2804 * otherwise the existing progress object is used. If Progress is NULL, then no
2805 * progress object is created/used at all. Note that @a aProgress cannot be
2806 * NULL when @a aWait is @c false (this method will assert in this case).
2807 *
2808 * @param aChain Merge chain created by #prepareMergeTo().
2809 * @param aProgress Where to find/store a Progress object to track operation
2810 * completion.
2811 * @param aWait @c true if this method should block instead of creating
2812 * an asynchronous thread.
2813 *
2814 * @note Locks the branch lock for writing. Locks the hard disks from the chain
2815 * for writing.
2816 */
2817HRESULT HardDisk::mergeTo(MergeChain *aChain,
2818 ComObjPtr<Progress> *aProgress,
2819 bool aWait)
2820{
2821 AssertReturn(aChain != NULL, E_FAIL);
2822 AssertReturn(aProgress != NULL || aWait == true, E_FAIL);
2823
2824 AutoCaller autoCaller(this);
2825 CheckComRCReturnRC(autoCaller.rc());
2826
2827 HRESULT rc = S_OK;
2828
2829 ComObjPtr<Progress> progress;
2830
2831 if (aProgress != NULL)
2832 {
2833 /* use the existing progress object... */
2834 progress = *aProgress;
2835
2836 /* ...but create a new one if it is null */
2837 if (progress.isNull())
2838 {
2839 AutoReadLock alock(this);
2840
2841 progress.createObject();
2842 rc = progress->init (mVirtualBox, static_cast<IHardDisk*>(this),
2843 BstrFmt (tr ("Merging hard disk '%s' to '%s'"),
2844 name().raw(), aChain->target()->name().raw()),
2845 TRUE /* aCancelable */);
2846 CheckComRCReturnRC(rc);
2847 }
2848 }
2849
2850 /* setup task object and thread to carry out the operation
2851 * asynchronously */
2852
2853 std::auto_ptr <Task> task (new Task (this, progress, Task::Merge));
2854 AssertComRCReturnRC(task->autoCaller.rc());
2855
2856 task->setData (aChain);
2857
2858 /* Note: task owns aChain (will delete it when not needed) in all cases
2859 * except when @a aWait is @c true and runNow() fails -- in this case
2860 * aChain will be left away because cancelMergeTo() will be applied by the
2861 * caller on it as it is required in the documentation above */
2862
2863 if (aWait)
2864 {
2865 rc = task->runNow();
2866 }
2867 else
2868 {
2869 rc = task->startThread();
2870 CheckComRCReturnRC(rc);
2871 }
2872
2873 /* task is now owned (or already deleted) by taskThread() so release it */
2874 task.release();
2875
2876 if (aProgress != NULL)
2877 {
2878 /* return progress to the caller */
2879 *aProgress = progress;
2880 }
2881
2882 return rc;
2883}
2884
2885/**
2886 * Undoes what #prepareMergeTo() did. Must be called if #mergeTo() is not called
2887 * or fails. Frees memory occupied by @a aChain.
2888 *
2889 * @param aChain Merge chain created by #prepareMergeTo().
2890 *
2891 * @note Locks the hard disks from the chain for writing.
2892 */
2893void HardDisk::cancelMergeTo (MergeChain *aChain)
2894{
2895 AutoCaller autoCaller(this);
2896 AssertComRCReturnVoid (autoCaller.rc());
2897
2898 AssertReturnVoid (aChain != NULL);
2899
2900 /* the destructor will do the thing */
2901 delete aChain;
2902}
2903
2904// private methods
2905////////////////////////////////////////////////////////////////////////////////
2906
2907/**
2908 * Sets the value of m.location and calculates the value of m.locationFull.
2909 *
2910 * Reimplements MediumBase::setLocation() to specially treat non-FS-path
2911 * locations and to prepend the default hard disk folder if the given location
2912 * string does not contain any path information at all.
2913 *
2914 * Also, if the specified location is a file path that ends with '/' then the
2915 * file name part will be generated by this method automatically in the format
2916 * '{<uuid>}.<ext>' where <uuid> is a fresh UUID that this method will generate
2917 * and assign to this medium, and <ext> is the default extension for this
2918 * medium's storage format. Note that this procedure requires the media state to
2919 * be NotCreated and will return a faiulre otherwise.
2920 *
2921 * @param aLocation Location of the storage unit. If the locaiton is a FS-path,
2922 * then it can be relative to the VirtualBox home directory.
2923 *
2924 * @note Must be called from under this object's write lock.
2925 */
2926HRESULT HardDisk::setLocation(const Utf8Str &aLocation)
2927{
2928 AssertReturn(!aLocation.isEmpty(), E_FAIL);
2929
2930 AutoCaller autoCaller(this);
2931 AssertComRCReturnRC(autoCaller.rc());
2932
2933 /* formatObj may be null only when initializing from an existing path and
2934 * no format is known yet */
2935 AssertReturn((!mm.format.isNull() && !mm.formatObj.isNull()) ||
2936 (autoCaller.state() == InInit &&
2937 m.state != MediaState_NotCreated && m.id.isEmpty() &&
2938 mm.format.isNull() && mm.formatObj.isNull()),
2939 E_FAIL);
2940
2941 /* are we dealing with a new hard disk constructed using the existing
2942 * location? */
2943 bool isImport = mm.format.isNull();
2944
2945 if (isImport ||
2946 (mm.formatObj->capabilities() & HardDiskFormatCapabilities_File))
2947 {
2948 Guid id;
2949
2950 Utf8Str location(aLocation);
2951
2952 if (m.state == MediaState_NotCreated)
2953 {
2954 /* must be a file (formatObj must be already known) */
2955 Assert (mm.formatObj->capabilities() & HardDiskFormatCapabilities_File);
2956
2957 if (RTPathFilename(location.c_str()) == NULL)
2958 {
2959 /* no file name is given (either an empty string or ends with a
2960 * slash), generate a new UUID + file name if the state allows
2961 * this */
2962
2963 ComAssertMsgRet(!mm.formatObj->fileExtensions().empty(),
2964 ("Must be at least one extension if it is HardDiskFormatCapabilities_File\n"),
2965 E_FAIL);
2966
2967 Bstr ext = mm.formatObj->fileExtensions().front();
2968 ComAssertMsgRet(!ext.isEmpty(),
2969 ("Default extension must not be empty\n"),
2970 E_FAIL);
2971
2972 id.create();
2973
2974 location = Utf8StrFmt("%s{%RTuuid}.%ls",
2975 location.raw(),
2976 id.raw(),
2977 ext.raw());
2978 }
2979 }
2980
2981 /* append the default folder if no path is given */
2982 if (!RTPathHavePath(location.c_str()))
2983 location = Utf8StrFmt("%s%c%s",
2984 mVirtualBox->getDefaultHardDiskFolder().raw(),
2985 RTPATH_DELIMITER,
2986 location.raw());
2987
2988 /* get the full file name */
2989 Utf8Str locationFull;
2990 int vrc = mVirtualBox->calculateFullPath (location, locationFull);
2991 if (RT_FAILURE(vrc))
2992 return setError (VBOX_E_FILE_ERROR,
2993 tr ("Invalid hard disk storage file location '%s' (%Rrc)"),
2994 location.raw(), vrc);
2995
2996 /* detect the backend from the storage unit if importing */
2997 if (isImport)
2998 {
2999 char *backendName = NULL;
3000
3001 /* is it a file? */
3002 {
3003 RTFILE file;
3004 vrc = RTFileOpen(&file, locationFull.c_str(), RTFILE_O_READ);
3005 if (RT_SUCCESS(vrc))
3006 RTFileClose (file);
3007 }
3008 if (RT_SUCCESS(vrc))
3009 {
3010 vrc = VDGetFormat(locationFull.c_str(), &backendName);
3011 }
3012 else if (vrc != VERR_FILE_NOT_FOUND && vrc != VERR_PATH_NOT_FOUND)
3013 {
3014 /* assume it's not a file, restore the original location */
3015 location = locationFull = aLocation;
3016 vrc = VDGetFormat(locationFull.c_str(), &backendName);
3017 }
3018
3019 if (RT_FAILURE(vrc))
3020 {
3021 if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
3022 return setError (VBOX_E_FILE_ERROR,
3023 tr ("Could not find file for the hard disk '%s' (%Rrc)"),
3024 locationFull.raw(), vrc);
3025 else
3026 return setError (VBOX_E_IPRT_ERROR,
3027 tr ("Could not get the storage format of the hard disk '%s' (%Rrc)"),
3028 locationFull.raw(), vrc);
3029 }
3030
3031 ComAssertRet (backendName != NULL && *backendName != '\0', E_FAIL);
3032
3033 HRESULT rc = setFormat (Bstr (backendName));
3034 RTStrFree (backendName);
3035
3036 /* setFormat() must not fail since we've just used the backend so
3037 * the format object must be there */
3038 AssertComRCReturnRC(rc);
3039 }
3040
3041 /* is it still a file? */
3042 if (mm.formatObj->capabilities() & HardDiskFormatCapabilities_File)
3043 {
3044 m.location = location;
3045 m.locationFull = locationFull;
3046
3047 if (m.state == MediaState_NotCreated)
3048 {
3049 /* assign a new UUID (this UUID will be used when calling
3050 * VDCreateBase/VDCreateDiff as a wanted UUID). Note that we
3051 * also do that if we didn't generate it to make sure it is
3052 * either generated by us or reset to null */
3053 unconst(m.id) = id;
3054 }
3055 }
3056 else
3057 {
3058 m.location = locationFull;
3059 m.locationFull = locationFull;
3060 }
3061 }
3062 else
3063 {
3064 m.location = aLocation;
3065 m.locationFull = aLocation;
3066 }
3067
3068 return S_OK;
3069}
3070
3071/**
3072 * Checks that the format ID is valid and sets it on success.
3073 *
3074 * Note that this method will caller-reference the format object on success!
3075 * This reference must be released somewhere to let the HardDiskFormat object be
3076 * uninitialized.
3077 *
3078 * @note Must be called from under this object's write lock.
3079 */
3080HRESULT HardDisk::setFormat (CBSTR aFormat)
3081{
3082 /* get the format object first */
3083 {
3084 AutoReadLock propsLock(mVirtualBox->systemProperties());
3085
3086 unconst(mm.formatObj)
3087 = mVirtualBox->systemProperties()->hardDiskFormat (aFormat);
3088 if (mm.formatObj.isNull())
3089 return setError (E_INVALIDARG,
3090 tr ("Invalid hard disk storage format '%ls'"), aFormat);
3091
3092 /* reference the format permanently to prevent its unexpected
3093 * uninitialization */
3094 HRESULT rc = mm.formatObj->addCaller();
3095 AssertComRCReturnRC(rc);
3096
3097 /* get properties (preinsert them as keys in the map). Note that the
3098 * map doesn't grow over the object life time since the set of
3099 * properties is meant to be constant. */
3100
3101 Assert (mm.properties.empty());
3102
3103 for (HardDiskFormat::PropertyList::const_iterator it =
3104 mm.formatObj->properties().begin();
3105 it != mm.formatObj->properties().end();
3106 ++ it)
3107 {
3108 mm.properties.insert (std::make_pair (it->name, Bstr::Null));
3109 }
3110 }
3111
3112 unconst(mm.format) = aFormat;
3113
3114 return S_OK;
3115}
3116
3117/**
3118 * Queries information from the image file.
3119 *
3120 * As a result of this call, the accessibility state and data members such as
3121 * size and description will be updated with the current information.
3122 *
3123 * Reimplements MediumBase::queryInfo() to query hard disk information using the
3124 * VD backend interface.
3125 *
3126 * @note This method may block during a system I/O call that checks storage
3127 * accessibility.
3128 *
3129 * @note Locks treeLock() for reading and writing (for new diff media checked
3130 * for the first time). Locks mParent for reading. Locks this object for
3131 * writing.
3132 */
3133HRESULT HardDisk::queryInfo()
3134{
3135 AutoWriteLock alock(this);
3136
3137 AssertReturn(m.state == MediaState_Created ||
3138 m.state == MediaState_Inaccessible ||
3139 m.state == MediaState_LockedRead ||
3140 m.state == MediaState_LockedWrite,
3141 E_FAIL);
3142
3143 HRESULT rc = S_OK;
3144
3145 int vrc = VINF_SUCCESS;
3146
3147 /* check if a blocking queryInfo() call is in progress on some other thread,
3148 * and wait for it to finish if so instead of querying data ourselves */
3149 if (m.queryInfoSem != NIL_RTSEMEVENTMULTI)
3150 {
3151 Assert (m.state == MediaState_LockedRead);
3152
3153 ++ m.queryInfoCallers;
3154 alock.leave();
3155
3156 vrc = RTSemEventMultiWait (m.queryInfoSem, RT_INDEFINITE_WAIT);
3157
3158 alock.enter();
3159 -- m.queryInfoCallers;
3160
3161 if (m.queryInfoCallers == 0)
3162 {
3163 /* last waiting caller deletes the semaphore */
3164 RTSemEventMultiDestroy (m.queryInfoSem);
3165 m.queryInfoSem = NIL_RTSEMEVENTMULTI;
3166 }
3167
3168 AssertRC (vrc);
3169
3170 return S_OK;
3171 }
3172
3173 /* lazily create a semaphore for possible callers */
3174 vrc = RTSemEventMultiCreate (&m.queryInfoSem);
3175 ComAssertRCRet (vrc, E_FAIL);
3176
3177 bool tempStateSet = false;
3178 if (m.state != MediaState_LockedRead &&
3179 m.state != MediaState_LockedWrite)
3180 {
3181 /* Cause other methods to prevent any modifications before leaving the
3182 * lock. Note that clients will never see this temporary state change
3183 * since any COMGETTER(State) is (or will be) blocked until we finish
3184 * and restore the actual state. */
3185 m.state = MediaState_LockedRead;
3186 tempStateSet = true;
3187 }
3188
3189 /* leave the lock before a blocking operation */
3190 alock.leave();
3191
3192 bool success = false;
3193 Utf8Str lastAccessError;
3194
3195 try
3196 {
3197 Utf8Str location (m.locationFull);
3198
3199 /* are we dealing with a new hard disk constructed using the existing
3200 * location? */
3201 bool isImport = m.id.isEmpty();
3202
3203 PVBOXHDD hdd;
3204 vrc = VDCreate (mm.vdDiskIfaces, &hdd);
3205 ComAssertRCThrow (vrc, E_FAIL);
3206
3207 try
3208 {
3209 unsigned flags = VD_OPEN_FLAGS_INFO;
3210
3211 /* Note that we don't use VD_OPEN_FLAGS_READONLY when opening new
3212 * hard disks because that would prevent necessary modifications
3213 * when opening hard disks of some third-party formats for the first
3214 * time in VirtualBox (such as VMDK for which VDOpen() needs to
3215 * generate an UUID if it is missing) */
3216 if ( (mm.hddOpenMode == OpenReadOnly)
3217 || !isImport
3218 )
3219 flags |= VD_OPEN_FLAGS_READONLY;
3220
3221 /** @todo This kind of opening of images is assuming that diff
3222 * images can be opened as base images. Should be fixed ASAP. */
3223 vrc = VDOpen(hdd,
3224 Utf8Str(mm.format).c_str(),
3225 location.c_str(),
3226 flags,
3227 mm.vdDiskIfaces);
3228 if (RT_FAILURE(vrc))
3229 {
3230 lastAccessError = Utf8StrFmt (
3231 tr ("Could not open the hard disk '%ls'%s"),
3232 m.locationFull.raw(), vdError (vrc).raw());
3233 throw S_OK;
3234 }
3235
3236 if (mm.formatObj->capabilities() & HardDiskFormatCapabilities_Uuid)
3237 {
3238 /* modify the UUIDs if necessary */
3239 if (mm.setImageId)
3240 {
3241 vrc = VDSetUuid(hdd, 0, mm.imageId);
3242 ComAssertRCThrow(vrc, E_FAIL);
3243 }
3244 if (mm.setParentId)
3245 {
3246 vrc = VDSetParentUuid(hdd, 0, mm.parentId);
3247 ComAssertRCThrow(vrc, E_FAIL);
3248 }
3249 /* zap the information, these are no long-term members */
3250 mm.setImageId = false;
3251 unconst(mm.imageId).clear();
3252 mm.setParentId = false;
3253 unconst(mm.parentId).clear();
3254
3255 /* check the UUID */
3256 RTUUID uuid;
3257 vrc = VDGetUuid(hdd, 0, &uuid);
3258 ComAssertRCThrow(vrc, E_FAIL);
3259
3260 if (isImport)
3261 {
3262 unconst(m.id) = uuid;
3263
3264 if (m.id.isEmpty() && (mm.hddOpenMode == OpenReadOnly))
3265 // only when importing a VDMK that has no UUID, create one in memory
3266 unconst(m.id).create();
3267 }
3268 else
3269 {
3270 Assert (!m.id.isEmpty());
3271
3272 if (m.id != uuid)
3273 {
3274 lastAccessError = Utf8StrFmt (
3275 tr ("UUID {%RTuuid} of the hard disk '%ls' does not match the value {%RTuuid} stored in the media registry ('%s')"),
3276 &uuid, m.locationFull.raw(), m.id.raw(),
3277 mVirtualBox->settingsFilePath().raw());
3278 throw S_OK;
3279 }
3280 }
3281 }
3282 else
3283 {
3284 /* the backend does not support storing UUIDs within the
3285 * underlying storage so use what we store in XML */
3286
3287 /* generate an UUID for an imported UUID-less hard disk */
3288 if (isImport)
3289 {
3290 if (mm.setImageId)
3291 unconst(m.id) = mm.imageId;
3292 else
3293 unconst(m.id).create();
3294 }
3295 }
3296
3297 /* check the type */
3298 unsigned uImageFlags;
3299 vrc = VDGetImageFlags (hdd, 0, &uImageFlags);
3300 ComAssertRCThrow (vrc, E_FAIL);
3301
3302 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
3303 {
3304 RTUUID parentId;
3305 vrc = VDGetParentUuid (hdd, 0, &parentId);
3306 ComAssertRCThrow (vrc, E_FAIL);
3307
3308 if (isImport)
3309 {
3310 /* the parent must be known to us. Note that we freely
3311 * call locking methods of mVirtualBox and parent from the
3312 * write lock (breaking the {parent,child} lock order)
3313 * because there may be no concurrent access to the just
3314 * opened hard disk on ther threads yet (and init() will
3315 * fail if this method reporst MediaState_Inaccessible) */
3316
3317 Guid id = parentId;
3318 ComObjPtr<HardDisk> parent;
3319 rc = mVirtualBox->findHardDisk(&id, NULL,
3320 false /* aSetError */,
3321 &parent);
3322 if (FAILED (rc))
3323 {
3324 lastAccessError = Utf8StrFmt (
3325 tr ("Parent hard disk with UUID {%RTuuid} of the hard disk '%ls' is not found in the media registry ('%s')"),
3326 &parentId, m.locationFull.raw(),
3327 mVirtualBox->settingsFilePath().raw());
3328 throw S_OK;
3329 }
3330
3331 /* deassociate from VirtualBox, associate with parent */
3332
3333 mVirtualBox->removeDependentChild (this);
3334
3335 /* we set mParent & children() */
3336 AutoWriteLock treeLock (this->treeLock());
3337
3338 Assert (mParent.isNull());
3339 mParent = parent;
3340 mParent->addDependentChild (this);
3341 }
3342 else
3343 {
3344 /* we access mParent */
3345 AutoReadLock treeLock (this->treeLock());
3346
3347 /* check that parent UUIDs match. Note that there's no need
3348 * for the parent's AutoCaller (our lifetime is bound to
3349 * it) */
3350
3351 if (mParent.isNull())
3352 {
3353 lastAccessError = Utf8StrFmt (
3354 tr ("Hard disk '%ls' is differencing but it is not associated with any parent hard disk in the media registry ('%s')"),
3355 m.locationFull.raw(),
3356 mVirtualBox->settingsFilePath().raw());
3357 throw S_OK;
3358 }
3359
3360 AutoReadLock parentLock (mParent);
3361 if (mParent->state() != MediaState_Inaccessible &&
3362 mParent->id() != parentId)
3363 {
3364 lastAccessError = Utf8StrFmt (
3365 tr ("Parent UUID {%RTuuid} of the hard disk '%ls' does not match UUID {%RTuuid} of its parent hard disk stored in the media registry ('%s')"),
3366 &parentId, m.locationFull.raw(),
3367 mParent->id().raw(),
3368 mVirtualBox->settingsFilePath().raw());
3369 throw S_OK;
3370 }
3371
3372 /// @todo NEWMEDIA what to do if the parent is not
3373 /// accessible while the diff is? Probably, nothing. The
3374 /// real code will detect the mismatch anyway.
3375 }
3376 }
3377
3378 m.size = VDGetFileSize (hdd, 0);
3379 mm.logicalSize = VDGetSize (hdd, 0) / _1M;
3380
3381 success = true;
3382 }
3383 catch (HRESULT aRC)
3384 {
3385 rc = aRC;
3386 }
3387
3388 VDDestroy (hdd);
3389
3390 }
3391 catch (HRESULT aRC)
3392 {
3393 rc = aRC;
3394 }
3395
3396 alock.enter();
3397
3398 if (success)
3399 m.lastAccessError.setNull();
3400 else
3401 {
3402 m.lastAccessError = lastAccessError;
3403 LogWarningFunc (("'%ls' is not accessible (error='%ls', rc=%Rhrc, vrc=%Rrc)\n",
3404 m.locationFull.raw(), m.lastAccessError.raw(),
3405 rc, vrc));
3406 }
3407
3408 /* inform other callers if there are any */
3409 if (m.queryInfoCallers > 0)
3410 {
3411 RTSemEventMultiSignal (m.queryInfoSem);
3412 }
3413 else
3414 {
3415 /* delete the semaphore ourselves */
3416 RTSemEventMultiDestroy (m.queryInfoSem);
3417 m.queryInfoSem = NIL_RTSEMEVENTMULTI;
3418 }
3419
3420 if (tempStateSet)
3421 {
3422 /* Set the proper state according to the result of the check */
3423 if (success)
3424 m.state = MediaState_Created;
3425 else
3426 m.state = MediaState_Inaccessible;
3427 }
3428 else
3429 {
3430 /* we're locked, use a special field to store the result */
3431 m.accessibleInLock = success;
3432 }
3433
3434 return rc;
3435}
3436
3437/**
3438 * @note Called from this object's AutoMayUninitSpan and from under mVirtualBox
3439 * write lock.
3440 *
3441 * @note Also reused by HardDisk::Reset().
3442 *
3443 * @note Locks treeLock() for reading.
3444 */
3445HRESULT HardDisk::canClose()
3446{
3447 /* we access children */
3448 AutoReadLock treeLock (this->treeLock());
3449
3450 if (children().size() != 0)
3451 return setError (E_FAIL,
3452 tr ("Hard disk '%ls' has %d child hard disks"),
3453 children().size());
3454
3455 return S_OK;
3456}
3457
3458/**
3459 * @note Called from within this object's AutoWriteLock.
3460 */
3461HRESULT HardDisk::canAttach(const Guid & /* aMachineId */,
3462 const Guid & /* aSnapshotId */)
3463{
3464 if (mm.numCreateDiffTasks > 0)
3465 return setError (E_FAIL,
3466 tr ("One or more differencing child hard disks are being created for the hard disk '%ls' (%u)"),
3467 m.locationFull.raw(), mm.numCreateDiffTasks);
3468
3469 return S_OK;
3470}
3471
3472/**
3473 * @note Called from within this object's AutoMayUninitSpan (or AutoCaller) and
3474 * from under mVirtualBox write lock.
3475 *
3476 * @note Locks treeLock() for writing.
3477 */
3478HRESULT HardDisk::unregisterWithVirtualBox()
3479{
3480 /* Note that we need to de-associate ourselves from the parent to let
3481 * unregisterHardDisk() properly save the registry */
3482
3483 /* we modify mParent and access children */
3484 AutoWriteLock treeLock (this->treeLock());
3485
3486 const ComObjPtr<HardDisk, ComWeakRef> parent = mParent;
3487
3488 AssertReturn(children().size() == 0, E_FAIL);
3489
3490 if (!mParent.isNull())
3491 {
3492 /* deassociate from the parent, associate with VirtualBox */
3493 mVirtualBox->addDependentChild (this);
3494 mParent->removeDependentChild (this);
3495 mParent.setNull();
3496 }
3497
3498 HRESULT rc = mVirtualBox->unregisterHardDisk(this);
3499
3500 if (FAILED (rc))
3501 {
3502 if (!parent.isNull())
3503 {
3504 /* re-associate with the parent as we are still relatives in the
3505 * registry */
3506 mParent = parent;
3507 mParent->addDependentChild (this);
3508 mVirtualBox->removeDependentChild (this);
3509 }
3510 }
3511
3512 return rc;
3513}
3514
3515/**
3516 * Returns the last error message collected by the vdErrorCall callback and
3517 * resets it.
3518 *
3519 * The error message is returned prepended with a dot and a space, like this:
3520 * <code>
3521 * ". <error_text> (%Rrc)"
3522 * </code>
3523 * to make it easily appendable to a more general error message. The @c %Rrc
3524 * format string is given @a aVRC as an argument.
3525 *
3526 * If there is no last error message collected by vdErrorCall or if it is a
3527 * null or empty string, then this function returns the following text:
3528 * <code>
3529 * " (%Rrc)"
3530 * </code>
3531 *
3532 * @note Doesn't do any object locking; it is assumed that the caller makes sure
3533 * the callback isn't called by more than one thread at a time.
3534 *
3535 * @param aVRC VBox error code to use when no error message is provided.
3536 */
3537Utf8Str HardDisk::vdError (int aVRC)
3538{
3539 Utf8Str error;
3540
3541 if (mm.vdError.isEmpty())
3542 error = Utf8StrFmt (" (%Rrc)", aVRC);
3543 else
3544 error = Utf8StrFmt (".\n%s", mm.vdError.raw());
3545
3546 mm.vdError.setNull();
3547
3548 return error;
3549}
3550
3551/**
3552 * Error message callback.
3553 *
3554 * Puts the reported error message to the mm.vdError field.
3555 *
3556 * @note Doesn't do any object locking; it is assumed that the caller makes sure
3557 * the callback isn't called by more than one thread at a time.
3558 *
3559 * @param pvUser The opaque data passed on container creation.
3560 * @param rc The VBox error code.
3561 * @param RT_SRC_POS_DECL Use RT_SRC_POS.
3562 * @param pszFormat Error message format string.
3563 * @param va Error message arguments.
3564 */
3565/*static*/
3566DECLCALLBACK(void) HardDisk::vdErrorCall(void *pvUser, int rc, RT_SRC_POS_DECL,
3567 const char *pszFormat, va_list va)
3568{
3569 NOREF(pszFile); NOREF(iLine); NOREF(pszFunction); /* RT_SRC_POS_DECL */
3570
3571 HardDisk *that = static_cast<HardDisk*>(pvUser);
3572 AssertReturnVoid (that != NULL);
3573
3574 if (that->mm.vdError.isEmpty())
3575 that->mm.vdError =
3576 Utf8StrFmt ("%s (%Rrc)", Utf8StrFmtVA (pszFormat, va).raw(), rc);
3577 else
3578 that->mm.vdError =
3579 Utf8StrFmt ("%s.\n%s (%Rrc)", that->mm.vdError.raw(),
3580 Utf8StrFmtVA (pszFormat, va).raw(), rc);
3581}
3582
3583/**
3584 * PFNVMPROGRESS callback handler for Task operations.
3585 *
3586 * @param uPercent Completetion precentage (0-100).
3587 * @param pvUser Pointer to the Progress instance.
3588 */
3589/*static*/
3590DECLCALLBACK(int) HardDisk::vdProgressCall(PVM /* pVM */, unsigned uPercent,
3591 void *pvUser)
3592{
3593 HardDisk *that = static_cast<HardDisk*>(pvUser);
3594 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
3595
3596 if (that->mm.vdProgress != NULL)
3597 {
3598 /* update the progress object, capping it at 99% as the final percent
3599 * is used for additional operations like setting the UUIDs and similar. */
3600 HRESULT rc = that->mm.vdProgress->setCurrentOperationProgress(uPercent * 99 / 100);
3601 if (FAILED(rc))
3602 {
3603 if (rc == E_FAIL)
3604 return VERR_CANCELLED;
3605 else
3606 return VERR_INVALID_STATE;
3607 }
3608 }
3609
3610 return VINF_SUCCESS;
3611}
3612
3613/* static */
3614DECLCALLBACK(bool) HardDisk::vdConfigAreKeysValid (void *pvUser,
3615 const char * /* pszzValid */)
3616{
3617 HardDisk *that = static_cast<HardDisk*>(pvUser);
3618 AssertReturn(that != NULL, false);
3619
3620 /* we always return true since the only keys we have are those found in
3621 * VDBACKENDINFO */
3622 return true;
3623}
3624
3625/* static */
3626DECLCALLBACK(int) HardDisk::vdConfigQuerySize(void *pvUser, const char *pszName,
3627 size_t *pcbValue)
3628{
3629 AssertReturn(VALID_PTR (pcbValue), VERR_INVALID_POINTER);
3630
3631 HardDisk *that = static_cast<HardDisk*>(pvUser);
3632 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
3633
3634 Data::PropertyMap::const_iterator it =
3635 that->mm.properties.find (Bstr (pszName));
3636 if (it == that->mm.properties.end())
3637 return VERR_CFGM_VALUE_NOT_FOUND;
3638
3639 /* we interpret null values as "no value" in HardDisk */
3640 if (it->second.isNull())
3641 return VERR_CFGM_VALUE_NOT_FOUND;
3642
3643 *pcbValue = it->second.length() + 1 /* include terminator */;
3644
3645 return VINF_SUCCESS;
3646}
3647
3648/* static */
3649DECLCALLBACK(int) HardDisk::vdConfigQuery(void *pvUser, const char *pszName,
3650 char *pszValue, size_t cchValue)
3651{
3652 AssertReturn(VALID_PTR (pszValue), VERR_INVALID_POINTER);
3653
3654 HardDisk *that = static_cast<HardDisk*>(pvUser);
3655 AssertReturn(that != NULL, VERR_GENERAL_FAILURE);
3656
3657 Data::PropertyMap::const_iterator it =
3658 that->mm.properties.find (Bstr (pszName));
3659 if (it == that->mm.properties.end())
3660 return VERR_CFGM_VALUE_NOT_FOUND;
3661
3662 Utf8Str value = it->second;
3663 if (value.length() >= cchValue)
3664 return VERR_CFGM_NOT_ENOUGH_SPACE;
3665
3666 /* we interpret null values as "no value" in HardDisk */
3667 if (it->second.isNull())
3668 return VERR_CFGM_VALUE_NOT_FOUND;
3669
3670 memcpy(pszValue, value.c_str(), value.length() + 1);
3671
3672 return VINF_SUCCESS;
3673}
3674
3675/**
3676 * Thread function for time-consuming tasks.
3677 *
3678 * The Task structure passed to @a pvUser must be allocated using new and will
3679 * be freed by this method before it returns.
3680 *
3681 * @param pvUser Pointer to the Task instance.
3682 */
3683/* static */
3684DECLCALLBACK(int) HardDisk::taskThread (RTTHREAD thread, void *pvUser)
3685{
3686 std::auto_ptr <Task> task (static_cast <Task *> (pvUser));
3687 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
3688
3689 bool isAsync = thread != NIL_RTTHREAD;
3690
3691 HardDisk *that = task->that;
3692
3693 /// @todo ugly hack, fix ComAssert... later
3694 #define setError that->setError
3695
3696 /* Note: no need in AutoCaller because Task does that */
3697
3698 LogFlowFuncEnter();
3699 LogFlowFunc (("{%p}: operation=%d\n", that, task->operation));
3700
3701 HRESULT rc = S_OK;
3702
3703 switch (task->operation)
3704 {
3705 ////////////////////////////////////////////////////////////////////////
3706
3707 case Task::CreateBase:
3708 {
3709 /* The lock is also used as a signal from the task initiator (which
3710 * releases it only after RTThreadCreate()) that we can start the job */
3711 AutoWriteLock thatLock (that);
3712
3713 /* these parameters we need after creation */
3714 uint64_t size = 0, logicalSize = 0;
3715
3716 /* The object may request a specific UUID (through a special form of
3717 * the setLocation() argument). Otherwise we have to generate it */
3718 Guid id = that->m.id;
3719 bool generateUuid = id.isEmpty();
3720 if (generateUuid)
3721 {
3722 id.create();
3723 /* VirtualBox::registerHardDisk() will need UUID */
3724 unconst(that->m.id) = id;
3725 }
3726
3727 try
3728 {
3729 PVBOXHDD hdd;
3730 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
3731 ComAssertRCThrow (vrc, E_FAIL);
3732
3733 Utf8Str format (that->mm.format);
3734 Utf8Str location (that->m.locationFull);
3735 /* uint64_t capabilities = */ that->mm.formatObj->capabilities();
3736
3737 /* unlock before the potentially lengthy operation */
3738 Assert (that->m.state == MediaState_Creating);
3739 thatLock.leave();
3740
3741 try
3742 {
3743 /* ensure the directory exists */
3744 rc = VirtualBox::ensureFilePathExists(location.c_str());
3745 CheckComRCThrowRC (rc);
3746
3747 PDMMEDIAGEOMETRY geo = { 0 }; /* auto-detect */
3748
3749 /* needed for vdProgressCallback */
3750 that->mm.vdProgress = task->progress;
3751
3752 vrc = VDCreateBase(hdd,
3753 format.c_str(),
3754 location.c_str(),
3755 task->d.size * _1M,
3756 task->d.variant,
3757 NULL,
3758 &geo,
3759 &geo,
3760 id.raw(),
3761 VD_OPEN_FLAGS_NORMAL,
3762 NULL,
3763 that->mm.vdDiskIfaces);
3764
3765 if (RT_FAILURE(vrc))
3766 {
3767 throw setError (E_FAIL,
3768 tr ("Could not create the hard disk storage unit '%s'%s"),
3769 location.raw(), that->vdError (vrc).raw());
3770 }
3771
3772 size = VDGetFileSize (hdd, 0);
3773 logicalSize = VDGetSize (hdd, 0) / _1M;
3774 }
3775 catch (HRESULT aRC) { rc = aRC; }
3776
3777 VDDestroy (hdd);
3778 }
3779 catch (HRESULT aRC) { rc = aRC; }
3780
3781 if (SUCCEEDED(rc))
3782 {
3783 /* register with mVirtualBox as the last step and move to
3784 * Created state only on success (leaving an orphan file is
3785 * better than breaking media registry consistency) */
3786 rc = that->mVirtualBox->registerHardDisk(that);
3787 }
3788
3789 thatLock.maybeEnter();
3790
3791 if (SUCCEEDED(rc))
3792 {
3793 that->m.state = MediaState_Created;
3794
3795 that->m.size = size;
3796 that->mm.logicalSize = logicalSize;
3797 }
3798 else
3799 {
3800 /* back to NotCreated on failure */
3801 that->m.state = MediaState_NotCreated;
3802
3803 /* reset UUID to prevent it from being reused next time */
3804 if (generateUuid)
3805 unconst(that->m.id).clear();
3806 }
3807
3808 break;
3809 }
3810
3811 ////////////////////////////////////////////////////////////////////////
3812
3813 case Task::CreateDiff:
3814 {
3815 ComObjPtr<HardDisk> &target = task->d.target;
3816
3817 /* Lock both in {parent,child} order. The lock is also used as a
3818 * signal from the task initiator (which releases it only after
3819 * RTThreadCreate()) that we can start the job*/
3820 AutoMultiWriteLock2 thatLock (that, target);
3821
3822 uint64_t size = 0, logicalSize = 0;
3823
3824 /* The object may request a specific UUID (through a special form of
3825 * the setLocation() argument). Otherwise we have to generate it */
3826 Guid targetId = target->m.id;
3827 bool generateUuid = targetId.isEmpty();
3828 if (generateUuid)
3829 {
3830 targetId.create();
3831 /* VirtualBox::registerHardDisk() will need UUID */
3832 unconst(target->m.id) = targetId;
3833 }
3834
3835 try
3836 {
3837 PVBOXHDD hdd;
3838 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
3839 ComAssertRCThrow (vrc, E_FAIL);
3840
3841 Guid id = that->m.id;
3842 Utf8Str format (that->mm.format);
3843 Utf8Str location (that->m.locationFull);
3844
3845 Utf8Str targetFormat (target->mm.format);
3846 Utf8Str targetLocation (target->m.locationFull);
3847
3848 Assert (target->m.state == MediaState_Creating);
3849
3850 /* Note: MediaState_LockedWrite is ok when taking an online
3851 * snapshot */
3852 Assert (that->m.state == MediaState_LockedRead ||
3853 that->m.state == MediaState_LockedWrite);
3854
3855 /* unlock before the potentially lengthy operation */
3856 thatLock.leave();
3857
3858 try
3859 {
3860 vrc = VDOpen(hdd,
3861 format.c_str(),
3862 location.c_str(),
3863 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
3864 that->mm.vdDiskIfaces);
3865 if (RT_FAILURE(vrc))
3866 {
3867 throw setError(E_FAIL,
3868 tr("Could not open the hard disk storage unit '%s'%s"),
3869 location.raw(), that->vdError (vrc).raw());
3870 }
3871
3872 /* ensure the target directory exists */
3873 rc = VirtualBox::ensureFilePathExists (targetLocation);
3874 CheckComRCThrowRC (rc);
3875
3876 /* needed for vdProgressCallback */
3877 that->mm.vdProgress = task->progress;
3878
3879 /** @todo add VD_IMAGE_FLAGS_DIFF to the image flags, to
3880 * be on the safe side. */
3881 vrc = VDCreateDiff(hdd,
3882 targetFormat.c_str(),
3883 targetLocation.c_str(),
3884 task->d.variant,
3885 NULL,
3886 targetId.raw(),
3887 id.raw(),
3888 VD_OPEN_FLAGS_NORMAL,
3889 target->mm.vdDiskIfaces,
3890 that->mm.vdDiskIfaces);
3891
3892 that->mm.vdProgress = NULL;
3893
3894 if (RT_FAILURE(vrc))
3895 {
3896 throw setError (E_FAIL,
3897 tr ("Could not create the differencing hard disk storage unit '%s'%s"),
3898 targetLocation.raw(), that->vdError (vrc).raw());
3899 }
3900
3901 size = VDGetFileSize (hdd, 1);
3902 logicalSize = VDGetSize (hdd, 1) / _1M;
3903 }
3904 catch (HRESULT aRC) { rc = aRC; }
3905
3906 VDDestroy (hdd);
3907 }
3908 catch (HRESULT aRC) { rc = aRC; }
3909
3910 if (SUCCEEDED(rc))
3911 {
3912 /* we set mParent & children() (note that thatLock is released
3913 * here), but lock VirtualBox first to follow the rule */
3914 AutoMultiWriteLock2 alock (that->mVirtualBox->lockHandle(),
3915 that->treeLock());
3916
3917 Assert (target->mParent.isNull());
3918
3919 /* associate the child with the parent and deassociate from
3920 * VirtualBox */
3921 target->mParent = that;
3922 that->addDependentChild (target);
3923 target->mVirtualBox->removeDependentChild (target);
3924
3925 /* diffs for immutable hard disks are auto-reset by default */
3926 target->mm.autoReset =
3927 that->root()->mm.type == HardDiskType_Immutable ?
3928 TRUE : FALSE;
3929
3930 /* register with mVirtualBox as the last step and move to
3931 * Created state only on success (leaving an orphan file is
3932 * better than breaking media registry consistency) */
3933 rc = that->mVirtualBox->registerHardDisk (target);
3934
3935 if (FAILED (rc))
3936 {
3937 /* break the parent association on failure to register */
3938 target->mVirtualBox->addDependentChild (target);
3939 that->removeDependentChild (target);
3940 target->mParent.setNull();
3941 }
3942 }
3943
3944 thatLock.maybeEnter();
3945
3946 if (SUCCEEDED(rc))
3947 {
3948 target->m.state = MediaState_Created;
3949
3950 target->m.size = size;
3951 target->mm.logicalSize = logicalSize;
3952 }
3953 else
3954 {
3955 /* back to NotCreated on failure */
3956 target->m.state = MediaState_NotCreated;
3957
3958 target->mm.autoReset = FALSE;
3959
3960 /* reset UUID to prevent it from being reused next time */
3961 if (generateUuid)
3962 unconst(target->m.id).clear();
3963 }
3964
3965 if (isAsync)
3966 {
3967 /* unlock ourselves when done (unless in MediaState_LockedWrite
3968 * state because of taking the online snapshot*/
3969 if (that->m.state != MediaState_LockedWrite)
3970 {
3971 HRESULT rc2 = that->UnlockRead (NULL);
3972 AssertComRC (rc2);
3973 }
3974 }
3975
3976 /* deregister the task registered in createDiffStorage() */
3977 Assert (that->mm.numCreateDiffTasks != 0);
3978 --that->mm.numCreateDiffTasks;
3979
3980 /* Note that in sync mode, it's the caller's responsibility to
3981 * unlock the hard disk */
3982
3983 break;
3984 }
3985
3986 ////////////////////////////////////////////////////////////////////////
3987
3988 case Task::Merge:
3989 {
3990 /* The lock is also used as a signal from the task initiator (which
3991 * releases it only after RTThreadCreate()) that we can start the
3992 * job. We don't actually need the lock for anything else since the
3993 * object is protected by MediaState_Deleting and we don't modify
3994 * its sensitive fields below */
3995 {
3996 AutoWriteLock thatLock (that);
3997 }
3998
3999 MergeChain *chain = task->d.chain.get();
4000
4001#if 0
4002 LogFlow (("*** MERGE forward = %RTbool\n", chain->isForward()));
4003#endif
4004
4005 try
4006 {
4007 PVBOXHDD hdd;
4008 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
4009 ComAssertRCThrow (vrc, E_FAIL);
4010
4011 try
4012 {
4013 /* Open all hard disks in the chain (they are in the
4014 * {parent,child} order in there. Note that we don't lock
4015 * objects in this chain since they must be in states
4016 * (Deleting and LockedWrite) that prevent from changing
4017 * their format and location fields from outside. */
4018
4019 for (MergeChain::const_iterator it = chain->begin();
4020 it != chain->end(); ++ it)
4021 {
4022 /* complex sanity (sane complexity) */
4023 Assert ((chain->isForward() &&
4024 ((*it != chain->back() &&
4025 (*it)->m.state == MediaState_Deleting) ||
4026 (*it == chain->back() &&
4027 (*it)->m.state == MediaState_LockedWrite))) ||
4028 (!chain->isForward() &&
4029 ((*it != chain->front() &&
4030 (*it)->m.state == MediaState_Deleting) ||
4031 (*it == chain->front() &&
4032 (*it)->m.state == MediaState_LockedWrite))));
4033
4034 Assert (*it == chain->target() ||
4035 (*it)->m.backRefs.size() == 0);
4036
4037 /* open the first image with VDOPEN_FLAGS_INFO because
4038 * it's not necessarily the base one */
4039 vrc = VDOpen(hdd,
4040 Utf8Str((*it)->mm.format).c_str(),
4041 Utf8Str((*it)->m.locationFull).c_str(),
4042 it == chain->begin() ?
4043 VD_OPEN_FLAGS_INFO : 0,
4044 (*it)->mm.vdDiskIfaces);
4045 if (RT_FAILURE(vrc))
4046 throw vrc;
4047#if 0
4048 LogFlow (("*** MERGE disk = %ls\n",
4049 (*it)->m.locationFull.raw()));
4050#endif
4051 }
4052
4053 /* needed for vdProgressCallback */
4054 that->mm.vdProgress = task->progress;
4055
4056 unsigned start = chain->isForward() ?
4057 0 : (unsigned)chain->size() - 1;
4058 unsigned end = chain->isForward() ?
4059 (unsigned)chain->size() - 1 : 0;
4060#if 0
4061 LogFlow (("*** MERGE from %d to %d\n", start, end));
4062#endif
4063 vrc = VDMerge (hdd, start, end, that->mm.vdDiskIfaces);
4064
4065 that->mm.vdProgress = NULL;
4066
4067 if (RT_FAILURE(vrc))
4068 throw vrc;
4069
4070 /* update parent UUIDs */
4071 /// @todo VDMerge should be taught to do so, including the
4072 /// multiple children case
4073 if (chain->isForward())
4074 {
4075 /* target's UUID needs to be updated (note that target
4076 * is the only image in the container on success) */
4077 vrc = VDSetParentUuid (hdd, 0, chain->parent()->m.id);
4078 if (RT_FAILURE(vrc))
4079 throw vrc;
4080 }
4081 else
4082 {
4083 /* we need to update UUIDs of all source's children
4084 * which cannot be part of the container at once so
4085 * add each one in there individually */
4086 if (chain->children().size() > 0)
4087 {
4088 for (List::const_iterator it = chain->children().begin();
4089 it != chain->children().end(); ++ it)
4090 {
4091 /* VD_OPEN_FLAGS_INFO since UUID is wrong yet */
4092 vrc = VDOpen(hdd,
4093 Utf8Str((*it)->mm.format).c_str(),
4094 Utf8Str((*it)->m.locationFull).c_str(),
4095 VD_OPEN_FLAGS_INFO,
4096 (*it)->mm.vdDiskIfaces);
4097 if (RT_FAILURE(vrc))
4098 throw vrc;
4099
4100 vrc = VDSetParentUuid (hdd, 1,
4101 chain->target()->m.id);
4102 if (RT_FAILURE(vrc))
4103 throw vrc;
4104
4105 vrc = VDClose (hdd, false /* fDelete */);
4106 if (RT_FAILURE(vrc))
4107 throw vrc;
4108 }
4109 }
4110 }
4111 }
4112 catch (HRESULT aRC) { rc = aRC; }
4113 catch (int aVRC)
4114 {
4115 throw setError (E_FAIL,
4116 tr ("Could not merge the hard disk '%ls' to '%ls'%s"),
4117 chain->source()->m.locationFull.raw(),
4118 chain->target()->m.locationFull.raw(),
4119 that->vdError (aVRC).raw());
4120 }
4121
4122 VDDestroy (hdd);
4123 }
4124 catch (HRESULT aRC) { rc = aRC; }
4125
4126 HRESULT rc2;
4127
4128 bool saveSettingsFailed = false;
4129
4130 if (SUCCEEDED(rc))
4131 {
4132 /* all hard disks but the target were successfully deleted by
4133 * VDMerge; reparent the last one and uninitialize deleted */
4134
4135 /* we set mParent & children() (note that thatLock is released
4136 * here), but lock VirtualBox first to follow the rule */
4137 AutoMultiWriteLock2 alock (that->mVirtualBox->lockHandle(),
4138 that->treeLock());
4139
4140 HardDisk *source = chain->source();
4141 HardDisk *target = chain->target();
4142
4143 if (chain->isForward())
4144 {
4145 /* first, unregister the target since it may become a base
4146 * hard disk which needs re-registration */
4147 rc2 = target->mVirtualBox->
4148 unregisterHardDisk (target, false /* aSaveSettings */);
4149 AssertComRC (rc2);
4150
4151 /* then, reparent it and disconnect the deleted branch at
4152 * both ends (chain->parent() is source's parent) */
4153 target->mParent->removeDependentChild (target);
4154 target->mParent = chain->parent();
4155 if (!target->mParent.isNull())
4156 {
4157 target->mParent->addDependentChild (target);
4158 target->mParent->removeDependentChild (source);
4159 source->mParent.setNull();
4160 }
4161 else
4162 {
4163 target->mVirtualBox->addDependentChild (target);
4164 target->mVirtualBox->removeDependentChild (source);
4165 }
4166
4167 /* then, register again */
4168 rc2 = target->mVirtualBox->
4169 registerHardDisk (target, false /* aSaveSettings */);
4170 AssertComRC (rc2);
4171 }
4172 else
4173 {
4174 Assert (target->children().size() == 1);
4175 HardDisk *targetChild = target->children().front();
4176
4177 /* disconnect the deleted branch at the elder end */
4178 target->removeDependentChild (targetChild);
4179 targetChild->mParent.setNull();
4180
4181 const List &children = chain->children();
4182
4183 /* reparent source's chidren and disconnect the deleted
4184 * branch at the younger end m*/
4185 if (children.size() > 0)
4186 {
4187 /* obey {parent,child} lock order */
4188 AutoWriteLock sourceLock (source);
4189
4190 for (List::const_iterator it = children.begin();
4191 it != children.end(); ++ it)
4192 {
4193 AutoWriteLock childLock (*it);
4194
4195 (*it)->mParent = target;
4196 (*it)->mParent->addDependentChild (*it);
4197 source->removeDependentChild (*it);
4198 }
4199 }
4200 }
4201
4202 /* try to save the hard disk registry */
4203 rc = that->mVirtualBox->saveSettings();
4204
4205 if (SUCCEEDED(rc))
4206 {
4207 /* unregister and uninitialize all hard disks in the chain
4208 * but the target */
4209
4210 for (MergeChain::iterator it = chain->begin();
4211 it != chain->end();)
4212 {
4213 if (*it == chain->target())
4214 {
4215 ++ it;
4216 continue;
4217 }
4218
4219 rc2 = (*it)->mVirtualBox->
4220 unregisterHardDisk(*it, false /* aSaveSettings */);
4221 AssertComRC (rc2);
4222
4223 /* now, uninitialize the deleted hard disk (note that
4224 * due to the Deleting state, uninit() will not touch
4225 * the parent-child relationship so we need to
4226 * uninitialize each disk individually) */
4227
4228 /* note that the operation initiator hard disk (which is
4229 * normally also the source hard disk) is a special case
4230 * -- there is one more caller added by Task to it which
4231 * we must release. Also, if we are in sync mode, the
4232 * caller may still hold an AutoCaller instance for it
4233 * and therefore we cannot uninit() it (it's therefore
4234 * the caller's responsibility) */
4235 if (*it == that)
4236 task->autoCaller.release();
4237
4238 /* release the caller added by MergeChain before
4239 * uninit() */
4240 (*it)->releaseCaller();
4241
4242 if (isAsync || *it != that)
4243 (*it)->uninit();
4244
4245 /* delete (to prevent uninitialization in MergeChain
4246 * dtor) and advance to the next item */
4247 it = chain->erase (it);
4248 }
4249
4250 /* Note that states of all other hard disks (target, parent,
4251 * children) will be restored by the MergeChain dtor */
4252 }
4253 else
4254 {
4255 /* too bad if we fail, but we'll need to rollback everything
4256 * we did above to at least keep the HD tree in sync with
4257 * the current registry on disk */
4258
4259 saveSettingsFailed = true;
4260
4261 /// @todo NEWMEDIA implement a proper undo
4262
4263 AssertFailed();
4264 }
4265 }
4266
4267 if (FAILED (rc))
4268 {
4269 /* Here we come if either VDMerge() failed (in which case we
4270 * assume that it tried to do everything to make a further
4271 * retry possible -- e.g. not deleted intermediate hard disks
4272 * and so on) or VirtualBox::saveSettings() failed (where we
4273 * should have the original tree but with intermediate storage
4274 * units deleted by VDMerge()). We have to only restore states
4275 * (through the MergeChain dtor) unless we are run synchronously
4276 * in which case it's the responsibility of the caller as stated
4277 * in the mergeTo() docs. The latter also implies that we
4278 * don't own the merge chain, so release it in this case. */
4279
4280 if (!isAsync)
4281 task->d.chain.release();
4282
4283 NOREF (saveSettingsFailed);
4284 }
4285
4286 break;
4287 }
4288
4289 ////////////////////////////////////////////////////////////////////////
4290
4291 case Task::Clone:
4292 {
4293 ComObjPtr<HardDisk> &target = task->d.target;
4294 ComObjPtr<HardDisk> &parent = task->d.parentDisk;
4295
4296 /* Lock all in {parent,child} order. The lock is also used as a
4297 * signal from the task initiator (which releases it only after
4298 * RTThreadCreate()) that we can start the job. */
4299 AutoMultiWriteLock3 thatLock (that, target, parent);
4300
4301 ImageChain *srcChain = task->d.source.get();
4302 ImageChain *parentChain = task->d.parent.get();
4303
4304 uint64_t size = 0, logicalSize = 0;
4305
4306 /* The object may request a specific UUID (through a special form of
4307 * the setLocation() argument). Otherwise we have to generate it */
4308 Guid targetId = target->m.id;
4309 bool generateUuid = targetId.isEmpty();
4310 if (generateUuid)
4311 {
4312 targetId.create();
4313 /* VirtualBox::registerHardDisk() will need UUID */
4314 unconst(target->m.id) = targetId;
4315 }
4316
4317 try
4318 {
4319 PVBOXHDD hdd;
4320 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
4321 ComAssertRCThrow (vrc, E_FAIL);
4322
4323 try
4324 {
4325 /* Open all hard disk images in the source chain. */
4326 for (List::const_iterator it = srcChain->begin();
4327 it != srcChain->end(); ++ it)
4328 {
4329 /* sanity check */
4330 Assert ((*it)->m.state == MediaState_LockedRead);
4331
4332 /** Open all images in read-only mode. */
4333 vrc = VDOpen(hdd,
4334 Utf8Str((*it)->mm.format).c_str(),
4335 Utf8Str((*it)->m.locationFull).c_str(),
4336 VD_OPEN_FLAGS_READONLY,
4337 (*it)->mm.vdDiskIfaces);
4338 if (RT_FAILURE(vrc))
4339 {
4340 throw setError (E_FAIL,
4341 tr ("Could not open the hard disk storage unit '%s'%s"),
4342 Utf8Str ((*it)->m.locationFull).raw(),
4343 that->vdError (vrc).raw());
4344 }
4345 }
4346
4347 /* unlock before the potentially lengthy operation */
4348 thatLock.leave();
4349
4350 Utf8Str targetFormat (target->mm.format);
4351 Utf8Str targetLocation (target->m.locationFull);
4352
4353 Assert ( target->m.state == MediaState_Creating
4354 || target->m.state == MediaState_LockedWrite);
4355 Assert (that->m.state == MediaState_LockedRead);
4356 Assert (parent.isNull() || parent->m.state == MediaState_LockedRead);
4357
4358 /* ensure the target directory exists */
4359 rc = VirtualBox::ensureFilePathExists (targetLocation);
4360 CheckComRCThrowRC (rc);
4361
4362 /* needed for vdProgressCallback */
4363 that->mm.vdProgress = task->progress;
4364
4365 PVBOXHDD targetHdd;
4366 int vrc = VDCreate (that->mm.vdDiskIfaces, &targetHdd);
4367 ComAssertRCThrow (vrc, E_FAIL);
4368
4369 try
4370 {
4371 /* Open all hard disk images in the parent chain. */
4372 for (List::const_iterator it = parentChain->begin();
4373 it != parentChain->end(); ++ it)
4374 {
4375 /* sanity check */
4376 Assert ( (*it)->m.state == MediaState_LockedRead
4377 || (*it)->m.state == MediaState_LockedWrite);
4378
4379 /* Open all images in appropriate mode. */
4380 vrc = VDOpen(targetHdd,
4381 Utf8Str((*it)->mm.format).c_str(),
4382 Utf8Str((*it)->m.locationFull).c_str(),
4383 ((*it)->m.state == MediaState_LockedWrite) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
4384 (*it)->mm.vdDiskIfaces);
4385 if (RT_FAILURE(vrc))
4386 {
4387 throw setError (E_FAIL,
4388 tr ("Could not open the hard disk storage unit '%s'%s"),
4389 Utf8Str ((*it)->m.locationFull).raw(),
4390 that->vdError (vrc).raw());
4391 }
4392 }
4393
4394 vrc = VDCopy(hdd,
4395 VD_LAST_IMAGE,
4396 targetHdd,
4397 targetFormat.c_str(),
4398 target->m.state == MediaState_Creating ? targetLocation.raw() : (char *)NULL,
4399 false, 0,
4400 task->d.variant, targetId.raw(), NULL,
4401 target->mm.vdDiskIfaces,
4402 that->mm.vdDiskIfaces);
4403
4404 that->mm.vdProgress = NULL;
4405
4406 if (RT_FAILURE(vrc))
4407 {
4408 throw setError (E_FAIL,
4409 tr ("Could not create the clone hard disk '%s'%s"),
4410 targetLocation.raw(), that->vdError (vrc).raw());
4411 }
4412 size = VDGetFileSize (targetHdd, 0);
4413 logicalSize = VDGetSize (targetHdd, 0) / _1M;
4414 }
4415 catch (HRESULT aRC) { rc = aRC; }
4416
4417 VDDestroy (targetHdd);
4418 }
4419 catch (HRESULT aRC) { rc = aRC; }
4420
4421 VDDestroy (hdd);
4422 }
4423 catch (HRESULT aRC) { rc = aRC; }
4424
4425 /* Only do the parent changes for newly created images. */
4426 if (target->m.state == MediaState_Creating)
4427 {
4428 if (SUCCEEDED(rc))
4429 {
4430 /* we set mParent & children() (note that thatLock is released
4431 * here), but lock VirtualBox first to follow the rule */
4432 AutoMultiWriteLock2 alock (that->mVirtualBox->lockHandle(),
4433 that->treeLock());
4434
4435 Assert (target->mParent.isNull());
4436
4437 if (parent)
4438 {
4439 /* associate the clone with the parent and deassociate
4440 * from VirtualBox */
4441 target->mParent = parent;
4442 parent->addDependentChild (target);
4443 target->mVirtualBox->removeDependentChild (target);
4444
4445 /* register with mVirtualBox as the last step and move to
4446 * Created state only on success (leaving an orphan file is
4447 * better than breaking media registry consistency) */
4448 rc = parent->mVirtualBox->registerHardDisk(target);
4449
4450 if (FAILED (rc))
4451 {
4452 /* break parent association on failure to register */
4453 target->mVirtualBox->addDependentChild (target);
4454 parent->removeDependentChild (target);
4455 target->mParent.setNull();
4456 }
4457 }
4458 else
4459 {
4460 /* just register */
4461 rc = that->mVirtualBox->registerHardDisk(target);
4462 }
4463 }
4464 }
4465
4466 thatLock.maybeEnter();
4467
4468 if (target->m.state == MediaState_Creating)
4469 {
4470 if (SUCCEEDED(rc))
4471 {
4472 target->m.state = MediaState_Created;
4473
4474 target->m.size = size;
4475 target->mm.logicalSize = logicalSize;
4476 }
4477 else
4478 {
4479 /* back to NotCreated on failure */
4480 target->m.state = MediaState_NotCreated;
4481
4482 /* reset UUID to prevent it from being reused next time */
4483 if (generateUuid)
4484 unconst(target->m.id).clear();
4485 }
4486 }
4487
4488 /* Everything is explicitly unlocked when the task exits,
4489 * as the task destruction also destroys the source chain. */
4490
4491 /* Make sure the source chain is released early. It could happen
4492 * that we get a deadlock in Appliance::Import when Medium::Close
4493 * is called & the source chain is released at the same time. */
4494 task->d.source.reset();
4495 break;
4496 }
4497
4498 ////////////////////////////////////////////////////////////////////////
4499
4500 case Task::Delete:
4501 {
4502 /* The lock is also used as a signal from the task initiator (which
4503 * releases it only after RTThreadCreate()) that we can start the job */
4504 AutoWriteLock thatLock (that);
4505
4506 try
4507 {
4508 PVBOXHDD hdd;
4509 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
4510 ComAssertRCThrow (vrc, E_FAIL);
4511
4512 Utf8Str format (that->mm.format);
4513 Utf8Str location (that->m.locationFull);
4514
4515 /* unlock before the potentially lengthy operation */
4516 Assert (that->m.state == MediaState_Deleting);
4517 thatLock.leave();
4518
4519 try
4520 {
4521 vrc = VDOpen(hdd,
4522 format.c_str(),
4523 location.c_str(),
4524 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
4525 that->mm.vdDiskIfaces);
4526 if (RT_SUCCESS(vrc))
4527 vrc = VDClose (hdd, true /* fDelete */);
4528
4529 if (RT_FAILURE(vrc))
4530 {
4531 throw setError (E_FAIL,
4532 tr ("Could not delete the hard disk storage unit '%s'%s"),
4533 location.raw(), that->vdError (vrc).raw());
4534 }
4535
4536 }
4537 catch (HRESULT aRC) { rc = aRC; }
4538
4539 VDDestroy (hdd);
4540 }
4541 catch (HRESULT aRC) { rc = aRC; }
4542
4543 thatLock.maybeEnter();
4544
4545 /* go to the NotCreated state even on failure since the storage
4546 * may have been already partially deleted and cannot be used any
4547 * more. One will be able to manually re-open the storage if really
4548 * needed to re-register it. */
4549 that->m.state = MediaState_NotCreated;
4550
4551 /* Reset UUID to prevent Create* from reusing it again */
4552 unconst(that->m.id).clear();
4553
4554 break;
4555 }
4556
4557 case Task::Reset:
4558 {
4559 /* The lock is also used as a signal from the task initiator (which
4560 * releases it only after RTThreadCreate()) that we can start the job */
4561 AutoWriteLock thatLock (that);
4562
4563 /// @todo Below we use a pair of delete/create operations to reset
4564 /// the diff contents but the most efficient way will of course be
4565 /// to add a VDResetDiff() API call
4566
4567 uint64_t size = 0, logicalSize = 0;
4568
4569 try
4570 {
4571 PVBOXHDD hdd;
4572 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
4573 ComAssertRCThrow (vrc, E_FAIL);
4574
4575 Guid id = that->m.id;
4576 Utf8Str format (that->mm.format);
4577 Utf8Str location (that->m.locationFull);
4578
4579 Guid parentId = that->mParent->m.id;
4580 Utf8Str parentFormat (that->mParent->mm.format);
4581 Utf8Str parentLocation (that->mParent->m.locationFull);
4582
4583 Assert (that->m.state == MediaState_LockedWrite);
4584
4585 /* unlock before the potentially lengthy operation */
4586 thatLock.leave();
4587
4588 try
4589 {
4590 /* first, delete the storage unit */
4591 vrc = VDOpen(hdd,
4592 format.c_str(),
4593 location.c_str(),
4594 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
4595 that->mm.vdDiskIfaces);
4596 if (RT_SUCCESS(vrc))
4597 vrc = VDClose (hdd, true /* fDelete */);
4598
4599 if (RT_FAILURE(vrc))
4600 {
4601 throw setError (E_FAIL,
4602 tr ("Could not delete the hard disk storage unit '%s'%s"),
4603 location.raw(), that->vdError (vrc).raw());
4604 }
4605
4606 /* next, create it again */
4607 vrc = VDOpen(hdd,
4608 parentFormat.c_str(),
4609 parentLocation.c_str(),
4610 VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO,
4611 that->mm.vdDiskIfaces);
4612 if (RT_FAILURE(vrc))
4613 {
4614 throw setError (E_FAIL,
4615 tr ("Could not open the hard disk storage unit '%s'%s"),
4616 parentLocation.raw(), that->vdError (vrc).raw());
4617 }
4618
4619 /* needed for vdProgressCallback */
4620 that->mm.vdProgress = task->progress;
4621
4622 vrc = VDCreateDiff(hdd,
4623 format.c_str(),
4624 location.c_str(),
4625 /// @todo use the same image variant as before
4626 VD_IMAGE_FLAGS_NONE,
4627 NULL,
4628 id.raw(),
4629 parentId.raw(),
4630 VD_OPEN_FLAGS_NORMAL,
4631 that->mm.vdDiskIfaces,
4632 that->mm.vdDiskIfaces);
4633
4634 that->mm.vdProgress = NULL;
4635
4636 if (RT_FAILURE(vrc))
4637 {
4638 throw setError (E_FAIL,
4639 tr ("Could not create the differencing hard disk storage unit '%s'%s"),
4640 location.raw(), that->vdError (vrc).raw());
4641 }
4642
4643 size = VDGetFileSize (hdd, 1);
4644 logicalSize = VDGetSize (hdd, 1) / _1M;
4645 }
4646 catch (HRESULT aRC) { rc = aRC; }
4647
4648 VDDestroy (hdd);
4649 }
4650 catch (HRESULT aRC) { rc = aRC; }
4651
4652 thatLock.enter();
4653
4654 that->m.size = size;
4655 that->mm.logicalSize = logicalSize;
4656
4657 if (isAsync)
4658 {
4659 /* unlock ourselves when done */
4660 HRESULT rc2 = that->UnlockWrite (NULL);
4661 AssertComRC (rc2);
4662 }
4663
4664 /* Note that in sync mode, it's the caller's responsibility to
4665 * unlock the hard disk */
4666
4667 break;
4668 }
4669
4670 ////////////////////////////////////////////////////////////////////////
4671
4672 case Task::Compact:
4673 {
4674 /* Lock all in {parent,child} order. The lock is also used as a
4675 * signal from the task initiator (which releases it only after
4676 * RTThreadCreate()) that we can start the job. */
4677 AutoWriteLock thatLock (that);
4678
4679 ImageChain *imgChain = task->d.images.get();
4680
4681 try
4682 {
4683 PVBOXHDD hdd;
4684 int vrc = VDCreate (that->mm.vdDiskIfaces, &hdd);
4685 ComAssertRCThrow (vrc, E_FAIL);
4686
4687 try
4688 {
4689 /* Open all hard disk images in the chain. */
4690 List::const_iterator last = imgChain->end();
4691 last--;
4692 for (List::const_iterator it = imgChain->begin();
4693 it != imgChain->end(); ++ it)
4694 {
4695 /* sanity check */
4696 if (it == last)
4697 Assert ((*it)->m.state == MediaState_LockedWrite);
4698 else
4699 Assert ((*it)->m.state == MediaState_LockedRead);
4700
4701 /** Open all images but last in read-only mode. */
4702 vrc = VDOpen(hdd,
4703 Utf8Str((*it)->mm.format).c_str(),
4704 Utf8Str((*it)->m.locationFull).c_str(),
4705 (it == last) ? VD_OPEN_FLAGS_NORMAL : VD_OPEN_FLAGS_READONLY,
4706 (*it)->mm.vdDiskIfaces);
4707 if (RT_FAILURE(vrc))
4708 {
4709 throw setError (E_FAIL,
4710 tr ("Could not open the hard disk storage unit '%s'%s"),
4711 Utf8Str ((*it)->m.locationFull).raw(),
4712 that->vdError (vrc).raw());
4713 }
4714 }
4715
4716 /* unlock before the potentially lengthy operation */
4717 thatLock.leave();
4718
4719 Assert (that->m.state == MediaState_LockedWrite);
4720
4721 /* needed for vdProgressCallback */
4722 that->mm.vdProgress = task->progress;
4723
4724 vrc = VDCompact (hdd, VD_LAST_IMAGE, that->mm.vdDiskIfaces);
4725
4726 that->mm.vdProgress = NULL;
4727
4728 if (RT_FAILURE(vrc))
4729 {
4730 if (vrc == VERR_NOT_SUPPORTED)
4731 throw setError(VBOX_E_NOT_SUPPORTED,
4732 tr("Compacting is not supported yet for hard disk '%s'"),
4733 Utf8Str (that->m.locationFull).raw());
4734 else if (vrc == VERR_NOT_IMPLEMENTED)
4735 throw setError(E_NOTIMPL,
4736 tr("Compacting is not implemented, hard disk '%s'"),
4737 Utf8Str (that->m.locationFull).raw());
4738 else
4739 throw setError (E_FAIL,
4740 tr ("Could not compact hard disk '%s'%s"),
4741 Utf8Str (that->m.locationFull).raw(),
4742 that->vdError (vrc).raw());
4743 }
4744 }
4745 catch (HRESULT aRC) { rc = aRC; }
4746
4747 VDDestroy (hdd);
4748 }
4749 catch (HRESULT aRC) { rc = aRC; }
4750
4751 /* Everything is explicitly unlocked when the task exits,
4752 * as the task destruction also destroys the image chain. */
4753
4754 break;
4755 }
4756
4757 default:
4758 AssertFailedReturn (VERR_GENERAL_FAILURE);
4759 }
4760
4761 /* complete the progress if run asynchronously */
4762 if (isAsync)
4763 {
4764 if (!task->progress.isNull())
4765 task->progress->notifyComplete (rc);
4766 }
4767 else
4768 {
4769 task->rc = rc;
4770 }
4771
4772 LogFlowFunc (("rc=%Rhrc\n", rc));
4773 LogFlowFuncLeave();
4774
4775 return VINF_SUCCESS;
4776
4777 /// @todo ugly hack, fix ComAssert... later
4778 #undef setError
4779}
4780/* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note: See TracBrowser for help on using the repository browser.

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