VirtualBox

source: vbox/trunk/src/VBox/Main/MachineImpl.cpp@ 703

Last change on this file since 703 was 636, checked in by vboxsync, 18 years ago

oops.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 304.5 KB
Line 
1/** @file
2 *
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22#if defined(__WIN__)
23#elif defined(__LINUX__)
24#endif
25
26#ifdef VBOX_WITH_SYS_V_IPC_SESSION_WATCHER
27# include <errno.h>
28# include <sys/types.h>
29# include <sys/stat.h>
30# include <sys/ipc.h>
31# include <sys/sem.h>
32#endif
33
34#include "VirtualBoxImpl.h"
35#include "MachineImpl.h"
36#include "HardDiskImpl.h"
37#include "HostDVDDriveImpl.h"
38#include "HostFloppyDriveImpl.h"
39#include "ProgressImpl.h"
40#include "HardDiskAttachmentImpl.h"
41#include "USBControllerImpl.h"
42#include "HostImpl.h"
43#include "SystemPropertiesImpl.h"
44#include "SharedFolderImpl.h"
45#include "GuestOSTypeImpl.h"
46#include "VirtualBoxErrorInfoImpl.h"
47
48#include "USBProxyService.h"
49
50#include "Logging.h"
51
52#include <stdio.h>
53#include <stdlib.h>
54#include <VBox/err.h>
55#include <VBox/cfgldr.h>
56#include <iprt/path.h>
57#include <iprt/dir.h>
58#include <iprt/asm.h>
59#include <iprt/process.h>
60#include <VBox/param.h>
61
62#include <algorithm>
63
64#if defined(__WIN__) || defined(__OS2__)
65#define HOSTSUFF_EXE ".exe"
66#else /* !__WIN__ */
67#define HOSTSUFF_EXE ""
68#endif /* !__WIN__ */
69
70// defines / prototypes
71/////////////////////////////////////////////////////////////////////////////
72
73/**
74 * Local mutability check macro for Machine implementation only.
75 */
76#define CHECK_SETTER() \
77 if (!isMutable()) \
78 return setError (E_ACCESSDENIED, tr ("The machine is not mutable"));
79
80// globals
81/////////////////////////////////////////////////////////////////////////////
82
83/**
84 * @note The template is NOT completely valid according to VBOX_XML_SCHEMA
85 * (when loading a newly created settings file, validation will be turned off)
86 */
87static const char DefaultMachineConfig[] =
88{
89 "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>" RTFILE_LINEFEED
90 "<!-- InnoTek VirtualBox Machine Configuration -->" RTFILE_LINEFEED
91 "<VirtualBox xmlns=\"" VBOX_XML_NAMESPACE "\" "
92 "version=\"" VBOX_XML_VERSION "-" VBOX_XML_PLATFORM "\">" RTFILE_LINEFEED
93 "</VirtualBox>" RTFILE_LINEFEED
94};
95
96/**
97 * Progress callback handler for lengthy operations
98 * (corresponds to the FNRTPROGRESS typedef).
99 *
100 * @param uPercentage Completetion precentage (0-100).
101 * @param pvUser Pointer to the Progress instance.
102 */
103static DECLCALLBACK(int) progressCallback (unsigned uPercentage, void *pvUser)
104{
105 Progress *progress = static_cast <Progress *> (pvUser);
106
107 /* update the progress object */
108 if (progress)
109 progress->notifyProgress (uPercentage);
110
111 return VINF_SUCCESS;
112}
113
114/////////////////////////////////////////////////////////////////////////////
115// Machine::Data structure
116/////////////////////////////////////////////////////////////////////////////
117
118Machine::Data::Data()
119{
120 mRegistered = FALSE;
121 /* mUuid is initialized in Machine::init() */
122
123 mMachineState = MachineState_PoweredOff;
124 RTTIMESPEC time;
125 mLastStateChange = RTTimeSpecGetMilli (RTTimeNow (&time));
126 mCurrentStateModified = TRUE;
127 mHandleCfgFile = NIL_RTFILE;
128
129 mSession.mPid = NIL_RTPROCESS;
130 mSession.mState = SessionState_SessionClosed;
131}
132
133Machine::Data::~Data()
134{
135}
136
137/////////////////////////////////////////////////////////////////////////////
138// Machine::UserData structure
139/////////////////////////////////////////////////////////////////////////////
140
141Machine::UserData::UserData()
142{
143 /* default values for a newly created machine */
144
145 mNameSync = TRUE;
146
147 /* mName, mOSType, mSnapshotFolder, mSnapshotFolderFull are initialized in
148 * Machine::init() */
149}
150
151Machine::UserData::~UserData()
152{
153}
154
155/////////////////////////////////////////////////////////////////////////////
156// Machine::HWData structure
157/////////////////////////////////////////////////////////////////////////////
158
159Machine::HWData::HWData()
160{
161 /* default values for a newly created machine */
162 mMemorySize = 128;
163 mVRAMSize = 8;
164 mHWVirtExEnabled = TriStateBool_False;
165
166 /* default boot order: floppy - DVD - HDD */
167 mBootOrder [0] = DeviceType_FloppyDevice;
168 mBootOrder [1] = DeviceType_DVDDevice;
169 mBootOrder [2] = DeviceType_HardDiskDevice;
170 for (size_t i = 3; i < ELEMENTS (mBootOrder); i++)
171 mBootOrder [i] = DeviceType_NoDevice;
172
173 mClipboardMode = ClipboardMode_ClipDisabled;
174}
175
176Machine::HWData::~HWData()
177{
178}
179
180bool Machine::HWData::operator== (const HWData &that) const
181{
182 if (this == &that)
183 return true;
184
185 if (mMemorySize != that.mMemorySize ||
186 mVRAMSize != that.mVRAMSize ||
187 mHWVirtExEnabled != that.mHWVirtExEnabled ||
188 mClipboardMode != that.mClipboardMode)
189 return false;
190
191 for (size_t i = 0; i < ELEMENTS (mBootOrder); ++ i)
192 if (mBootOrder [i] != that.mBootOrder [i])
193 return false;
194
195 if (mSharedFolders.size() != that.mSharedFolders.size())
196 return false;
197
198 if (mSharedFolders.size() == 0)
199 return true;
200
201 /* Make copies to speed up comparison */
202 SharedFolderList folders = mSharedFolders;
203 SharedFolderList thatFolders = that.mSharedFolders;
204
205 SharedFolderList::iterator it = folders.begin();
206 while (it != folders.end())
207 {
208 bool found = false;
209 SharedFolderList::iterator thatIt = thatFolders.begin();
210 while (thatIt != thatFolders.end())
211 {
212 if ((*it)->name() == (*thatIt)->name() &&
213 RTPathCompare (Utf8Str ((*it)->hostPath()),
214 Utf8Str ((*thatIt)->hostPath())) == 0)
215 {
216 thatFolders.erase (thatIt);
217 found = true;
218 break;
219 }
220 else
221 ++ thatIt;
222 }
223 if (found)
224 it = folders.erase (it);
225 else
226 return false;
227 }
228
229 Assert (folders.size() == 0 && thatFolders.size() == 0);
230
231 return true;
232}
233
234/////////////////////////////////////////////////////////////////////////////
235// Machine::HDData structure
236/////////////////////////////////////////////////////////////////////////////
237
238Machine::HDData::HDData()
239{
240 /* default values for a newly created machine */
241 mHDAttachmentsChanged = false;
242}
243
244Machine::HDData::~HDData()
245{
246}
247
248bool Machine::HDData::operator== (const HDData &that) const
249{
250 if (this == &that)
251 return true;
252
253 if (mHDAttachments.size() != that.mHDAttachments.size())
254 return false;
255
256 if (mHDAttachments.size() == 0)
257 return true;
258
259 /* Make copies to speed up comparison */
260 HDAttachmentList atts = mHDAttachments;
261 HDAttachmentList thatAtts = that.mHDAttachments;
262
263 HDAttachmentList::iterator it = atts.begin();
264 while (it != atts.end())
265 {
266 bool found = false;
267 HDAttachmentList::iterator thatIt = thatAtts.begin();
268 while (thatIt != thatAtts.end())
269 {
270 if ((*it)->deviceNumber() == (*thatIt)->deviceNumber() &&
271 (*it)->controller() == (*thatIt)->controller() &&
272 (*it)->hardDisk().equalsTo ((*thatIt)->hardDisk()))
273 {
274 thatAtts.erase (thatIt);
275 found = true;
276 break;
277 }
278 else
279 ++ thatIt;
280 }
281 if (found)
282 it = atts.erase (it);
283 else
284 return false;
285 }
286
287 Assert (atts.size() == 0 && thatAtts.size() == 0);
288
289 return true;
290}
291
292/////////////////////////////////////////////////////////////////////////////
293// Machine class
294/////////////////////////////////////////////////////////////////////////////
295
296// constructor / destructor
297/////////////////////////////////////////////////////////////////////////////
298
299Machine::Machine() : mType (IsMachine) {}
300
301Machine::~Machine() {}
302
303HRESULT Machine::FinalConstruct()
304{
305 LogFlowThisFunc (("\n"));
306 return S_OK;
307}
308
309void Machine::FinalRelease()
310{
311 LogFlowThisFunc (("\n"));
312 uninit();
313}
314
315/**
316 * Initializes the instance.
317 *
318 * @param aParent Associated parent object
319 * @param aConfigFile Local file system path to the VM settings file (can
320 * be relative to the VirtualBox config directory).
321 * @param aMode Init_New, Init_Existing or Init_Registered
322 * @param aName name for the machine when aMode is Init_New
323 * (ignored otherwise)
324 * @param aNameSync |TRUE| to authomatically sync settings dir and file
325 * name with the machine name. |FALSE| is used for legacy
326 * machines where the file name is specified by the
327 * user and should never change. Used only in Init_New
328 * mode (ignored otherwise).
329 * @param aId UUID of the machine (used only for consistency
330 * check when aMode is Init_Registered; must match UUID
331 * stored in the settings file).
332 *
333 * @return Success indicator. if not S_OK, the machine object is invalid
334 */
335HRESULT Machine::init (VirtualBox *aParent, const BSTR aConfigFile,
336 InitMode aMode, const BSTR aName /* = NULL */,
337 BOOL aNameSync /* = TRUE */,
338 const Guid *aId /* = NULL */)
339{
340 LogFlowThisFuncEnter();
341 LogFlowThisFunc (("aConfigFile='%ls', aMode=%d\n", aConfigFile, aMode));
342
343 AssertReturn (aParent, E_INVALIDARG);
344 AssertReturn (aConfigFile, E_INVALIDARG);
345 AssertReturn (aMode != Init_New || (aName != NULL && *aName != '\0'),
346 E_INVALIDARG);
347 AssertReturn (aMode != Init_Registered || aId != NULL, E_FAIL);
348
349 /* Enclose the state transition NotReady->InInit->Ready */
350 AutoInitSpan autoInitSpan (this);
351 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
352
353 HRESULT rc = S_OK;
354
355 /* share the parent weakly */
356 unconst (mParent) = aParent;
357
358 /* register with parent early, since uninit() will unconditionally
359 * unregister on failure */
360 mParent->addDependentChild (this);
361
362 /* create machine data structures */
363 mData.allocate();
364 mSSData.allocate();
365
366 mUserData.allocate();
367 mHWData.allocate();
368 mHDData.allocate();
369
370 char configFileFull [RTPATH_MAX] = {0};
371
372 /* memorize the config file name (as provided) */
373 mData->mConfigFile = aConfigFile;
374
375 /* get the full file name */
376 int vrc = RTPathAbsEx (mParent->homeDir(), Utf8Str (aConfigFile),
377 configFileFull, sizeof (configFileFull));
378 if (VBOX_FAILURE (vrc))
379 return setError (E_FAIL,
380 tr ("Invalid settings file name: '%ls' (%Vrc)"),
381 aConfigFile, vrc);
382 mData->mConfigFileFull = configFileFull;
383
384 mData->mAccessible = TRUE;
385
386 if (aMode != Init_New)
387 {
388 /* lock the settings file */
389 rc = lockConfig();
390
391 if (aMode == Init_Registered && FAILED (rc))
392 {
393 /* If the machine is registered, then, instead of returning a
394 * failure, we mark it as inaccessible and set the result to
395 * success to give it a try later */
396 mData->mAccessible = FALSE;
397 /* fetch the current error info */
398 mData->mAccessError = com::ErrorInfo();
399 LogWarning (("Machine {%Vuuid} is inaccessible! [%ls]\n",
400 mData->mUuid.raw(),
401 mData->mAccessError.getText().raw()));
402 rc = S_OK;
403 }
404 }
405 else
406 {
407 /* check for the file existence */
408 RTFILE f = NIL_RTFILE;
409 int vrc = RTFileOpen (&f, configFileFull, RTFILE_O_READ);
410 if (VBOX_SUCCESS (vrc) || vrc == VERR_SHARING_VIOLATION)
411 {
412 rc = setError (E_FAIL,
413 tr ("Settings file '%s' already exists"), configFileFull);
414 if (VBOX_SUCCESS (vrc))
415 RTFileClose (f);
416 }
417 else
418 {
419 if (vrc != VERR_FILE_NOT_FOUND && vrc != VERR_PATH_NOT_FOUND)
420 rc = setError (E_FAIL,
421 tr ("Invalid settings file name: '%ls' (%Vrc)"),
422 mData->mConfigFileFull.raw(), vrc);
423 }
424 }
425
426 CheckComRCReturnRC (rc);
427
428 /* initialize mOSType */
429 mUserData->mOSType = mParent->getUnknownOSType();
430
431 /* create associated BIOS settings object */
432 unconst (mBIOSSettings).createObject();
433 mBIOSSettings->init(this);
434
435#ifdef VBOX_VRDP
436 /* create an associated VRDPServer object (default is disabled) */
437 unconst (mVRDPServer).createObject();
438 mVRDPServer->init(this);
439#endif
440
441 /* create an associated DVD drive object */
442 unconst (mDVDDrive).createObject();
443 mDVDDrive->init (this);
444
445 /* create an associated floppy drive object */
446 unconst (mFloppyDrive).createObject();
447 mFloppyDrive->init (this);
448
449 /* create the audio adapter object (always present, default is disabled) */
450 unconst (mAudioAdapter).createObject();
451 mAudioAdapter->init(this);
452
453 /* create the USB controller object (always present, default is disabled) */
454 unconst (mUSBController).createObject();
455 mUSBController->init(this);
456
457 /* create associated network adapter objects */
458 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
459 {
460 unconst (mNetworkAdapters [slot]).createObject();
461 mNetworkAdapters [slot]->init (this, slot);
462 }
463
464 if (aMode == Init_Registered)
465 {
466 /* store the supplied UUID (will be used to check for UUID consistency
467 * in loadSettings() */
468 unconst (mData->mUuid) = *aId;
469 /* try to load settings only if the settings file is accessible */
470 if (mData->mAccessible)
471 rc = registeredInit();
472 }
473 else
474 {
475 if (aMode != Init_New)
476 {
477 rc = loadSettings (false /* aRegistered */);
478 }
479 else
480 {
481 /* create the machine UUID */
482 unconst (mData->mUuid).create();
483
484 /* initialize the default snapshots folder */
485 rc = COMSETTER(SnapshotFolder) (NULL);
486 AssertComRC (rc);
487
488 /* memorize the provided new machine's name */
489 mUserData->mName = aName;
490 mUserData->mNameSync = aNameSync;
491 }
492
493 /* commit all changes made during the initialization */
494 if (SUCCEEDED (rc))
495 commit();
496 }
497
498 /* Confirm a successful initialization when it's the case */
499 if (SUCCEEDED (rc))
500 {
501 if (mData->mAccessible)
502 autoInitSpan.setSucceeded();
503 else
504 autoInitSpan.setLimited();
505 }
506
507 LogFlowThisFunc (("mName='%ls', mRegistered=%RTbool, mAccessible=%RTbool "
508 "rc=%08X\n",
509 mUserData->mName.raw(), mData->mRegistered,
510 mData->mAccessible, rc));
511
512 LogFlowThisFuncLeave();
513
514 return rc;
515}
516
517/**
518 * Initializes the registered machine by loading the settings file.
519 * This method is separated from #init() in order to make it possible to
520 * retry the operation after VirtualBox startup instead of refusing to
521 * startup the whole VirtualBox server in case if the settings file of some
522 * registered VM is invalid or inaccessible.
523 *
524 * @note Must be always called from this object's write lock
525 * (unless called from #init() that doesn't need any locking).
526 * @note Locks the mUSBController method for writing.
527 * @note Subclasses must not call this method.
528 */
529HRESULT Machine::registeredInit()
530{
531 AssertReturn (mType == IsMachine, E_FAIL);
532 AssertReturn (!mData->mUuid.isEmpty(), E_FAIL);
533
534 HRESULT rc = S_OK;
535
536 if (!mData->mAccessible)
537 rc = lockConfig();
538
539 /* Temporarily reset the registered flag in order to let setters potentially
540 * called from loadSettings() succeed (isMutable() used in all setters
541 * will return FALSE for a Machine instance if mRegistered is TRUE). */
542 mData->mRegistered = FALSE;
543
544 if (SUCCEEDED (rc))
545 {
546 rc = loadSettings (true /* aRegistered */);
547
548 if (FAILED (rc))
549 unlockConfig();
550 }
551
552 if (SUCCEEDED (rc))
553 {
554 mData->mAccessible = TRUE;
555
556 /* commit all changes made during loading the settings file */
557 commit();
558
559 /* VirtualBox will not call trySetRegistered(), so
560 * inform the USB proxy about all attached USB filters */
561 mUSBController->onMachineRegistered (TRUE);
562 }
563 else
564 {
565 /* If the machine is registered, then, instead of returning a
566 * failure, we mark it as inaccessible and set the result to
567 * success to give it a try later */
568 mData->mAccessible = FALSE;
569 /* fetch the current error info */
570 mData->mAccessError = com::ErrorInfo();
571 LogWarning (("Machine {%Vuuid} is inaccessible! [%ls]\n",
572 mData->mUuid.raw(),
573 mData->mAccessError.getText().raw()));
574
575 /* rollback all changes */
576 rollback (false /* aNotify */);
577
578 rc = S_OK;
579 }
580
581 /* Restore the registered flag (even on failure) */
582 mData->mRegistered = TRUE;
583
584 return rc;
585}
586
587/**
588 * Uninitializes the instance.
589 * Called either from FinalRelease() or by the parent when it gets destroyed.
590 *
591 * @note The caller of this method must make sure that this object
592 * a) doesn't have active callers on the current thread and b) is not locked
593 * by the current thread; otherwise uninit() will hang either a) due to
594 * AutoUninitSpan waiting for a number of calls to drop to zero or b) due to
595 * a dead-lock caused by this thread waiting for all callers on the other
596 * threads are are done but preventing them from doing so by holding a lock.
597 */
598void Machine::uninit()
599{
600 LogFlowThisFuncEnter();
601
602 Assert (!isLockedOnCurrentThread());
603
604 /* Enclose the state transition Ready->InUninit->NotReady */
605 AutoUninitSpan autoUninitSpan (this);
606 if (autoUninitSpan.uninitDone())
607 return;
608
609 Assert (mType == IsMachine);
610 Assert (!!mData && !!mUserData && !!mHWData && !!mHDData && !!mSSData);
611
612 LogFlowThisFunc (("initFailed()=%d\n", autoUninitSpan.initFailed()));
613 LogFlowThisFunc (("mRegistered=%d\n", mData->mRegistered));
614
615 /*
616 * Enter this object's lock because there may be a SessionMachine instance
617 * somewhere around, that shares our data and lock but doesn't use our
618 * addCaller()/removeCaller(), and it may be also accessing the same
619 * data members. mParent lock is necessary as well because of
620 * SessionMachine::uninit(), etc.
621 */
622 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
623
624 if (!mData->mSession.mMachine.isNull())
625 {
626 /*
627 * Theoretically, this can only happen if the VirtualBox server has
628 * been terminated while there were clients running that owned open
629 * direct sessions. Since in this case we are definitely called by
630 * VirtualBox::uninit(), we may be sure that SessionMachine::uninit()
631 * won't happen on the client watcher thread (because it does
632 * VirtualBox::addCaller() for the duration of the
633 * SessionMachine::checkForDeath() call, so that VirtualBox::uninit()
634 * cannot happen until the VirtualBox caller is released). This is
635 * important, because SessionMachine::uninit() cannot correctly operate
636 * after we return from this method (it expects the Machine instance
637 * is still valid). We'll call it ourselves below.
638 */
639 LogWarningThisFunc (("Session machine is not NULL (%p), "
640 "the direct session is still open!\n",
641 (SessionMachine *) mData->mSession.mMachine));
642
643 if (mData->mMachineState >= MachineState_Running)
644 {
645 LogWarningThisFunc (("Setting state to Aborted!\n"));
646 /* set machine state using SessionMachine reimplementation */
647 static_cast <Machine *> (mData->mSession.mMachine)
648 ->setMachineState (MachineState_Aborted);
649 }
650
651 /*
652 * Uninitialize SessionMachine using public uninit() to indicate
653 * an unexpected uninitialization.
654 */
655 mData->mSession.mMachine->uninit();
656 /* SessionMachine::uninit() must set mSession.mMachine to null */
657 Assert (mData->mSession.mMachine.isNull());
658 }
659
660 /* the lock is no more necessary (SessionMachine is uninitialized) */
661 alock.leave();
662
663 /* make sure the configuration is unlocked */
664 unlockConfig();
665
666 if (isModified())
667 {
668 LogWarningThisFunc (("Discarding unsaved settings changes!\n"));
669 rollback (false /* aNotify */);
670 }
671
672 uninitDataAndChildObjects();
673
674 mParent->removeDependentChild (this);
675
676 LogFlowThisFuncLeave();
677}
678
679// IMachine properties
680/////////////////////////////////////////////////////////////////////////////
681
682STDMETHODIMP Machine::COMGETTER(Parent) (IVirtualBox **aParent)
683{
684 if (!aParent)
685 return E_POINTER;
686
687 AutoLimitedCaller autoCaller (this);
688 CheckComRCReturnRC (autoCaller.rc());
689
690 /* mParent is constant during life time, no need to lock */
691 mParent.queryInterfaceTo (aParent);
692
693 return S_OK;
694}
695
696STDMETHODIMP Machine::COMGETTER(Accessible) (BOOL *aAccessible)
697{
698 if (!aAccessible)
699 return E_POINTER;
700
701 AutoLimitedCaller autoCaller (this);
702 CheckComRCReturnRC (autoCaller.rc());
703
704 AutoLock alock (this);
705
706 HRESULT rc = S_OK;
707
708 if (!mData->mAccessible)
709 {
710 /* try to initialize the VM once more if not accessible */
711
712 AutoReadySpan autoReadySpan (this);
713 AssertReturn (autoReadySpan.isOk(), E_FAIL);
714
715 rc = registeredInit();
716
717 if (mData->mAccessible)
718 autoReadySpan.setSucceeded();
719 }
720
721 if (SUCCEEDED (rc))
722 *aAccessible = mData->mAccessible;
723
724 return rc;
725}
726
727STDMETHODIMP Machine::COMGETTER(AccessError) (IVirtualBoxErrorInfo **aAccessError)
728{
729 if (!aAccessError)
730 return E_POINTER;
731
732 AutoLimitedCaller autoCaller (this);
733 CheckComRCReturnRC (autoCaller.rc());
734
735 AutoReaderLock alock (this);
736
737 if (mData->mAccessible || !mData->mAccessError.isBasicAvailable())
738 {
739 /* return shortly */
740 aAccessError = NULL;
741 return S_OK;
742 }
743
744 HRESULT rc = S_OK;
745
746 ComObjPtr <VirtualBoxErrorInfo> errorInfo;
747 rc = errorInfo.createObject();
748 if (SUCCEEDED (rc))
749 {
750 errorInfo->init (mData->mAccessError.getResultCode(),
751 mData->mAccessError.getInterfaceID(),
752 mData->mAccessError.getComponent(),
753 mData->mAccessError.getText());
754 rc = errorInfo.queryInterfaceTo (aAccessError);
755 }
756
757 return rc;
758}
759
760STDMETHODIMP Machine::COMGETTER(Name) (BSTR *aName)
761{
762 if (!aName)
763 return E_POINTER;
764
765 AutoCaller autoCaller (this);
766 CheckComRCReturnRC (autoCaller.rc());
767
768 AutoReaderLock alock (this);
769
770 mUserData->mName.cloneTo (aName);
771
772 return S_OK;
773}
774
775STDMETHODIMP Machine::COMSETTER(Name) (INPTR BSTR aName)
776{
777 if (!aName)
778 return E_INVALIDARG;
779
780 if (!*aName)
781 return setError (E_INVALIDARG,
782 tr ("Machine name cannot be empty"));
783
784 AutoCaller autoCaller (this);
785 CheckComRCReturnRC (autoCaller.rc());
786
787 AutoLock alock (this);
788
789 CHECK_SETTER();
790
791 mUserData.backup();
792 mUserData->mName = aName;
793
794 return S_OK;
795}
796
797STDMETHODIMP Machine::COMGETTER(Id) (GUIDPARAMOUT aId)
798{
799 if (!aId)
800 return E_POINTER;
801
802 AutoLimitedCaller autoCaller (this);
803 CheckComRCReturnRC (autoCaller.rc());
804
805 AutoReaderLock alock (this);
806
807 mData->mUuid.cloneTo (aId);
808
809 return S_OK;
810}
811
812STDMETHODIMP Machine::COMGETTER(OSType) (IGuestOSType **aOSType)
813{
814 if (!aOSType)
815 return E_POINTER;
816
817 AutoCaller autoCaller (this);
818 CheckComRCReturnRC (autoCaller.rc());
819
820 AutoReaderLock alock (this);
821
822 mUserData->mOSType.queryInterfaceTo (aOSType);
823
824 return S_OK;
825}
826
827STDMETHODIMP Machine::COMSETTER(OSType) (IGuestOSType *aOSType)
828{
829 if (!aOSType)
830 return E_INVALIDARG;
831
832 AutoCaller autoCaller (this);
833 CheckComRCReturnRC (autoCaller.rc());
834
835 AutoLock alock (this);
836
837 CHECK_SETTER();
838
839 mUserData.backup();
840 mUserData->mOSType = aOSType;
841
842 return S_OK;
843}
844
845STDMETHODIMP Machine::COMGETTER(MemorySize) (ULONG *memorySize)
846{
847 if (!memorySize)
848 return E_POINTER;
849
850 AutoCaller autoCaller (this);
851 CheckComRCReturnRC (autoCaller.rc());
852
853 AutoReaderLock alock (this);
854
855 *memorySize = mHWData->mMemorySize;
856
857 return S_OK;
858}
859
860STDMETHODIMP Machine::COMSETTER(MemorySize) (ULONG memorySize)
861{
862 /* check RAM limits */
863 if (memorySize < SchemaDefs::MinGuestRAM ||
864 memorySize > SchemaDefs::MaxGuestRAM)
865 return setError (E_INVALIDARG,
866 tr ("Invalid RAM size: %lu MB (must be in range [%lu, %lu] MB)"),
867 memorySize, SchemaDefs::MinGuestRAM, SchemaDefs::MaxGuestRAM);
868
869 AutoCaller autoCaller (this);
870 CheckComRCReturnRC (autoCaller.rc());
871
872 AutoLock alock (this);
873
874 CHECK_SETTER();
875
876 mHWData.backup();
877 mHWData->mMemorySize = memorySize;
878
879 return S_OK;
880}
881
882STDMETHODIMP Machine::COMGETTER(VRAMSize) (ULONG *memorySize)
883{
884 if (!memorySize)
885 return E_POINTER;
886
887 AutoCaller autoCaller (this);
888 CheckComRCReturnRC (autoCaller.rc());
889
890 AutoReaderLock alock (this);
891
892 *memorySize = mHWData->mVRAMSize;
893
894 return S_OK;
895}
896
897STDMETHODIMP Machine::COMSETTER(VRAMSize) (ULONG memorySize)
898{
899 /* check VRAM limits */
900 if (memorySize < SchemaDefs::MinGuestVRAM ||
901 memorySize > SchemaDefs::MaxGuestVRAM)
902 return setError (E_INVALIDARG,
903 tr ("Invalid VRAM size: %lu MB (must be in range [%lu, %lu] MB)"),
904 memorySize, SchemaDefs::MinGuestVRAM, SchemaDefs::MaxGuestVRAM);
905
906 AutoCaller autoCaller (this);
907 CheckComRCReturnRC (autoCaller.rc());
908
909 AutoLock alock (this);
910
911 CHECK_SETTER();
912
913 mHWData.backup();
914 mHWData->mVRAMSize = memorySize;
915
916 return S_OK;
917}
918
919STDMETHODIMP Machine::COMGETTER(BIOSSettings)(IBIOSSettings **biosSettings)
920{
921 if (!biosSettings)
922 return E_POINTER;
923
924 AutoCaller autoCaller (this);
925 CheckComRCReturnRC (autoCaller.rc());
926
927 /* mBIOSSettings is constant during life time, no need to lock */
928 mBIOSSettings.queryInterfaceTo (biosSettings);
929
930 return S_OK;
931}
932
933STDMETHODIMP Machine::COMGETTER(HWVirtExEnabled)(TriStateBool_T *enabled)
934{
935 if (!enabled)
936 return E_POINTER;
937
938 AutoCaller autoCaller (this);
939 CheckComRCReturnRC (autoCaller.rc());
940
941 AutoReaderLock alock (this);
942
943 *enabled = mHWData->mHWVirtExEnabled;
944
945 return S_OK;
946}
947
948STDMETHODIMP Machine::COMSETTER(HWVirtExEnabled)(TriStateBool_T enable)
949{
950 AutoCaller autoCaller (this);
951 CheckComRCReturnRC (autoCaller.rc());
952
953 AutoLock alock (this);
954
955 CHECK_SETTER();
956
957 /** @todo check validity! */
958
959 mHWData.backup();
960 mHWData->mHWVirtExEnabled = enable;
961
962 return S_OK;
963}
964
965STDMETHODIMP Machine::COMGETTER(SnapshotFolder) (BSTR *aSnapshotFolder)
966{
967 if (!aSnapshotFolder)
968 return E_POINTER;
969
970 AutoCaller autoCaller (this);
971 CheckComRCReturnRC (autoCaller.rc());
972
973 AutoReaderLock alock (this);
974
975 mUserData->mSnapshotFolderFull.cloneTo (aSnapshotFolder);
976
977 return S_OK;
978}
979
980STDMETHODIMP Machine::COMSETTER(SnapshotFolder) (INPTR BSTR aSnapshotFolder)
981{
982 /// @todo (r=dmik):
983 // 1. Allow to change the name of the snapshot folder containing snapshots
984 // 2. Rename the folder on disk instead of just changing the property
985 // value (to be smart and not to leave garbage). Note that it cannot be
986 // done here because the change may be rolled back. Thus, the right
987 // place is #saveSettings().
988
989 AutoCaller autoCaller (this);
990 CheckComRCReturnRC (autoCaller.rc());
991
992 AutoLock alock (this);
993
994 CHECK_SETTER();
995
996 if (!mData->mCurrentSnapshot.isNull())
997 return setError (E_FAIL,
998 tr ("The snapshot folder of a machine with snapshots cannot "
999 "be changed (please discard all snapshots first)"));
1000
1001 Utf8Str snapshotFolder = aSnapshotFolder;
1002
1003 if (snapshotFolder.isEmpty())
1004 {
1005 if (isInOwnDir())
1006 {
1007 /* the default snapshots folder is 'Snapshots' in the machine dir */
1008 snapshotFolder = Utf8Str ("Snapshots");
1009 }
1010 else
1011 {
1012 /* the default snapshots folder is {UUID}, for backwards
1013 * compatibility and to resolve conflicts */
1014 snapshotFolder = Utf8StrFmt ("{%Vuuid}", mData->mUuid.raw());
1015 }
1016 }
1017
1018 int vrc = calculateFullPath (snapshotFolder, snapshotFolder);
1019 if (VBOX_FAILURE (vrc))
1020 return setError (E_FAIL,
1021 tr ("Invalid snapshot folder: '%ls' (%Vrc)"),
1022 aSnapshotFolder, vrc);
1023
1024 mUserData.backup();
1025 mUserData->mSnapshotFolder = aSnapshotFolder;
1026 mUserData->mSnapshotFolderFull = snapshotFolder;
1027
1028 return S_OK;
1029}
1030
1031STDMETHODIMP Machine::COMGETTER(HardDiskAttachments) (IHardDiskAttachmentCollection **attachments)
1032{
1033 if (!attachments)
1034 return E_POINTER;
1035
1036 AutoCaller autoCaller (this);
1037 CheckComRCReturnRC (autoCaller.rc());
1038
1039 AutoReaderLock alock (this);
1040
1041 ComObjPtr <HardDiskAttachmentCollection> collection;
1042 collection.createObject();
1043 collection->init (mHDData->mHDAttachments);
1044 collection.queryInterfaceTo (attachments);
1045
1046 return S_OK;
1047}
1048
1049STDMETHODIMP Machine::COMGETTER(VRDPServer)(IVRDPServer **vrdpServer)
1050{
1051#ifdef VBOX_VRDP
1052 if (!vrdpServer)
1053 return E_POINTER;
1054
1055 AutoCaller autoCaller (this);
1056 CheckComRCReturnRC (autoCaller.rc());
1057
1058 AutoReaderLock alock (this);
1059
1060 Assert (!!mVRDPServer);
1061 mVRDPServer.queryInterfaceTo (vrdpServer);
1062
1063 return S_OK;
1064#else
1065 return E_NOTIMPL;
1066#endif
1067}
1068
1069STDMETHODIMP Machine::COMGETTER(DVDDrive) (IDVDDrive **dvdDrive)
1070{
1071 if (!dvdDrive)
1072 return E_POINTER;
1073
1074 AutoCaller autoCaller (this);
1075 CheckComRCReturnRC (autoCaller.rc());
1076
1077 AutoReaderLock alock (this);
1078
1079 Assert (!!mDVDDrive);
1080 mDVDDrive.queryInterfaceTo (dvdDrive);
1081 return S_OK;
1082}
1083
1084STDMETHODIMP Machine::COMGETTER(FloppyDrive) (IFloppyDrive **floppyDrive)
1085{
1086 if (!floppyDrive)
1087 return E_POINTER;
1088
1089 AutoCaller autoCaller (this);
1090 CheckComRCReturnRC (autoCaller.rc());
1091
1092 AutoReaderLock alock (this);
1093
1094 Assert (!!mFloppyDrive);
1095 mFloppyDrive.queryInterfaceTo (floppyDrive);
1096 return S_OK;
1097}
1098
1099STDMETHODIMP Machine::COMGETTER(AudioAdapter)(IAudioAdapter **audioAdapter)
1100{
1101 if (!audioAdapter)
1102 return E_POINTER;
1103
1104 AutoCaller autoCaller (this);
1105 CheckComRCReturnRC (autoCaller.rc());
1106
1107 AutoReaderLock alock (this);
1108
1109 mAudioAdapter.queryInterfaceTo (audioAdapter);
1110 return S_OK;
1111}
1112
1113STDMETHODIMP Machine::COMGETTER(USBController)(IUSBController * *a_ppUSBController)
1114{
1115#ifdef VBOX_WITH_USB
1116 if (!a_ppUSBController)
1117 return E_POINTER;
1118
1119 AutoCaller autoCaller (this);
1120 CheckComRCReturnRC (autoCaller.rc());
1121
1122 AutoReaderLock alock (this);
1123
1124 USBProxyService *usbProxyService = mParent->host()->usbProxyService();
1125 AssertReturn (usbProxyService, E_FAIL);
1126 if (!usbProxyService->isActive())
1127 {
1128 /* disable the USB controller completely to avoid assertions if the
1129 * USB proxy service could not start. */
1130
1131 Assert (VBOX_FAILURE (usbProxyService->getLastError()));
1132 if (usbProxyService->getLastError() == VERR_FILE_NOT_FOUND)
1133 return setError (E_FAIL,
1134 tr ("Could not load the USB proxy service (%Vrc), "
1135 "the virtual USB Controller is not available."
1136 "The service may not be installed on the host computer."),
1137 usbProxyService->getLastError());
1138 else
1139 return setError (E_FAIL,
1140 tr ("Could not load the USB proxy service (%Vrc), "
1141 "the virtual USB Controller is not available."),
1142 usbProxyService->getLastError());
1143 }
1144
1145 mUSBController.queryInterfaceTo (a_ppUSBController);
1146 return S_OK;
1147#else
1148 return E_NOTIMPL;
1149#endif
1150}
1151
1152STDMETHODIMP Machine::COMGETTER(SettingsFilePath) (BSTR *filePath)
1153{
1154 if (!filePath)
1155 return E_POINTER;
1156
1157 AutoLimitedCaller autoCaller (this);
1158 CheckComRCReturnRC (autoCaller.rc());
1159
1160 AutoReaderLock alock (this);
1161
1162 mData->mConfigFileFull.cloneTo (filePath);
1163 return S_OK;
1164}
1165
1166STDMETHODIMP Machine::COMGETTER(SettingsModified) (BOOL *modified)
1167{
1168 if (!modified)
1169 return E_POINTER;
1170
1171 AutoCaller autoCaller (this);
1172 CheckComRCReturnRC (autoCaller.rc());
1173
1174 AutoLock alock (this);
1175
1176 CHECK_SETTER();
1177
1178 if (!isConfigLocked())
1179 {
1180 /*
1181 * if we're ready and isConfigLocked() is FALSE then it means
1182 * that no config file exists yet, so always return TRUE
1183 */
1184 *modified = TRUE;
1185 }
1186 else
1187 {
1188 *modified = isModified();
1189 }
1190
1191 return S_OK;
1192}
1193
1194STDMETHODIMP Machine::COMGETTER(SessionState) (SessionState_T *sessionState)
1195{
1196 if (!sessionState)
1197 return E_POINTER;
1198
1199 AutoCaller autoCaller (this);
1200 CheckComRCReturnRC (autoCaller.rc());
1201
1202 AutoReaderLock alock (this);
1203
1204 *sessionState = mData->mSession.mState;
1205
1206 return S_OK;
1207}
1208
1209STDMETHODIMP Machine::COMGETTER(State) (MachineState_T *machineState)
1210{
1211 if (!machineState)
1212 return E_POINTER;
1213
1214 AutoCaller autoCaller (this);
1215 CheckComRCReturnRC (autoCaller.rc());
1216
1217 AutoReaderLock alock (this);
1218
1219 *machineState = mData->mMachineState;
1220
1221 return S_OK;
1222}
1223
1224STDMETHODIMP Machine::COMGETTER(LastStateChange) (LONG64 *aLastStateChange)
1225{
1226 if (!aLastStateChange)
1227 return E_POINTER;
1228
1229 AutoCaller autoCaller (this);
1230 CheckComRCReturnRC (autoCaller.rc());
1231
1232 AutoReaderLock alock (this);
1233
1234 *aLastStateChange = mData->mLastStateChange;
1235
1236 return S_OK;
1237}
1238
1239STDMETHODIMP Machine::COMGETTER(StateFilePath) (BSTR *aStateFilePath)
1240{
1241 if (!aStateFilePath)
1242 return E_POINTER;
1243
1244 AutoCaller autoCaller (this);
1245 CheckComRCReturnRC (autoCaller.rc());
1246
1247 AutoReaderLock alock (this);
1248
1249 mSSData->mStateFilePath.cloneTo (aStateFilePath);
1250
1251 return S_OK;
1252}
1253
1254STDMETHODIMP Machine::COMGETTER(CurrentSnapshot) (ISnapshot **aCurrentSnapshot)
1255{
1256 if (!aCurrentSnapshot)
1257 return E_POINTER;
1258
1259 AutoCaller autoCaller (this);
1260 CheckComRCReturnRC (autoCaller.rc());
1261
1262 AutoReaderLock alock (this);
1263
1264 mData->mCurrentSnapshot.queryInterfaceTo (aCurrentSnapshot);
1265
1266 return S_OK;
1267}
1268
1269STDMETHODIMP Machine::COMGETTER(SnapshotCount) (ULONG *aSnapshotCount)
1270{
1271 if (!aSnapshotCount)
1272 return E_POINTER;
1273
1274 AutoCaller autoCaller (this);
1275 CheckComRCReturnRC (autoCaller.rc());
1276
1277 AutoReaderLock alock (this);
1278
1279 *aSnapshotCount = !mData->mFirstSnapshot ? 0 :
1280 mData->mFirstSnapshot->descendantCount() + 1 /* self */;
1281
1282 return S_OK;
1283}
1284
1285STDMETHODIMP Machine::COMGETTER(CurrentStateModified) (BOOL *aCurrentStateModified)
1286{
1287 if (!aCurrentStateModified)
1288 return E_POINTER;
1289
1290 AutoCaller autoCaller (this);
1291 CheckComRCReturnRC (autoCaller.rc());
1292
1293 AutoReaderLock alock (this);
1294
1295 /*
1296 * Note: for machines with no snapshots, we always return FALSE
1297 * (mData->mCurrentStateModified will be TRUE in this case, for historical
1298 * reasons :)
1299 */
1300
1301 *aCurrentStateModified = !mData->mFirstSnapshot ? FALSE :
1302 mData->mCurrentStateModified;
1303
1304 return S_OK;
1305}
1306
1307STDMETHODIMP
1308Machine::COMGETTER(SharedFolders) (ISharedFolderCollection **aSharedFolders)
1309{
1310 if (!aSharedFolders)
1311 return E_POINTER;
1312
1313 AutoCaller autoCaller (this);
1314 CheckComRCReturnRC (autoCaller.rc());
1315
1316 AutoReaderLock alock (this);
1317
1318 ComObjPtr <SharedFolderCollection> coll;
1319 coll.createObject();
1320 coll->init (mHWData->mSharedFolders);
1321 coll.queryInterfaceTo (aSharedFolders);
1322
1323 return S_OK;
1324}
1325
1326STDMETHODIMP
1327Machine::COMGETTER(ClipboardMode) (ClipboardMode_T *aClipboardMode)
1328{
1329 if (!aClipboardMode)
1330 return E_POINTER;
1331
1332 AutoCaller autoCaller (this);
1333 CheckComRCReturnRC (autoCaller.rc());
1334
1335 AutoReaderLock alock (this);
1336
1337 *aClipboardMode = mHWData->mClipboardMode;
1338
1339 return S_OK;
1340}
1341
1342STDMETHODIMP
1343Machine::COMSETTER(ClipboardMode) (ClipboardMode_T aClipboardMode)
1344{
1345 AutoCaller autoCaller (this);
1346 CheckComRCReturnRC (autoCaller.rc());
1347
1348 AutoLock alock (this);
1349
1350 CHECK_SETTER();
1351
1352 mHWData.backup();
1353 mHWData->mClipboardMode = aClipboardMode;
1354
1355 return S_OK;
1356}
1357
1358// IMachine methods
1359/////////////////////////////////////////////////////////////////////////////
1360
1361STDMETHODIMP Machine::SetBootOrder (ULONG aPosition, DeviceType_T aDevice)
1362{
1363 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
1364 return setError (E_INVALIDARG,
1365 tr ("Invalid boot position: %lu (must be in range [1, %lu])"),
1366 aPosition, SchemaDefs::MaxBootPosition);
1367
1368 if (aDevice == DeviceType_USBDevice)
1369 return setError (E_FAIL,
1370 tr ("Booting from USB devices is not currently supported"));
1371
1372 AutoCaller autoCaller (this);
1373 CheckComRCReturnRC (autoCaller.rc());
1374
1375 AutoLock alock (this);
1376
1377 CHECK_SETTER();
1378
1379 mHWData.backup();
1380 mHWData->mBootOrder [aPosition - 1] = aDevice;
1381
1382 return S_OK;
1383}
1384
1385STDMETHODIMP Machine::GetBootOrder (ULONG aPosition, DeviceType_T *aDevice)
1386{
1387 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
1388 return setError (E_INVALIDARG,
1389 tr ("Invalid boot position: %lu (must be in range [1, %lu])"),
1390 aPosition, SchemaDefs::MaxBootPosition);
1391
1392 AutoCaller autoCaller (this);
1393 CheckComRCReturnRC (autoCaller.rc());
1394
1395 AutoReaderLock alock (this);
1396
1397 *aDevice = mHWData->mBootOrder [aPosition - 1];
1398
1399 return S_OK;
1400}
1401
1402STDMETHODIMP Machine::AttachHardDisk (INPTR GUIDPARAM aId,
1403 DiskControllerType_T aCtl, LONG aDev)
1404{
1405 Guid id = aId;
1406
1407 if (id.isEmpty() ||
1408 aCtl == DiskControllerType_InvalidController ||
1409 aDev < 0 || aDev > 1)
1410 return E_INVALIDARG;
1411
1412 AutoCaller autoCaller (this);
1413 CheckComRCReturnRC (autoCaller.rc());
1414
1415 AutoLock alock (this);
1416
1417 CHECK_SETTER();
1418
1419 if (!mData->mRegistered)
1420 return setError (E_FAIL,
1421 tr ("Cannot attach hard disks to an unregistered machine"));
1422
1423 AssertReturn (mData->mMachineState != MachineState_Saved, E_FAIL);
1424
1425 if (mData->mMachineState >= MachineState_Running)
1426 return setError (E_FAIL,
1427 tr ("Invalid machine state: %d"), mData->mMachineState);
1428
1429 /* see if the device on the controller is already busy */
1430 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
1431 it != mHDData->mHDAttachments.end(); ++ it)
1432 {
1433 ComObjPtr <HardDiskAttachment> hda = *it;
1434 if (hda->controller() == aCtl && hda->deviceNumber() == aDev)
1435 {
1436 ComObjPtr <HardDisk> hd = hda->hardDisk();
1437 AutoLock hdLock (hd);
1438 return setError (E_FAIL,
1439 tr ("Hard disk '%ls' is already attached to device slot %d "
1440 "on controller %d"),
1441 hd->toString().raw(), aDev, aCtl);
1442 }
1443 }
1444
1445 /* find a hard disk by UUID */
1446 ComObjPtr <HardDisk> hd;
1447 HRESULT rc = mParent->getHardDisk (id, hd);
1448 if (FAILED (rc))
1449 return rc;
1450
1451 AutoLock hdLock (hd);
1452
1453 if (hd->isDifferencing())
1454 return setError (E_FAIL,
1455 tr ("Cannot attach the differencing hard disk '%ls'"),
1456 hd->toString().raw());
1457
1458 bool dirty = false;
1459
1460 switch (hd->type())
1461 {
1462 case HardDiskType_ImmutableHardDisk:
1463 {
1464 Assert (hd->machineId().isEmpty());
1465 /*
1466 * increase readers to protect from unregistration
1467 * until rollback()/commit() is done
1468 */
1469 hd->addReader();
1470 // LogTraceMsg (("A: %ls proteced\n", hd->toString().raw()));
1471 dirty = true;
1472 break;
1473 }
1474 case HardDiskType_WritethroughHardDisk:
1475 {
1476 Assert (hd->children().size() == 0);
1477 Assert (hd->snapshotId().isEmpty());
1478 /* fall through */
1479 }
1480 case HardDiskType_NormalHardDisk:
1481 {
1482 if (hd->machineId().isEmpty())
1483 {
1484 /* attach directly */
1485 hd->setMachineId (mData->mUuid);
1486 // LogTraceMsg (("A: %ls associated with %Vuuid\n",
1487 // hd->toString().raw(), mData->mUuid.raw()));
1488 dirty = true;
1489 }
1490 else
1491 {
1492 /* determine what the hard disk is already attached to */
1493 if (hd->snapshotId().isEmpty())
1494 {
1495 /* attached to some VM in its current state */
1496 if (hd->machineId() == mData->mUuid)
1497 {
1498 /*
1499 * attached to us, either in the backed up list of the
1500 * attachments or in the current one; the former is ok
1501 * (reattachment takes place within the same
1502 * "transaction") the latter is an error so check for it
1503 */
1504 for (HDData::HDAttachmentList::const_iterator it =
1505 mHDData->mHDAttachments.begin();
1506 it != mHDData->mHDAttachments.end(); ++ it)
1507 {
1508 if ((*it)->hardDisk().equalsTo (hd))
1509 {
1510 return setError (E_FAIL,
1511 tr ("Normal/Writethrough hard disk '%ls' is "
1512 "currently attached to device slot %d "
1513 "on controller %d of this machine"),
1514 hd->toString().raw(),
1515 (*it)->deviceNumber(), (*it)->controller());
1516 }
1517 }
1518 /*
1519 * dirty = false to indicate we didn't set machineId
1520 * and prevent it from being reset in DetachHardDisk()
1521 */
1522 // LogTraceMsg (("A: %ls found in old\n", hd->toString().raw()));
1523 }
1524 else
1525 {
1526 /* attached to other VM */
1527 return setError (E_FAIL,
1528 tr ("Normal/Writethrough hard disk '%ls' is "
1529 "currently attached to a machine with "
1530 "UUID {%Vuuid}"),
1531 hd->toString().raw(), hd->machineId().raw());
1532 }
1533 }
1534 else
1535 {
1536 /*
1537 * here we go when the HardDiskType_NormalHardDisk
1538 * is attached to some VM (probably to this one, too)
1539 * at some particular snapshot, so we can create a diff
1540 * based on it
1541 */
1542 Assert (!hd->machineId().isEmpty());
1543 /*
1544 * increase readers to protect from unregistration
1545 * until rollback()/commit() is done
1546 */
1547 hd->addReader();
1548 // LogTraceMsg (("A: %ls proteced\n", hd->toString().raw()));
1549 dirty = true;
1550 }
1551 }
1552
1553 break;
1554 }
1555 }
1556
1557 ComObjPtr <HardDiskAttachment> attachment;
1558 attachment.createObject();
1559 attachment->init (hd, aCtl, aDev, dirty);
1560
1561 mHDData.backup();
1562 mHDData->mHDAttachments.push_back (attachment);
1563 // LogTraceMsg (("A: %ls attached\n", hd->toString().raw()));
1564
1565 /* note: diff images are actually created only in commit() */
1566
1567 return S_OK;
1568}
1569
1570STDMETHODIMP Machine::GetHardDisk (DiskControllerType_T aCtl,
1571 LONG aDev, IHardDisk **aHardDisk)
1572{
1573 if (aCtl == DiskControllerType_InvalidController ||
1574 aDev < 0 || aDev > 1)
1575 return E_INVALIDARG;
1576
1577 AutoCaller autoCaller (this);
1578 CheckComRCReturnRC (autoCaller.rc());
1579
1580 AutoReaderLock alock (this);
1581
1582 *aHardDisk = NULL;
1583
1584 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
1585 it != mHDData->mHDAttachments.end(); ++ it)
1586 {
1587 ComObjPtr <HardDiskAttachment> hda = *it;
1588 if (hda->controller() == aCtl && hda->deviceNumber() == aDev)
1589 {
1590 hda->hardDisk().queryInterfaceTo (aHardDisk);
1591 return S_OK;
1592 }
1593 }
1594
1595 return setError (E_INVALIDARG,
1596 tr ("No hard disk attached to device slot %d on controller %d"),
1597 aDev, aCtl);
1598}
1599
1600STDMETHODIMP Machine::DetachHardDisk (DiskControllerType_T aCtl, LONG aDev)
1601{
1602 if (aCtl == DiskControllerType_InvalidController ||
1603 aDev < 0 || aDev > 1)
1604 return E_INVALIDARG;
1605
1606 AutoCaller autoCaller (this);
1607 CheckComRCReturnRC (autoCaller.rc());
1608
1609 AutoLock alock (this);
1610
1611 CHECK_SETTER();
1612
1613 AssertReturn (mData->mMachineState != MachineState_Saved, E_FAIL);
1614
1615 if (mData->mMachineState >= MachineState_Running)
1616 return setError (E_FAIL,
1617 tr ("Invalid machine state: %d"), mData->mMachineState);
1618
1619 for (HDData::HDAttachmentList::iterator it = mHDData->mHDAttachments.begin();
1620 it != mHDData->mHDAttachments.end(); ++ it)
1621 {
1622 ComObjPtr <HardDiskAttachment> hda = *it;
1623 if (hda->controller() == aCtl && hda->deviceNumber() == aDev)
1624 {
1625 ComObjPtr <HardDisk> hd = hda->hardDisk();
1626 AutoLock hdLock (hd);
1627
1628 ComAssertRet (hd->children().size() == 0 &&
1629 hd->machineId() == mData->mUuid, E_FAIL);
1630
1631 if (hda->isDirty())
1632 {
1633 switch (hd->type())
1634 {
1635 case HardDiskType_ImmutableHardDisk:
1636 {
1637 /* decrease readers increased in AttachHardDisk() */
1638 hd->releaseReader();
1639 // LogTraceMsg (("D: %ls released\n", hd->toString().raw()));
1640 break;
1641 }
1642 case HardDiskType_WritethroughHardDisk:
1643 {
1644 /* deassociate from this machine */
1645 hd->setMachineId (Guid());
1646 // LogTraceMsg (("D: %ls deassociated\n", hd->toString().raw()));
1647 break;
1648 }
1649 case HardDiskType_NormalHardDisk:
1650 {
1651 if (hd->snapshotId().isEmpty())
1652 {
1653 /* deassociate from this machine */
1654 hd->setMachineId (Guid());
1655 // LogTraceMsg (("D: %ls deassociated\n", hd->toString().raw()));
1656 }
1657 else
1658 {
1659 /* decrease readers increased in AttachHardDisk() */
1660 hd->releaseReader();
1661 // LogTraceMsg (("%ls released\n", hd->toString().raw()));
1662 }
1663
1664 break;
1665 }
1666 }
1667 }
1668
1669 mHDData.backup();
1670 /*
1671 * we cannot use erase (it) below because backup() above will create
1672 * a copy of the list and make this copy active, but the iterator
1673 * still refers to the original and is not valid for a copy
1674 */
1675 mHDData->mHDAttachments.remove (hda);
1676 // LogTraceMsg (("D: %ls detached\n", hd->toString().raw()));
1677
1678 /*
1679 * note: Non-dirty hard disks are actually deassociated
1680 * and diff images are deleted only in commit()
1681 */
1682
1683 return S_OK;
1684 }
1685 }
1686
1687 return setError (E_INVALIDARG,
1688 tr ("No hard disk attached to device slot %d on controller %d"),
1689 aDev, aCtl);
1690}
1691
1692STDMETHODIMP Machine::GetNetworkAdapter (ULONG slot, INetworkAdapter **adapter)
1693{
1694 if (!adapter)
1695 return E_POINTER;
1696 if (slot >= ELEMENTS (mNetworkAdapters))
1697 return setError (E_INVALIDARG, tr ("Invalid slot number: %d"), slot);
1698
1699 AutoCaller autoCaller (this);
1700 CheckComRCReturnRC (autoCaller.rc());
1701
1702 AutoReaderLock alock (this);
1703
1704 mNetworkAdapters [slot].queryInterfaceTo (adapter);
1705
1706 return S_OK;
1707}
1708
1709/**
1710 * Returns the extra data key name following the given key. If the key
1711 * is not found, an error is returned. If NULL is supplied, the first
1712 * key will be returned. If key is the last item, NULL will be returned.
1713 *
1714 * @returns COM status code
1715 * @param key extra data key name
1716 * @param nextKey name of the key following "key". NULL if "key" is the last.
1717 * @param nextValue value of the key following "key". Optional parameter.
1718 */
1719STDMETHODIMP Machine::GetNextExtraDataKey(INPTR BSTR key, BSTR *nextKey, BSTR *nextValue)
1720{
1721 if (!nextKey)
1722 return E_POINTER;
1723
1724 AutoCaller autoCaller (this);
1725 CheckComRCReturnRC (autoCaller.rc());
1726
1727 AutoReaderLock alock (this);
1728
1729 /* start with nothing found */
1730 *nextKey = NULL;
1731
1732 /*
1733 * if we're ready and isConfigLocked() is FALSE then it means
1734 * that no config file exists yet, so return shortly
1735 */
1736 if (!isConfigLocked())
1737 return S_OK;
1738
1739 HRESULT rc = S_OK;
1740
1741 /* load the config file */
1742 CFGHANDLE configLoader = 0;
1743 rc = openConfigLoader (&configLoader);
1744 if (FAILED (rc))
1745 return E_FAIL;
1746
1747 CFGNODE machineNode;
1748 CFGNODE extraDataNode;
1749
1750 /* navigate to the right position */
1751 if (VBOX_SUCCESS(CFGLDRGetNode(configLoader, "VirtualBox/Machine", 0, &machineNode)) &&
1752 VBOX_SUCCESS(CFGLDRGetChildNode(machineNode, "ExtraData", 0, &extraDataNode)))
1753 {
1754 /* check if it exists */
1755 bool found = false;
1756 unsigned count;
1757 CFGNODE extraDataItemNode;
1758 CFGLDRCountChildren(extraDataNode, "ExtraDataItem", &count);
1759 for (unsigned i = 0; (i < count) && (found == false); i++)
1760 {
1761 Bstr name;
1762 CFGLDRGetChildNode(extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1763 CFGLDRQueryBSTR(extraDataItemNode, "name", name.asOutParam());
1764
1765 /* if we're supposed to return the first one */
1766 if (key == NULL)
1767 {
1768 name.cloneTo(nextKey);
1769 if (nextValue)
1770 CFGLDRQueryBSTR(extraDataItemNode, "value", nextValue);
1771 found = true;
1772 }
1773 /* did we find the key we're looking for? */
1774 else if (name == key)
1775 {
1776 found = true;
1777 /* is there another item? */
1778 if (i + 1 < count)
1779 {
1780 CFGLDRGetChildNode(extraDataNode, "ExtraDataItem", i + 1, &extraDataItemNode);
1781 CFGLDRQueryBSTR(extraDataItemNode, "name", name.asOutParam());
1782 name.cloneTo(nextKey);
1783 if (nextValue)
1784 CFGLDRQueryBSTR(extraDataItemNode, "value", nextValue);
1785 found = true;
1786 }
1787 else
1788 {
1789 /* it's the last one */
1790 *nextKey = NULL;
1791 }
1792 }
1793 CFGLDRReleaseNode(extraDataItemNode);
1794 }
1795
1796 /* if we haven't found the key, it's an error */
1797 if (!found)
1798 rc = setError(E_FAIL, tr("Could not find extra data key"));
1799
1800 CFGLDRReleaseNode(extraDataNode);
1801 CFGLDRReleaseNode(machineNode);
1802 }
1803
1804 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
1805
1806 return rc;
1807}
1808
1809/**
1810 * Returns associated extra data from the configuration. If the key does
1811 * not exist, NULL will be stored in the output pointer.
1812 *
1813 * @returns COM status code
1814 * @param key extra data key
1815 * @param value address of result pointer
1816 */
1817STDMETHODIMP Machine::GetExtraData(INPTR BSTR key, BSTR *value)
1818{
1819 if (!key)
1820 return E_INVALIDARG;
1821 if (!value)
1822 return E_POINTER;
1823
1824 AutoCaller autoCaller (this);
1825 CheckComRCReturnRC (autoCaller.rc());
1826
1827 AutoReaderLock alock (this);
1828
1829 /* start with nothing found */
1830 *value = NULL;
1831
1832 /*
1833 * if we're ready and isConfigLocked() is FALSE then it means
1834 * that no config file exists yet, so return shortly
1835 */
1836 if (!isConfigLocked())
1837 return S_OK;
1838
1839 HRESULT rc = S_OK;
1840
1841 /* load the config file */
1842 CFGHANDLE configLoader = 0;
1843 rc = openConfigLoader (&configLoader);
1844 if (FAILED (rc))
1845 return E_FAIL;
1846
1847 CFGNODE machineNode;
1848 CFGNODE extraDataNode;
1849
1850 /* navigate to the right position */
1851 if (VBOX_SUCCESS(CFGLDRGetNode(configLoader, "VirtualBox/Machine", 0, &machineNode)) &&
1852 VBOX_SUCCESS(CFGLDRGetChildNode(machineNode, "ExtraData", 0, &extraDataNode)))
1853 {
1854 /* check if it exists */
1855 bool found = false;
1856 unsigned count;
1857 CFGNODE extraDataItemNode;
1858 CFGLDRCountChildren(extraDataNode, "ExtraDataItem", &count);
1859 for (unsigned i = 0; (i < count) && (found == false); i++)
1860 {
1861 Bstr name;
1862 CFGLDRGetChildNode(extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1863 CFGLDRQueryBSTR(extraDataItemNode, "name", name.asOutParam());
1864 if (name == key)
1865 {
1866 found = true;
1867 CFGLDRQueryBSTR(extraDataItemNode, "value", value);
1868 }
1869 CFGLDRReleaseNode(extraDataItemNode);
1870 }
1871
1872 CFGLDRReleaseNode(extraDataNode);
1873 CFGLDRReleaseNode(machineNode);
1874 }
1875
1876 rc = closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
1877
1878 return rc;
1879}
1880
1881/**
1882 * Stores associated extra data in the configuration. If the data value is NULL
1883 * then the corresponding extra data item is deleted. This method can be called
1884 * outside a session and therefore belongs to the non protected machine data.
1885 *
1886 * @param key extra data key
1887 * @param value extra data value
1888 *
1889 * @note Locks mParent for reading + this object for writing.
1890 */
1891STDMETHODIMP Machine::SetExtraData (INPTR BSTR key, INPTR BSTR value)
1892{
1893 if (!key)
1894 return E_INVALIDARG;
1895
1896 AutoCaller autoCaller (this);
1897 CheckComRCReturnRC (autoCaller.rc());
1898
1899 /* VirtualBox::onExtraDataCanChange() needs mParent lock */
1900 AutoMultiLock <2> alock (mParent->rlock(), this->wlock());
1901
1902 if (mType == IsSnapshotMachine)
1903 CHECK_SETTER();
1904
1905 bool changed = false;
1906 HRESULT rc = S_OK;
1907
1908 /*
1909 * if we're ready and isConfigLocked() is FALSE then it means
1910 * that no config file exists yet, so call saveSettings() to create one
1911 */
1912 if (!isConfigLocked())
1913 {
1914 rc = saveSettings (false /* aMarkCurStateAsModified */);
1915 if (FAILED (rc))
1916 return rc;
1917 }
1918
1919 /* load the config file */
1920 CFGHANDLE configLoader = 0;
1921 rc = openConfigLoader (&configLoader);
1922 if (FAILED (rc))
1923 return rc;
1924
1925 CFGNODE machineNode = 0;
1926 CFGNODE extraDataNode = 0;
1927
1928 int vrc = CFGLDRGetNode (configLoader, "VirtualBox/Machine", 0, &machineNode);
1929 if (VBOX_FAILURE (vrc))
1930 vrc = CFGLDRCreateNode (configLoader, "VirtualBox/Machine", &machineNode);
1931
1932 vrc = CFGLDRGetChildNode (machineNode, "ExtraData", 0, &extraDataNode);
1933 if (VBOX_FAILURE (vrc) && value)
1934 vrc = CFGLDRCreateChildNode (machineNode, "ExtraData", &extraDataNode);
1935
1936 if (extraDataNode)
1937 {
1938 CFGNODE extraDataItemNode = 0;
1939 Bstr oldVal;
1940
1941 unsigned count;
1942 CFGLDRCountChildren (extraDataNode, "ExtraDataItem", &count);
1943
1944 for (unsigned i = 0; i < count; i++)
1945 {
1946 CFGLDRGetChildNode (extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1947 Bstr name;
1948 CFGLDRQueryBSTR (extraDataItemNode, "name", name.asOutParam());
1949 if (name == key)
1950 {
1951 CFGLDRQueryBSTR (extraDataItemNode, "value", oldVal.asOutParam());
1952 break;
1953 }
1954 CFGLDRReleaseNode (extraDataItemNode);
1955 extraDataItemNode = 0;
1956 }
1957
1958 /*
1959 * When no key is found, oldVal is null
1960 * Note:
1961 * 1. when oldVal is null, |oldVal == (BSTR) NULL| is true
1962 * 2. we cannot do |oldVal != value| because it will compare
1963 * BSTR pointers instead of strings (due to type conversion ops)
1964 */
1965 changed = !(oldVal == value);
1966
1967 if (changed)
1968 {
1969 /* ask for permission from all listeners */
1970 if (!mParent->onExtraDataCanChange (mData->mUuid, key, value))
1971 {
1972 LogWarningFunc (("Someone vetoed! Change refused!\n"));
1973 rc = setError (E_ACCESSDENIED,
1974 tr ("Could not set extra data because someone refused "
1975 "the requested change of '%ls' to '%ls'"), key, value);
1976 }
1977 else
1978 {
1979 if (value)
1980 {
1981 if (!extraDataItemNode)
1982 {
1983 /* create a new item */
1984 CFGLDRAppendChildNode (extraDataNode, "ExtraDataItem",
1985 &extraDataItemNode);
1986 CFGLDRSetBSTR (extraDataItemNode, "name", key);
1987 }
1988 CFGLDRSetBSTR (extraDataItemNode, "value", value);
1989 }
1990 else
1991 {
1992 /* an old value does for sure exist here */
1993 CFGLDRDeleteNode (extraDataItemNode);
1994 extraDataItemNode = 0;
1995 }
1996 }
1997 }
1998
1999 if (extraDataItemNode)
2000 CFGLDRReleaseNode (extraDataItemNode);
2001
2002 CFGLDRReleaseNode (extraDataNode);
2003 }
2004
2005 CFGLDRReleaseNode (machineNode);
2006
2007 if (SUCCEEDED (rc) && changed)
2008 rc = closeConfigLoader (configLoader, true /* aSaveBeforeClose */);
2009 else
2010 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
2011
2012 /* fire an event */
2013 if (SUCCEEDED (rc) && changed)
2014 {
2015 mParent->onExtraDataChange (mData->mUuid, key, value);
2016 }
2017
2018 return rc;
2019}
2020
2021STDMETHODIMP Machine::SaveSettings()
2022{
2023 AutoCaller autoCaller (this);
2024 CheckComRCReturnRC (autoCaller.rc());
2025
2026 /* Under some circumstancies, saveSettings() needs mParent lock */
2027 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
2028
2029 CHECK_SETTER();
2030
2031 /* the settings file path may never be null */
2032 ComAssertRet (mData->mConfigFileFull, E_FAIL);
2033
2034 /* save all VM data excluding snapshots */
2035 return saveSettings();
2036}
2037
2038STDMETHODIMP Machine::DiscardSettings()
2039{
2040 AutoCaller autoCaller (this);
2041 CheckComRCReturnRC (autoCaller.rc());
2042
2043 AutoLock alock (this);
2044
2045 CHECK_SETTER();
2046
2047 /*
2048 * during this rollback, the session will be notified if data has
2049 * been actually changed
2050 */
2051 rollback (true /* aNotify */);
2052
2053 return S_OK;
2054}
2055
2056STDMETHODIMP Machine::DeleteSettings()
2057{
2058 AutoCaller autoCaller (this);
2059 CheckComRCReturnRC (autoCaller.rc());
2060
2061 AutoLock alock (this);
2062
2063 CHECK_SETTER();
2064
2065 if (mData->mRegistered)
2066 return setError (E_FAIL,
2067 tr ("Cannot delete settings of a registered machine"));
2068
2069 /* delete the settings only when the file actually exists */
2070 if (isConfigLocked())
2071 {
2072 unlockConfig();
2073 int vrc = RTFileDelete (Utf8Str (mData->mConfigFileFull));
2074 if (VBOX_FAILURE (vrc))
2075 return setError (E_FAIL,
2076 tr ("Could not delete the settings file '%ls' (%Vrc)"),
2077 mData->mConfigFileFull.raw(), vrc);
2078
2079 /* delete the Logs folder, nothing important should be left
2080 * there (we don't check for errors because the user might have
2081 * some private files there that we don't want to delete) */
2082 Utf8Str logFolder;
2083 getLogFolder (logFolder);
2084 Assert (!logFolder.isEmpty());
2085 if (RTDirExists (logFolder))
2086 {
2087 /* delete all VBox.log[.N] files from the Logs folder
2088 * (this must be in sync with the rotation logic in
2089 * Console::powerUpThread()) */
2090 Utf8Str log = Utf8StrFmt ("%s/VBox.log", logFolder.raw());
2091 RTFileDelete (log);
2092 for (int i = 3; i >= 0; i--)
2093 {
2094 log = Utf8StrFmt ("%s/VBox.log.%d", logFolder.raw(), i);
2095 RTFileDelete (log);
2096 }
2097
2098 RTDirRemove (logFolder);
2099 }
2100
2101 /* delete the Snapshots folder, nothing important should be left
2102 * there (we don't check for errors because the user might have
2103 * some private files there that we don't want to delete) */
2104 Utf8Str snapshotFolder = mUserData->mSnapshotFolderFull;
2105 Assert (!snapshotFolder.isEmpty());
2106 if (RTDirExists (snapshotFolder))
2107 RTDirRemove (snapshotFolder);
2108
2109 /* delete the directory that contains the settings file, but only
2110 * if it matches the VM name (i.e. a structure created by default in
2111 * openConfigLoader()) */
2112 {
2113 Utf8Str settingsDir;
2114 if (isInOwnDir (&settingsDir))
2115 RTDirRemove (settingsDir);
2116 }
2117 }
2118
2119 return S_OK;
2120}
2121
2122STDMETHODIMP Machine::GetSnapshot (INPTR GUIDPARAM aId, ISnapshot **aSnapshot)
2123{
2124 if (!aSnapshot)
2125 return E_POINTER;
2126
2127 AutoCaller autoCaller (this);
2128 CheckComRCReturnRC (autoCaller.rc());
2129
2130 AutoReaderLock alock (this);
2131
2132 Guid id = aId;
2133 ComObjPtr <Snapshot> snapshot;
2134
2135 HRESULT rc = findSnapshot (id, snapshot, true /* aSetError */);
2136 snapshot.queryInterfaceTo (aSnapshot);
2137
2138 return rc;
2139}
2140
2141STDMETHODIMP Machine::FindSnapshot (INPTR BSTR aName, ISnapshot **aSnapshot)
2142{
2143 if (!aName)
2144 return E_INVALIDARG;
2145 if (!aSnapshot)
2146 return E_POINTER;
2147
2148 AutoCaller autoCaller (this);
2149 CheckComRCReturnRC (autoCaller.rc());
2150
2151 AutoReaderLock alock (this);
2152
2153 ComObjPtr <Snapshot> snapshot;
2154
2155 HRESULT rc = findSnapshot (aName, snapshot, true /* aSetError */);
2156 snapshot.queryInterfaceTo (aSnapshot);
2157
2158 return rc;
2159}
2160
2161STDMETHODIMP Machine::SetCurrentSnapshot (INPTR GUIDPARAM aId)
2162{
2163 /// @todo (dmik) don't forget to set
2164 // mData->mCurrentStateModified to FALSE
2165
2166 return setError (E_NOTIMPL, "Not implemented");
2167}
2168
2169STDMETHODIMP
2170Machine::CreateSharedFolder (INPTR BSTR aName, INPTR BSTR aHostPath)
2171{
2172 if (!aName || !aHostPath)
2173 return E_INVALIDARG;
2174
2175 AutoCaller autoCaller (this);
2176 CheckComRCReturnRC (autoCaller.rc());
2177
2178 AutoLock alock (this);
2179
2180 CHECK_SETTER();
2181
2182 /// @todo (dmik) check global shared folders when they are done
2183
2184 ComObjPtr <SharedFolder> sharedFolder;
2185 HRESULT rc = findSharedFolder (aName, sharedFolder, false /* aSetError */);
2186 if (SUCCEEDED (rc))
2187 return setError (E_FAIL,
2188 tr ("Shared folder named '%ls' already exists"), aName);
2189
2190 sharedFolder.createObject();
2191 rc = sharedFolder->init (machine(), aName, aHostPath);
2192 if (FAILED (rc))
2193 return rc;
2194
2195 BOOL accessible = FALSE;
2196 rc = sharedFolder->COMGETTER(Accessible) (&accessible);
2197 if (FAILED (rc))
2198 return rc;
2199
2200 if (!accessible)
2201 return setError (E_FAIL,
2202 tr ("Shared folder path '%ls' is not accessible"), aHostPath);
2203
2204 mHWData.backup();
2205 mHWData->mSharedFolders.push_back (sharedFolder);
2206
2207 return S_OK;
2208}
2209
2210STDMETHODIMP Machine::RemoveSharedFolder (INPTR BSTR aName)
2211{
2212 if (!aName)
2213 return E_INVALIDARG;
2214
2215 AutoCaller autoCaller (this);
2216 CheckComRCReturnRC (autoCaller.rc());
2217
2218 AutoReaderLock alock (this);
2219
2220 CHECK_SETTER();
2221
2222 ComObjPtr <SharedFolder> sharedFolder;
2223 HRESULT rc = findSharedFolder (aName, sharedFolder, true /* aSetError */);
2224 if (FAILED (rc))
2225 return rc;
2226
2227 mHWData.backup();
2228 mHWData->mSharedFolders.remove (sharedFolder);
2229
2230 return S_OK;
2231}
2232
2233// public methods for internal purposes
2234/////////////////////////////////////////////////////////////////////////////
2235
2236/**
2237 * Returns the session machine object associated with the this machine.
2238 * The returned session machine is null if no direct session is currently open.
2239 *
2240 * @Note locks this object for reading.
2241 */
2242ComObjPtr <SessionMachine> Machine::sessionMachine()
2243{
2244 ComObjPtr <SessionMachine> sm;
2245
2246 AutoCaller autoCaller (this);
2247 /* the machine may be inaccessible, so don't assert below */
2248 if (FAILED (autoCaller.rc()))
2249 return sm;
2250
2251 AutoReaderLock alock (this);
2252
2253 sm = mData->mSession.mMachine;
2254 Assert (!sm.isNull() ||
2255 mData->mSession.mState != SessionState_SessionOpen);
2256
2257 return sm;
2258}
2259
2260/**
2261 * Calculates the absolute path of the given path taking the directory of
2262 * the machine settings file as the current directory.
2263 *
2264 * @param aPath path to calculate the absolute path for
2265 * @param aResult where to put the result (used only on success,
2266 * so can be the same Utf8Str instance as passed as \a aPath)
2267 * @return VirtualBox result
2268 *
2269 * @note Locks this object for reading.
2270 */
2271int Machine::calculateFullPath (const char *aPath, Utf8Str &aResult)
2272{
2273 AutoCaller autoCaller (this);
2274 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
2275
2276 AutoReaderLock alock (this);
2277
2278 AssertReturn (!mData->mConfigFileFull.isNull(), VERR_GENERAL_FAILURE);
2279
2280 Utf8Str settingsDir = mData->mConfigFileFull;
2281
2282 RTPathStripFilename (settingsDir.mutableRaw());
2283 char folder [RTPATH_MAX];
2284 int vrc = RTPathAbsEx (settingsDir, aPath,
2285 folder, sizeof (folder));
2286 if (VBOX_SUCCESS (vrc))
2287 aResult = folder;
2288
2289 return vrc;
2290}
2291
2292/**
2293 * Tries to calculate the relative path of the given absolute path using the
2294 * directory of the machine settings file as the base directory.
2295 *
2296 * @param aPath absolute path to calculate the relative path for
2297 * @param aResult where to put the result (used only when it's possible to
2298 * make a relative path from the given absolute path;
2299 * otherwise left untouched)
2300 *
2301 * @note Locks this object for reading.
2302 */
2303void Machine::calculateRelativePath (const char *aPath, Utf8Str &aResult)
2304{
2305 AutoCaller autoCaller (this);
2306 AssertComRCReturn (autoCaller.rc(), (void) 0);
2307
2308 AutoReaderLock alock (this);
2309
2310 AssertReturnVoid (!mData->mConfigFileFull.isNull());
2311
2312 Utf8Str settingsDir = mData->mConfigFileFull;
2313
2314 RTPathStripFilename (settingsDir.mutableRaw());
2315 if (RTPathStartsWith (aPath, settingsDir))
2316 {
2317 /* when assigning, we create a separate Utf8Str instance because both
2318 * aPath and aResult can point to the same memory location when this
2319 * func is called (if we just do aResult = aPath, aResult will be freed
2320 * first, and since its the same as aPath, an attempt to copy garbage
2321 * will be made. */
2322 aResult = Utf8Str (aPath + settingsDir.length() + 1);
2323 }
2324}
2325
2326/**
2327 * Returns the full path to the machine's log folder in the
2328 * \a aLogFolder argument.
2329 */
2330void Machine::getLogFolder (Utf8Str &aLogFolder)
2331{
2332 AutoCaller autoCaller (this);
2333 AssertComRCReturn (autoCaller.rc(), (void) 0);
2334
2335 AutoReaderLock alock (this);
2336
2337 Utf8Str settingsDir;
2338 if (isInOwnDir (&settingsDir))
2339 {
2340 /* Log folder is <Machines>/<VM_Name>/Logs */
2341 aLogFolder = Utf8StrFmt ("%s%cLogs", settingsDir.raw(), RTPATH_DELIMITER);
2342 }
2343 else
2344 {
2345 /* Log folder is <Machines>/<VM_SnapshotFolder>/Logs */
2346 Assert (!mUserData->mSnapshotFolderFull.isEmpty());
2347 aLogFolder = Utf8StrFmt ("%ls%cLogs", mUserData->mSnapshotFolderFull.raw(),
2348 RTPATH_DELIMITER);
2349 }
2350}
2351
2352/**
2353 * @note Locks mParent and this object for writing,
2354 * calls the client process (outside the lock).
2355 */
2356HRESULT Machine::openSession (IInternalSessionControl *aControl)
2357{
2358 LogFlowThisFuncEnter();
2359
2360 AssertReturn (aControl, E_FAIL);
2361
2362 AutoCaller autoCaller (this);
2363 CheckComRCReturnRC (autoCaller.rc());
2364
2365 /* We need VirtualBox lock because of Progress::notifyComplete() */
2366 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
2367
2368 if (!mData->mRegistered)
2369 return setError (E_UNEXPECTED,
2370 tr ("The machine '%ls' is not registered"), mUserData->mName.raw());
2371
2372 LogFlowThisFunc (("mSession.mState=%d\n", mData->mSession.mState));
2373
2374 if (mData->mSession.mState == SessionState_SessionOpen ||
2375 mData->mSession.mState == SessionState_SessionClosing)
2376 return setError (E_ACCESSDENIED,
2377 tr ("A session for the machine '%ls' is currently open "
2378 "(or being closed)"),
2379 mUserData->mName.raw());
2380
2381 /* may not be Running */
2382 AssertReturn (mData->mMachineState < MachineState_Running, E_FAIL);
2383
2384 if (mData->mSession.mState == SessionState_SessionSpawning)
2385 {
2386 /*
2387 * this machine awaits for a spawning session to be opened,
2388 * so reject any other open attempts from processes other than
2389 * one started by #openRemoteSession().
2390 */
2391
2392 RTPROCESS pid = NIL_RTPROCESS; AssertCompile (sizeof (ULONG) == sizeof (RTPROCESS));
2393 aControl->GetPID ((ULONG *)&pid);
2394
2395 LogFlowThisFunc (("mSession.mPid=%d(0x%x)\n",
2396 mData->mSession.mPid, mData->mSession.mPid));
2397 LogFlowThisFunc (("session.pid=%d(0x%x)\n", pid, pid));
2398
2399 if (mData->mSession.mPid != pid)
2400 return setError (E_ACCESSDENIED,
2401 tr ("An unexpected process (PID=0x%08X) has tried to open a direct "
2402 "session with the machine named '%ls', while only a process "
2403 "started by OpenRemoteSession (PID=0x%08X) is allowed"),
2404 pid, mUserData->mName.raw(), mData->mSession.mPid);
2405 }
2406
2407 /* create a SessionMachine object */
2408 ComObjPtr <SessionMachine> sessionMachine;
2409 sessionMachine.createObject();
2410 HRESULT rc = sessionMachine->init (this);
2411 AssertComRC (rc);
2412
2413 if (SUCCEEDED (rc))
2414 {
2415 /*
2416 * Set the session state to Spawning to protect against subsequent
2417 * attempts to open a session and to unregister the machine after
2418 * we leave the lock.
2419 */
2420 SessionState_T origState = mData->mSession.mState;
2421 mData->mSession.mState = SessionState_SessionSpawning;
2422
2423 /*
2424 * Leave the lock before calling the client process -- it will call
2425 * Machine/SessionMachine methods. Leaving the lock here is quite safe
2426 * because the state is Spawning, so that openRemotesession() and
2427 * openExistingSession() calls will fail. This method, called before we
2428 * enter the lock again, will fail because of the wrong PID.
2429 *
2430 * Note that mData->mSession.mRemoteControls accessed outside
2431 * the lock may not be modified when state is Spawning, so it's safe.
2432 */
2433 alock.leave();
2434
2435 LogFlowThisFunc (("Calling AssignMachine()...\n"));
2436 rc = aControl->AssignMachine (sessionMachine);
2437 LogFlowThisFunc (("AssignMachine() returned %08X\n", rc));
2438
2439 /* The failure may w/o any error info (from RPC), so provide one */
2440 if (FAILED (rc))
2441 setError (rc,
2442 tr ("Failed to assign the machine to the session"));
2443
2444 if (SUCCEEDED (rc) && origState == SessionState_SessionSpawning)
2445 {
2446 /* complete the remote session initialization */
2447
2448 /* get the console from the direct session */
2449 ComPtr <IConsole> console;
2450 rc = aControl->GetRemoteConsole (console.asOutParam());
2451 ComAssertComRC (rc);
2452
2453 if (SUCCEEDED (rc) && !console)
2454 {
2455 ComAssert (!!console);
2456 rc = E_FAIL;
2457 }
2458
2459 /* assign machine & console to the remote sesion */
2460 if (SUCCEEDED (rc))
2461 {
2462 /*
2463 * after openRemoteSession(), the first and the only
2464 * entry in remoteControls is that remote session
2465 */
2466 LogFlowThisFunc (("Calling AssignRemoteMachine()...\n"));
2467 rc = mData->mSession.mRemoteControls.front()->
2468 AssignRemoteMachine (sessionMachine, console);
2469 LogFlowThisFunc (("AssignRemoteMachine() returned %08X\n", rc));
2470
2471 /* The failure may w/o any error info (from RPC), so provide one */
2472 if (FAILED (rc))
2473 setError (rc,
2474 tr ("Failed to assign the machine to the remote session"));
2475 }
2476
2477 if (FAILED (rc))
2478 aControl->Uninitialize();
2479 }
2480
2481 /* enter the lock again */
2482 alock.enter();
2483
2484 /* Restore the session state */
2485 mData->mSession.mState = origState;
2486 }
2487
2488 /* finalize spawning amyway (this is why we don't return on errors above) */
2489 if (mData->mSession.mState == SessionState_SessionSpawning)
2490 {
2491 /* Note that the progress object is finalized later */
2492
2493 /* We don't reset mSession.mPid here because it is necessary for
2494 * SessionMachine::uninit() to reap the child process later. */
2495
2496 if (FAILED (rc))
2497 {
2498 /* Remove the remote control from the list on failure
2499 * and reset session state to Closed. */
2500 mData->mSession.mRemoteControls.clear();
2501 mData->mSession.mState = SessionState_SessionClosed;
2502 }
2503 }
2504
2505 if (SUCCEEDED (rc))
2506 {
2507 /* memorize the direct session control */
2508 mData->mSession.mDirectControl = aControl;
2509 mData->mSession.mState = SessionState_SessionOpen;
2510 /* associate the SessionMachine with this Machine */
2511 mData->mSession.mMachine = sessionMachine;
2512 }
2513
2514 if (mData->mSession.mProgress)
2515 {
2516 /* finalize the progress after setting the state, for consistency */
2517 mData->mSession.mProgress->notifyComplete (rc);
2518 mData->mSession.mProgress.setNull();
2519 }
2520
2521 /* uninitialize the created session machine on failure */
2522 if (FAILED (rc))
2523 sessionMachine->uninit();
2524
2525 LogFlowThisFunc (("rc=%08X\n", rc));
2526 LogFlowThisFuncLeave();
2527 return rc;
2528}
2529
2530/**
2531 * @note Locks this object for writing, calls the client process
2532 * (inside the lock).
2533 */
2534HRESULT Machine::openRemoteSession (IInternalSessionControl *aControl,
2535 INPTR BSTR aType, Progress *aProgress)
2536{
2537 LogFlowThisFuncEnter();
2538
2539 AssertReturn (aControl, E_FAIL);
2540 AssertReturn (aProgress, E_FAIL);
2541
2542 AutoCaller autoCaller (this);
2543 CheckComRCReturnRC (autoCaller.rc());
2544
2545 AutoLock alock (this);
2546
2547 if (!mData->mRegistered)
2548 return setError (E_UNEXPECTED,
2549 tr ("The machine '%ls' is not registered"), mUserData->mName.raw());
2550
2551 LogFlowThisFunc (("mSession.mState=%d\n", mData->mSession.mState));
2552
2553 if (mData->mSession.mState == SessionState_SessionOpen ||
2554 mData->mSession.mState == SessionState_SessionSpawning ||
2555 mData->mSession.mState == SessionState_SessionClosing)
2556 return setError (E_ACCESSDENIED,
2557 tr ("A session for the machine '%ls' is currently open "
2558 "(or being opened or closed)"),
2559 mUserData->mName.raw());
2560
2561 /* may not be Running */
2562 AssertReturn (mData->mMachineState < MachineState_Running, E_FAIL);
2563
2564 /* get the path to the executable */
2565 char path [RTPATH_MAX];
2566 RTPathProgram (path, RTPATH_MAX);
2567 size_t sz = strlen (path);
2568 path [sz++] = RTPATH_DELIMITER;
2569 path [sz] = 0;
2570 char *cmd = path + sz;
2571 sz = RTPATH_MAX - sz;
2572
2573 int vrc = VINF_SUCCESS;
2574 RTPROCESS pid = NIL_RTPROCESS;
2575
2576 Bstr type (aType);
2577 if (type == "gui")
2578 {
2579 const char VirtualBox_exe[] = "VirtualBox" HOSTSUFF_EXE;
2580 Assert (sz >= sizeof (VirtualBox_exe));
2581 strcpy (cmd, VirtualBox_exe);
2582
2583 Utf8Str idStr = mData->mUuid.toString();
2584 const char * args[] = {path, "-startvm", idStr, 0 };
2585 vrc = RTProcCreate (path, args, NULL, 0, &pid);
2586 }
2587 else
2588#ifdef VBOX_VRDP
2589 if (type == "vrdp")
2590 {
2591 const char VBoxVRDP_exe[] = "VBoxVRDP" HOSTSUFF_EXE;
2592 Assert (sz >= sizeof (VBoxVRDP_exe));
2593 strcpy (cmd, VBoxVRDP_exe);
2594
2595 Utf8Str idStr = mData->mUuid.toString();
2596 const char * args[] = {path, "-startvm", idStr, 0 };
2597 vrc = RTProcCreate (path, args, NULL, 0, &pid);
2598 }
2599 else
2600#endif /* VBOX_VRDP */
2601 if (type == "capture")
2602 {
2603 const char VBoxVRDP_exe[] = "VBoxVRDP" HOSTSUFF_EXE;
2604 Assert (sz >= sizeof (VBoxVRDP_exe));
2605 strcpy (cmd, VBoxVRDP_exe);
2606
2607 Utf8Str idStr = mData->mUuid.toString();
2608 const char * args[] = {path, "-startvm", idStr, "-capture", 0 };
2609 vrc = RTProcCreate (path, args, NULL, 0, &pid);
2610 }
2611 else
2612 {
2613 return setError (E_INVALIDARG,
2614 tr ("Invalid session type: '%ls'"), aType);
2615 }
2616
2617 if (VBOX_FAILURE (vrc))
2618 return setError (E_FAIL,
2619 tr ("Could not launch a process for the machine '%ls' (%Vrc)"),
2620 mUserData->mName.raw(), vrc);
2621
2622 LogFlowThisFunc (("launched.pid=%d(0x%x)\n", pid, pid));
2623
2624 /*
2625 * Note that we don't leave the lock here before calling the client,
2626 * because it doesn't need to call us back if called with a NULL argument.
2627 * Leaving the lock herer is dangerous because we didn't prepare the
2628 * launch data yet, but the client we've just started may happen to be
2629 * too fast and call openSession() that will fail (because of PID, etc.),
2630 * so that the Machine will never get out of the Spawning session state.
2631 */
2632
2633 /* inform the session that it will be a remote one */
2634 LogFlowThisFunc (("Calling AssignMachine (NULL)...\n"));
2635 HRESULT rc = aControl->AssignMachine (NULL);
2636 LogFlowThisFunc (("AssignMachine (NULL) returned %08X\n", rc));
2637
2638 if (FAILED (rc))
2639 {
2640 /* restore the session state */
2641 mData->mSession.mState = SessionState_SessionClosed;
2642 /* The failure may w/o any error info (from RPC), so provide one */
2643 return setError (rc,
2644 tr ("Failed to assign the machine to the session"));
2645 }
2646
2647 /* attach launch data to the machine */
2648 Assert (mData->mSession.mPid == NIL_RTPROCESS);
2649 mData->mSession.mRemoteControls.push_back (aControl);
2650 mData->mSession.mProgress = aProgress;
2651 mData->mSession.mPid = pid;
2652 mData->mSession.mState = SessionState_SessionSpawning;
2653
2654 LogFlowThisFuncLeave();
2655 return S_OK;
2656}
2657
2658/**
2659 * @note Locks this object for writing, calls the client process
2660 * (outside the lock).
2661 */
2662HRESULT Machine::openExistingSession (IInternalSessionControl *aControl)
2663{
2664 LogFlowThisFuncEnter();
2665
2666 AssertReturn (aControl, E_FAIL);
2667
2668 AutoCaller autoCaller (this);
2669 CheckComRCReturnRC (autoCaller.rc());
2670
2671 AutoLock alock (this);
2672
2673 if (!mData->mRegistered)
2674 return setError (E_UNEXPECTED,
2675 tr ("The machine '%ls' is not registered"), mUserData->mName.raw());
2676
2677 LogFlowThisFunc (("mSession.state=%d\n", mData->mSession.mState));
2678
2679 if (mData->mSession.mState != SessionState_SessionOpen)
2680 return setError (E_ACCESSDENIED,
2681 tr ("The machine '%ls' does not have an open session"),
2682 mUserData->mName.raw());
2683
2684 ComAssertRet (!mData->mSession.mDirectControl.isNull(), E_FAIL);
2685
2686 /*
2687 * Get the console from the direct session (note that we don't leave the
2688 * lock here because GetRemoteConsole must not call us back).
2689 */
2690 ComPtr <IConsole> console;
2691 HRESULT rc = mData->mSession.mDirectControl->
2692 GetRemoteConsole (console.asOutParam());
2693 if (FAILED (rc))
2694 {
2695 /* The failure may w/o any error info (from RPC), so provide one */
2696 return setError (rc,
2697 tr ("Failed to get a console object from the direct session"));
2698 }
2699
2700 ComAssertRet (!console.isNull(), E_FAIL);
2701
2702 ComObjPtr <SessionMachine> sessionMachine = mData->mSession.mMachine;
2703 AssertReturn (!sessionMachine.isNull(), E_FAIL);
2704
2705 /*
2706 * Leave the lock before calling the client process. It's safe here
2707 * since the only thing to do after we get the lock again is to add
2708 * the remote control to the list (which doesn't directly influence
2709 * anything).
2710 */
2711 alock.leave();
2712
2713 /* attach the remote session to the machine */
2714 LogFlowThisFunc (("Calling AssignRemoteMachine()...\n"));
2715 rc = aControl->AssignRemoteMachine (sessionMachine, console);
2716 LogFlowThisFunc (("AssignRemoteMachine() returned %08X\n", rc));
2717
2718 /* The failure may w/o any error info (from RPC), so provide one */
2719 if (FAILED (rc))
2720 return setError (rc,
2721 tr ("Failed to assign the machine to the session"));
2722
2723 alock.enter();
2724
2725 /* need to revalidate the state after entering the lock again */
2726 if (mData->mSession.mState != SessionState_SessionOpen)
2727 {
2728 aControl->Uninitialize();
2729
2730 return setError (E_ACCESSDENIED,
2731 tr ("The machine '%ls' does not have an open session"),
2732 mUserData->mName.raw());
2733 }
2734
2735 /* store the control in the list */
2736 mData->mSession.mRemoteControls.push_back (aControl);
2737
2738 LogFlowThisFuncLeave();
2739 return S_OK;
2740}
2741
2742/**
2743 * Checks that the registered flag of the machine can be set according to
2744 * the argument and sets it. On success, commits and saves all settings.
2745 *
2746 * @note When this machine is inaccessible, the only valid value for \a
2747 * aRegistered is FALSE (i.e. unregister the machine) because unregistered
2748 * inaccessible machines are not currently supported. Note that unregistering
2749 * an inaccessible machine will \b uninitialize this machine object. Therefore,
2750 * the caller must make sure there are no active Machine::addCaller() calls
2751 * on the current thread because this will block Machine::uninit().
2752 *
2753 * @note Locks this object and children for writing!
2754 */
2755HRESULT Machine::trySetRegistered (BOOL aRegistered)
2756{
2757 AutoLimitedCaller autoCaller (this);
2758 AssertComRCReturnRC (autoCaller.rc());
2759
2760 AutoLock alock (this);
2761
2762 ComAssertRet (mData->mRegistered != aRegistered, E_FAIL);
2763
2764 if (!mData->mAccessible)
2765 {
2766 /* A special case: the machine is not accessible. */
2767
2768 /* inaccessible machines can only be unregistered */
2769 AssertReturn (!aRegistered, E_FAIL);
2770
2771 /* Uninitialize ourselves here because currently there may be no
2772 * unregistered that are inaccessible (this state combination is not
2773 * supported). Note releasing the caller and leaving the lock before
2774 * calling uninit() */
2775
2776 alock.leave();
2777 autoCaller.release();
2778
2779 uninit();
2780
2781 return S_OK;
2782 }
2783
2784 AssertReturn (autoCaller.state() == Ready, E_FAIL);
2785
2786 if (aRegistered)
2787 {
2788 if (mData->mRegistered)
2789 return setError (E_FAIL,
2790 tr ("The machine '%ls' with UUID {%s} is already registered"),
2791 mUserData->mName.raw(),
2792 mData->mUuid.toString().raw());
2793 }
2794 else
2795 {
2796 if (mData->mMachineState == MachineState_Saved)
2797 return setError (E_FAIL,
2798 tr ("Cannot unregister the machine '%ls' because it "
2799 "is in the Saved state"),
2800 mUserData->mName.raw());
2801
2802 size_t snapshotCount = 0;
2803 if (mData->mFirstSnapshot)
2804 snapshotCount = mData->mFirstSnapshot->descendantCount() + 1;
2805 if (snapshotCount)
2806 return setError (E_FAIL,
2807 tr ("Cannot unregister the machine '%ls' because it "
2808 "has %d snapshots"),
2809 mUserData->mName.raw(), snapshotCount);
2810
2811 if (mData->mSession.mState != SessionState_SessionClosed)
2812 return setError (E_FAIL,
2813 tr ("Cannot unregister the machine '%ls' because it has an "
2814 "open session"),
2815 mUserData->mName.raw());
2816
2817 if (mHDData->mHDAttachments.size() != 0)
2818 return setError (E_FAIL,
2819 tr ("Cannot unregister the machine '%ls' because it "
2820 "has %d hard disks attached"),
2821 mUserData->mName.raw(), mHDData->mHDAttachments.size());
2822 }
2823
2824 /* Ensure the settings are saved. If we are going to be registered and
2825 * isConfigLocked() is FALSE then it means that no config file exists yet,
2826 * so create it. */
2827 if (isModified() || (aRegistered && !isConfigLocked()))
2828 {
2829 HRESULT rc = saveSettings();
2830 CheckComRCReturnRC (rc);
2831 }
2832
2833 mData->mRegistered = aRegistered;
2834
2835 /* inform the USB proxy about all attached/detached USB filters */
2836 mUSBController->onMachineRegistered (aRegistered);
2837
2838 return S_OK;
2839}
2840
2841// protected methods
2842/////////////////////////////////////////////////////////////////////////////
2843
2844/**
2845 * Helper to uninitialize all associated child objects
2846 * and to free all data structures.
2847 *
2848 * This method must be called as a part of the object's uninitialization
2849 * procedure (usually done in the uninit() method).
2850 *
2851 * @note Must be called only from uninit().
2852 */
2853void Machine::uninitDataAndChildObjects()
2854{
2855 AutoCaller autoCaller (this);
2856 AssertComRCReturn (autoCaller.rc(), (void) 0);
2857 AssertComRCReturn (autoCaller.state( ) == InUninit, (void) 0);
2858
2859 /* tell all our child objects we've been uninitialized */
2860
2861 /*
2862 * uninit all children using addDependentChild()/removeDependentChild()
2863 * in their init()/uninit() methods
2864 */
2865 uninitDependentChildren();
2866
2867 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
2868 {
2869 if (mNetworkAdapters [slot])
2870 {
2871 mNetworkAdapters [slot]->uninit();
2872 unconst (mNetworkAdapters [slot]).setNull();
2873 }
2874 }
2875
2876 if (mUSBController)
2877 {
2878 mUSBController->uninit();
2879 unconst (mUSBController).setNull();
2880 }
2881
2882 if (mAudioAdapter)
2883 {
2884 mAudioAdapter->uninit();
2885 unconst (mAudioAdapter).setNull();
2886 }
2887
2888 if (mFloppyDrive)
2889 {
2890 mFloppyDrive->uninit();
2891 unconst (mFloppyDrive).setNull();
2892 }
2893
2894 if (mDVDDrive)
2895 {
2896 mDVDDrive->uninit();
2897 unconst (mDVDDrive).setNull();
2898 }
2899
2900#ifdef VBOX_VRDP
2901 if (mVRDPServer)
2902 {
2903 mVRDPServer->uninit();
2904 unconst (mVRDPServer).setNull();
2905 }
2906#endif
2907
2908 if (mBIOSSettings)
2909 {
2910 mBIOSSettings->uninit();
2911 unconst (mBIOSSettings).setNull();
2912 }
2913
2914 /* free data structures */
2915 mSSData.free();
2916 mHDData.free();
2917 mHWData.free();
2918 mUserData.free();
2919 mData.free();
2920}
2921
2922/**
2923 * Helper to change the machine state.
2924 *
2925 * @note Locks this object for writing.
2926 */
2927HRESULT Machine::setMachineState (MachineState_T aMachineState)
2928{
2929 LogFlowThisFuncEnter();
2930 LogFlowThisFunc (("aMachineState=%d\n", aMachineState));
2931
2932 AutoCaller autoCaller (this);
2933 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
2934
2935 AutoLock alock (this);
2936
2937 if (mData->mMachineState != aMachineState)
2938 {
2939 mData->mMachineState = aMachineState;
2940
2941 RTTIMESPEC time;
2942 mData->mLastStateChange = RTTimeSpecGetMilli(RTTimeNow(&time));
2943
2944 mParent->onMachineStateChange (mData->mUuid, aMachineState);
2945 }
2946
2947 LogFlowThisFuncLeave();
2948 return S_OK;
2949}
2950
2951/**
2952 * Searches for a shared folder with the given logical name
2953 * in the collection of shared folders.
2954 *
2955 * @param aName logical name of the shared folder
2956 * @param aSharedFolder where to return the found object
2957 * @param aSetError whether to set the error info if the folder is
2958 * not found
2959 * @return
2960 * S_OK when found or E_INVALIDARG when not found
2961 *
2962 * @note
2963 * must be called from under the object's lock!
2964 */
2965HRESULT Machine::findSharedFolder (const BSTR aName,
2966 ComObjPtr <SharedFolder> &aSharedFolder,
2967 bool aSetError /* = false */)
2968{
2969 bool found = false;
2970 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
2971 !found && it != mHWData->mSharedFolders.end();
2972 ++ it)
2973 {
2974 AutoLock alock (*it);
2975 found = (*it)->name() == aName;
2976 if (found)
2977 aSharedFolder = *it;
2978 }
2979
2980 HRESULT rc = found ? S_OK : E_INVALIDARG;
2981
2982 if (aSetError && !found)
2983 setError (rc, tr ("Could not find a shared folder named '%ls'"), aName);
2984
2985 return rc;
2986}
2987
2988/**
2989 * Loads all the VM settings by walking down the <Machine> node.
2990 *
2991 * @param aRegistered true when the machine is being loaded on VirtualBox
2992 * startup
2993 *
2994 * @note This method is intended to be called only from init(), so it assumes
2995 * all machine data fields have appropriate default values when it is called.
2996 *
2997 * @note Doesn't lock any objects.
2998 */
2999HRESULT Machine::loadSettings (bool aRegistered)
3000{
3001 LogFlowThisFuncEnter();
3002 AssertReturn (mType == IsMachine, E_FAIL);
3003
3004 AutoCaller autoCaller (this);
3005 AssertReturn (autoCaller.state() == InInit, E_FAIL);
3006
3007 HRESULT rc = S_OK;
3008
3009 CFGHANDLE configLoader = NULL;
3010 char *loaderError = NULL;
3011 int vrc = CFGLDRLoad (&configLoader,
3012 Utf8Str (mData->mConfigFileFull), mData->mHandleCfgFile,
3013 XmlSchemaNS, true, cfgLdrEntityResolver,
3014 &loaderError);
3015 if (VBOX_FAILURE (vrc))
3016 {
3017 rc = setError (E_FAIL,
3018 tr ("Could not load the settings file '%ls' (%Vrc)%s%s"),
3019 mData->mConfigFileFull.raw(), vrc,
3020 loaderError ? ".\n" : "", loaderError ? loaderError : "");
3021
3022 if (loaderError)
3023 RTMemTmpFree (loaderError);
3024
3025 LogFlowThisFuncLeave();
3026 return rc;
3027 }
3028
3029 /*
3030 * When reading the XML, we assume it has been validated, so we don't
3031 * do any structural checks here, Just Assert() some things.
3032 */
3033
3034 CFGNODE machineNode = 0;
3035 CFGLDRGetNode (configLoader, "VirtualBox/Machine", 0, &machineNode);
3036
3037 do
3038 {
3039 ComAssertBreak (machineNode, rc = E_FAIL);
3040
3041 /* uuid (required) */
3042 Guid id;
3043 CFGLDRQueryUUID (machineNode, "uuid", id.ptr());
3044
3045 /* If the stored UUID is not empty, it means the registered machine
3046 * is being loaded. Compare the loaded UUID with the stored one taken
3047 * from the global registry. */
3048 if (!mData->mUuid.isEmpty())
3049 {
3050 if (mData->mUuid != id)
3051 {
3052 rc = setError (E_FAIL,
3053 tr ("Machine UUID {%Vuuid} in '%ls' doesn't match its "
3054 "UUID {%s} in the registry file '%ls'"),
3055 id.raw(), mData->mConfigFileFull.raw(),
3056 mData->mUuid.toString().raw(),
3057 mParent->settingsFileName().raw());
3058 break;
3059 }
3060 }
3061 else
3062 unconst (mData->mUuid) = id;
3063
3064 /* name (required) */
3065 CFGLDRQueryBSTR (machineNode, "name", mUserData->mName.asOutParam());
3066
3067 /* nameSync (optional, default is true) */
3068 {
3069 bool nameSync = true;
3070 CFGLDRQueryBool (machineNode, "nameSync", &nameSync);
3071 mUserData->mNameSync = nameSync;
3072 }
3073
3074 /* OSType (required) */
3075 {
3076 Bstr osTypeId;
3077 CFGLDRQueryBSTR (machineNode, "OSType", osTypeId.asOutParam());
3078
3079 /* look up the object in our list */
3080 ComPtr <IGuestOSType> guestOSType;
3081 rc = mParent->FindGuestOSType (osTypeId, guestOSType.asOutParam());
3082 if (FAILED (rc))
3083 break;
3084
3085 mUserData->mOSType = guestOSType;
3086 }
3087
3088 /* stateFile (optional) */
3089 {
3090 Bstr stateFilePath;
3091 CFGLDRQueryBSTR (machineNode, "stateFile", stateFilePath.asOutParam());
3092 if (stateFilePath)
3093 {
3094 Utf8Str stateFilePathFull = stateFilePath;
3095 int vrc = calculateFullPath (stateFilePathFull, stateFilePathFull);
3096 if (VBOX_FAILURE (vrc))
3097 {
3098 rc = setError (E_FAIL,
3099 tr ("Invalid saved state file path: '%ls' (%Vrc)"),
3100 stateFilePath.raw(), vrc);
3101 break;
3102 }
3103 mSSData->mStateFilePath = stateFilePathFull;
3104 }
3105 else
3106 mSSData->mStateFilePath.setNull();
3107 }
3108
3109 /*
3110 * currentSnapshot ID (optional)
3111 * Note that due to XML Schema constaraints this attribute, when present,
3112 * will guaranteedly refer to an existing snapshot definition in XML
3113 */
3114 Guid currentSnapshotId;
3115 CFGLDRQueryUUID (machineNode, "currentSnapshot", currentSnapshotId.ptr());
3116
3117 /* snapshotFolder (optional) */
3118 {
3119 Bstr folder;
3120 CFGLDRQueryBSTR (machineNode, "snapshotFolder", folder.asOutParam());
3121 rc = COMSETTER(SnapshotFolder) (folder);
3122 if (FAILED (rc))
3123 break;
3124 }
3125
3126 /* lastStateChange (optional, for compatiblity) */
3127 {
3128 int64_t lastStateChange = 0;
3129 CFGLDRQueryDateTime (machineNode, "lastStateChange", &lastStateChange);
3130 if (lastStateChange == 0)
3131 {
3132 /// @todo (dmik) until lastStateChange is the required attribute,
3133 // we simply set it to the current time if missing in the config
3134 RTTIMESPEC time;
3135 lastStateChange = RTTimeSpecGetMilli (RTTimeNow (&time));
3136 }
3137 mData->mLastStateChange = lastStateChange;
3138 }
3139
3140 /* aborted (optional) */
3141 bool aborted = false;
3142 CFGLDRQueryBool (machineNode, "aborted", &aborted);
3143
3144 /* currentStateModified (optional, default is true) */
3145 mData->mCurrentStateModified = TRUE;
3146 {
3147 bool val = true;
3148 CFGLDRQueryBool (machineNode, "currentStateModified", &val);
3149 mData->mCurrentStateModified = val;
3150 }
3151
3152 /*
3153 * note: all mUserData members must be assigned prior this point because
3154 * we need to commit changes in order to let mUserData be shared by all
3155 * snapshot machine instances.
3156 */
3157 mUserData.commitCopy();
3158
3159 /* Snapshot node (optional) */
3160 {
3161 CFGNODE snapshotNode = 0;
3162 CFGLDRGetChildNode (machineNode, "Snapshot", 0, &snapshotNode);
3163 if (snapshotNode)
3164 {
3165 /* read all snapshots recursively */
3166 rc = loadSnapshot (snapshotNode, currentSnapshotId, NULL);
3167 CFGLDRReleaseNode (snapshotNode);
3168 if (FAILED (rc))
3169 break;
3170 }
3171 }
3172
3173 /* Hardware node (required) */
3174 {
3175 CFGNODE hardwareNode = 0;
3176 CFGLDRGetChildNode (machineNode, "Hardware", 0, &hardwareNode);
3177 ComAssertBreak (hardwareNode, rc = E_FAIL);
3178 rc = loadHardware (hardwareNode);
3179 CFGLDRReleaseNode (hardwareNode);
3180 if (FAILED (rc))
3181 break;
3182 }
3183
3184 /* HardDiskAttachments node (required) */
3185 {
3186 CFGNODE hdasNode = 0;
3187 CFGLDRGetChildNode (machineNode, "HardDiskAttachments", 0, &hdasNode);
3188 ComAssertBreak (hdasNode, rc = E_FAIL);
3189
3190 rc = loadHardDisks (hdasNode, aRegistered);
3191 CFGLDRReleaseNode (hdasNode);
3192 if (FAILED (rc))
3193 break;
3194 }
3195
3196 /*
3197 * NOTE: the assignment below must be the last thing to do,
3198 * otherwise it will be not possible to change the settings
3199 * somewehere in the code above because all setters will be
3200 * blocked by CHECK_SETTER()
3201 */
3202
3203 /* set the machine state to Aborted or Saved when appropriate */
3204 if (aborted)
3205 {
3206 Assert (!mSSData->mStateFilePath);
3207 mSSData->mStateFilePath.setNull();
3208
3209 mData->mMachineState = MachineState_Aborted;
3210 }
3211 else if (mSSData->mStateFilePath)
3212 {
3213 mData->mMachineState = MachineState_Saved;
3214 }
3215 }
3216 while (0);
3217
3218 if (machineNode)
3219 CFGLDRReleaseNode (machineNode);
3220
3221 CFGLDRFree (configLoader);
3222
3223 LogFlowThisFuncLeave();
3224 return rc;
3225}
3226
3227/**
3228 * Recursively loads all snapshots starting from the given.
3229 *
3230 * @param aNode <Snapshot> node
3231 * @param aCurSnapshotId current snapshot ID from the settings file
3232 * @param aParentSnapshot parent snapshot
3233 */
3234HRESULT Machine::loadSnapshot (CFGNODE aNode, const Guid &aCurSnapshotId,
3235 Snapshot *aParentSnapshot)
3236{
3237 AssertReturn (aNode, E_INVALIDARG);
3238 AssertReturn (mType == IsMachine, E_FAIL);
3239
3240 // create a snapshot machine object
3241 ComObjPtr <SnapshotMachine> snapshotMachine;
3242 snapshotMachine.createObject();
3243
3244 HRESULT rc = S_OK;
3245
3246 Guid uuid; // required
3247 CFGLDRQueryUUID (aNode, "uuid", uuid.ptr());
3248
3249 Bstr stateFilePath; // optional
3250 CFGLDRQueryBSTR (aNode, "stateFile", stateFilePath.asOutParam());
3251 if (stateFilePath)
3252 {
3253 Utf8Str stateFilePathFull = stateFilePath;
3254 int vrc = calculateFullPath (stateFilePathFull, stateFilePathFull);
3255 if (VBOX_FAILURE (vrc))
3256 return setError (E_FAIL,
3257 tr ("Invalid saved state file path: '%ls' (%Vrc)"),
3258 stateFilePath.raw(), vrc);
3259
3260 stateFilePath = stateFilePathFull;
3261 }
3262
3263 do
3264 {
3265 // Hardware node (required)
3266 CFGNODE hardwareNode = 0;
3267 CFGLDRGetChildNode (aNode, "Hardware", 0, &hardwareNode);
3268 ComAssertBreak (hardwareNode, rc = E_FAIL);
3269
3270 do
3271 {
3272 // HardDiskAttachments node (required)
3273 CFGNODE hdasNode = 0;
3274 CFGLDRGetChildNode (aNode, "HardDiskAttachments", 0, &hdasNode);
3275 ComAssertBreak (hdasNode, rc = E_FAIL);
3276
3277 // initialize the snapshot machine
3278 rc = snapshotMachine->init (this, hardwareNode, hdasNode,
3279 uuid, stateFilePath);
3280
3281 CFGLDRReleaseNode (hdasNode);
3282 }
3283 while (0);
3284
3285 CFGLDRReleaseNode (hardwareNode);
3286 }
3287 while (0);
3288
3289 if (FAILED (rc))
3290 return rc;
3291
3292 // create a snapshot object
3293 ComObjPtr <Snapshot> snapshot;
3294 snapshot.createObject();
3295
3296 {
3297 Bstr name; // required
3298 CFGLDRQueryBSTR (aNode, "name", name.asOutParam());
3299
3300 LONG64 timeStamp = 0; // required
3301 CFGLDRQueryDateTime (aNode, "timeStamp", &timeStamp);
3302
3303 Bstr description; // optional
3304 {
3305 CFGNODE descNode = 0;
3306 CFGLDRGetChildNode (aNode, "Description", 0, &descNode);
3307 CFGLDRQueryBSTR (descNode, NULL, description.asOutParam());
3308 CFGLDRReleaseNode (descNode);
3309 }
3310
3311 // initialize the snapshot
3312 rc = snapshot->init (uuid, name, description, timeStamp,
3313 snapshotMachine, aParentSnapshot);
3314 if (FAILED (rc))
3315 return rc;
3316 }
3317
3318 // memorize the first snapshot if necessary
3319 if (!mData->mFirstSnapshot)
3320 mData->mFirstSnapshot = snapshot;
3321
3322 // memorize the current snapshot when appropriate
3323 if (!mData->mCurrentSnapshot && snapshot->data().mId == aCurSnapshotId)
3324 mData->mCurrentSnapshot = snapshot;
3325
3326 // Snapshots node (optional)
3327 {
3328 CFGNODE snapshotsNode = 0;
3329 CFGLDRGetChildNode (aNode, "Snapshots", 0, &snapshotsNode);
3330 if (snapshotsNode)
3331 {
3332 unsigned cbDisks = 0;
3333 CFGLDRCountChildren (snapshotsNode, "Snapshot", &cbDisks);
3334 for (unsigned i = 0; i < cbDisks && SUCCEEDED (rc); i++)
3335 {
3336 CFGNODE snapshotNode;
3337 CFGLDRGetChildNode (snapshotsNode, "Snapshot", i, &snapshotNode);
3338 ComAssertBreak (snapshotNode, rc = E_FAIL);
3339
3340 rc = loadSnapshot (snapshotNode, aCurSnapshotId, snapshot);
3341
3342 CFGLDRReleaseNode (snapshotNode);
3343 }
3344
3345 CFGLDRReleaseNode (snapshotsNode);
3346 }
3347 }
3348
3349 return rc;
3350}
3351
3352/**
3353 * @param aNode <Hardware> node
3354 */
3355HRESULT Machine::loadHardware (CFGNODE aNode)
3356{
3357 AssertReturn (aNode, E_INVALIDARG);
3358 AssertReturn (mType == IsMachine || mType == IsSnapshotMachine, E_FAIL);
3359
3360 /* CPU node (currently not required) */
3361 {
3362 /* default value in case the node is not there */
3363 mHWData->mHWVirtExEnabled = TriStateBool_Default;
3364
3365 CFGNODE cpuNode = 0;
3366 CFGLDRGetChildNode (aNode, "CPU", 0, &cpuNode);
3367 if (cpuNode)
3368 {
3369 CFGNODE hwVirtExNode = 0;
3370 CFGLDRGetChildNode (cpuNode, "HardwareVirtEx", 0, &hwVirtExNode);
3371 if (hwVirtExNode)
3372 {
3373 Bstr hwVirtExEnabled;
3374 CFGLDRQueryBSTR (hwVirtExNode, "enabled", hwVirtExEnabled.asOutParam());
3375 if (hwVirtExEnabled == L"false")
3376 mHWData->mHWVirtExEnabled = TriStateBool_False;
3377 else if (hwVirtExEnabled == L"true")
3378 mHWData->mHWVirtExEnabled = TriStateBool_True;
3379 else
3380 mHWData->mHWVirtExEnabled = TriStateBool_Default;
3381 CFGLDRReleaseNode (hwVirtExNode);
3382 }
3383 CFGLDRReleaseNode (cpuNode);
3384 }
3385 }
3386
3387 /* Memory node (required) */
3388 {
3389 CFGNODE memoryNode = 0;
3390 CFGLDRGetChildNode (aNode, "Memory", 0, &memoryNode);
3391 ComAssertRet (memoryNode, E_FAIL);
3392
3393 uint32_t RAMSize;
3394 CFGLDRQueryUInt32 (memoryNode, "RAMSize", &RAMSize);
3395 mHWData->mMemorySize = RAMSize;
3396 CFGLDRReleaseNode (memoryNode);
3397 }
3398
3399 /* Boot node (required) */
3400 {
3401 /* reset all boot order positions to NoDevice */
3402 for (size_t i = 0; i < ELEMENTS (mHWData->mBootOrder); i++)
3403 mHWData->mBootOrder [i] = DeviceType_NoDevice;
3404
3405 CFGNODE bootNode = 0;
3406 CFGLDRGetChildNode (aNode, "Boot", 0, &bootNode);
3407 ComAssertRet (bootNode, E_FAIL);
3408
3409 HRESULT rc = S_OK;
3410
3411 unsigned cOrder;
3412 CFGLDRCountChildren (bootNode, "Order", &cOrder);
3413 for (unsigned i = 0; i < cOrder; i++)
3414 {
3415 CFGNODE orderNode = 0;
3416 CFGLDRGetChildNode (bootNode, "Order", i, &orderNode);
3417 ComAssertBreak (orderNode, rc = E_FAIL);
3418
3419 /* position (required) */
3420 /* position unicity is guaranteed by XML Schema */
3421 uint32_t position = 0;
3422 CFGLDRQueryUInt32 (orderNode, "position", &position);
3423 -- position;
3424 Assert (position < ELEMENTS (mHWData->mBootOrder));
3425
3426 /* device (required) */
3427 Bstr device;
3428 CFGLDRQueryBSTR (orderNode, "device", device.asOutParam());
3429 if (device == L"None")
3430 mHWData->mBootOrder [position] = DeviceType_NoDevice;
3431 else if (device == L"Floppy")
3432 mHWData->mBootOrder [position] = DeviceType_FloppyDevice;
3433 else if (device == L"DVD")
3434 mHWData->mBootOrder [position] = DeviceType_DVDDevice;
3435 else if (device == L"HardDisk")
3436 mHWData->mBootOrder [position] = DeviceType_HardDiskDevice;
3437 else if (device == L"Network")
3438 mHWData->mBootOrder [position] = DeviceType_NetworkDevice;
3439 else
3440 ComAssertMsgFailed (("Invalid device: %ls\n", device.raw()));
3441
3442 CFGLDRReleaseNode (orderNode);
3443 }
3444
3445 CFGLDRReleaseNode (bootNode);
3446 if (FAILED (rc))
3447 return rc;
3448 }
3449
3450 /* Display node (required) */
3451 {
3452 CFGNODE displayNode = 0;
3453 CFGLDRGetChildNode (aNode, "Display", 0, &displayNode);
3454 ComAssertRet (displayNode, E_FAIL);
3455
3456 uint32_t VRAMSize;
3457 CFGLDRQueryUInt32 (displayNode, "VRAMSize", &VRAMSize);
3458 mHWData->mVRAMSize = VRAMSize;
3459 CFGLDRReleaseNode (displayNode);
3460 }
3461
3462#ifdef VBOX_VRDP
3463 /* RemoteDisplay node (optional) */
3464 /// @todo (dmik) move the code to VRDPServer
3465 /// @todo r=sunlover: moved. dmik, please review.
3466 {
3467 CFGNODE remoteDisplayNode = 0;
3468 CFGLDRGetChildNode (aNode, "RemoteDisplay", 0, &remoteDisplayNode);
3469 if (remoteDisplayNode)
3470 {
3471 mVRDPServer->loadConfig (remoteDisplayNode);
3472 CFGLDRReleaseNode (remoteDisplayNode);
3473 }
3474 }
3475#endif
3476
3477 /* BIOS node (required) */
3478 {
3479 CFGNODE biosNode = 0;
3480 CFGLDRGetChildNode (aNode, "BIOS", 0, &biosNode);
3481 ComAssertRet (biosNode, E_FAIL);
3482
3483 HRESULT rc = S_OK;
3484
3485 do
3486 {
3487 /* ACPI */
3488 {
3489 CFGNODE acpiNode = 0;
3490 CFGLDRGetChildNode (biosNode, "ACPI", 0, &acpiNode);
3491 ComAssertBreak (acpiNode, rc = E_FAIL);
3492
3493 bool enabled;
3494 CFGLDRQueryBool (acpiNode, "enabled", &enabled);
3495 mBIOSSettings->COMSETTER(ACPIEnabled)(enabled);
3496 CFGLDRReleaseNode (acpiNode);
3497 }
3498
3499 /* IOAPIC */
3500 {
3501 CFGNODE ioapicNode = 0;
3502 CFGLDRGetChildNode (biosNode, "IOAPIC", 0, &ioapicNode);
3503 if (ioapicNode)
3504 {
3505 bool enabled;
3506 CFGLDRQueryBool (ioapicNode, "enabled", &enabled);
3507 mBIOSSettings->COMSETTER(IOAPICEnabled)(enabled);
3508 CFGLDRReleaseNode (ioapicNode);
3509 }
3510 }
3511
3512 /* Logo (optional) */
3513 {
3514 CFGNODE logoNode = 0;
3515 CFGLDRGetChildNode (biosNode, "Logo", 0, &logoNode);
3516 if (logoNode)
3517 {
3518 bool enabled = false;
3519 CFGLDRQueryBool (logoNode, "fadeIn", &enabled);
3520 mBIOSSettings->COMSETTER(LogoFadeIn)(enabled);
3521 CFGLDRQueryBool (logoNode, "fadeOut", &enabled);
3522 mBIOSSettings->COMSETTER(LogoFadeOut)(enabled);
3523
3524 uint32_t BIOSLogoDisplayTime;
3525 CFGLDRQueryUInt32 (logoNode, "displayTime", &BIOSLogoDisplayTime);
3526 mBIOSSettings->COMSETTER(LogoDisplayTime)(BIOSLogoDisplayTime);
3527
3528 Bstr logoPath;
3529 CFGLDRQueryBSTR (logoNode, "imagePath", logoPath.asOutParam());
3530 mBIOSSettings->COMSETTER(LogoImagePath)(logoPath);
3531
3532 CFGLDRReleaseNode (logoNode);
3533 }
3534 }
3535
3536 /* boot menu (optional) */
3537 {
3538 CFGNODE bootMenuNode = 0;
3539 CFGLDRGetChildNode (biosNode, "BootMenu", 0, &bootMenuNode);
3540 if (bootMenuNode)
3541 {
3542 Bstr modeStr;
3543 BIOSBootMenuMode_T mode;
3544 CFGLDRQueryBSTR (bootMenuNode, "mode", modeStr.asOutParam());
3545 if (modeStr == L"disabled")
3546 mode = BIOSBootMenuMode_Disabled;
3547 else if (modeStr == L"menuonly")
3548 mode = BIOSBootMenuMode_MenuOnly;
3549 else
3550 mode = BIOSBootMenuMode_MessageAndMenu;
3551 mBIOSSettings->COMSETTER(BootMenuMode)(mode);
3552
3553 CFGLDRReleaseNode (bootMenuNode);
3554 }
3555 }
3556 }
3557 while (0);
3558
3559 CFGLDRReleaseNode (biosNode);
3560 if (FAILED (rc))
3561 return rc;
3562 }
3563
3564 /* DVD drive (contains either Image or HostDrive or nothing) */
3565 /// @todo (dmik) move the code to DVDDrive
3566 {
3567 HRESULT rc = S_OK;
3568
3569 CFGNODE dvdDriveNode = 0;
3570 CFGLDRGetChildNode (aNode, "DVDDrive", 0, &dvdDriveNode);
3571 ComAssertRet (dvdDriveNode, E_FAIL);
3572
3573 bool fPassthrough;
3574 CFGLDRQueryBool(dvdDriveNode, "passthrough", &fPassthrough);
3575 mDVDDrive->COMSETTER(Passthrough)(fPassthrough);
3576
3577 CFGNODE typeNode = 0;
3578
3579 do
3580 {
3581 CFGLDRGetChildNode (dvdDriveNode, "Image", 0, &typeNode);
3582 if (typeNode)
3583 {
3584 Guid uuid;
3585 CFGLDRQueryUUID (typeNode, "uuid", uuid.ptr());
3586 rc = mDVDDrive->MountImage (uuid);
3587 }
3588 else
3589 {
3590 CFGLDRGetChildNode (dvdDriveNode, "HostDrive", 0, &typeNode);
3591 if (typeNode)
3592 {
3593 Bstr src;
3594 CFGLDRQueryBSTR (typeNode, "src", src.asOutParam());
3595
3596 /* find the correspoding object */
3597 ComPtr <IHost> host;
3598 rc = mParent->COMGETTER(Host) (host.asOutParam());
3599 ComAssertComRCBreak (rc, rc = rc);
3600
3601 ComPtr <IHostDVDDriveCollection> coll;
3602 rc = host->COMGETTER(DVDDrives) (coll.asOutParam());
3603 ComAssertComRCBreak (rc, rc = rc);
3604
3605 ComPtr <IHostDVDDrive> drive;
3606 rc = coll->FindByName (src, drive.asOutParam());
3607 if (SUCCEEDED (rc))
3608 rc = mDVDDrive->CaptureHostDrive (drive);
3609 else if (rc == E_INVALIDARG)
3610 {
3611 /* the host DVD drive is not currently available. we
3612 * assume it will be available later and create an
3613 * extra object now */
3614 ComObjPtr <HostDVDDrive> hostDrive;
3615 hostDrive.createObject();
3616 rc = hostDrive->init (src);
3617 ComAssertComRCBreak (rc, rc = rc);
3618 rc = mDVDDrive->CaptureHostDrive (hostDrive);
3619 }
3620 else
3621 ComAssertComRCBreak (rc, rc = rc);
3622 }
3623 }
3624 }
3625 while (0);
3626
3627 if (typeNode)
3628 CFGLDRReleaseNode (typeNode);
3629 CFGLDRReleaseNode (dvdDriveNode);
3630
3631 if (FAILED (rc))
3632 return rc;
3633 }
3634
3635 /* Floppy drive (contains either Image or HostDrive or nothing) */
3636 /// @todo (dmik) move the code to FloppyDrive
3637 {
3638 HRESULT rc = S_OK;
3639
3640 CFGNODE driveNode = 0;
3641 CFGLDRGetChildNode (aNode, "FloppyDrive", 0, &driveNode);
3642 ComAssertRet (driveNode, E_FAIL);
3643
3644 BOOL fFloppyEnabled = TRUE;
3645 CFGLDRQueryBool (driveNode, "enabled", (bool*)&fFloppyEnabled);
3646 rc = mFloppyDrive->COMSETTER(Enabled)(fFloppyEnabled);
3647
3648 CFGNODE typeNode = 0;
3649 do
3650 {
3651 CFGLDRGetChildNode (driveNode, "Image", 0, &typeNode);
3652 if (typeNode)
3653 {
3654 Guid uuid;
3655 CFGLDRQueryUUID (typeNode, "uuid", uuid.ptr());
3656 rc = mFloppyDrive->MountImage (uuid);
3657 }
3658 else
3659 {
3660 CFGLDRGetChildNode (driveNode, "HostDrive", 0, &typeNode);
3661 if (typeNode)
3662 {
3663 Bstr src;
3664 CFGLDRQueryBSTR (typeNode, "src", src.asOutParam());
3665
3666 /* find the correspoding object */
3667 ComPtr <IHost> host;
3668 rc = mParent->COMGETTER(Host) (host.asOutParam());
3669 ComAssertComRCBreak (rc, rc = rc);
3670
3671 ComPtr <IHostFloppyDriveCollection> coll;
3672 rc = host->COMGETTER(FloppyDrives) (coll.asOutParam());
3673 ComAssertComRCBreak (rc, rc = rc);
3674
3675 ComPtr <IHostFloppyDrive> drive;
3676 rc = coll->FindByName (src, drive.asOutParam());
3677 if (SUCCEEDED (rc))
3678 rc = mFloppyDrive->CaptureHostDrive (drive);
3679 else if (rc == E_INVALIDARG)
3680 {
3681 /* the host Floppy drive is not currently available. we
3682 * assume it will be available later and create an
3683 * extra object now */
3684 ComObjPtr <HostFloppyDrive> hostDrive;
3685 hostDrive.createObject();
3686 rc = hostDrive->init (src);
3687 ComAssertComRCBreak (rc, rc = rc);
3688 rc = mFloppyDrive->CaptureHostDrive (hostDrive);
3689 }
3690 else
3691 ComAssertComRCBreak (rc, rc = rc);
3692 }
3693 }
3694 }
3695 while (0);
3696
3697 if (typeNode)
3698 CFGLDRReleaseNode (typeNode);
3699 CFGLDRReleaseNode (driveNode);
3700
3701 if (FAILED (rc))
3702 return rc;
3703 }
3704
3705 /* USB Controller */
3706 {
3707 HRESULT rc = mUSBController->loadSettings (aNode);
3708 if (FAILED (rc))
3709 return rc;
3710 }
3711
3712 /* Network node (required) */
3713 /// @todo (dmik) move the code to NetworkAdapter
3714 {
3715 /* we assume that all network adapters are initially disabled
3716 * and detached */
3717
3718 CFGNODE networkNode = 0;
3719 CFGLDRGetChildNode (aNode, "Network", 0, &networkNode);
3720 ComAssertRet (networkNode, E_FAIL);
3721
3722 HRESULT rc = S_OK;
3723
3724 unsigned cAdapters = 0;
3725 CFGLDRCountChildren (networkNode, "Adapter", &cAdapters);
3726 for (unsigned i = 0; i < cAdapters; i++)
3727 {
3728 CFGNODE adapterNode = 0;
3729 CFGLDRGetChildNode (networkNode, "Adapter", i, &adapterNode);
3730 ComAssertBreak (adapterNode, rc = E_FAIL);
3731
3732 /* slot number (required) */
3733 /* slot unicity is guaranteed by XML Schema */
3734 uint32_t slot = 0;
3735 CFGLDRQueryUInt32 (adapterNode, "slot", &slot);
3736 Assert (slot < ELEMENTS (mNetworkAdapters));
3737
3738 /* type */
3739 Bstr adapterType;
3740 CFGLDRQueryBSTR (adapterNode, "type", adapterType.asOutParam());
3741 ComAssertBreak (adapterType, rc = E_FAIL);
3742
3743 /* enabled (required) */
3744 bool enabled = false;
3745 CFGLDRQueryBool (adapterNode, "enabled", &enabled);
3746 /* MAC address (can be null) */
3747 Bstr macAddr;
3748 CFGLDRQueryBSTR (adapterNode, "MACAddress", macAddr.asOutParam());
3749 /* cable (required) */
3750 bool cableConnected;
3751 CFGLDRQueryBool (adapterNode, "cable", &cableConnected);
3752 /* tracing (defaults to false) */
3753 bool traceEnabled;
3754 CFGLDRQueryBool (adapterNode, "trace", &traceEnabled);
3755 Bstr traceFile;
3756 CFGLDRQueryBSTR (adapterNode, "tracefile", traceFile.asOutParam());
3757
3758 mNetworkAdapters [slot]->COMSETTER(Enabled) (enabled);
3759 mNetworkAdapters [slot]->COMSETTER(MACAddress) (macAddr);
3760 mNetworkAdapters [slot]->COMSETTER(CableConnected) (cableConnected);
3761 mNetworkAdapters [slot]->COMSETTER(TraceEnabled) (traceEnabled);
3762 mNetworkAdapters [slot]->COMSETTER(TraceFile) (traceFile);
3763
3764 if (adapterType.compare(Bstr("Am79C970A")) == 0)
3765 mNetworkAdapters [slot]->COMSETTER(AdapterType)(NetworkAdapterType_NetworkAdapterAm79C970A);
3766 else if (adapterType.compare(Bstr("Am79C973")) == 0)
3767 mNetworkAdapters [slot]->COMSETTER(AdapterType)(NetworkAdapterType_NetworkAdapterAm79C973);
3768 else
3769 ComAssertBreak (0, rc = E_FAIL);
3770
3771 CFGNODE attachmentNode = 0;
3772 if (CFGLDRGetChildNode (adapterNode, "NAT", 0, &attachmentNode), attachmentNode)
3773 {
3774 mNetworkAdapters [slot]->AttachToNAT();
3775 }
3776 else
3777 if (CFGLDRGetChildNode (adapterNode, "HostInterface", 0, &attachmentNode), attachmentNode)
3778 {
3779 /* Host Interface Networking */
3780 Bstr name;
3781 CFGLDRQueryBSTR (attachmentNode, "name", name.asOutParam());
3782#ifdef __WIN__
3783 /* @name can be empty on Win32, but not null */
3784 ComAssertBreak (!name.isNull(), rc = E_FAIL);
3785#endif
3786 mNetworkAdapters [slot]->COMSETTER(HostInterface) (name);
3787#ifdef VBOX_WITH_UNIXY_TAP_NETWORKING
3788 Bstr tapSetupApp;
3789 CFGLDRQueryBSTR (attachmentNode, "TAPSetup", tapSetupApp.asOutParam());
3790 Bstr tapTerminateApp;
3791 CFGLDRQueryBSTR (attachmentNode, "TAPTerminate", tapTerminateApp.asOutParam());
3792
3793 mNetworkAdapters [slot]->COMSETTER(TAPSetupApplication) (tapSetupApp);
3794 mNetworkAdapters [slot]->COMSETTER(TAPTerminateApplication) (tapTerminateApp);
3795#endif // VBOX_WITH_UNIXY_TAP_NETWORKING
3796 mNetworkAdapters [slot]->AttachToHostInterface();
3797 }
3798 else
3799 if (CFGLDRGetChildNode(adapterNode, "InternalNetwork", 0, &attachmentNode), attachmentNode)
3800 {
3801 /* Internal Networking */
3802 Bstr name;
3803 CFGLDRQueryBSTR (attachmentNode, "name", name.asOutParam());
3804 ComAssertBreak (!name.isNull(), rc = E_FAIL);
3805 mNetworkAdapters[slot]->AttachToInternalNetwork();
3806 mNetworkAdapters[slot]->COMSETTER(InternalNetwork) (name);
3807 }
3808 else
3809 {
3810 /* Adapter has no children */
3811 mNetworkAdapters [slot]->Detach();
3812 }
3813 if (attachmentNode)
3814 CFGLDRReleaseNode (attachmentNode);
3815
3816 CFGLDRReleaseNode (adapterNode);
3817 }
3818
3819 CFGLDRReleaseNode (networkNode);
3820 if (FAILED (rc))
3821 return rc;
3822 }
3823
3824 /* AudioAdapter node (required) */
3825 /// @todo (dmik) move the code to AudioAdapter
3826 {
3827 CFGNODE audioAdapterNode = 0;
3828 CFGLDRGetChildNode (aNode, "AudioAdapter", 0, &audioAdapterNode);
3829 ComAssertRet (audioAdapterNode, E_FAIL);
3830
3831 // is the adapter enabled?
3832 bool enabled = false;
3833 CFGLDRQueryBool (audioAdapterNode, "enabled", &enabled);
3834 mAudioAdapter->COMSETTER(Enabled) (enabled);
3835 // now check the audio driver
3836 Bstr driver;
3837 CFGLDRQueryBSTR (audioAdapterNode, "driver", driver.asOutParam());
3838 AudioDriverType_T audioDriver;
3839 audioDriver = AudioDriverType_NullAudioDriver;
3840 if (driver == L"null")
3841 ; // Null has been set above
3842#ifdef __WIN__
3843 else if (driver == L"winmm")
3844 audioDriver = AudioDriverType_WINMMAudioDriver;
3845 else if (driver == L"dsound")
3846 audioDriver = AudioDriverType_DSOUNDAudioDriver;
3847#endif // __WIN__
3848#ifdef __LINUX__
3849 else if (driver == L"oss")
3850 audioDriver = AudioDriverType_OSSAudioDriver;
3851 else if (driver == L"alsa")
3852#ifdef VBOX_WITH_ALSA
3853 audioDriver = AudioDriverType_ALSAAudioDriver;
3854#else
3855 // fall back to OSS
3856 audioDriver = AudioDriverType_OSSAudioDriver;
3857#endif
3858#endif // __LINUX__
3859 else
3860 AssertMsgFailed (("Invalid driver: %ls\n", driver.raw()));
3861 mAudioAdapter->COMSETTER(AudioDriver) (audioDriver);
3862
3863 CFGLDRReleaseNode (audioAdapterNode);
3864 }
3865
3866 /* Shared folders (optional) */
3867 /// @todo (dmik) make required on next format change!
3868 do
3869 {
3870 CFGNODE sharedFoldersNode = 0;
3871 CFGLDRGetChildNode (aNode, "SharedFolders", 0, &sharedFoldersNode);
3872
3873 if (!sharedFoldersNode)
3874 break;
3875
3876 HRESULT rc = S_OK;
3877
3878 unsigned cFolders = 0;
3879 CFGLDRCountChildren (sharedFoldersNode, "SharedFolder", &cFolders);
3880
3881 for (unsigned i = 0; i < cFolders; i++)
3882 {
3883 CFGNODE folderNode = 0;
3884 CFGLDRGetChildNode (sharedFoldersNode, "SharedFolder", i, &folderNode);
3885 ComAssertBreak (folderNode, rc = E_FAIL);
3886
3887 // folder logical name (required)
3888 Bstr name;
3889 CFGLDRQueryBSTR (folderNode, "name", name.asOutParam());
3890
3891 // folder host path (required)
3892 Bstr hostPath;
3893 CFGLDRQueryBSTR (folderNode, "hostPath", hostPath.asOutParam());
3894
3895 rc = CreateSharedFolder (name, hostPath);
3896 if (FAILED (rc))
3897 break;
3898
3899 CFGLDRReleaseNode (folderNode);
3900 }
3901
3902 CFGLDRReleaseNode (sharedFoldersNode);
3903 if (FAILED (rc))
3904 return rc;
3905 }
3906 while (0);
3907
3908 /* Clipboard node (currently not required) */
3909 /// @todo (dmik) make required on next format change!
3910 {
3911 /* default value in case the node is not there */
3912 mHWData->mClipboardMode = ClipboardMode_ClipDisabled;
3913
3914 CFGNODE clipNode = 0;
3915 CFGLDRGetChildNode (aNode, "Clipboard", 0, &clipNode);
3916 if (clipNode)
3917 {
3918 Bstr mode;
3919 CFGLDRQueryBSTR (clipNode, "mode", mode.asOutParam());
3920 if (mode == L"Disabled")
3921 mHWData->mClipboardMode = ClipboardMode_ClipDisabled;
3922 else if (mode == L"HostToGuest")
3923 mHWData->mClipboardMode = ClipboardMode_ClipHostToGuest;
3924 else if (mode == L"GuestToHost")
3925 mHWData->mClipboardMode = ClipboardMode_ClipGuestToHost;
3926 else if (mode == L"Bidirectional")
3927 mHWData->mClipboardMode = ClipboardMode_ClipBidirectional;
3928 else
3929 AssertMsgFailed (("%ls clipboard mode is invalid\n", mode.raw()));
3930 CFGLDRReleaseNode (clipNode);
3931 }
3932 }
3933
3934 return S_OK;
3935}
3936
3937/**
3938 * @param aNode <HardDiskAttachments> node
3939 * @param aRegistered true when the machine is being loaded on VirtualBox
3940 * startup, or when a snapshot is being loaded (wchich
3941 * currently can happen on startup only)
3942 * @param aSnapshotId pointer to the snapshot ID if this is a snapshot machine
3943 */
3944HRESULT Machine::loadHardDisks (CFGNODE aNode, bool aRegistered,
3945 const Guid *aSnapshotId /* = NULL */)
3946{
3947 AssertReturn (aNode, E_INVALIDARG);
3948 AssertReturn ((mType == IsMachine && aSnapshotId == NULL) ||
3949 (mType == IsSnapshotMachine && aSnapshotId != NULL), E_FAIL);
3950
3951 HRESULT rc = S_OK;
3952
3953 unsigned cbDisks = 0;
3954 CFGLDRCountChildren (aNode, "HardDiskAttachment", &cbDisks);
3955
3956 if (!aRegistered && cbDisks > 0)
3957 {
3958 /* when the machine is being loaded (opened) from a file, it cannot
3959 * have hard disks attached (this should not happen normally,
3960 * because we don't allow to attach hard disks to an unregistered
3961 * VM at all */
3962 return setError (E_FAIL,
3963 tr ("Unregistered machine '%ls' cannot have hard disks attached "
3964 "(found %d hard disk attachments)"),
3965 mUserData->mName.raw(), cbDisks);
3966 }
3967
3968 for (unsigned i = 0; i < cbDisks && SUCCEEDED (rc); ++ i)
3969 {
3970 CFGNODE hdNode;
3971 CFGLDRGetChildNode (aNode, "HardDiskAttachment", i, &hdNode);
3972 ComAssertRet (hdNode, E_FAIL);
3973
3974 do
3975 {
3976 /* hardDisk uuid (required) */
3977 Guid uuid;
3978 CFGLDRQueryUUID (hdNode, "hardDisk", uuid.ptr());
3979 /* bus (controller) type (required) */
3980 Bstr bus;
3981 CFGLDRQueryBSTR (hdNode, "bus", bus.asOutParam());
3982 /* device (required) */
3983 Bstr device;
3984 CFGLDRQueryBSTR (hdNode, "device", device.asOutParam());
3985
3986 /* find a hard disk by UUID */
3987 ComObjPtr <HardDisk> hd;
3988 rc = mParent->getHardDisk (uuid, hd);
3989 if (FAILED (rc))
3990 break;
3991
3992 AutoLock hdLock (hd);
3993
3994 if (!hd->machineId().isEmpty())
3995 {
3996 rc = setError (E_FAIL,
3997 tr ("Hard disk '%ls' with UUID {%s} is already "
3998 "attached to a machine with UUID {%s} (see '%ls')"),
3999 hd->toString().raw(), uuid.toString().raw(),
4000 hd->machineId().toString().raw(),
4001 mData->mConfigFileFull.raw());
4002 break;
4003 }
4004
4005 if (hd->type() == HardDiskType_ImmutableHardDisk)
4006 {
4007 rc = setError (E_FAIL,
4008 tr ("Immutable hard disk '%ls' with UUID {%s} cannot be "
4009 "directly attached to a machine (see '%ls')"),
4010 hd->toString().raw(), uuid.toString().raw(),
4011 mData->mConfigFileFull.raw());
4012 break;
4013 }
4014
4015 /* attach the device */
4016 DiskControllerType_T ctl = DiskControllerType_InvalidController;
4017 LONG dev = -1;
4018
4019 if (bus == L"ide0")
4020 {
4021 ctl = DiskControllerType_IDE0Controller;
4022 if (device == L"master")
4023 dev = 0;
4024 else if (device == L"slave")
4025 dev = 1;
4026 else
4027 ComAssertMsgFailedBreak (("Invalid device: %ls\n", device.raw()),
4028 rc = E_FAIL);
4029 }
4030 else if (bus == L"ide1")
4031 {
4032 ctl = DiskControllerType_IDE1Controller;
4033 if (device == L"master")
4034 rc = setError (E_FAIL, tr("Could not attach a disk as a master "
4035 "device on the secondary controller"));
4036 else if (device == L"slave")
4037 dev = 1;
4038 else
4039 ComAssertMsgFailedBreak (("Invalid device: %ls\n", device.raw()),
4040 rc = E_FAIL);
4041 }
4042 else
4043 ComAssertMsgFailedBreak (("Invalid bus: %ls\n", bus.raw()),
4044 rc = E_FAIL);
4045
4046 ComObjPtr <HardDiskAttachment> attachment;
4047 attachment.createObject();
4048 rc = attachment->init (hd, ctl, dev, false /* aDirty */);
4049 if (FAILED (rc))
4050 break;
4051
4052 /* associate the hard disk with this machine */
4053 hd->setMachineId (mData->mUuid);
4054
4055 /* associate the hard disk with the given snapshot ID */
4056 if (mType == IsSnapshotMachine)
4057 hd->setSnapshotId (*aSnapshotId);
4058
4059 mHDData->mHDAttachments.push_back (attachment);
4060 }
4061 while (0);
4062
4063 CFGLDRReleaseNode (hdNode);
4064 }
4065
4066 return rc;
4067}
4068
4069/**
4070 * Creates a config loader and loads the settings file.
4071 *
4072 * @param aIsNew |true| if a newly created settings file is to be opened
4073 * (must be the case only when called from #saveSettings())
4074 *
4075 * @note
4076 * XML Schema errors are not detected by this method because
4077 * it assumes that it will load settings from an exclusively locked
4078 * file (using a file handle) that was previously validated when opened
4079 * for the first time. Thus, this method should be used only when
4080 * it's necessary to modify (save) the settings file.
4081 *
4082 * @note The object must be locked at least for reading before calling
4083 * this method.
4084 */
4085HRESULT Machine::openConfigLoader (CFGHANDLE *aLoader, bool aIsNew /* = false */)
4086{
4087 AssertReturn (aLoader, E_FAIL);
4088
4089 /* The settings file must be created and locked at this point */
4090 ComAssertRet (isConfigLocked(), E_FAIL);
4091
4092 /* load the config file */
4093 int vrc = CFGLDRLoad (aLoader,
4094 Utf8Str (mData->mConfigFileFull), mData->mHandleCfgFile,
4095 aIsNew ? NULL : XmlSchemaNS, true, cfgLdrEntityResolver,
4096 NULL);
4097 ComAssertRCRet (vrc, E_FAIL);
4098
4099 return S_OK;
4100}
4101
4102/**
4103 * Closes the config loader previously created by #openConfigLoader().
4104 * If \a aSaveBeforeClose is true, then the config is saved to the settings file
4105 * before closing. If saving fails, a proper error message is set.
4106 *
4107 * @param aSaveBeforeClose whether to save the config before closing or not
4108 */
4109HRESULT Machine::closeConfigLoader (CFGHANDLE aLoader, bool aSaveBeforeClose)
4110{
4111 HRESULT rc = S_OK;
4112
4113 if (aSaveBeforeClose)
4114 {
4115 char *loaderError = NULL;
4116 int vrc = CFGLDRSave (aLoader, &loaderError);
4117 if (VBOX_FAILURE (vrc))
4118 {
4119 rc = setError (E_FAIL,
4120 tr ("Could not save the settings file '%ls' (%Vrc)%s%s"),
4121 mData->mConfigFileFull.raw(), vrc,
4122 loaderError ? ".\n" : "", loaderError ? loaderError : "");
4123 if (loaderError)
4124 RTMemTmpFree (loaderError);
4125 }
4126 }
4127
4128 CFGLDRFree (aLoader);
4129
4130 return rc;
4131}
4132
4133/**
4134 * Searches for a <Snapshot> node for the given snapshot.
4135 * If the search is successful, \a aSnapshotNode will contain the found node.
4136 * In this case, \a aSnapshotsNode can be NULL meaning the found node is a
4137 * direct child of \a aMachineNode.
4138 *
4139 * If the search fails, a failure is returned and both \a aSnapshotsNode and
4140 * \a aSnapshotNode are set to 0.
4141 *
4142 * @param aSnapshot snapshot to search for
4143 * @param aMachineNode <Machine> node to start from
4144 * @param aSnapshotsNode <Snapshots> node containing the found <Snapshot> node
4145 * (may be NULL if the caller is not interested)
4146 * @param aSnapshotNode found <Snapshot> node
4147 */
4148HRESULT Machine::findSnapshotNode (Snapshot *aSnapshot, CFGNODE aMachineNode,
4149 CFGNODE *aSnapshotsNode, CFGNODE *aSnapshotNode)
4150{
4151 AssertReturn (aSnapshot && aMachineNode && aSnapshotNode, E_FAIL);
4152
4153 if (aSnapshotsNode)
4154 *aSnapshotsNode = 0;
4155 *aSnapshotNode = 0;
4156
4157 // build the full uuid path (from the fist parent to the given snapshot)
4158 std::list <Guid> path;
4159 {
4160 ComObjPtr <Snapshot> parent = aSnapshot;
4161 while (parent)
4162 {
4163 path.push_front (parent->data().mId);
4164 parent = parent->parent();
4165 }
4166 }
4167
4168 CFGNODE snapshotsNode = aMachineNode;
4169 CFGNODE snapshotNode = 0;
4170
4171 for (std::list <Guid>::const_iterator it = path.begin();
4172 it != path.end();
4173 ++ it)
4174 {
4175 if (snapshotNode)
4176 {
4177 // proceed to the nested <Snapshots> node
4178 Assert (snapshotsNode);
4179 if (snapshotsNode != aMachineNode)
4180 {
4181 CFGLDRReleaseNode (snapshotsNode);
4182 snapshotsNode = 0;
4183 }
4184 CFGLDRGetChildNode (snapshotNode, "Snapshots", 0, &snapshotsNode);
4185 CFGLDRReleaseNode (snapshotNode);
4186 snapshotNode = 0;
4187 }
4188
4189 AssertReturn (snapshotsNode, E_FAIL);
4190
4191 unsigned count = 0, i = 0;
4192 CFGLDRCountChildren (snapshotsNode, "Snapshot", &count);
4193 for (; i < count; ++ i)
4194 {
4195 snapshotNode = 0;
4196 CFGLDRGetChildNode (snapshotsNode, "Snapshot", i, &snapshotNode);
4197 Guid id;
4198 CFGLDRQueryUUID (snapshotNode, "uuid", id.ptr());
4199 if (id == (*it))
4200 {
4201 // we keep (don't release) snapshotNode and snapshotsNode
4202 break;
4203 }
4204 CFGLDRReleaseNode (snapshotNode);
4205 snapshotNode = 0;
4206 }
4207
4208 if (i == count)
4209 {
4210 // the next uuid is not found, no need to continue...
4211 AssertFailed();
4212 if (snapshotsNode != aMachineNode)
4213 {
4214 CFGLDRReleaseNode (snapshotsNode);
4215 snapshotsNode = 0;
4216 }
4217 break;
4218 }
4219 }
4220
4221 // we must always succesfully find the node
4222 AssertReturn (snapshotNode, E_FAIL);
4223 AssertReturn (snapshotsNode, E_FAIL);
4224
4225 if (aSnapshotsNode)
4226 *aSnapshotsNode = snapshotsNode != aMachineNode ? snapshotsNode : 0;
4227 *aSnapshotNode = snapshotNode;
4228
4229 return S_OK;
4230}
4231
4232/**
4233 * Returns the snapshot with the given UUID or fails of no such snapshot.
4234 *
4235 * @param aId snapshot UUID to find (empty UUID refers the first snapshot)
4236 * @param aSnapshot where to return the found snapshot
4237 * @param aSetError true to set extended error info on failure
4238 */
4239HRESULT Machine::findSnapshot (const Guid &aId, ComObjPtr <Snapshot> &aSnapshot,
4240 bool aSetError /* = false */)
4241{
4242 if (!mData->mFirstSnapshot)
4243 {
4244 if (aSetError)
4245 return setError (E_FAIL,
4246 tr ("This machine does not have any snapshots"));
4247 return E_FAIL;
4248 }
4249
4250 if (aId.isEmpty())
4251 aSnapshot = mData->mFirstSnapshot;
4252 else
4253 aSnapshot = mData->mFirstSnapshot->findChildOrSelf (aId);
4254
4255 if (!aSnapshot)
4256 {
4257 if (aSetError)
4258 return setError (E_FAIL,
4259 tr ("Could not find a snapshot with UUID {%s}"),
4260 aId.toString().raw());
4261 return E_FAIL;
4262 }
4263
4264 return S_OK;
4265}
4266
4267/**
4268 * Returns the snapshot with the given name or fails of no such snapshot.
4269 *
4270 * @param aName snapshot name to find
4271 * @param aSnapshot where to return the found snapshot
4272 * @param aSetError true to set extended error info on failure
4273 */
4274HRESULT Machine::findSnapshot (const BSTR aName, ComObjPtr <Snapshot> &aSnapshot,
4275 bool aSetError /* = false */)
4276{
4277 AssertReturn (aName, E_INVALIDARG);
4278
4279 if (!mData->mFirstSnapshot)
4280 {
4281 if (aSetError)
4282 return setError (E_FAIL,
4283 tr ("This machine does not have any snapshots"));
4284 return E_FAIL;
4285 }
4286
4287 aSnapshot = mData->mFirstSnapshot->findChildOrSelf (aName);
4288
4289 if (!aSnapshot)
4290 {
4291 if (aSetError)
4292 return setError (E_FAIL,
4293 tr ("Could not find a snapshot named '%ls'"), aName);
4294 return E_FAIL;
4295 }
4296
4297 return S_OK;
4298}
4299
4300/**
4301 * Searches for an attachment that contains the given hard disk.
4302 * The hard disk must be associated with some VM and can be optionally
4303 * associated with some snapshot. If the attachment is stored in the snapshot
4304 * (i.e. the hard disk is associated with some snapshot), @a aSnapshot
4305 * will point to a non-null object on output.
4306 *
4307 * @param aHd hard disk to search an attachment for
4308 * @param aMachine where to store the hard disk's machine (can be NULL)
4309 * @param aSnapshot where to store the hard disk's snapshot (can be NULL)
4310 * @param aHda where to store the hard disk's attachment (can be NULL)
4311 *
4312 *
4313 * @note
4314 * It is assumed that the machine where the attachment is found,
4315 * is already placed to the Discarding state, when this method is called.
4316 * @note
4317 * The object returned in @a aHda is the attachment from the snapshot
4318 * machine if the hard disk is associated with the snapshot, not from the
4319 * primary machine object returned returned in @a aMachine.
4320 */
4321HRESULT Machine::findHardDiskAttachment (const ComObjPtr <HardDisk> &aHd,
4322 ComObjPtr <Machine> *aMachine,
4323 ComObjPtr <Snapshot> *aSnapshot,
4324 ComObjPtr <HardDiskAttachment> *aHda)
4325{
4326 AssertReturn (!aHd.isNull(), E_INVALIDARG);
4327
4328 Guid mid = aHd->machineId();
4329 Guid sid = aHd->snapshotId();
4330
4331 AssertReturn (!mid.isEmpty(), E_INVALIDARG);
4332
4333 ComObjPtr <Machine> m;
4334 mParent->getMachine (mid, m);
4335 ComAssertRet (!m.isNull(), E_FAIL);
4336
4337 HDData::HDAttachmentList *attachments = &m->mHDData->mHDAttachments;
4338
4339 ComObjPtr <Snapshot> s;
4340 if (!sid.isEmpty())
4341 {
4342 m->findSnapshot (sid, s);
4343 ComAssertRet (!s.isNull(), E_FAIL);
4344 attachments = &s->data().mMachine->mHDData->mHDAttachments;
4345 }
4346
4347 AssertReturn (attachments, E_FAIL);
4348
4349 for (HDData::HDAttachmentList::const_iterator it = attachments->begin();
4350 it != attachments->end();
4351 ++ it)
4352 {
4353 if ((*it)->hardDisk() == aHd)
4354 {
4355 if (aMachine) *aMachine = m;
4356 if (aSnapshot) *aSnapshot = s;
4357 if (aHda) *aHda = (*it);
4358 return S_OK;
4359 }
4360 }
4361
4362 ComAssertFailed();
4363 return E_FAIL;
4364}
4365
4366/**
4367 * Helper for #saveSettings. Cares about renaming the settings directory and
4368 * file if the machine name was changed and about creating a new settings file
4369 * if this is a new machine.
4370 *
4371 * @note Must be never called directly.
4372 *
4373 * @param aRenamed receives |true| if the name was changed and the settings
4374 * file was renamed as a result, or |false| otherwise. The
4375 * value makes sense only on success.
4376 * @param aNew receives |true| if a virgin settings file was created.
4377 */
4378HRESULT Machine::prepareSaveSettings (bool &aRenamed, bool &aNew)
4379{
4380 HRESULT rc = S_OK;
4381
4382 aRenamed = false;
4383
4384 /* if we're ready and isConfigLocked() is FALSE then it means
4385 * that no config file exists yet (we will create a virgin one) */
4386 aNew = !isConfigLocked();
4387
4388 /* attempt to rename the settings file if machine name is changed */
4389 if (mUserData->mNameSync &&
4390 mUserData.isBackedUp() &&
4391 mUserData.backedUpData()->mName != mUserData->mName)
4392 {
4393 aRenamed = true;
4394
4395 if (!aNew)
4396 {
4397 /* unlock the old config file */
4398 rc = unlockConfig();
4399 CheckComRCReturnRC (rc);
4400 }
4401
4402 bool dirRenamed = false;
4403 bool fileRenamed = false;
4404
4405 Utf8Str configFile, newConfigFile;
4406 Utf8Str configDir, newConfigDir;
4407
4408 do
4409 {
4410 int vrc = VINF_SUCCESS;
4411
4412 Utf8Str name = mUserData.backedUpData()->mName;
4413 Utf8Str newName = mUserData->mName;
4414
4415 configFile = mData->mConfigFileFull;
4416
4417 /* first, rename the directory if it matches the machine name */
4418 configDir = configFile;
4419 RTPathStripFilename (configDir.mutableRaw());
4420 newConfigDir = configDir;
4421 if (RTPathFilename (configDir) == name)
4422 {
4423 RTPathStripFilename (newConfigDir.mutableRaw());
4424 newConfigDir = Utf8StrFmt ("%s%c%s",
4425 newConfigDir.raw(), RTPATH_DELIMITER, newName.raw());
4426 /* new dir and old dir cannot be equal here because of 'if'
4427 * above and because name != newName */
4428 Assert (configDir != newConfigDir);
4429 if (!aNew)
4430 {
4431 /* perform real rename only if the machine is not new */
4432 vrc = RTPathRename (configDir.raw(), newConfigDir.raw(), 0);
4433 if (VBOX_FAILURE (vrc))
4434 {
4435 rc = setError (E_FAIL,
4436 tr ("Could not rename the directory '%s' to '%s' "
4437 "to save the settings file (%Vrc)"),
4438 configDir.raw(), newConfigDir.raw(), vrc);
4439 break;
4440 }
4441 dirRenamed = true;
4442 }
4443 }
4444
4445 newConfigFile = Utf8StrFmt ("%s%c%s.xml",
4446 newConfigDir.raw(), RTPATH_DELIMITER, newName.raw());
4447
4448 /* then try to rename the settings file itself */
4449 if (newConfigFile != configFile)
4450 {
4451 /* get the path to old settings file in renamed directory */
4452 configFile = Utf8StrFmt ("%s%c%s",
4453 newConfigDir.raw(), RTPATH_DELIMITER,
4454 RTPathFilename (configFile));
4455 if (!aNew)
4456 {
4457 /* perform real rename only if the machine is not new */
4458 vrc = RTFileRename (configFile.raw(), newConfigFile.raw(), 0);
4459 if (VBOX_FAILURE (vrc))
4460 {
4461 rc = setError (E_FAIL,
4462 tr ("Could not rename the settings file '%s' to '%s' "
4463 "(%Vrc)"),
4464 configFile.raw(), newConfigFile.raw(), vrc);
4465 break;
4466 }
4467 fileRenamed = true;
4468 }
4469 }
4470
4471 /* update mConfigFileFull amd mConfigFile */
4472 Bstr oldConfigFileFull = mData->mConfigFileFull;
4473 Bstr oldConfigFile = mData->mConfigFile;
4474 mData->mConfigFileFull = newConfigFile;
4475 /* try to get the relative path for mConfigFile */
4476 Utf8Str path = newConfigFile;
4477 mParent->calculateRelativePath (path, path);
4478 mData->mConfigFile = path;
4479
4480 /* last, try to update the global settings with the new path */
4481 if (mData->mRegistered)
4482 {
4483 rc = mParent->updateSettings (configDir, newConfigDir);
4484 if (FAILED (rc))
4485 {
4486 /* revert to old values */
4487 mData->mConfigFileFull = oldConfigFileFull;
4488 mData->mConfigFile = oldConfigFile;
4489 break;
4490 }
4491 }
4492
4493 /* update the snapshot folder */
4494 path = mUserData->mSnapshotFolderFull;
4495 if (RTPathStartsWith (path, configDir))
4496 {
4497 path = Utf8StrFmt ("%s%s", newConfigDir.raw(),
4498 path.raw() + configDir.length());
4499 mUserData->mSnapshotFolderFull = path;
4500 calculateRelativePath (path, path);
4501 mUserData->mSnapshotFolder = path;
4502 }
4503
4504 /* update the saved state file path */
4505 path = mSSData->mStateFilePath;
4506 if (RTPathStartsWith (path, configDir))
4507 {
4508 path = Utf8StrFmt ("%s%s", newConfigDir.raw(),
4509 path.raw() + configDir.length());
4510 mSSData->mStateFilePath = path;
4511 }
4512
4513 /* Update saved state file paths of all online snapshots.
4514 * Note that saveSettings() will recognize name change
4515 * and will save all snapshots in this case. */
4516 if (mData->mFirstSnapshot)
4517 mData->mFirstSnapshot->updateSavedStatePaths (configDir,
4518 newConfigDir);
4519 }
4520 while (0);
4521
4522 if (FAILED (rc))
4523 {
4524 /* silently try to rename everything back */
4525 if (fileRenamed)
4526 RTFileRename (newConfigFile.raw(), configFile.raw(), 0);
4527 if (dirRenamed)
4528 RTPathRename (newConfigDir.raw(), configDir.raw(), 0);
4529 }
4530
4531 if (!aNew)
4532 {
4533 /* lock the config again */
4534 HRESULT rc2 = lockConfig();
4535 if (SUCCEEDED (rc))
4536 rc = rc2;
4537 }
4538
4539 CheckComRCReturnRC (rc);
4540 }
4541
4542 if (aNew)
4543 {
4544 /* create a virgin config file */
4545 int vrc = VINF_SUCCESS;
4546
4547 /* ensure the settings directory exists */
4548 Utf8Str path = mData->mConfigFileFull;
4549 RTPathStripFilename (path.mutableRaw());
4550 if (!RTDirExists (path))
4551 {
4552 vrc = RTDirCreateFullPath (path, 0777);
4553 if (VBOX_FAILURE (vrc))
4554 {
4555 return setError (E_FAIL,
4556 tr ("Could not create a directory '%s' "
4557 "to save the settings file (%Vrc)"),
4558 path.raw(), vrc);
4559 }
4560 }
4561
4562 /* Note: open flags must correlated with RTFileOpen() in lockConfig() */
4563 path = Utf8Str (mData->mConfigFileFull);
4564 vrc = RTFileOpen (&mData->mHandleCfgFile, path,
4565 RTFILE_O_READWRITE | RTFILE_O_CREATE |
4566 RTFILE_O_DENY_WRITE);
4567 if (VBOX_SUCCESS (vrc))
4568 {
4569 vrc = RTFileWrite (mData->mHandleCfgFile,
4570 (void *) DefaultMachineConfig,
4571 sizeof (DefaultMachineConfig), NULL);
4572 }
4573 if (VBOX_FAILURE (vrc))
4574 {
4575 mData->mHandleCfgFile = NIL_RTFILE;
4576 return setError (E_FAIL,
4577 tr ("Could not create the settings file '%s' (%Vrc)"),
4578 path.raw(), vrc);
4579 }
4580 /* we do not close the file to simulate lockConfig() */
4581 }
4582
4583 return rc;
4584}
4585
4586/**
4587 * Saves machine data, user data and hardware data.
4588 *
4589 * @param aMarkCurStateAsModified
4590 * if true (default), mData->mCurrentStateModified will be set to
4591 * what #isReallyModified() returns prior to saving settings to a file,
4592 * otherwise the current value of mData->mCurrentStateModified will be
4593 * saved.
4594 * @param aInformCallbacksAnyway
4595 * if true, callbacks will be informed even if #isReallyModified()
4596 * returns false. This is necessary for cases when we change machine data
4597 * diectly, not through the backup()/commit() mechanism.
4598 *
4599 * @note Locks mParent (only in some cases, and only when #isConfigLocked() is
4600 * |TRUE|, see the #prepareSaveSettings() code for details) +
4601 * this object + children for writing.
4602 */
4603HRESULT Machine::saveSettings (bool aMarkCurStateAsModified /* = true */,
4604 bool aInformCallbacksAnyway /* = false */)
4605{
4606 /// @todo (dmik) I guess we should lock all our child objects here
4607 // (such as mVRDPServer etc.) to ensure they are not changed
4608 // until completely saved to disk and committed
4609
4610 /// @todo (dmik) also, we need to delegate saving child objects' settings
4611 // to objects themselves to ensure operations 'commit + save changes'
4612 // are atomic (amd done from the object's lock so that nobody can change
4613 // settings again until completely saved).
4614
4615 AssertReturn (mType == IsMachine || mType == IsSessionMachine, E_FAIL);
4616
4617 bool wasModified;
4618
4619 if (aMarkCurStateAsModified)
4620 {
4621 /*
4622 * We ignore changes to user data when setting mCurrentStateModified
4623 * because the current state will not differ from the current snapshot
4624 * if only user data has been changed (user data is shared by all
4625 * snapshots).
4626 */
4627 mData->mCurrentStateModified = isReallyModified (true /* aIgnoreUserData */);
4628 wasModified = mUserData.hasActualChanges() || mData->mCurrentStateModified;
4629 }
4630 else
4631 {
4632 wasModified = isReallyModified();
4633 }
4634
4635 HRESULT rc = S_OK;
4636
4637 /* First, prepare to save settings. It will will care about renaming the
4638 * settings directory and file if the machine name was changed and about
4639 * creating a new settings file if this is a new machine. */
4640 bool isRenamed = false;
4641 bool isNew = false;
4642 rc = prepareSaveSettings (isRenamed, isNew);
4643 CheckComRCReturnRC (rc);
4644
4645 /* then, open the settings file */
4646 CFGHANDLE configLoader = 0;
4647 rc = openConfigLoader (&configLoader, isNew);
4648 CheckComRCReturnRC (rc);
4649
4650 /* save all snapshots when the machine name was changed since
4651 * it may affect saved state file paths for online snapshots (see
4652 * #openConfigLoader() for details) */
4653 bool updateAllSnapshots = isRenamed;
4654
4655 /* commit before saving, since it may change settings
4656 * (for example, perform fixup of lazy hard disk changes) */
4657 rc = commit();
4658 if (FAILED (rc))
4659 {
4660 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
4661 return rc;
4662 }
4663
4664 /* include hard disk changes to the modified flag */
4665 wasModified |= mHDData->mHDAttachmentsChanged;
4666 if (aMarkCurStateAsModified)
4667 mData->mCurrentStateModified |= BOOL (mHDData->mHDAttachmentsChanged);
4668
4669
4670 CFGNODE machineNode = 0;
4671 /* create if not exists */
4672 CFGLDRCreateNode (configLoader, "VirtualBox/Machine", &machineNode);
4673
4674 do
4675 {
4676 ComAssertBreak (machineNode, rc = E_FAIL);
4677
4678 /* uuid (required) */
4679 Assert (mData->mUuid);
4680 CFGLDRSetUUID (machineNode, "uuid", mData->mUuid.raw());
4681
4682 /* name (required) */
4683 Assert (!mUserData->mName.isEmpty());
4684 CFGLDRSetBSTR (machineNode, "name", mUserData->mName);
4685
4686 /* nameSync (optional, default is true) */
4687 if (!mUserData->mNameSync)
4688 CFGLDRSetBool (machineNode, "nameSync", false);
4689 else
4690 CFGLDRDeleteAttribute (machineNode, "nameSync");
4691
4692 /* OSType (required) */
4693 {
4694 Bstr osTypeID;
4695 rc = mUserData->mOSType->COMGETTER(Id) (osTypeID.asOutParam());
4696 ComAssertComRCBreak (rc, rc = rc);
4697 Assert (!osTypeID.isNull());
4698 CFGLDRSetBSTR (machineNode, "OSType", osTypeID);
4699 }
4700
4701 /* stateFile (optional) */
4702 if (mData->mMachineState == MachineState_Saved)
4703 {
4704 Assert (!mSSData->mStateFilePath.isEmpty());
4705 /* try to make the file name relative to the settings file dir */
4706 Utf8Str stateFilePath = mSSData->mStateFilePath;
4707 calculateRelativePath (stateFilePath, stateFilePath);
4708 CFGLDRSetString (machineNode, "stateFile", stateFilePath);
4709 }
4710 else
4711 {
4712 Assert (mSSData->mStateFilePath.isNull());
4713 CFGLDRDeleteAttribute (machineNode, "stateFile");
4714 }
4715
4716 /* currentSnapshot ID (optional) */
4717 if (!mData->mCurrentSnapshot.isNull())
4718 {
4719 Assert (!mData->mFirstSnapshot.isNull());
4720 CFGLDRSetUUID (machineNode, "currentSnapshot",
4721 mData->mCurrentSnapshot->data().mId);
4722 }
4723 else
4724 {
4725 Assert (mData->mFirstSnapshot.isNull());
4726 CFGLDRDeleteAttribute (machineNode, "currentSnapshot");
4727 }
4728
4729 /* snapshotFolder (optional) */
4730 if (mUserData->mSnapshotFolder)
4731 CFGLDRSetBSTR (machineNode, "snapshotFolder", mUserData->mSnapshotFolder);
4732 else
4733 CFGLDRDeleteAttribute (machineNode, "snapshotFolder");
4734
4735 /* currentStateModified (optional, default is yes) */
4736 if (!mData->mCurrentStateModified)
4737 CFGLDRSetBool (machineNode, "currentStateModified", false);
4738 else
4739 CFGLDRDeleteAttribute (machineNode, "currentStateModified");
4740
4741 /* lastStateChange */
4742 CFGLDRSetDateTime (machineNode, "lastStateChange",
4743 mData->mLastStateChange);
4744
4745 /* Hardware node (required) */
4746 {
4747 CFGNODE hwNode = 0;
4748 CFGLDRGetChildNode (machineNode, "Hardware", 0, &hwNode);
4749 /* first, delete the entire node if exists */
4750 if (hwNode)
4751 CFGLDRDeleteNode (hwNode);
4752 /* then recreate it */
4753 hwNode = 0;
4754 CFGLDRCreateChildNode (machineNode, "Hardware", &hwNode);
4755 ComAssertBreak (hwNode, rc = E_FAIL);
4756
4757 rc = saveHardware (hwNode);
4758
4759 CFGLDRReleaseNode (hwNode);
4760 if (FAILED (rc))
4761 break;
4762 }
4763
4764 /* HardDiskAttachments node (required) */
4765 {
4766 CFGNODE hdasNode = 0;
4767 CFGLDRGetChildNode (machineNode, "HardDiskAttachments", 0, &hdasNode);
4768 /* first, delete the entire node if exists */
4769 if (hdasNode)
4770 CFGLDRDeleteNode (hdasNode);
4771 /* then recreate it */
4772 hdasNode = 0;
4773 CFGLDRCreateChildNode (machineNode, "HardDiskAttachments", &hdasNode);
4774 ComAssertBreak (hdasNode, rc = E_FAIL);
4775
4776 rc = saveHardDisks (hdasNode);
4777
4778 CFGLDRReleaseNode (hdasNode);
4779 if (FAILED (rc))
4780 break;
4781 }
4782
4783 /* update all snapshots if requested */
4784 if (updateAllSnapshots)
4785 rc = saveSnapshotSettingsWorker (machineNode, NULL,
4786 SaveSS_UpdateAllOp);
4787 }
4788 while (0);
4789
4790 if (machineNode)
4791 CFGLDRReleaseNode (machineNode);
4792
4793 if (SUCCEEDED (rc))
4794 rc = closeConfigLoader (configLoader, true /* aSaveBeforeClose */);
4795 else
4796 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
4797
4798 if (FAILED (rc))
4799 {
4800 /*
4801 * backup arbitrary data item to cause #isModified() to still return
4802 * true in case of any error
4803 */
4804 mHWData.backup();
4805 }
4806
4807 if (wasModified || aInformCallbacksAnyway)
4808 {
4809 /*
4810 * Fire the data change event, even on failure (since we've already
4811 * committed all data). This is done only for SessionMachines because
4812 * mutable Machine instances are always not registered (i.e. private
4813 * to the client process that creates them) and thus don't need to
4814 * inform callbacks.
4815 */
4816 if (mType == IsSessionMachine)
4817 mParent->onMachineDataChange (mData->mUuid);
4818 }
4819
4820 return rc;
4821}
4822
4823/**
4824 * Wrapper for #saveSnapshotSettingsWorker() that opens the settings file
4825 * and locates the <Machine> node in there. See #saveSnapshotSettingsWorker()
4826 * for more details.
4827 *
4828 * @param aSnapshot Snapshot to operate on
4829 * @param aOpFlags Operation to perform, one of SaveSS_NoOp, SaveSS_AddOp
4830 * or SaveSS_UpdateAttrsOp possibly combined with
4831 * SaveSS_UpdateCurrentId.
4832 *
4833 * @note Locks this object for writing + other child objects.
4834 */
4835HRESULT Machine::saveSnapshotSettings (Snapshot *aSnapshot, int aOpFlags)
4836{
4837 AutoCaller autoCaller (this);
4838 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4839
4840 AssertReturn (mType == IsMachine || mType == IsSessionMachine, E_FAIL);
4841
4842 AutoLock alock (this);
4843
4844 AssertReturn (isConfigLocked(), E_FAIL);
4845
4846 HRESULT rc = S_OK;
4847
4848 /* load the config file */
4849 CFGHANDLE configLoader = 0;
4850 rc = openConfigLoader (&configLoader);
4851 if (FAILED (rc))
4852 return rc;
4853
4854 CFGNODE machineNode = 0;
4855 CFGLDRGetNode (configLoader, "VirtualBox/Machine", 0, &machineNode);
4856
4857 do
4858 {
4859 ComAssertBreak (machineNode, rc = E_FAIL);
4860
4861 rc = saveSnapshotSettingsWorker (machineNode, aSnapshot, aOpFlags);
4862
4863 CFGLDRReleaseNode (machineNode);
4864 }
4865 while (0);
4866
4867 if (SUCCEEDED (rc))
4868 rc = closeConfigLoader (configLoader, true /* aSaveBeforeClose */);
4869 else
4870 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
4871
4872 return rc;
4873}
4874
4875/**
4876 * Performs the specified operation on the given snapshot
4877 * in the settings file represented by \a aMachineNode.
4878 *
4879 * If \a aOpFlags = SaveSS_UpdateAllOp, \a aSnapshot can be NULL to indicate
4880 * that the whole tree of the snapshots should be updated in <Machine>.
4881 * One particular case is when the last (and the only) snapshot should be
4882 * removed (it is so when both mCurrentSnapshot and mFirstSnapshot are NULL).
4883 *
4884 * \a aOp may be just SaveSS_UpdateCurrentId if only the currentSnapshot
4885 * attribute of <Machine> needs to be updated.
4886 *
4887 * @param aMachineNode <Machine> node in the opened settings file
4888 * @param aSnapshot Snapshot to operate on
4889 * @param aOpFlags Operation to perform, one of SaveSS_NoOp, SaveSS_AddOp
4890 * or SaveSS_UpdateAttrsOp possibly combined with
4891 * SaveSS_UpdateCurrentId.
4892 *
4893 * @note Must be called with this object locked for writing.
4894 * Locks child objects.
4895 */
4896HRESULT Machine::saveSnapshotSettingsWorker (CFGNODE aMachineNode,
4897 Snapshot *aSnapshot, int aOpFlags)
4898{
4899 AssertReturn (aMachineNode, E_FAIL);
4900
4901 AssertReturn (isLockedOnCurrentThread(), E_FAIL);
4902
4903 int op = aOpFlags & SaveSS_OpMask;
4904 AssertReturn (
4905 (aSnapshot && (op == SaveSS_AddOp || op == SaveSS_UpdateAttrsOp ||
4906 op == SaveSS_UpdateAllOp)) ||
4907 (!aSnapshot && ((op == SaveSS_NoOp && (aOpFlags & SaveSS_UpdateCurrentId)) ||
4908 op == SaveSS_UpdateAllOp)),
4909 E_FAIL);
4910
4911 HRESULT rc = S_OK;
4912
4913 bool recreateWholeTree = false;
4914
4915 do
4916 {
4917 if (op == SaveSS_NoOp)
4918 break;
4919
4920 /* quick path: recreate the whole tree of the snapshots */
4921 if (op == SaveSS_UpdateAllOp && !aSnapshot)
4922 {
4923 /* first, delete the entire root snapshot node if it exists */
4924 CFGNODE snapshotNode = 0;
4925 CFGLDRGetChildNode (aMachineNode, "Snapshot", 0, &snapshotNode);
4926 if (snapshotNode)
4927 CFGLDRDeleteNode (snapshotNode);
4928
4929 /*
4930 * second, if we have any snapshots left, substitute aSnapshot with
4931 * the first snapshot to recreate the whole tree, otherwise break
4932 */
4933 if (mData->mFirstSnapshot)
4934 {
4935 aSnapshot = mData->mFirstSnapshot;
4936 recreateWholeTree = true;
4937 }
4938 else
4939 break;
4940 }
4941
4942 Assert (!!aSnapshot);
4943 ComObjPtr <Snapshot> parent = aSnapshot->parent();
4944
4945 if (op == SaveSS_AddOp)
4946 {
4947 CFGNODE parentNode = 0;
4948
4949 if (parent)
4950 {
4951 rc = findSnapshotNode (parent, aMachineNode, NULL, &parentNode);
4952 if (FAILED (rc))
4953 break;
4954 ComAssertBreak (parentNode, rc = E_FAIL);
4955 }
4956
4957 do
4958 {
4959 CFGNODE snapshotsNode = 0;
4960
4961 if (parentNode)
4962 {
4963 CFGLDRCreateChildNode (parentNode, "Snapshots", &snapshotsNode);
4964 ComAssertBreak (snapshotsNode, rc = E_FAIL);
4965 }
4966 else
4967 snapshotsNode = aMachineNode;
4968 do
4969 {
4970 CFGNODE snapshotNode = 0;
4971 CFGLDRAppendChildNode (snapshotsNode, "Snapshot", &snapshotNode);
4972 ComAssertBreak (snapshotNode, rc = E_FAIL);
4973 rc = saveSnapshot (snapshotNode, aSnapshot, false /* aAttrsOnly */);
4974 CFGLDRReleaseNode (snapshotNode);
4975
4976 if (FAILED (rc))
4977 break;
4978
4979 /*
4980 * when a new snapshot is added, this means diffs were created
4981 * for every normal/immutable hard disk of the VM, so we need to
4982 * save the current hard disk attachments
4983 */
4984
4985 CFGNODE hdasNode = 0;
4986 CFGLDRGetChildNode (aMachineNode, "HardDiskAttachments", 0, &hdasNode);
4987 if (hdasNode)
4988 CFGLDRDeleteNode (hdasNode);
4989 CFGLDRCreateChildNode (aMachineNode, "HardDiskAttachments", &hdasNode);
4990 ComAssertBreak (hdasNode, rc = E_FAIL);
4991
4992 rc = saveHardDisks (hdasNode);
4993
4994 if (mHDData->mHDAttachments.size() != 0)
4995 {
4996 /*
4997 * If we have one or more attachments then we definitely
4998 * created diffs for them and associated new diffs with
4999 * current settngs. So, since we don't use saveSettings(),
5000 * we need to inform callbacks manually.
5001 */
5002 if (mType == IsSessionMachine)
5003 mParent->onMachineDataChange (mData->mUuid);
5004 }
5005
5006 CFGLDRReleaseNode (hdasNode);
5007 }
5008 while (0);
5009
5010 if (snapshotsNode != aMachineNode)
5011 CFGLDRReleaseNode (snapshotsNode);
5012 }
5013 while (0);
5014
5015 if (parentNode)
5016 CFGLDRReleaseNode (parentNode);
5017
5018 break;
5019 }
5020
5021 Assert (op == SaveSS_UpdateAttrsOp && !recreateWholeTree ||
5022 op == SaveSS_UpdateAllOp);
5023
5024 CFGNODE snapshotsNode = 0;
5025 CFGNODE snapshotNode = 0;
5026
5027 if (!recreateWholeTree)
5028 {
5029 rc = findSnapshotNode (aSnapshot, aMachineNode,
5030 &snapshotsNode, &snapshotNode);
5031 if (FAILED (rc))
5032 break;
5033 ComAssertBreak (snapshotNode, rc = E_FAIL);
5034 }
5035
5036 if (!snapshotsNode)
5037 snapshotsNode = aMachineNode;
5038
5039 if (op == SaveSS_UpdateAttrsOp)
5040 rc = saveSnapshot (snapshotNode, aSnapshot, true /* aAttrsOnly */);
5041 else do
5042 {
5043 if (snapshotNode)
5044 {
5045 CFGLDRDeleteNode (snapshotNode);
5046 snapshotNode = 0;
5047 }
5048 CFGLDRAppendChildNode (snapshotsNode, "Snapshot", &snapshotNode);
5049 ComAssertBreak (snapshotNode, rc = E_FAIL);
5050 rc = saveSnapshot (snapshotNode, aSnapshot, false /* aAttrsOnly */);
5051 }
5052 while (0);
5053
5054 CFGLDRReleaseNode (snapshotNode);
5055 if (snapshotsNode != aMachineNode)
5056 CFGLDRReleaseNode (snapshotsNode);
5057 }
5058 while (0);
5059
5060 if (SUCCEEDED (rc))
5061 {
5062 /* update currentSnapshot when appropriate */
5063 if (aOpFlags & SaveSS_UpdateCurrentId)
5064 {
5065 if (!mData->mCurrentSnapshot.isNull())
5066 CFGLDRSetUUID (aMachineNode, "currentSnapshot",
5067 mData->mCurrentSnapshot->data().mId);
5068 else
5069 CFGLDRDeleteAttribute (aMachineNode, "currentSnapshot");
5070 }
5071 if (aOpFlags & SaveSS_UpdateCurStateModified)
5072 {
5073 if (!mData->mCurrentStateModified)
5074 CFGLDRSetBool (aMachineNode, "currentStateModified", false);
5075 else
5076 CFGLDRDeleteAttribute (aMachineNode, "currentStateModified");
5077 }
5078 }
5079
5080 return rc;
5081}
5082
5083/**
5084 * Saves the given snapshot and all its children (unless \a aAttrsOnly is true).
5085 * It is assumed that the given node is empty (unless \a aAttrsOnly is true).
5086 *
5087 * @param aNode <Snapshot> node to save the snapshot to
5088 * @param aSnapshot snapshot to save
5089 * @param aAttrsOnly if true, only updatge user-changeable attrs
5090 */
5091HRESULT Machine::saveSnapshot (CFGNODE aNode, Snapshot *aSnapshot, bool aAttrsOnly)
5092{
5093 AssertReturn (aNode && aSnapshot, E_INVALIDARG);
5094 AssertReturn (mType == IsMachine || mType == IsSessionMachine, E_FAIL);
5095
5096 /* uuid (required) */
5097 if (!aAttrsOnly)
5098 CFGLDRSetUUID (aNode, "uuid", aSnapshot->data().mId);
5099
5100 /* name (required) */
5101 CFGLDRSetBSTR (aNode, "name", aSnapshot->data().mName);
5102
5103 /* timeStamp (required) */
5104 CFGLDRSetDateTime (aNode, "timeStamp", aSnapshot->data().mTimeStamp);
5105
5106 /* Description node (optional) */
5107 {
5108 CFGNODE descNode = 0;
5109 CFGLDRCreateChildNode (aNode, "Description", &descNode);
5110 CFGLDRSetBSTR (descNode, NULL, aSnapshot->data().mDescription);
5111 CFGLDRReleaseNode (descNode);
5112 }
5113
5114 if (aAttrsOnly)
5115 return S_OK;
5116
5117 /* stateFile (optional) */
5118 if (aSnapshot->stateFilePath())
5119 {
5120 /* try to make the file name relative to the settings file dir */
5121 Utf8Str stateFilePath = aSnapshot->stateFilePath();
5122 calculateRelativePath (stateFilePath, stateFilePath);
5123 CFGLDRSetString (aNode, "stateFile", stateFilePath);
5124 }
5125
5126 {
5127 ComObjPtr <SnapshotMachine> snapshotMachine = aSnapshot->data().mMachine;
5128 ComAssertRet (!snapshotMachine.isNull(), E_FAIL);
5129
5130 /* save hardware */
5131 {
5132 CFGNODE hwNode = 0;
5133 CFGLDRCreateChildNode (aNode, "Hardware", &hwNode);
5134
5135 HRESULT rc = snapshotMachine->saveHardware (hwNode);
5136
5137 CFGLDRReleaseNode (hwNode);
5138 if (FAILED (rc))
5139 return rc;
5140 }
5141
5142 /* save hard disks */
5143 {
5144 CFGNODE hdasNode = 0;
5145 CFGLDRCreateChildNode (aNode, "HardDiskAttachments", &hdasNode);
5146
5147 HRESULT rc = snapshotMachine->saveHardDisks (hdasNode);
5148
5149 CFGLDRReleaseNode (hdasNode);
5150 if (FAILED (rc))
5151 return rc;
5152 }
5153 }
5154
5155 /* save children */
5156 {
5157 AutoLock listLock (aSnapshot->childrenLock());
5158
5159 if (aSnapshot->children().size())
5160 {
5161 CFGNODE snapshotsNode = 0;
5162 CFGLDRCreateChildNode (aNode, "Snapshots", &snapshotsNode);
5163
5164 HRESULT rc = S_OK;
5165
5166 for (Snapshot::SnapshotList::const_iterator it = aSnapshot->children().begin();
5167 it != aSnapshot->children().end() && SUCCEEDED (rc);
5168 ++ it)
5169 {
5170 CFGNODE snapshotNode = 0;
5171 CFGLDRCreateChildNode (snapshotsNode, "Snapshot", &snapshotNode);
5172
5173 rc = saveSnapshot (snapshotNode, (*it), aAttrsOnly);
5174
5175 CFGLDRReleaseNode (snapshotNode);
5176 }
5177
5178 CFGLDRReleaseNode (snapshotsNode);
5179 if (FAILED (rc))
5180 return rc;
5181 }
5182 }
5183
5184 return S_OK;
5185}
5186
5187/**
5188 * Creates Saves the VM hardware configuration.
5189 * It is assumed that the given node is empty.
5190 *
5191 * @param aNode <Hardware> node to save the VM hardware confguration to
5192 */
5193HRESULT Machine::saveHardware (CFGNODE aNode)
5194{
5195 AssertReturn (aNode, E_INVALIDARG);
5196
5197 HRESULT rc = S_OK;
5198
5199 /* CPU */
5200 {
5201 CFGNODE cpuNode = 0;
5202 CFGLDRCreateChildNode (aNode, "CPU", &cpuNode);
5203 CFGNODE hwVirtExNode = 0;
5204 CFGLDRCreateChildNode (cpuNode, "HardwareVirtEx", &hwVirtExNode);
5205 char *value = NULL;
5206 switch (mHWData->mHWVirtExEnabled)
5207 {
5208 case TriStateBool_False:
5209 value = "false";
5210 break;
5211 case TriStateBool_True:
5212 value = "true";
5213 break;
5214 default:
5215 value = "default";
5216 }
5217 CFGLDRSetString (hwVirtExNode, "enabled", value);
5218 CFGLDRReleaseNode (hwVirtExNode);
5219 CFGLDRReleaseNode (cpuNode);
5220 }
5221
5222 /* memory (required) */
5223 {
5224 CFGNODE memoryNode = 0;
5225 CFGLDRCreateChildNode (aNode, "Memory", &memoryNode);
5226 CFGLDRSetUInt32 (memoryNode, "RAMSize", mHWData->mMemorySize);
5227 CFGLDRReleaseNode (memoryNode);
5228 }
5229
5230 /* boot (required) */
5231 do
5232 {
5233 CFGNODE bootNode = 0;
5234 CFGLDRCreateChildNode (aNode, "Boot", &bootNode);
5235
5236 for (ULONG pos = 0; pos < ELEMENTS (mHWData->mBootOrder); pos ++)
5237 {
5238 const char *device = NULL;
5239 switch (mHWData->mBootOrder [pos])
5240 {
5241 case DeviceType_NoDevice:
5242 /* skip, this is allowed for <Order> nodes
5243 * when loading, the default value NoDevice will remain */
5244 continue;
5245 case DeviceType_FloppyDevice: device = "Floppy"; break;
5246 case DeviceType_DVDDevice: device = "DVD"; break;
5247 case DeviceType_HardDiskDevice: device = "HardDisk"; break;
5248 case DeviceType_NetworkDevice: device = "Network"; break;
5249 default:
5250 ComAssertMsgFailedBreak (("Invalid boot device: %d\n",
5251 mHWData->mBootOrder [pos]),
5252 rc = E_FAIL);
5253 }
5254 if (FAILED (rc))
5255 break;
5256
5257 CFGNODE orderNode = 0;
5258 CFGLDRAppendChildNode (bootNode, "Order", &orderNode);
5259
5260 CFGLDRSetUInt32 (orderNode, "position", pos + 1);
5261 CFGLDRSetString (orderNode, "device", device);
5262
5263 CFGLDRReleaseNode (orderNode);
5264 }
5265
5266 CFGLDRReleaseNode (bootNode);
5267 }
5268 while (0);
5269
5270 if (FAILED (rc))
5271 return rc;
5272
5273 /* display (required) */
5274 {
5275 CFGNODE displayNode = 0;
5276 CFGLDRCreateChildNode (aNode, "Display", &displayNode);
5277 CFGLDRSetUInt32 (displayNode, "VRAMSize", mHWData->mVRAMSize);
5278 CFGLDRReleaseNode (displayNode);
5279 }
5280
5281#ifdef VBOX_VRDP
5282 /* VRDP settings (optional) */
5283 /// @todo (dmik) move the code to VRDPServer
5284 /// @todo r=sunlover: moved. dmik, please review.
5285 {
5286 CFGNODE remoteDisplayNode = 0;
5287 CFGLDRCreateChildNode (aNode, "RemoteDisplay", &remoteDisplayNode);
5288
5289 if (remoteDisplayNode)
5290 {
5291 mVRDPServer->saveConfig (remoteDisplayNode);
5292 CFGLDRReleaseNode (remoteDisplayNode);
5293 }
5294 }
5295#endif
5296
5297 /* BIOS (required) */
5298 {
5299 CFGNODE biosNode = 0;
5300 CFGLDRCreateChildNode (aNode, "BIOS", &biosNode);
5301 {
5302 BOOL fSet;
5303 /* ACPI */
5304 CFGNODE acpiNode = 0;
5305 CFGLDRCreateChildNode (biosNode, "ACPI", &acpiNode);
5306 mBIOSSettings->COMGETTER(ACPIEnabled)(&fSet);
5307 CFGLDRSetBool (acpiNode, "enabled", !!fSet);
5308 CFGLDRReleaseNode (acpiNode);
5309
5310 /* IOAPIC */
5311 CFGNODE ioapicNode = 0;
5312 CFGLDRCreateChildNode (biosNode, "IOAPIC", &ioapicNode);
5313 mBIOSSettings->COMGETTER(IOAPICEnabled)(&fSet);
5314 CFGLDRSetBool (ioapicNode, "enabled", !!fSet);
5315 CFGLDRReleaseNode (ioapicNode);
5316
5317 /* BIOS logo (optional) **/
5318 CFGNODE logoNode = 0;
5319 CFGLDRCreateChildNode (biosNode, "Logo", &logoNode);
5320 mBIOSSettings->COMGETTER(LogoFadeIn)(&fSet);
5321 CFGLDRSetBool (logoNode, "fadeIn", !!fSet);
5322 mBIOSSettings->COMGETTER(LogoFadeOut)(&fSet);
5323 CFGLDRSetBool (logoNode, "fadeOut", !!fSet);
5324 ULONG ulDisplayTime;
5325 mBIOSSettings->COMGETTER(LogoDisplayTime)(&ulDisplayTime);
5326 CFGLDRSetUInt32 (logoNode, "displayTime", ulDisplayTime);
5327 Bstr logoPath;
5328 mBIOSSettings->COMGETTER(LogoImagePath)(logoPath.asOutParam());
5329 if (logoPath)
5330 CFGLDRSetBSTR (logoNode, "imagePath", logoPath);
5331 else
5332 CFGLDRDeleteAttribute (logoNode, "imagePath");
5333 CFGLDRReleaseNode (logoNode);
5334
5335 /* boot menu (optional) */
5336 CFGNODE bootMenuNode = 0;
5337 CFGLDRCreateChildNode (biosNode, "BootMenu", &bootMenuNode);
5338 BIOSBootMenuMode_T bootMenuMode;
5339 Bstr bootMenuModeStr;
5340 mBIOSSettings->COMGETTER(BootMenuMode)(&bootMenuMode);
5341 switch (bootMenuMode)
5342 {
5343 case BIOSBootMenuMode_Disabled:
5344 bootMenuModeStr = "disabled";
5345 break;
5346 case BIOSBootMenuMode_MenuOnly:
5347 bootMenuModeStr = "menuonly";
5348 break;
5349 default:
5350 bootMenuModeStr = "messageandmenu";
5351 }
5352 CFGLDRSetBSTR (bootMenuNode, "mode", bootMenuModeStr);
5353 CFGLDRReleaseNode (bootMenuNode);
5354 }
5355 CFGLDRReleaseNode(biosNode);
5356 }
5357
5358 /* DVD drive (required) */
5359 /// @todo (dmik) move the code to DVDDrive
5360 do
5361 {
5362 CFGNODE dvdNode = 0;
5363 CFGLDRCreateChildNode (aNode, "DVDDrive", &dvdNode);
5364
5365 BOOL fPassthrough;
5366 mDVDDrive->COMGETTER(Passthrough)(&fPassthrough);
5367 CFGLDRSetBool(dvdNode, "passthrough", !!fPassthrough);
5368
5369 switch (mDVDDrive->data()->mDriveState)
5370 {
5371 case DriveState_ImageMounted:
5372 {
5373 Assert (!mDVDDrive->data()->mDVDImage.isNull());
5374
5375 Guid id;
5376 rc = mDVDDrive->data()->mDVDImage->COMGETTER(Id) (id.asOutParam());
5377 Assert (!id.isEmpty());
5378
5379 CFGNODE imageNode = 0;
5380 CFGLDRCreateChildNode (dvdNode, "Image", &imageNode);
5381 CFGLDRSetUUID (imageNode, "uuid", id.ptr());
5382 CFGLDRReleaseNode (imageNode);
5383 break;
5384 }
5385 case DriveState_HostDriveCaptured:
5386 {
5387 Assert (!mDVDDrive->data()->mHostDrive.isNull());
5388
5389 Bstr name;
5390 rc = mDVDDrive->data()->mHostDrive->COMGETTER(Name) (name.asOutParam());
5391 Assert (!name.isEmpty());
5392
5393 CFGNODE hostDriveNode = 0;
5394 CFGLDRCreateChildNode (dvdNode, "HostDrive", &hostDriveNode);
5395 CFGLDRSetBSTR (hostDriveNode, "src", name);
5396 CFGLDRReleaseNode (hostDriveNode);
5397 break;
5398 }
5399 case DriveState_NotMounted:
5400 /* do nothing, i.e.leave the DVD drive node empty */
5401 break;
5402 default:
5403 ComAssertMsgFailedBreak (("Invalid DVD drive state: %d\n",
5404 mDVDDrive->data()->mDriveState),
5405 rc = E_FAIL);
5406 }
5407
5408 CFGLDRReleaseNode (dvdNode);
5409 }
5410 while (0);
5411
5412 if (FAILED (rc))
5413 return rc;
5414
5415 /* Flooppy drive (required) */
5416 /// @todo (dmik) move the code to DVDDrive
5417 do
5418 {
5419 CFGNODE floppyNode = 0;
5420 CFGLDRCreateChildNode (aNode, "FloppyDrive", &floppyNode);
5421
5422 BOOL fFloppyEnabled;
5423 rc = mFloppyDrive->COMGETTER(Enabled)(&fFloppyEnabled);
5424 CFGLDRSetBool (floppyNode, "enabled", !!fFloppyEnabled);
5425
5426 switch (mFloppyDrive->data()->mDriveState)
5427 {
5428 case DriveState_ImageMounted:
5429 {
5430 Assert (!mFloppyDrive->data()->mFloppyImage.isNull());
5431
5432 Guid id;
5433 rc = mFloppyDrive->data()->mFloppyImage->COMGETTER(Id) (id.asOutParam());
5434 Assert (!id.isEmpty());
5435
5436 CFGNODE imageNode = 0;
5437 CFGLDRCreateChildNode (floppyNode, "Image", &imageNode);
5438 CFGLDRSetUUID (imageNode, "uuid", id.ptr());
5439 CFGLDRReleaseNode (imageNode);
5440 break;
5441 }
5442 case DriveState_HostDriveCaptured:
5443 {
5444 Assert (!mFloppyDrive->data()->mHostDrive.isNull());
5445
5446 Bstr name;
5447 rc = mFloppyDrive->data()->mHostDrive->COMGETTER(Name) (name.asOutParam());
5448 Assert (!name.isEmpty());
5449
5450 CFGNODE hostDriveNode = 0;
5451 CFGLDRCreateChildNode (floppyNode, "HostDrive", &hostDriveNode);
5452 CFGLDRSetBSTR (hostDriveNode, "src", name);
5453 CFGLDRReleaseNode (hostDriveNode);
5454 break;
5455 }
5456 case DriveState_NotMounted:
5457 /* do nothing, i.e.leave the Floppy drive node empty */
5458 break;
5459 default:
5460 ComAssertMsgFailedBreak (("Invalid Floppy drive state: %d\n",
5461 mFloppyDrive->data()->mDriveState),
5462 rc = E_FAIL);
5463 }
5464
5465 CFGLDRReleaseNode (floppyNode);
5466 }
5467 while (0);
5468
5469 if (FAILED (rc))
5470 return rc;
5471
5472
5473 /* USB Controller (required) */
5474 rc = mUSBController->saveSettings (aNode);
5475 if (FAILED (rc))
5476 return rc;
5477
5478 /* Network adapters (required) */
5479 do
5480 {
5481 CFGNODE nwNode = 0;
5482 CFGLDRCreateChildNode (aNode, "Network", &nwNode);
5483
5484 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
5485 {
5486 CFGNODE networkAdapterNode = 0;
5487 CFGLDRAppendChildNode (nwNode, "Adapter", &networkAdapterNode);
5488
5489 CFGLDRSetUInt32 (networkAdapterNode, "slot", slot);
5490 CFGLDRSetBool (networkAdapterNode, "enabled",
5491 !!mNetworkAdapters [slot]->data()->mEnabled);
5492 CFGLDRSetBSTR (networkAdapterNode, "MACAddress",
5493 mNetworkAdapters [slot]->data()->mMACAddress);
5494 CFGLDRSetBool (networkAdapterNode, "cable",
5495 !!mNetworkAdapters [slot]->data()->mCableConnected);
5496
5497 if (mNetworkAdapters [slot]->data()->mTraceEnabled)
5498 CFGLDRSetBool (networkAdapterNode, "trace", true);
5499
5500 CFGLDRSetBSTR (networkAdapterNode, "tracefile",
5501 mNetworkAdapters [slot]->data()->mTraceFile);
5502
5503 switch (mNetworkAdapters [slot]->data()->mAdapterType)
5504 {
5505 case NetworkAdapterType_NetworkAdapterAm79C970A:
5506 CFGLDRSetString (networkAdapterNode, "type", "Am79C970A");
5507 break;
5508 case NetworkAdapterType_NetworkAdapterAm79C973:
5509 CFGLDRSetString (networkAdapterNode, "type", "Am79C973");
5510 break;
5511 default:
5512 ComAssertMsgFailedBreak (("Invalid network adapter type: %d\n",
5513 mNetworkAdapters [slot]->data()->mAdapterType),
5514 rc = E_FAIL);
5515 }
5516
5517 CFGNODE attachmentNode = 0;
5518 switch (mNetworkAdapters [slot]->data()->mAttachmentType)
5519 {
5520 case NetworkAttachmentType_NoNetworkAttachment:
5521 {
5522 /* do nothing -- empty content */
5523 break;
5524 }
5525 case NetworkAttachmentType_NATNetworkAttachment:
5526 {
5527 CFGLDRAppendChildNode (networkAdapterNode, "NAT", &attachmentNode);
5528 break;
5529 }
5530 case NetworkAttachmentType_HostInterfaceNetworkAttachment:
5531 {
5532 CFGLDRAppendChildNode (networkAdapterNode, "HostInterface", &attachmentNode);
5533 const Bstr &name = mNetworkAdapters [slot]->data()->mHostInterface;
5534#ifdef __WIN__
5535 Assert (!name.isNull());
5536#endif
5537#ifdef VBOX_WITH_UNIXY_TAP_NETWORKING
5538 if (!name.isEmpty())
5539#endif
5540 CFGLDRSetBSTR (attachmentNode, "name", name);
5541#ifdef VBOX_WITH_UNIXY_TAP_NETWORKING
5542 const Bstr &tapSetupApp =
5543 mNetworkAdapters [slot]->data()->mTAPSetupApplication;
5544 if (!tapSetupApp.isEmpty())
5545 CFGLDRSetBSTR (attachmentNode, "TAPSetup", tapSetupApp);
5546 const Bstr &tapTerminateApp =
5547 mNetworkAdapters [slot]->data()->mTAPTerminateApplication;
5548 if (!tapTerminateApp.isEmpty())
5549 CFGLDRSetBSTR (attachmentNode, "TAPTerminate", tapTerminateApp);
5550#endif /* VBOX_WITH_UNIXY_TAP_NETWORKING */
5551 break;
5552 }
5553 case NetworkAttachmentType_InternalNetworkAttachment:
5554 {
5555 CFGLDRAppendChildNode (networkAdapterNode, "InternalNetwork", &attachmentNode);
5556 const Bstr &name = mNetworkAdapters[slot]->data()->mInternalNetwork;
5557 Assert(!name.isNull());
5558 CFGLDRSetBSTR (attachmentNode, "name", name);
5559 break;
5560 }
5561 default:
5562 {
5563 ComAssertFailedBreak (rc = E_FAIL);
5564 break;
5565 }
5566 }
5567 if (attachmentNode)
5568 CFGLDRReleaseNode (attachmentNode);
5569
5570 CFGLDRReleaseNode (networkAdapterNode);
5571 }
5572
5573 CFGLDRReleaseNode (nwNode);
5574 }
5575 while (0);
5576
5577 if (FAILED (rc))
5578 return rc;
5579
5580 /* Audio adapter */
5581 do
5582 {
5583 CFGNODE adapterNode = 0;
5584 CFGLDRCreateChildNode (aNode, "AudioAdapter", &adapterNode);
5585
5586 switch (mAudioAdapter->data()->mAudioDriver)
5587 {
5588 case AudioDriverType_NullAudioDriver:
5589 {
5590 CFGLDRSetString (adapterNode, "driver", "null");
5591 break;
5592 }
5593#ifdef __WIN__
5594 case AudioDriverType_WINMMAudioDriver:
5595 {
5596 CFGLDRSetString (adapterNode, "driver", "winmm");
5597 break;
5598 }
5599 case AudioDriverType_DSOUNDAudioDriver:
5600 {
5601 CFGLDRSetString (adapterNode, "driver", "dsound");
5602 break;
5603 }
5604#endif /* __WIN__ */
5605#ifdef VBOX_WITH_ALSA
5606 case AudioDriverType_ALSAAudioDriver:
5607 {
5608 CFGLDRSetString (adapterNode, "driver", "alsa");
5609 break;
5610 }
5611#else
5612 /* fall back to OSS */
5613 case AudioDriverType_ALSAAudioDriver:
5614#endif
5615#ifdef __LINUX__
5616 case AudioDriverType_OSSAudioDriver:
5617 {
5618 CFGLDRSetString (adapterNode, "driver", "oss");
5619 break;
5620 }
5621#endif /* __LINUX__ */
5622 default:
5623 ComAssertMsgFailedBreak (("Wrong audio driver type! driver = %d\n",
5624 mAudioAdapter->data()->mAudioDriver),
5625 rc = E_FAIL);
5626 }
5627
5628 CFGLDRSetBool (adapterNode, "enabled", !!mAudioAdapter->data()->mEnabled);
5629
5630 CFGLDRReleaseNode (adapterNode);
5631 }
5632 while (0);
5633
5634 if (FAILED (rc))
5635 return rc;
5636
5637 /* Shared folders */
5638 do
5639 {
5640 CFGNODE sharedFoldersNode = 0;
5641 CFGLDRCreateChildNode (aNode, "SharedFolders", &sharedFoldersNode);
5642
5643 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
5644 it != mHWData->mSharedFolders.end();
5645 ++ it)
5646 {
5647 ComObjPtr <SharedFolder> folder = *it;
5648
5649 CFGNODE folderNode = 0;
5650 CFGLDRAppendChildNode (sharedFoldersNode, "SharedFolder", &folderNode);
5651
5652 /* all are mandatory */
5653 CFGLDRSetBSTR (folderNode, "name", folder->name());
5654 CFGLDRSetBSTR (folderNode, "hostPath", folder->hostPath());
5655
5656 CFGLDRReleaseNode (folderNode);
5657 }
5658
5659 CFGLDRReleaseNode (sharedFoldersNode);
5660 }
5661 while (0);
5662
5663 /* Clipboard */
5664 {
5665 CFGNODE clipNode = 0;
5666 CFGLDRCreateChildNode (aNode, "Clipboard", &clipNode);
5667
5668 char *mode = "Disabled";
5669 switch (mHWData->mClipboardMode)
5670 {
5671 case ClipboardMode_ClipDisabled:
5672 /* already assigned */
5673 break;
5674 case ClipboardMode_ClipHostToGuest:
5675 mode = "HostToGuest";
5676 break;
5677 case ClipboardMode_ClipGuestToHost:
5678 mode = "GuestToHost";
5679 break;
5680 case ClipboardMode_ClipBidirectional:
5681 mode = "Bidirectional";
5682 break;
5683 default:
5684 AssertMsgFailed (("Clipboard mode %d is invalid",
5685 mHWData->mClipboardMode));
5686 break;
5687 }
5688 CFGLDRSetString (clipNode, "mode", mode);
5689
5690 CFGLDRReleaseNode (clipNode);
5691 }
5692
5693 return rc;
5694}
5695
5696/**
5697 * Saves the hard disk confguration.
5698 * It is assumed that the given node is empty.
5699 *
5700 * @param aNode <HardDiskAttachments> node to save the hard disk confguration to
5701 */
5702HRESULT Machine::saveHardDisks (CFGNODE aNode)
5703{
5704 AssertReturn (aNode, E_INVALIDARG);
5705
5706 HRESULT rc = S_OK;
5707
5708 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
5709 it != mHDData->mHDAttachments.end() && SUCCEEDED (rc);
5710 ++ it)
5711 {
5712 ComObjPtr <HardDiskAttachment> att = *it;
5713
5714 CFGNODE hdNode = 0;
5715 CFGLDRAppendChildNode (aNode, "HardDiskAttachment", &hdNode);
5716
5717 do
5718 {
5719 const char *bus = NULL;
5720 switch (att->controller())
5721 {
5722 case DiskControllerType_IDE0Controller: bus = "ide0"; break;
5723 case DiskControllerType_IDE1Controller: bus = "ide1"; break;
5724 default:
5725 ComAssertFailedBreak (rc = E_FAIL);
5726 }
5727 if (FAILED (rc))
5728 break;
5729
5730 const char *dev = NULL;
5731 switch (att->deviceNumber())
5732 {
5733 case 0: dev = "master"; break;
5734 case 1: dev = "slave"; break;
5735 default:
5736 ComAssertFailedBreak (rc = E_FAIL);
5737 }
5738 if (FAILED (rc))
5739 break;
5740
5741 CFGLDRSetUUID (hdNode, "hardDisk", att->hardDisk()->id());
5742 CFGLDRSetString (hdNode, "bus", bus);
5743 CFGLDRSetString (hdNode, "device", dev);
5744 }
5745 while (0);
5746
5747 CFGLDRReleaseNode (hdNode);
5748 }
5749
5750 return rc;
5751}
5752
5753/**
5754 * Saves machine state settings as defined by aFlags
5755 * (SaveSTS_* values).
5756 *
5757 * @param aFlags a combination of SaveSTS_* flags
5758 *
5759 * @note Locks objects!
5760 */
5761HRESULT Machine::saveStateSettings (int aFlags)
5762{
5763 if (aFlags == 0)
5764 return S_OK;
5765
5766 AutoCaller autoCaller (this);
5767 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
5768
5769 AutoLock alock (this);
5770
5771 /* load the config file */
5772 CFGHANDLE configLoader = 0;
5773 HRESULT rc = openConfigLoader (&configLoader);
5774 if (FAILED (rc))
5775 return rc;
5776
5777 CFGNODE machineNode = 0;
5778 CFGLDRGetNode (configLoader, "VirtualBox/Machine", 0, &machineNode);
5779
5780 do
5781 {
5782 ComAssertBreak (machineNode, rc = E_FAIL);
5783
5784 if (aFlags & SaveSTS_CurStateModified)
5785 {
5786 if (!mData->mCurrentStateModified)
5787 CFGLDRSetBool (machineNode, "currentStateModified", false);
5788 else
5789 CFGLDRDeleteAttribute (machineNode, "currentStateModified");
5790 }
5791
5792 if (aFlags & SaveSTS_StateFilePath)
5793 {
5794 if (mSSData->mStateFilePath)
5795 CFGLDRSetBSTR (machineNode, "stateFile", mSSData->mStateFilePath);
5796 else
5797 CFGLDRDeleteAttribute (machineNode, "stateFile");
5798 }
5799
5800 if (aFlags & SaveSTS_StateTimeStamp)
5801 {
5802 Assert (mData->mMachineState != MachineState_Aborted ||
5803 mSSData->mStateFilePath.isNull());
5804
5805 CFGLDRSetDateTime (machineNode, "lastStateChange",
5806 mData->mLastStateChange);
5807
5808 // set the aborted attribute when appropriate
5809 if (mData->mMachineState == MachineState_Aborted)
5810 CFGLDRSetBool (machineNode, "aborted", true);
5811 else
5812 CFGLDRDeleteAttribute (machineNode, "aborted");
5813 }
5814 }
5815 while (0);
5816
5817 if (machineNode)
5818 CFGLDRReleaseNode (machineNode);
5819
5820 if (SUCCEEDED (rc))
5821 rc = closeConfigLoader (configLoader, true /* aSaveBeforeClose */);
5822 else
5823 closeConfigLoader (configLoader, false /* aSaveBeforeClose */);
5824
5825 return rc;
5826}
5827
5828/**
5829 * Cleans up all differencing hard disks based on immutable hard disks.
5830 *
5831 * @note Locks objects!
5832 */
5833HRESULT Machine::wipeOutImmutableDiffs()
5834{
5835 AutoCaller autoCaller (this);
5836 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
5837
5838 AutoReaderLock alock (this);
5839
5840 AssertReturn (mData->mMachineState == MachineState_PoweredOff ||
5841 mData->mMachineState == MachineState_Aborted, E_FAIL);
5842
5843 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
5844 it != mHDData->mHDAttachments.end();
5845 ++ it)
5846 {
5847 ComObjPtr <HardDisk> hd = (*it)->hardDisk();
5848 AutoLock hdLock (hd);
5849
5850 if(hd->isParentImmutable())
5851 {
5852 /// @todo (dmik) no error handling for now
5853 // (need async error reporting for this)
5854 hd->asVDI()->wipeOutImage();
5855 }
5856 }
5857
5858 return S_OK;
5859}
5860
5861/**
5862 * Fixes up lazy hard disk attachments by creating or deleting differencing
5863 * hard disks when machine settings are being committed.
5864 * Must be called only from #commit().
5865 *
5866 * @note Locks objects!
5867 */
5868HRESULT Machine::fixupHardDisks (bool aCommit)
5869{
5870 AutoCaller autoCaller (this);
5871 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
5872
5873 AutoLock alock (this);
5874
5875 /* no attac/detach operations -- nothing to do */
5876 if (!mHDData.isBackedUp())
5877 {
5878 mHDData->mHDAttachmentsChanged = false;
5879 return S_OK;
5880 }
5881
5882 AssertReturn (mData->mRegistered, E_FAIL);
5883
5884 if (aCommit)
5885 {
5886 /*
5887 * changes are being committed,
5888 * perform actual diff image creation, deletion etc.
5889 */
5890
5891 /* take a copy of backed up attachments (will modify it) */
5892 HDData::HDAttachmentList backedUp = mHDData.backedUpData()->mHDAttachments;
5893 /* list of new diffs created */
5894 std::list <ComObjPtr <HardDisk> > newDiffs;
5895
5896 HRESULT rc = S_OK;
5897
5898 /* go through current attachments */
5899 for (HDData::HDAttachmentList::const_iterator
5900 it = mHDData->mHDAttachments.begin();
5901 it != mHDData->mHDAttachments.end();
5902 ++ it)
5903 {
5904 ComObjPtr <HardDiskAttachment> hda = *it;
5905 ComObjPtr <HardDisk> hd = hda->hardDisk();
5906 AutoLock hdLock (hd);
5907
5908 if (!hda->isDirty())
5909 {
5910 /*
5911 * not dirty, therefore was either attached before backing up
5912 * or doesn't need any fixup (already fixed up); try to locate
5913 * this hard disk among backed up attachments and remove from
5914 * there to prevent it from being deassociated/deleted
5915 */
5916 HDData::HDAttachmentList::iterator oldIt;
5917 for (oldIt = backedUp.begin(); oldIt != backedUp.end(); ++ oldIt)
5918 if ((*oldIt)->hardDisk().equalsTo (hd))
5919 break;
5920 if (oldIt != backedUp.end())
5921 {
5922 /* remove from there */
5923 backedUp.erase (oldIt);
5924 // LogTraceMsg (("FC: %ls found in old\n", hd->toString().raw()));
5925 }
5926 }
5927 else
5928 {
5929 /* dirty, determine what to do */
5930
5931 bool needDiff = false;
5932 bool searchAmongSnapshots = false;
5933
5934 switch (hd->type())
5935 {
5936 case HardDiskType_ImmutableHardDisk:
5937 {
5938 /* decrease readers increased in AttachHardDisk() */
5939 hd->releaseReader();
5940 // LogTraceMsg (("FC: %ls released\n", hd->toString().raw()));
5941 /* indicate we need a diff (indirect attachment) */
5942 needDiff = true;
5943 break;
5944 }
5945 case HardDiskType_WritethroughHardDisk:
5946 {
5947 /* reset the dirty flag */
5948 hda->updateHardDisk (hd, false /* aDirty */);
5949 // LogTraceMsg (("FC: %ls updated\n", hd->toString().raw()));
5950 break;
5951 }
5952 case HardDiskType_NormalHardDisk:
5953 {
5954 if (hd->snapshotId().isEmpty())
5955 {
5956 /* reset the dirty flag */
5957 hda->updateHardDisk (hd, false /* aDirty */);
5958 // LogTraceMsg (("FC: %ls updated\n", hd->toString().raw()));
5959 }
5960 else
5961 {
5962 /* decrease readers increased in AttachHardDisk() */
5963 hd->releaseReader();
5964 // LogTraceMsg (("FC: %ls released\n", hd->toString().raw()));
5965 /* indicate we need a diff (indirect attachment) */
5966 needDiff = true;
5967 /* search for the most recent base among snapshots */
5968 searchAmongSnapshots = true;
5969 }
5970 break;
5971 }
5972 }
5973
5974 if (!needDiff)
5975 continue;
5976
5977 bool createDiff = false;
5978
5979 /*
5980 * see whether any previously attached hard disk has the
5981 * the currently attached one (Normal or Independent) as
5982 * the root
5983 */
5984
5985 HDData::HDAttachmentList::iterator foundIt = backedUp.end();
5986
5987 for (HDData::HDAttachmentList::iterator it = backedUp.begin();
5988 it != backedUp.end();
5989 ++ it)
5990 {
5991 if ((*it)->hardDisk()->root().equalsTo (hd))
5992 {
5993 /*
5994 * matched dev and ctl (i.e. attached to the same place)
5995 * will win and immediately stop the search; otherwise
5996 * the first attachment that matched the hd only will
5997 * be used
5998 */
5999 if ((*it)->deviceNumber() == hda->deviceNumber() &&
6000 (*it)->controller() == hda->controller())
6001 {
6002 foundIt = it;
6003 break;
6004 }
6005 else
6006 if (foundIt == backedUp.end())
6007 {
6008 /*
6009 * not an exact match; ensure there is no exact match
6010 * among other current attachments referring the same
6011 * root (to prevent this attachmend from reusing the
6012 * hard disk of the other attachment that will later
6013 * give the exact match or already gave it before)
6014 */
6015 bool canReuse = true;
6016 for (HDData::HDAttachmentList::const_iterator
6017 it2 = mHDData->mHDAttachments.begin();
6018 it2 != mHDData->mHDAttachments.end();
6019 ++ it2)
6020 {
6021 if ((*it2)->deviceNumber() == (*it)->deviceNumber() &&
6022 (*it2)->controller() == (*it)->controller() &&
6023 (*it2)->hardDisk()->root().equalsTo (hd))
6024 {
6025 /*
6026 * the exact match, either non-dirty or dirty
6027 * one refers the same root: in both cases
6028 * we cannot reuse the hard disk, so break
6029 */
6030 canReuse = false;
6031 break;
6032 }
6033 }
6034
6035 if (canReuse)
6036 foundIt = it;
6037 }
6038 }
6039 }
6040
6041 if (foundIt != backedUp.end())
6042 {
6043 /* found either one or another, reuse the diff */
6044 hda->updateHardDisk ((*foundIt)->hardDisk(),
6045 false /* aDirty */);
6046 // LogTraceMsg (("FC: %ls reused as %ls\n", hd->toString().raw(),
6047 // (*foundIt)->hardDisk()->toString().raw()));
6048 /* remove from there */
6049 backedUp.erase (foundIt);
6050 }
6051 else
6052 {
6053 /* was not attached, need a diff */
6054 createDiff = true;
6055 }
6056
6057 if (!createDiff)
6058 continue;
6059
6060 ComObjPtr <HardDisk> baseHd = hd;
6061
6062 if (searchAmongSnapshots)
6063 {
6064 /*
6065 * find the most recent diff based on the currently
6066 * attached root (Normal hard disk) among snapshots
6067 */
6068
6069 ComObjPtr <Snapshot> snap = mData->mCurrentSnapshot;
6070
6071 while (snap)
6072 {
6073 AutoLock snapLock (snap);
6074
6075 const HDData::HDAttachmentList &snapAtts =
6076 snap->data().mMachine->hdData()->mHDAttachments;
6077
6078 HDData::HDAttachmentList::const_iterator foundIt = snapAtts.end();
6079
6080 for (HDData::HDAttachmentList::const_iterator
6081 it = snapAtts.begin(); it != snapAtts.end(); ++ it)
6082 {
6083 if ((*it)->hardDisk()->root().equalsTo (hd))
6084 {
6085 /*
6086 * matched dev and ctl (i.e. attached to the same place)
6087 * will win and immediately stop the search; otherwise
6088 * the first attachment that matched the hd only will
6089 * be used
6090 */
6091 if ((*it)->deviceNumber() == hda->deviceNumber() &&
6092 (*it)->controller() == hda->controller())
6093 {
6094 foundIt = it;
6095 break;
6096 }
6097 else
6098 if (foundIt == snapAtts.end())
6099 foundIt = it;
6100 }
6101 }
6102
6103 if (foundIt != snapAtts.end())
6104 {
6105 /* the most recent diff has been found, use as a base */
6106 baseHd = (*foundIt)->hardDisk();
6107 // LogTraceMsg (("FC: %ls: recent found %ls\n",
6108 // hd->toString().raw(), baseHd->toString().raw()));
6109 break;
6110 }
6111
6112 snap = snap->parent();
6113 }
6114 }
6115
6116 /* create a new diff for the hard disk being indirectly attached */
6117
6118 AutoLock baseHdLock (baseHd);
6119 baseHd->addReader();
6120
6121 ComObjPtr <HVirtualDiskImage> vdi;
6122 rc = baseHd->createDiffHardDisk (mUserData->mSnapshotFolderFull,
6123 mData->mUuid, vdi, NULL);
6124 baseHd->releaseReader();
6125 CheckComRCBreakRC (rc);
6126
6127 newDiffs.push_back (ComObjPtr <HardDisk> (vdi));
6128
6129 /* update the attachment and reset the dirty flag */
6130 hda->updateHardDisk (ComObjPtr <HardDisk> (vdi),
6131 false /* aDirty */);
6132 // LogTraceMsg (("FC: %ls: diff created %ls\n",
6133 // baseHd->toString().raw(), vdi->toString().raw()));
6134 }
6135 }
6136
6137 if (FAILED (rc))
6138 {
6139 /* delete diffs we created */
6140 for (std::list <ComObjPtr <HardDisk> >::const_iterator
6141 it = newDiffs.begin(); it != newDiffs.end(); ++ it)
6142 {
6143 /*
6144 * unregisterDiffHardDisk() is supposed to delete and uninit
6145 * the differencing hard disk
6146 */
6147 mParent->unregisterDiffHardDisk (*it);
6148 /* too bad if we fail here, but nothing to do, just continue */
6149 }
6150
6151 /* the best is to rollback the changes... */
6152 mHDData.rollback();
6153 mHDData->mHDAttachmentsChanged = false;
6154 // LogTraceMsg (("FC: ROLLED BACK\n"));
6155 return rc;
6156 }
6157
6158 /*
6159 * go through the rest of old attachments and delete diffs
6160 * or deassociate hard disks from machines (they will become detached)
6161 */
6162 for (HDData::HDAttachmentList::iterator
6163 it = backedUp.begin(); it != backedUp.end(); ++ it)
6164 {
6165 ComObjPtr <HardDiskAttachment> hda = *it;
6166 ComObjPtr <HardDisk> hd = hda->hardDisk();
6167 AutoLock hdLock (hd);
6168
6169 if (hd->isDifferencing())
6170 {
6171 /*
6172 * unregisterDiffHardDisk() is supposed to delete and uninit
6173 * the differencing hard disk
6174 */
6175 // LogTraceMsg (("FC: %ls diff deleted\n", hd->toString().raw()));
6176 rc = mParent->unregisterDiffHardDisk (hd);
6177 /*
6178 * too bad if we fail here, but nothing to do, just continue
6179 * (the last rc will be returned to the caller though)
6180 */
6181 }
6182 else
6183 {
6184 /* deassociate from this machine */
6185 // LogTraceMsg (("FC: %ls deassociated\n", hd->toString().raw()));
6186 hd->setMachineId (Guid());
6187 }
6188 }
6189
6190 /* commit all the changes */
6191 mHDData->mHDAttachmentsChanged = mHDData.hasActualChanges();
6192 mHDData.commit();
6193 // LogTraceMsg (("FC: COMMITTED\n"));
6194
6195 return rc;
6196 }
6197
6198 /*
6199 * changes are being rolled back,
6200 * go trhough all current attachments and fix up dirty ones
6201 * the way it is done in DetachHardDisk()
6202 */
6203
6204 for (HDData::HDAttachmentList::iterator it = mHDData->mHDAttachments.begin();
6205 it != mHDData->mHDAttachments.end();
6206 ++ it)
6207 {
6208 ComObjPtr <HardDiskAttachment> hda = *it;
6209 ComObjPtr <HardDisk> hd = hda->hardDisk();
6210 AutoLock hdLock (hd);
6211
6212 if (hda->isDirty())
6213 {
6214 switch (hd->type())
6215 {
6216 case HardDiskType_ImmutableHardDisk:
6217 {
6218 /* decrease readers increased in AttachHardDisk() */
6219 hd->releaseReader();
6220 // LogTraceMsg (("FR: %ls released\n", hd->toString().raw()));
6221 break;
6222 }
6223 case HardDiskType_WritethroughHardDisk:
6224 {
6225 /* deassociate from this machine */
6226 hd->setMachineId (Guid());
6227 // LogTraceMsg (("FR: %ls deassociated\n", hd->toString().raw()));
6228 break;
6229 }
6230 case HardDiskType_NormalHardDisk:
6231 {
6232 if (hd->snapshotId().isEmpty())
6233 {
6234 /* deassociate from this machine */
6235 hd->setMachineId (Guid());
6236 // LogTraceMsg (("FR: %ls deassociated\n", hd->toString().raw()));
6237 }
6238 else
6239 {
6240 /* decrease readers increased in AttachHardDisk() */
6241 hd->releaseReader();
6242 // LogTraceMsg (("FR: %ls released\n", hd->toString().raw()));
6243 }
6244
6245 break;
6246 }
6247 }
6248 }
6249 }
6250
6251 /* rollback all the changes */
6252 mHDData.rollback();
6253 // LogTraceMsg (("FR: ROLLED BACK\n"));
6254
6255 return S_OK;
6256}
6257
6258/**
6259 * Creates differencing hard disks for all normal hard disks
6260 * and replaces attachments to refer to created disks.
6261 * Used when taking a snapshot or when discarding the current state.
6262 *
6263 * @param aSnapshotId ID of the snapshot being taken
6264 * or NULL if the current state is being discarded
6265 * @param aFolder folder where to create diff. hard disks
6266 * @param aProgress progress object to run (must contain at least as
6267 * many operations left as the number of VDIs attached)
6268 * @param aOnline whether the machine is online (i.e., when the EMT
6269 * thread is paused, OR when current hard disks are
6270 * marked as busy for some other reason)
6271 *
6272 * @note
6273 * The progress object is not marked as completed, neither on success
6274 * nor on failure. This is a responsibility of the caller.
6275 *
6276 * @note Locks mParent + this object for writing
6277 */
6278HRESULT Machine::createSnapshotDiffs (const Guid *aSnapshotId,
6279 const Bstr &aFolder,
6280 const ComObjPtr <Progress> &aProgress,
6281 bool aOnline)
6282{
6283 AssertReturn (!aFolder.isEmpty(), E_FAIL);
6284
6285 AutoCaller autoCaller (this);
6286 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
6287
6288 /* accessing mParent methods below needs mParent lock */
6289 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
6290
6291 HRESULT rc = S_OK;
6292
6293 // first pass: check accessibility before performing changes
6294 if (!aOnline)
6295 {
6296 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
6297 it != mHDData->mHDAttachments.end();
6298 ++ it)
6299 {
6300 ComObjPtr <HardDiskAttachment> hda = *it;
6301 ComObjPtr <HardDisk> hd = hda->hardDisk();
6302 AutoLock hdLock (hd);
6303
6304 ComAssertMsgBreak (hd->type() == HardDiskType_NormalHardDisk,
6305 ("Invalid hard disk type %d\n", hd->type()),
6306 rc = E_FAIL);
6307
6308 ComAssertMsgBreak (!hd->isParentImmutable() ||
6309 hd->storageType() == HardDiskStorageType_VirtualDiskImage,
6310 ("Invalid hard disk storage type %d\n", hd->storageType()),
6311 rc = E_FAIL);
6312
6313 Bstr accessError;
6314 rc = hd->getAccessible (accessError);
6315 CheckComRCBreakRC (rc);
6316
6317 if (!accessError.isNull())
6318 {
6319 rc = setError (E_FAIL,
6320 tr ("Hard disk '%ls' is not accessible (%ls)"),
6321 hd->toString().raw(), accessError.raw());
6322 break;
6323 }
6324 }
6325 CheckComRCReturnRC (rc);
6326 }
6327
6328 HDData::HDAttachmentList attachments;
6329
6330 // second pass: perform changes
6331 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
6332 it != mHDData->mHDAttachments.end();
6333 ++ it)
6334 {
6335 ComObjPtr <HardDiskAttachment> hda = *it;
6336 ComObjPtr <HardDisk> hd = hda->hardDisk();
6337 AutoLock hdLock (hd);
6338
6339 ComObjPtr <HardDisk> parent = hd->parent();
6340 AutoLock parentHdLock (parent);
6341
6342 ComObjPtr <HardDisk> newHd;
6343
6344 // clear busy flag if the VM is online
6345 if (aOnline)
6346 hd->clearBusy();
6347 // increase readers
6348 hd->addReader();
6349
6350 if (hd->isParentImmutable())
6351 {
6352 aProgress->advanceOperation (Bstr (Utf8StrFmt (
6353 tr ("Preserving immutable hard disk '%ls'"),
6354 parent->toString (true /* aShort */).raw())));
6355
6356 parentHdLock.unlock();
6357 alock.leave();
6358
6359 // create a copy of the independent diff
6360 ComObjPtr <HVirtualDiskImage> vdi;
6361 rc = hd->asVDI()->cloneDiffImage (aFolder, mData->mUuid, vdi,
6362 aProgress);
6363 newHd = vdi;
6364
6365 alock.enter();
6366 parentHdLock.lock();
6367
6368 // decrease readers (hd is no more used for reading in any case)
6369 hd->releaseReader();
6370 }
6371 else
6372 {
6373 // checked in the first pass
6374 Assert (hd->type() == HardDiskType_NormalHardDisk);
6375
6376 aProgress->advanceOperation (Bstr (Utf8StrFmt (
6377 tr ("Creating a differencing hard disk for '%ls'"),
6378 hd->root()->toString (true /* aShort */).raw())));
6379
6380 parentHdLock.unlock();
6381 alock.leave();
6382
6383 // create a new diff for the image being attached
6384 ComObjPtr <HVirtualDiskImage> vdi;
6385 rc = hd->createDiffHardDisk (aFolder, mData->mUuid, vdi, aProgress);
6386 newHd = vdi;
6387
6388 alock.enter();
6389 parentHdLock.lock();
6390
6391 if (SUCCEEDED (rc))
6392 {
6393 // if online, hd must keep a reader referece
6394 if (!aOnline)
6395 hd->releaseReader();
6396 }
6397 else
6398 {
6399 // decrease readers
6400 hd->releaseReader();
6401 }
6402 }
6403
6404 if (SUCCEEDED (rc))
6405 {
6406 ComObjPtr <HardDiskAttachment> newHda;
6407 newHda.createObject();
6408 rc = newHda->init (newHd, hda->controller(), hda->deviceNumber(),
6409 false /* aDirty */);
6410
6411 if (SUCCEEDED (rc))
6412 {
6413 // associate the snapshot id with the old hard disk
6414 if (hd->type() != HardDiskType_WritethroughHardDisk && aSnapshotId)
6415 hd->setSnapshotId (*aSnapshotId);
6416
6417 // add the new attachment
6418 attachments.push_back (newHda);
6419
6420 // if online, newHd must be marked as busy
6421 if (aOnline)
6422 newHd->setBusy();
6423 }
6424 }
6425
6426 if (FAILED (rc))
6427 {
6428 // set busy flag back if the VM is online
6429 if (aOnline)
6430 hd->setBusy();
6431 break;
6432 }
6433 }
6434
6435 if (SUCCEEDED (rc))
6436 {
6437 // replace the whole list of attachments with the new one
6438 mHDData->mHDAttachments = attachments;
6439 }
6440 else
6441 {
6442 // delete those diffs we've just created
6443 for (HDData::HDAttachmentList::const_iterator it = attachments.begin();
6444 it != attachments.end();
6445 ++ it)
6446 {
6447 ComObjPtr <HardDisk> hd = (*it)->hardDisk();
6448 AutoLock hdLock (hd);
6449 Assert (hd->children().size() == 0);
6450 Assert (hd->isDifferencing());
6451 // unregisterDiffHardDisk() is supposed to delete and uninit
6452 // the differencing hard disk
6453 mParent->unregisterDiffHardDisk (hd);
6454 }
6455 }
6456
6457 return rc;
6458}
6459
6460/**
6461 * Deletes differencing hard disks created by createSnapshotDiffs() in case
6462 * if snapshot creation was failed.
6463 *
6464 * @param aSnapshot failed snapshot
6465 *
6466 * @note Locks mParent + this object for writing.
6467 */
6468HRESULT Machine::deleteSnapshotDiffs (const ComObjPtr <Snapshot> &aSnapshot)
6469{
6470 AssertReturn (!aSnapshot.isNull(), E_FAIL);
6471
6472 AutoCaller autoCaller (this);
6473 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
6474
6475 /* accessing mParent methods below needs mParent lock */
6476 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
6477
6478 /* short cut: check whether attachments are all the same */
6479 if (mHDData->mHDAttachments == aSnapshot->data().mMachine->mHDData->mHDAttachments)
6480 return S_OK;
6481
6482 HRESULT rc = S_OK;
6483
6484 for (HDData::HDAttachmentList::const_iterator it = mHDData->mHDAttachments.begin();
6485 it != mHDData->mHDAttachments.end();
6486 ++ it)
6487 {
6488 ComObjPtr <HardDiskAttachment> hda = *it;
6489 ComObjPtr <HardDisk> hd = hda->hardDisk();
6490 AutoLock hdLock (hd);
6491
6492 ComObjPtr <HardDisk> parent = hd->parent();
6493 AutoLock parentHdLock (parent);
6494
6495 if (!parent || parent->snapshotId() != aSnapshot->data().mId)
6496 continue;
6497
6498 /* must not have children */
6499 ComAssertRet (hd->children().size() == 0, E_FAIL);
6500
6501 /* deassociate the old hard disk from the given snapshot's ID */
6502 parent->setSnapshotId (Guid());
6503
6504 /* unregisterDiffHardDisk() is supposed to delete and uninit
6505 * the differencing hard disk */
6506 rc = mParent->unregisterDiffHardDisk (hd);
6507 /* continue on error */
6508 }
6509
6510 /* restore the whole list of attachments from the failed snapshot */
6511 mHDData->mHDAttachments = aSnapshot->data().mMachine->mHDData->mHDAttachments;
6512
6513 return rc;
6514}
6515
6516/**
6517 * Helper to lock the machine configuration for write access.
6518 *
6519 * @return S_OK or E_FAIL and sets error info on failure
6520 *
6521 * @note Doesn't lock anything (must be called from this object's lock)
6522 */
6523HRESULT Machine::lockConfig()
6524{
6525 HRESULT rc = S_OK;
6526
6527 if (!isConfigLocked())
6528 {
6529 /* open the associated config file */
6530 int vrc = RTFileOpen (&mData->mHandleCfgFile,
6531 Utf8Str (mData->mConfigFileFull),
6532 RTFILE_O_READWRITE | RTFILE_O_OPEN |
6533 RTFILE_O_DENY_WRITE);
6534 if (VBOX_FAILURE (vrc))
6535 mData->mHandleCfgFile = NIL_RTFILE;
6536 }
6537
6538 LogFlowThisFunc (("mConfigFile={%ls}, mHandleCfgFile=%d, rc=%08X\n",
6539 mData->mConfigFileFull.raw(), mData->mHandleCfgFile, rc));
6540 return rc;
6541}
6542
6543/**
6544 * Helper to unlock the machine configuration from write access
6545 *
6546 * @return S_OK
6547 *
6548 * @note Doesn't lock anything.
6549 * @note Not thread safe (must be called from this object's lock).
6550 */
6551HRESULT Machine::unlockConfig()
6552{
6553 HRESULT rc = S_OK;
6554
6555 if (isConfigLocked())
6556 {
6557 RTFileClose(mData->mHandleCfgFile);
6558 mData->mHandleCfgFile = NIL_RTFILE;
6559 }
6560
6561 LogFlowThisFunc (("\n"));
6562
6563 return rc;
6564}
6565
6566/**
6567 * Returns true if the settings file is located in the directory named exactly
6568 * as the machine. This will be true if the machine settings structure was
6569 * created by default in #openConfigLoader().
6570 *
6571 * @param aSettingsDir if not NULL, the full machine settings file directory
6572 * name will be assigned there.
6573 *
6574 * @note Doesn't lock anything.
6575 * @note Not thread safe (must be called from this object's lock).
6576 */
6577bool Machine::isInOwnDir (Utf8Str *aSettingsDir /* = NULL */)
6578{
6579 Utf8Str settingsDir = mData->mConfigFileFull;
6580 RTPathStripFilename (settingsDir.mutableRaw());
6581 char *dirName = RTPathFilename (settingsDir);
6582
6583 AssertReturn (dirName, false);
6584
6585 /* if we don't rename anything on name change, return false shorlty */
6586 if (!mUserData->mNameSync)
6587 return false;
6588
6589 if (aSettingsDir)
6590 *aSettingsDir = settingsDir;
6591
6592 return Bstr (dirName) == mUserData->mName;
6593}
6594
6595/**
6596 * @note Locks objects for reading!
6597 */
6598bool Machine::isModified()
6599{
6600 AutoCaller autoCaller (this);
6601 AssertComRCReturn (autoCaller.rc(), false);
6602
6603 AutoReaderLock alock (this);
6604
6605 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
6606 if (mNetworkAdapters [slot] && mNetworkAdapters [slot]->isModified())
6607 return true;
6608
6609 return
6610 mUserData.isBackedUp() ||
6611 mHWData.isBackedUp() ||
6612 mHDData.isBackedUp() ||
6613#ifdef VBOX_VRDP
6614 (mVRDPServer && mVRDPServer->isModified()) ||
6615#endif
6616 (mDVDDrive && mDVDDrive->isModified()) ||
6617 (mFloppyDrive && mFloppyDrive->isModified()) ||
6618 (mAudioAdapter && mAudioAdapter->isModified()) ||
6619 (mUSBController && mUSBController->isModified()) ||
6620 (mBIOSSettings && mBIOSSettings->isModified());
6621}
6622
6623/**
6624 * @note This method doesn't check (ignores) actual changes to mHDData.
6625 * Use mHDData.mHDAttachmentsChanged right after #commit() instead.
6626 *
6627 * @param aIgnoreUserData |true| to ignore changes to mUserData
6628 *
6629 * @note Locks objects for reading!
6630 */
6631bool Machine::isReallyModified (bool aIgnoreUserData /* = false */)
6632{
6633 AutoCaller autoCaller (this);
6634 AssertComRCReturn (autoCaller.rc(), false);
6635
6636 AutoReaderLock alock (this);
6637
6638 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
6639 if (mNetworkAdapters [slot] && mNetworkAdapters [slot]->isReallyModified())
6640 return true;
6641
6642 return
6643 (!aIgnoreUserData && mUserData.hasActualChanges()) ||
6644 mHWData.hasActualChanges() ||
6645 /* ignore mHDData */
6646 //mHDData.hasActualChanges() ||
6647#ifdef VBOX_VRDP
6648 (mVRDPServer && mVRDPServer->isReallyModified()) ||
6649#endif
6650 (mDVDDrive && mDVDDrive->isReallyModified()) ||
6651 (mFloppyDrive && mFloppyDrive->isReallyModified()) ||
6652 (mAudioAdapter && mAudioAdapter->isReallyModified()) ||
6653 (mUSBController && mUSBController->isReallyModified()) ||
6654 (mBIOSSettings && mBIOSSettings->isReallyModified());
6655}
6656
6657/**
6658 * Discards all changes to machine settings.
6659 *
6660 * @param aNotify whether to notify the direct session about changes or not
6661 *
6662 * @note Locks objects!
6663 */
6664void Machine::rollback (bool aNotify)
6665{
6666 AutoCaller autoCaller (this);
6667 AssertComRCReturn (autoCaller.rc(), (void) 0);
6668
6669 AutoLock alock (this);
6670
6671 mUserData.rollback();
6672
6673 mHWData.rollback();
6674
6675 if (mHDData.isBackedUp())
6676 fixupHardDisks (false /* aCommit */);
6677
6678 bool vrdpChanged = false, dvdChanged = false, floppyChanged = false,
6679 usbChanged = false;
6680 ComPtr <INetworkAdapter> networkAdapters [ELEMENTS (mNetworkAdapters)];
6681
6682 if (mBIOSSettings)
6683 mBIOSSettings->rollback();
6684
6685#ifdef VBOX_VRDP
6686 if (mVRDPServer)
6687 vrdpChanged = mVRDPServer->rollback();
6688#endif
6689
6690 if (mDVDDrive)
6691 dvdChanged = mDVDDrive->rollback();
6692
6693 if (mFloppyDrive)
6694 floppyChanged = mFloppyDrive->rollback();
6695
6696 if (mAudioAdapter)
6697 mAudioAdapter->rollback();
6698
6699 if (mUSBController)
6700 usbChanged = mUSBController->rollback();
6701
6702 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
6703 if (mNetworkAdapters [slot])
6704 if (mNetworkAdapters [slot]->rollback())
6705 networkAdapters [slot] = mNetworkAdapters [slot];
6706
6707 if (aNotify)
6708 {
6709 // inform the direct session about changes
6710
6711 ComObjPtr <Machine> that = this;
6712 alock.leave();
6713
6714 if (vrdpChanged)
6715 that->onVRDPServerChange();
6716 if (dvdChanged)
6717 that->onDVDDriveChange();
6718 if (floppyChanged)
6719 that->onFloppyDriveChange();
6720 if (usbChanged)
6721 that->onUSBControllerChange();
6722 for (ULONG slot = 0; slot < ELEMENTS (networkAdapters); slot ++)
6723 if (networkAdapters [slot])
6724 that->onNetworkAdapterChange (networkAdapters [slot]);
6725 }
6726}
6727
6728/**
6729 * Commits all the changes to machine settings.
6730 *
6731 * Note that when committing fails at some stage, it still continues
6732 * until the end. So, all data will either be actually committed or rolled
6733 * back (for failed cases) and the returned result code will describe the
6734 * first failure encountered. However, #isModified() will still return true
6735 * in case of failure, to indicade that settings in memory and on disk are
6736 * out of sync.
6737 *
6738 * @note Locks objects!
6739 */
6740HRESULT Machine::commit()
6741{
6742 AutoCaller autoCaller (this);
6743 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
6744
6745 AutoLock alock (this);
6746
6747 HRESULT rc = S_OK;
6748
6749 /*
6750 * use safe commit to ensure Snapshot machines (that share mUserData)
6751 * will still refer to a valid memory location
6752 */
6753 mUserData.commitCopy();
6754
6755 mHWData.commit();
6756
6757 if (mHDData.isBackedUp())
6758 rc = fixupHardDisks (true /* aCommit */);
6759
6760 mBIOSSettings->commit();
6761#ifdef VBOX_VRDP
6762 mVRDPServer->commit();
6763#endif
6764 mDVDDrive->commit();
6765 mFloppyDrive->commit();
6766 mAudioAdapter->commit();
6767 mUSBController->commit();
6768
6769 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
6770 mNetworkAdapters [slot]->commit();
6771
6772 if (mType == IsSessionMachine)
6773 {
6774 /* attach new data to the primary machine and reshare it */
6775 mPeer->mUserData.attach (mUserData);
6776 mPeer->mHWData.attach (mHWData);
6777 mPeer->mHDData.attach (mHDData);
6778 }
6779
6780 if (FAILED (rc))
6781 {
6782 /*
6783 * backup arbitrary data item to cause #isModified() to still return
6784 * true in case of any error
6785 */
6786 mHWData.backup();
6787 }
6788
6789 return rc;
6790}
6791
6792/**
6793 * Copies all the hardware data from the given machine.
6794 *
6795 * @note
6796 * This method must be called from under this object's lock.
6797 * @note
6798 * This method doesn't call #commit(), so all data remains backed up
6799 * and unsaved.
6800 */
6801void Machine::copyFrom (Machine *aThat)
6802{
6803 AssertReturn (mType == IsMachine || mType == IsSessionMachine, (void) 0);
6804 AssertReturn (aThat->mType == IsSnapshotMachine, (void) 0);
6805
6806 mHWData.assignCopy (aThat->mHWData);
6807
6808 // create copies of all shared folders (mHWData after attiching a copy
6809 // contains just references to original objects)
6810 for (HWData::SharedFolderList::iterator it = mHWData->mSharedFolders.begin();
6811 it != mHWData->mSharedFolders.end();
6812 ++ it)
6813 {
6814 ComObjPtr <SharedFolder> folder;
6815 folder.createObject();
6816 HRESULT rc = folder->initCopy (machine(), *it);
6817 AssertComRC (rc);
6818 *it = folder;
6819 }
6820
6821 mBIOSSettings->copyFrom (aThat->mBIOSSettings);
6822#ifdef VBOX_VRDP
6823 mVRDPServer->copyFrom (aThat->mVRDPServer);
6824#endif
6825 mDVDDrive->copyFrom (aThat->mDVDDrive);
6826 mFloppyDrive->copyFrom (aThat->mFloppyDrive);
6827 mAudioAdapter->copyFrom (aThat->mAudioAdapter);
6828 mUSBController->copyFrom (aThat->mUSBController);
6829
6830 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
6831 mNetworkAdapters [slot]->copyFrom (aThat->mNetworkAdapters [slot]);
6832}
6833
6834/////////////////////////////////////////////////////////////////////////////
6835// SessionMachine class
6836/////////////////////////////////////////////////////////////////////////////
6837
6838/** Task structure for asynchronous VM operations */
6839struct SessionMachine::Task
6840{
6841 Task (SessionMachine *m, Progress *p)
6842 : machine (m), progress (p)
6843 , state (m->data()->mMachineState) // save the current machine state
6844 , subTask (false), settingsChanged (false)
6845 {}
6846
6847 void modifyLastState (MachineState_T s)
6848 {
6849 *const_cast <MachineState_T *> (&state) = s;
6850 }
6851
6852 virtual void handler() = 0;
6853
6854 const ComObjPtr <SessionMachine> machine;
6855 const ComObjPtr <Progress> progress;
6856 const MachineState_T state;
6857
6858 bool subTask : 1;
6859 bool settingsChanged : 1;
6860};
6861
6862/** Take snapshot task */
6863struct SessionMachine::TakeSnapshotTask : public SessionMachine::Task
6864{
6865 TakeSnapshotTask (SessionMachine *m)
6866 : Task (m, NULL) {}
6867
6868 void handler() { machine->takeSnapshotHandler (*this); }
6869};
6870
6871/** Discard snapshot task */
6872struct SessionMachine::DiscardSnapshotTask : public SessionMachine::Task
6873{
6874 DiscardSnapshotTask (SessionMachine *m, Progress *p, Snapshot *s)
6875 : Task (m, p)
6876 , snapshot (s) {}
6877
6878 DiscardSnapshotTask (const Task &task, Snapshot *s)
6879 : Task (task)
6880 , snapshot (s) {}
6881
6882 void handler() { machine->discardSnapshotHandler (*this); }
6883
6884 const ComObjPtr <Snapshot> snapshot;
6885};
6886
6887/** Discard current state task */
6888struct SessionMachine::DiscardCurrentStateTask : public SessionMachine::Task
6889{
6890 DiscardCurrentStateTask (SessionMachine *m, Progress *p,
6891 bool discardCurSnapshot)
6892 : Task (m, p), discardCurrentSnapshot (discardCurSnapshot) {}
6893
6894 void handler() { machine->discardCurrentStateHandler (*this); }
6895
6896 const bool discardCurrentSnapshot;
6897};
6898
6899////////////////////////////////////////////////////////////////////////////////
6900
6901DEFINE_EMPTY_CTOR_DTOR (SessionMachine)
6902
6903HRESULT SessionMachine::FinalConstruct()
6904{
6905 LogFlowThisFunc (("\n"));
6906
6907 /* set the proper type to indicate we're the SessionMachine instance */
6908 unconst (mType) = IsSessionMachine;
6909
6910#if defined(__WIN__)
6911 mIPCSem = NULL;
6912#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
6913 mIPCSem = -1;
6914#endif
6915
6916 return S_OK;
6917}
6918
6919void SessionMachine::FinalRelease()
6920{
6921 LogFlowThisFunc (("\n"));
6922
6923 uninit (Uninit::Unexpected);
6924}
6925
6926/**
6927 * @note Must be called only by Machine::openSession() from its own write lock.
6928 */
6929HRESULT SessionMachine::init (Machine *aMachine)
6930{
6931 LogFlowThisFuncEnter();
6932 LogFlowThisFunc (("mName={%ls}\n", aMachine->mUserData->mName.raw()));
6933
6934 AssertReturn (aMachine, E_INVALIDARG);
6935
6936 AssertReturn (aMachine->lockHandle()->isLockedOnCurrentThread(), E_FAIL);
6937
6938 /* Enclose the state transition NotReady->InInit->Ready */
6939 AutoInitSpan autoInitSpan (this);
6940 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
6941
6942 /* create the interprocess semaphore */
6943#if defined(__WIN__)
6944 mIPCSemName = aMachine->mData->mConfigFileFull;
6945 for (size_t i = 0; i < mIPCSemName.length(); i++)
6946 if (mIPCSemName[i] == '\\')
6947 mIPCSemName[i] = '/';
6948 mIPCSem = ::CreateMutex (NULL, FALSE, mIPCSemName);
6949 ComAssertMsgRet (mIPCSem, ("Cannot create IPC mutex, err=0x%08X", ::GetLastError()),
6950 E_FAIL);
6951#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
6952 Utf8Str configFile = aMachine->mData->mConfigFileFull;
6953 char *pszConfigFile = NULL;
6954 RTStrUtf8ToCurrentCP (&pszConfigFile, configFile);
6955 key_t key = ::ftok (pszConfigFile, 0);
6956 RTStrFree (pszConfigFile);
6957 mIPCSem = ::semget (key, 1, S_IRWXU | S_IRWXG | S_IRWXO | IPC_CREAT);
6958 ComAssertMsgRet (mIPCSem >= 0, ("Cannot create IPC semaphore, errno=%d", errno),
6959 E_FAIL);
6960 /* set the initial value to 1 */
6961 int rv = ::semctl (mIPCSem, 0, SETVAL, 1);
6962 ComAssertMsgRet (rv == 0, ("Cannot init IPC semaphore, errno=%d", errno),
6963 E_FAIL);
6964#endif
6965
6966 /* memorize the peer Machine */
6967 unconst (mPeer) = aMachine;
6968 /* share the parent pointer */
6969 unconst (mParent) = aMachine->mParent;
6970
6971 /* take the pointers to data to share */
6972 mData.share (aMachine->mData);
6973 mSSData.share (aMachine->mSSData);
6974
6975 mUserData.share (aMachine->mUserData);
6976 mHWData.share (aMachine->mHWData);
6977 mHDData.share (aMachine->mHDData);
6978
6979 unconst (mBIOSSettings).createObject();
6980 mBIOSSettings->init (this, aMachine->mBIOSSettings);
6981#ifdef VBOX_VRDP
6982 /* create another VRDPServer object that will be mutable */
6983 unconst (mVRDPServer).createObject();
6984 mVRDPServer->init (this, aMachine->mVRDPServer);
6985#endif
6986 /* create another DVD drive object that will be mutable */
6987 unconst (mDVDDrive).createObject();
6988 mDVDDrive->init (this, aMachine->mDVDDrive);
6989 /* create another floppy drive object that will be mutable */
6990 unconst (mFloppyDrive).createObject();
6991 mFloppyDrive->init (this, aMachine->mFloppyDrive);
6992 /* create another audio adapter object that will be mutable */
6993 unconst (mAudioAdapter).createObject();
6994 mAudioAdapter->init (this, aMachine->mAudioAdapter);
6995 /* create another USB controller object that will be mutable */
6996 unconst (mUSBController).createObject();
6997 mUSBController->init (this, aMachine->mUSBController);
6998 /* create a list of network adapters that will be mutable */
6999 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
7000 {
7001 unconst (mNetworkAdapters [slot]).createObject();
7002 mNetworkAdapters [slot]->init (this, aMachine->mNetworkAdapters [slot]);
7003 }
7004
7005 /* Confirm a successful initialization when it's the case */
7006 autoInitSpan.setSucceeded();
7007
7008 LogFlowThisFuncLeave();
7009 return S_OK;
7010}
7011
7012/**
7013 * Uninitializes this session object. If the reason is other than
7014 * Uninit::Unexpected, then this method MUST be called from #checkForDeath().
7015 *
7016 * @param aReason uninitialization reason
7017 *
7018 * @note Locks mParent + this object for writing.
7019 */
7020void SessionMachine::uninit (Uninit::Reason aReason)
7021{
7022 LogFlowThisFuncEnter();
7023 LogFlowThisFunc (("reason=%d\n", aReason));
7024
7025 /*
7026 * Strongly reference ourselves to prevent this object deletion after
7027 * mData->mSession.mMachine.setNull() below (which can release the last
7028 * reference and call the destructor). Important: this must be done before
7029 * accessing any members (and before AutoUninitSpan that does it as well).
7030 * This self reference will be released as the very last step on return.
7031 */
7032 ComObjPtr <SessionMachine> selfRef = this;
7033
7034 /* Enclose the state transition Ready->InUninit->NotReady */
7035 AutoUninitSpan autoUninitSpan (this);
7036 if (autoUninitSpan.uninitDone())
7037 {
7038 LogFlowThisFunc (("Already uninitialized\n"));
7039 LogFlowThisFuncLeave();
7040 return;
7041 }
7042
7043 if (autoUninitSpan.initFailed())
7044 {
7045 /*
7046 * We've been called by init() because it's failed. It's not really
7047 * necessary (nor it's safe) to perform the regular uninit sequence
7048 * below, the following is enough.
7049 */
7050 LogFlowThisFunc (("Initialization failed\n"));
7051#if defined(__WIN__)
7052 if (mIPCSem)
7053 ::CloseHandle (mIPCSem);
7054 mIPCSem = NULL;
7055#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
7056 if (mIPCSem >= 0)
7057 ::semctl (mIPCSem, 0, IPC_RMID);
7058 mIPCSem = -1;
7059#endif
7060 uninitDataAndChildObjects();
7061 unconst (mParent).setNull();
7062 unconst (mPeer).setNull();
7063 LogFlowThisFuncLeave();
7064 return;
7065 }
7066
7067 /*
7068 * We need to lock this object in uninit() because the lock is shared
7069 * with mPeer (as well as data we modify below).
7070 * mParent->addProcessToReap() and others need mParent lock.
7071 */
7072 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7073
7074 if (isModified())
7075 {
7076 LogWarningThisFunc (("Discarding unsaved settings changes!\n"));
7077 rollback (false /* aNotify */);
7078 }
7079
7080 Assert (!mSnapshotData.mStateFilePath || !mSnapshotData.mSnapshot);
7081 if (mSnapshotData.mStateFilePath)
7082 {
7083 LogWarningThisFunc (("canceling failed save state request!\n"));
7084 endSavingState (FALSE /* aSuccess */);
7085 }
7086 else if (!!mSnapshotData.mSnapshot)
7087 {
7088 LogWarningThisFunc (("canceling untaken snapshot!\n"));
7089 endTakingSnapshot (FALSE /* aSuccess */);
7090 }
7091
7092 /* release all captured USB devices */
7093 mParent->host()->releaseAllUSBDevices (this);
7094
7095 if (mData->mSession.mPid != NIL_RTPROCESS)
7096 {
7097 /*
7098 * pid is not NIL, meaning this machine's process has been started
7099 * using VirtualBox::OpenRemoteSession(), thus it is our child.
7100 * we need to queue this pid to be reaped (to avoid zombies on Linux)
7101 */
7102 mParent->addProcessToReap (mData->mSession.mPid);
7103 mData->mSession.mPid = NIL_RTPROCESS;
7104 }
7105
7106 if (aReason == Uninit::Unexpected)
7107 {
7108 /*
7109 * uninitialization didn't come from #checkForDeath(), so tell the
7110 * client watcher thread to update the set of machines that have
7111 * open sessions.
7112 */
7113 mParent->updateClientWatcher();
7114 }
7115
7116 /* uninitialize all remote controls */
7117 if (mData->mSession.mRemoteControls.size())
7118 {
7119 LogFlowThisFunc (("Closing remote sessions (%d):\n",
7120 mData->mSession.mRemoteControls.size()));
7121
7122 Data::Session::RemoteControlList::iterator it =
7123 mData->mSession.mRemoteControls.begin();
7124 while (it != mData->mSession.mRemoteControls.end())
7125 {
7126 LogFlowThisFunc ((" Calling remoteControl->Uninitialize()...\n"));
7127 HRESULT rc = (*it)->Uninitialize();
7128 LogFlowThisFunc ((" remoteControl->Uninitialize() returned %08X\n", rc));
7129 if (FAILED (rc))
7130 LogWarningThisFunc (("Forgot to close the remote session?\n"));
7131 ++ it;
7132 }
7133 mData->mSession.mRemoteControls.clear();
7134 }
7135
7136 /*
7137 * An expected uninitialization can come only from #checkForDeath().
7138 * Otherwise it means that something's got really wrong (for examlple,
7139 * the Session implementation has released the VirtualBox reference
7140 * before it triggered #OnSessionEnd(), or before releasing IPC semaphore,
7141 * etc). However, it's also possible, that the client releases the IPC
7142 * semaphore correctly (i.e. before it releases the VirtualBox reference),
7143 * but but the VirtualBox release event comes first to the server process.
7144 * This case is practically possible, so we should not assert on an
7145 * unexpected uninit, just log a warning.
7146 */
7147
7148 if ((aReason == Uninit::Unexpected))
7149 LogWarningThisFunc (("Unexpected SessionMachine uninitialization!\n"));
7150
7151 if (aReason != Uninit::Normal)
7152 mData->mSession.mDirectControl.setNull();
7153 else
7154 {
7155 /* this must be null here (see #OnSessionEnd()) */
7156 Assert (mData->mSession.mDirectControl.isNull());
7157 Assert (mData->mSession.mState == SessionState_SessionClosing);
7158 Assert (!mData->mSession.mProgress.isNull());
7159
7160 mData->mSession.mProgress->notifyComplete (S_OK);
7161 mData->mSession.mProgress.setNull();
7162 }
7163
7164 /* remove the association between the peer machine and this session machine */
7165 Assert (mData->mSession.mMachine == this ||
7166 aReason == Uninit::Unexpected);
7167
7168 /* reset the rest of session data */
7169 mData->mSession.mMachine.setNull();
7170 mData->mSession.mState = SessionState_SessionClosed;
7171
7172 /* close the interprocess semaphore before leaving the shared lock */
7173#if defined(__WIN__)
7174 if (mIPCSem)
7175 ::CloseHandle (mIPCSem);
7176 mIPCSem = NULL;
7177#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
7178 if (mIPCSem >= 0)
7179 ::semctl (mIPCSem, 0, IPC_RMID);
7180 mIPCSem = -1;
7181#endif
7182
7183 /* fire an event */
7184 mParent->onSessionStateChange (mData->mUuid, SessionState_SessionClosed);
7185
7186 uninitDataAndChildObjects();
7187
7188 /* leave the shared lock before setting the above two to NULL */
7189 alock.leave();
7190
7191 unconst (mParent).setNull();
7192 unconst (mPeer).setNull();
7193
7194 LogFlowThisFuncLeave();
7195}
7196
7197// AutoLock::Lockable interface
7198////////////////////////////////////////////////////////////////////////////////
7199
7200/**
7201 * Overrides VirtualBoxBase::lockHandle() in order to share the lock handle
7202 * with the primary Machine instance (mPeer).
7203 */
7204AutoLock::Handle *SessionMachine::lockHandle() const
7205{
7206 AssertReturn (!mPeer.isNull(), NULL);
7207 return mPeer->lockHandle();
7208}
7209
7210// IInternalMachineControl methods
7211////////////////////////////////////////////////////////////////////////////////
7212
7213/**
7214 * @note Locks the same as #setMachineState() does.
7215 */
7216STDMETHODIMP SessionMachine::UpdateState (MachineState_T machineState)
7217{
7218 return setMachineState (machineState);
7219}
7220
7221/**
7222 * @note Locks this object for reading.
7223 */
7224STDMETHODIMP SessionMachine::GetIPCId (BSTR *id)
7225{
7226 AutoCaller autoCaller (this);
7227 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7228
7229 AutoReaderLock alock (this);
7230
7231#if defined(__WIN__)
7232 mIPCSemName.cloneTo (id);
7233 return S_OK;
7234#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
7235 mData->mConfigFileFull.cloneTo (id);
7236 return S_OK;
7237#else
7238 return S_FAIL;
7239#endif
7240}
7241
7242/**
7243 * @note Locks this object for reading.
7244 */
7245STDMETHODIMP SessionMachine::GetLogFolder (BSTR *aLogFolder)
7246{
7247 AutoCaller autoCaller (this);
7248 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7249
7250 AutoReaderLock alock (this);
7251
7252 Utf8Str logFolder;
7253 getLogFolder (logFolder);
7254
7255 Bstr (logFolder).cloneTo (aLogFolder);
7256
7257 return S_OK;
7258}
7259
7260/**
7261 * Goes through the USB filters of the given machine to see if the given
7262 * device matches any filter or not.
7263 *
7264 * @note Locks the same as USBController::hasMatchingFilter() does.
7265 */
7266STDMETHODIMP SessionMachine::RunUSBDeviceFilters (IUSBDevice *aUSBDevice,
7267 BOOL *aMatched)
7268{
7269 if (!aUSBDevice)
7270 return E_INVALIDARG;
7271 if (!aMatched)
7272 return E_POINTER;
7273
7274 AutoCaller autoCaller (this);
7275 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7276
7277 *aMatched = mUSBController->hasMatchingFilter (aUSBDevice);
7278
7279 return S_OK;
7280}
7281
7282/**
7283 * @note Locks the same as Host::captureUSBDevice() does.
7284 */
7285STDMETHODIMP SessionMachine::CaptureUSBDevice (INPTR GUIDPARAM aId,
7286 IUSBDevice **aHostDevice)
7287{
7288 if (!aHostDevice)
7289 return E_POINTER;
7290
7291 AutoCaller autoCaller (this);
7292 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7293
7294 // if cautureUSBDevice() fails, it must have set extended error info
7295 return mParent->host()->captureUSBDevice (this, aId, aHostDevice);
7296}
7297
7298/**
7299 * @note Locks the same as Host::releaseUSBDevice() does.
7300 */
7301STDMETHODIMP SessionMachine::ReleaseUSBDevice (INPTR GUIDPARAM aId)
7302{
7303 AutoCaller autoCaller (this);
7304 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7305
7306 return mParent->host()->releaseUSBDevice (this, aId);
7307}
7308
7309/**
7310 * @note Locks the same as Host::autoCaptureUSBDevices() does.
7311 */
7312STDMETHODIMP SessionMachine::AutoCaptureUSBDevices (IUSBDeviceCollection **aHostDevices)
7313{
7314 AutoCaller autoCaller (this);
7315 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7316
7317 return mParent->host()->autoCaptureUSBDevices (this, aHostDevices);
7318}
7319
7320/**
7321 * @note Locks the same as Host::releaseAllUSBDevices() does.
7322 */
7323STDMETHODIMP SessionMachine::ReleaseAllUSBDevices()
7324{
7325 AutoCaller autoCaller (this);
7326 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7327
7328 return mParent->host()->releaseAllUSBDevices (this);
7329}
7330
7331/**
7332 * @note Locks mParent + this object for writing.
7333 */
7334STDMETHODIMP SessionMachine::OnSessionEnd (ISession *aSession,
7335 IProgress **aProgress)
7336{
7337 LogFlowThisFuncEnter();
7338
7339 AssertReturn (aSession, E_INVALIDARG);
7340 AssertReturn (aProgress, E_INVALIDARG);
7341
7342 AutoCaller autoCaller (this);
7343
7344 LogFlowThisFunc (("state=%d\n", autoCaller.state()));
7345 /*
7346 * We don't assert below because it might happen that a non-direct session
7347 * informs us it is closed right after we've been uninitialized -- it's ok.
7348 */
7349 CheckComRCReturnRC (autoCaller.rc());
7350
7351 /* get IInternalSessionControl interface */
7352 ComPtr <IInternalSessionControl> control (aSession);
7353
7354 ComAssertRet (!control.isNull(), E_INVALIDARG);
7355
7356 /* Progress::init() needs mParent lock */
7357 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7358
7359 if (control.equalsTo (mData->mSession.mDirectControl))
7360 {
7361 ComAssertRet (aProgress, E_POINTER);
7362
7363 /* The direct session is being normally closed by the client process
7364 * ----------------------------------------------------------------- */
7365
7366 /* go to the closing state (essential for all open*Session() calls and
7367 * for #checkForDeath()) */
7368 Assert (mData->mSession.mState == SessionState_SessionOpen);
7369 mData->mSession.mState = SessionState_SessionClosing;
7370
7371 /* set direct control to NULL to release the remote instance */
7372 mData->mSession.mDirectControl.setNull();
7373 LogFlowThisFunc (("Direct control is set to NULL\n"));
7374
7375 /*
7376 * Create the progress object the client will use to wait until
7377 * #checkForDeath() is called to uninitialize this session object
7378 * after it releases the IPC semaphore.
7379 */
7380 ComObjPtr <Progress> progress;
7381 progress.createObject();
7382 progress->init (mParent, (IMachine *) mPeer, Bstr (tr ("Closing session")),
7383 FALSE /* aCancelable */);
7384 progress.queryInterfaceTo (aProgress);
7385 mData->mSession.mProgress = progress;
7386 }
7387 else
7388 {
7389 /* the remote session is being normally closed */
7390 Data::Session::RemoteControlList::iterator it =
7391 mData->mSession.mRemoteControls.begin();
7392 while (it != mData->mSession.mRemoteControls.end())
7393 {
7394 if (control.equalsTo (*it))
7395 break;
7396 ++it;
7397 }
7398 BOOL found = it != mData->mSession.mRemoteControls.end();
7399 ComAssertMsgRet (found, ("The session is not found in the session list!"),
7400 E_INVALIDARG);
7401 mData->mSession.mRemoteControls.remove (*it);
7402 }
7403
7404 LogFlowThisFuncLeave();
7405 return S_OK;
7406}
7407
7408/**
7409 * @note Locks mParent + this object for writing.
7410 */
7411STDMETHODIMP SessionMachine::BeginSavingState (IProgress *aProgress, BSTR *aStateFilePath)
7412{
7413 LogFlowThisFuncEnter();
7414
7415 AssertReturn (aProgress, E_INVALIDARG);
7416 AssertReturn (aStateFilePath, E_POINTER);
7417
7418 AutoCaller autoCaller (this);
7419 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7420
7421 /* mParent->addProgress() needs mParent lock */
7422 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7423
7424 AssertReturn (mData->mMachineState == MachineState_Paused &&
7425 mSnapshotData.mLastState == MachineState_InvalidMachineState &&
7426 mSnapshotData.mProgressId.isEmpty() &&
7427 mSnapshotData.mStateFilePath.isNull(),
7428 E_FAIL);
7429
7430 /* memorize the progress ID and add it to the global collection */
7431 Guid progressId;
7432 HRESULT rc = aProgress->COMGETTER(Id) (progressId.asOutParam());
7433 AssertComRCReturn (rc, rc);
7434 rc = mParent->addProgress (aProgress);
7435 AssertComRCReturn (rc, rc);
7436
7437 Bstr stateFilePath;
7438 /* stateFilePath is null when the machine is not running */
7439 if (mData->mMachineState == MachineState_Paused)
7440 {
7441 stateFilePath = Utf8StrFmt ("%ls%c{%Vuuid}.sav",
7442 mUserData->mSnapshotFolderFull.raw(),
7443 RTPATH_DELIMITER, mData->mUuid.raw());
7444 }
7445
7446 /* fill in the snapshot data */
7447 mSnapshotData.mLastState = mData->mMachineState;
7448 mSnapshotData.mProgressId = progressId;
7449 mSnapshotData.mStateFilePath = stateFilePath;
7450
7451 /* set the state to Saving (this is expected by Console::SaveState()) */
7452 setMachineState (MachineState_Saving);
7453
7454 stateFilePath.cloneTo (aStateFilePath);
7455
7456 return S_OK;
7457}
7458
7459/**
7460 * @note Locks mParent + this objects for writing.
7461 */
7462STDMETHODIMP SessionMachine::EndSavingState (BOOL aSuccess)
7463{
7464 LogFlowThisFunc (("\n"));
7465
7466 AutoCaller autoCaller (this);
7467 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7468
7469 /* endSavingState() need mParent lock */
7470 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7471
7472 AssertReturn (mData->mMachineState == MachineState_Saving &&
7473 mSnapshotData.mLastState != MachineState_InvalidMachineState &&
7474 !mSnapshotData.mProgressId.isEmpty() &&
7475 !mSnapshotData.mStateFilePath.isNull(),
7476 E_FAIL);
7477
7478 /*
7479 * on success, set the state to Saved;
7480 * on failure, set the state to the state we had when BeginSavingState() was
7481 * called (this is expected by Console::SaveState() and
7482 * Console::saveStateThread())
7483 */
7484 if (aSuccess)
7485 setMachineState (MachineState_Saved);
7486 else
7487 setMachineState (mSnapshotData.mLastState);
7488
7489 return endSavingState (aSuccess);
7490}
7491
7492/**
7493 * @note Locks mParent + this objects for writing.
7494 */
7495STDMETHODIMP SessionMachine::BeginTakingSnapshot (
7496 IConsole *aInitiator, INPTR BSTR aName, INPTR BSTR aDescription,
7497 IProgress *aProgress, BSTR *aStateFilePath,
7498 IProgress **aServerProgress)
7499{
7500 LogFlowThisFuncEnter();
7501
7502 AssertReturn (aInitiator && aName, E_INVALIDARG);
7503 AssertReturn (aStateFilePath && aServerProgress, E_POINTER);
7504
7505 LogFlowThisFunc (("aName='%ls'\n", aName));
7506
7507 AutoCaller autoCaller (this);
7508 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7509
7510 /* Progress::init() needs mParent lock */
7511 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7512
7513 AssertReturn ((mData->mMachineState < MachineState_Running ||
7514 mData->mMachineState == MachineState_Paused) &&
7515 mSnapshotData.mLastState == MachineState_InvalidMachineState &&
7516 mSnapshotData.mSnapshot.isNull() &&
7517 mSnapshotData.mServerProgress.isNull() &&
7518 mSnapshotData.mCombinedProgress.isNull(),
7519 E_FAIL);
7520
7521 bool takingSnapshotOnline = mData->mMachineState == MachineState_Paused;
7522
7523 if (!takingSnapshotOnline && mData->mMachineState != MachineState_Saved)
7524 {
7525 /*
7526 * save all current settings to ensure current changes are committed
7527 * and hard disks are fixed up
7528 */
7529 HRESULT rc = saveSettings();
7530 CheckComRCReturnRC (rc);
7531 }
7532
7533 /* check that there are no Writethrough hard disks attached */
7534 for (HDData::HDAttachmentList::const_iterator
7535 it = mHDData->mHDAttachments.begin();
7536 it != mHDData->mHDAttachments.end();
7537 ++ it)
7538 {
7539 ComObjPtr <HardDisk> hd = (*it)->hardDisk();
7540 AutoLock hdLock (hd);
7541 if (hd->type() == HardDiskType_WritethroughHardDisk)
7542 return setError (E_FAIL,
7543 tr ("Cannot take a snapshot when there is a Writethrough hard "
7544 " disk attached ('%ls')"), hd->toString().raw());
7545 }
7546
7547 AssertReturn (aProgress || !takingSnapshotOnline, E_FAIL);
7548
7549 /* create an ID for the snapshot */
7550 Guid snapshotId;
7551 snapshotId.create();
7552
7553 Bstr stateFilePath;
7554 /* stateFilePath is null when the machine is not online nor saved */
7555 if (takingSnapshotOnline || mData->mMachineState == MachineState_Saved)
7556 stateFilePath = Utf8StrFmt ("%ls%c{%Vuuid}.sav",
7557 mUserData->mSnapshotFolderFull.raw(),
7558 RTPATH_DELIMITER,
7559 snapshotId.ptr());
7560
7561 /* ensure the directory for the saved state file exists */
7562 if (stateFilePath)
7563 {
7564 Utf8Str dir = stateFilePath;
7565 RTPathStripFilename (dir.mutableRaw());
7566 if (!RTDirExists (dir))
7567 {
7568 int vrc = RTDirCreateFullPath (dir, 0777);
7569 if (VBOX_FAILURE (vrc))
7570 return setError (E_FAIL,
7571 tr ("Could not create a directory '%s' to save the "
7572 "VM state to (%Vrc)"),
7573 dir.raw(), vrc);
7574 }
7575 }
7576
7577 /* create a snapshot machine object */
7578 ComObjPtr <SnapshotMachine> snapshotMachine;
7579 snapshotMachine.createObject();
7580 HRESULT rc = snapshotMachine->init (this, snapshotId, stateFilePath);
7581 AssertComRCReturn (rc, rc);
7582
7583 Bstr progressDesc = Bstr (tr ("Taking snapshot of virtual machine"));
7584 Bstr firstOpDesc = Bstr (tr ("Preparing to take snapshot"));
7585
7586 /*
7587 * create a server-side progress object (it will be descriptionless
7588 * when we need to combine it with the VM-side progress, i.e. when we're
7589 * taking a snapshot online). The number of operations is:
7590 * 1 (preparing) + # of VDIs + 1 (if the state is saved so we need to copy it)
7591 */
7592 ComObjPtr <Progress> serverProgress;
7593 {
7594 ULONG opCount = 1 + mHDData->mHDAttachments.size();
7595 if (mData->mMachineState == MachineState_Saved)
7596 opCount ++;
7597 serverProgress.createObject();
7598 if (takingSnapshotOnline)
7599 rc = serverProgress->init (FALSE, opCount, firstOpDesc);
7600 else
7601 rc = serverProgress->init (mParent, aInitiator, progressDesc, FALSE,
7602 opCount, firstOpDesc);
7603 AssertComRCReturn (rc, rc);
7604 }
7605
7606 /* create a combined server-side progress object when necessary */
7607 ComObjPtr <CombinedProgress> combinedProgress;
7608 if (takingSnapshotOnline)
7609 {
7610 combinedProgress.createObject();
7611 rc = combinedProgress->init (mParent, aInitiator, progressDesc,
7612 serverProgress, aProgress);
7613 AssertComRCReturn (rc, rc);
7614 }
7615
7616 /* create a snapshot object */
7617 RTTIMESPEC time;
7618 ComObjPtr <Snapshot> snapshot;
7619 snapshot.createObject();
7620 rc = snapshot->init (snapshotId, aName, aDescription,
7621 RTTimeSpecGetMilli (RTTimeNow (&time)),
7622 snapshotMachine, mData->mCurrentSnapshot);
7623 AssertComRCReturn (rc, rc);
7624
7625 /*
7626 * create and start the task on a separate thread
7627 * (note that it will not start working until we release alock)
7628 */
7629 TakeSnapshotTask *task = new TakeSnapshotTask (this);
7630 int vrc = RTThreadCreate (NULL, taskHandler,
7631 (void *) task,
7632 0, RTTHREADTYPE_MAIN_WORKER, 0, "TakeSnapshot");
7633 if (VBOX_FAILURE (vrc))
7634 {
7635 snapshot->uninit();
7636 delete task;
7637 ComAssertFailedRet (E_FAIL);
7638 }
7639
7640 /* fill in the snapshot data */
7641 mSnapshotData.mLastState = mData->mMachineState;
7642 mSnapshotData.mSnapshot = snapshot;
7643 mSnapshotData.mServerProgress = serverProgress;
7644 mSnapshotData.mCombinedProgress = combinedProgress;
7645
7646 /* set the state to Saving (this is expected by Console::TakeSnapshot()) */
7647 setMachineState (MachineState_Saving);
7648
7649 if (takingSnapshotOnline)
7650 stateFilePath.cloneTo (aStateFilePath);
7651 else
7652 *aStateFilePath = NULL;
7653
7654 serverProgress.queryInterfaceTo (aServerProgress);
7655
7656 LogFlowThisFuncLeave();
7657 return S_OK;
7658}
7659
7660/**
7661 * @note Locks mParent + this objects for writing.
7662 */
7663STDMETHODIMP SessionMachine::EndTakingSnapshot (BOOL aSuccess)
7664{
7665 LogFlowThisFunc (("\n"));
7666
7667 AutoCaller autoCaller (this);
7668 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7669
7670 /* Lock mParent because of endTakingSnapshot() */
7671 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7672
7673 AssertReturn (!aSuccess ||
7674 (mData->mMachineState == MachineState_Saving &&
7675 mSnapshotData.mLastState != MachineState_InvalidMachineState &&
7676 !mSnapshotData.mSnapshot.isNull() &&
7677 !mSnapshotData.mServerProgress.isNull() &&
7678 !mSnapshotData.mCombinedProgress.isNull()),
7679 E_FAIL);
7680
7681 /*
7682 * set the state to the state we had when BeginTakingSnapshot() was called
7683 * (this is expected by Console::TakeSnapshot() and
7684 * Console::saveStateThread())
7685 */
7686 setMachineState (mSnapshotData.mLastState);
7687
7688 return endTakingSnapshot (aSuccess);
7689}
7690
7691/**
7692 * @note Locks mParent + this + children objects for writing!
7693 */
7694STDMETHODIMP SessionMachine::DiscardSnapshot (
7695 IConsole *aInitiator, INPTR GUIDPARAM aId,
7696 MachineState_T *aMachineState, IProgress **aProgress)
7697{
7698 LogFlowThisFunc (("\n"));
7699
7700 Guid id = aId;
7701 AssertReturn (aInitiator && !id.isEmpty(), E_INVALIDARG);
7702 AssertReturn (aMachineState && aProgress, E_POINTER);
7703
7704 AutoCaller autoCaller (this);
7705 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7706
7707 /* Progress::init() needs mParent lock */
7708 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7709
7710 ComAssertRet (mData->mMachineState < MachineState_Running, E_FAIL);
7711
7712 ComObjPtr <Snapshot> snapshot;
7713 HRESULT rc = findSnapshot (id, snapshot, true /* aSetError */);
7714 CheckComRCReturnRC (rc);
7715
7716 AutoLock snapshotLock (snapshot);
7717 if (snapshot == mData->mFirstSnapshot)
7718 {
7719 AutoLock chLock (mData->mFirstSnapshot->childrenLock());
7720 size_t childrenCount = mData->mFirstSnapshot->children().size();
7721 if (childrenCount > 1)
7722 return setError (E_FAIL,
7723 tr ("Cannot discard the snapshot '%ls' because it is the first "
7724 "snapshot of the machine '%ls' and it has more than one "
7725 "child snapshot (%d)"),
7726 snapshot->data().mName.raw(), mUserData->mName.raw(),
7727 childrenCount);
7728 }
7729
7730 /*
7731 * If the snapshot being discarded is the current one, ensure current
7732 * settings are committed and saved.
7733 */
7734 if (snapshot == mData->mCurrentSnapshot)
7735 {
7736 if (isModified())
7737 {
7738 rc = saveSettings();
7739 CheckComRCReturnRC (rc);
7740 }
7741 }
7742
7743 /*
7744 * create a progress object. The number of operations is:
7745 * 1 (preparing) + # of VDIs
7746 */
7747 ComObjPtr <Progress> progress;
7748 progress.createObject();
7749 rc = progress->init (mParent, aInitiator,
7750 Bstr (Utf8StrFmt (tr ("Discarding snapshot '%ls'"),
7751 snapshot->data().mName.raw())),
7752 FALSE /* aCancelable */,
7753 1 + snapshot->data().mMachine->mHDData->mHDAttachments.size(),
7754 Bstr (tr ("Preparing to discard snapshot")));
7755 AssertComRCReturn (rc, rc);
7756
7757 /* create and start the task on a separate thread */
7758 DiscardSnapshotTask *task = new DiscardSnapshotTask (this, progress, snapshot);
7759 int vrc = RTThreadCreate (NULL, taskHandler,
7760 (void *) task,
7761 0, RTTHREADTYPE_MAIN_WORKER, 0, "DiscardSnapshot");
7762 if (VBOX_FAILURE (vrc))
7763 delete task;
7764 ComAssertRCRet (vrc, E_FAIL);
7765
7766 /* set the proper machine state (note: after creating a Task instance) */
7767 setMachineState (MachineState_Discarding);
7768
7769 /* return the progress to the caller */
7770 progress.queryInterfaceTo (aProgress);
7771
7772 /* return the new state to the caller */
7773 *aMachineState = mData->mMachineState;
7774
7775 return S_OK;
7776}
7777
7778/**
7779 * @note Locks mParent + this + children objects for writing!
7780 */
7781STDMETHODIMP SessionMachine::DiscardCurrentState (
7782 IConsole *aInitiator, MachineState_T *aMachineState, IProgress **aProgress)
7783{
7784 LogFlowThisFunc (("\n"));
7785
7786 AssertReturn (aInitiator, E_INVALIDARG);
7787 AssertReturn (aMachineState && aProgress, E_POINTER);
7788
7789 AutoCaller autoCaller (this);
7790 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7791
7792 /* Progress::init() needs mParent lock */
7793 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7794
7795 ComAssertRet (mData->mMachineState < MachineState_Running, E_FAIL);
7796
7797 if (mData->mCurrentSnapshot.isNull())
7798 return setError (E_FAIL,
7799 tr ("Could not discard the current state of the machine '%ls' "
7800 "because it doesn't have any snapshots"),
7801 mUserData->mName.raw());
7802
7803 /*
7804 * create a progress object. The number of operations is:
7805 * 1 (preparing) + # of VDIs + 1 (if we need to copy the saved state file)
7806 */
7807 ComObjPtr <Progress> progress;
7808 progress.createObject();
7809 {
7810 ULONG opCount = 1 + mData->mCurrentSnapshot->data()
7811 .mMachine->mHDData->mHDAttachments.size();
7812 if (mData->mCurrentSnapshot->stateFilePath())
7813 ++ opCount;
7814 progress->init (mParent, aInitiator,
7815 Bstr (tr ("Discarding current machine state")),
7816 FALSE /* aCancelable */, opCount,
7817 Bstr (tr ("Preparing to discard current state")));
7818 }
7819
7820 /* create and start the task on a separate thread */
7821 DiscardCurrentStateTask *task =
7822 new DiscardCurrentStateTask (this, progress, false /* discardCurSnapshot */);
7823 int vrc = RTThreadCreate (NULL, taskHandler,
7824 (void *) task,
7825 0, RTTHREADTYPE_MAIN_WORKER, 0, "DiscardCurState");
7826 if (VBOX_FAILURE (vrc))
7827 delete task;
7828 ComAssertRCRet (vrc, E_FAIL);
7829
7830 /* set the proper machine state (note: after creating a Task instance) */
7831 setMachineState (MachineState_Discarding);
7832
7833 /* return the progress to the caller */
7834 progress.queryInterfaceTo (aProgress);
7835
7836 /* return the new state to the caller */
7837 *aMachineState = mData->mMachineState;
7838
7839 return S_OK;
7840}
7841
7842/**
7843 * @note Locks mParent + other objects for writing!
7844 */
7845STDMETHODIMP SessionMachine::DiscardCurrentSnapshotAndState (
7846 IConsole *aInitiator, MachineState_T *aMachineState, IProgress **aProgress)
7847{
7848 LogFlowThisFunc (("\n"));
7849
7850 AssertReturn (aInitiator, E_INVALIDARG);
7851 AssertReturn (aMachineState && aProgress, E_POINTER);
7852
7853 AutoCaller autoCaller (this);
7854 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7855
7856 /* Progress::init() needs mParent lock */
7857 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
7858
7859 ComAssertRet (mData->mMachineState < MachineState_Running, E_FAIL);
7860
7861 if (mData->mCurrentSnapshot.isNull())
7862 return setError (E_FAIL,
7863 tr ("Could not discard the current state of the machine '%ls' "
7864 "because it doesn't have any snapshots"),
7865 mUserData->mName.raw());
7866
7867 /*
7868 * create a progress object. The number of operations is:
7869 * 1 (preparing) + # of VDIs in the current snapshot +
7870 * # of VDIs in the previous snapshot +
7871 * 1 (if we need to copy the saved state file of the previous snapshot)
7872 * or (if there is no previous snapshot):
7873 * 1 (preparing) + # of VDIs in the current snapshot * 2 +
7874 * 1 (if we need to copy the saved state file of the current snapshot)
7875 */
7876 ComObjPtr <Progress> progress;
7877 progress.createObject();
7878 {
7879 ComObjPtr <Snapshot> curSnapshot = mData->mCurrentSnapshot;
7880 ComObjPtr <Snapshot> prevSnapshot = mData->mCurrentSnapshot->parent();
7881
7882 ULONG opCount = 1;
7883 if (prevSnapshot)
7884 {
7885 opCount += curSnapshot->data().mMachine->mHDData->mHDAttachments.size();
7886 opCount += prevSnapshot->data().mMachine->mHDData->mHDAttachments.size();
7887 if (prevSnapshot->stateFilePath())
7888 ++ opCount;
7889 }
7890 else
7891 {
7892 opCount += curSnapshot->data().mMachine->mHDData->mHDAttachments.size() * 2;
7893 if (curSnapshot->stateFilePath())
7894 ++ opCount;
7895 }
7896
7897 progress->init (mParent, aInitiator,
7898 Bstr (tr ("Discarding current machine snapshot and state")),
7899 FALSE /* aCancelable */, opCount,
7900 Bstr (tr ("Preparing to discard current snapshot and state")));
7901 }
7902
7903 /* create and start the task on a separate thread */
7904 DiscardCurrentStateTask *task =
7905 new DiscardCurrentStateTask (this, progress, true /* discardCurSnapshot */);
7906 int vrc = RTThreadCreate (NULL, taskHandler,
7907 (void *) task,
7908 0, RTTHREADTYPE_MAIN_WORKER, 0, "DiscardCurState");
7909 if (VBOX_FAILURE (vrc))
7910 delete task;
7911 ComAssertRCRet (vrc, E_FAIL);
7912
7913 /* set the proper machine state (note: after creating a Task instance) */
7914 setMachineState (MachineState_Discarding);
7915
7916 /* return the progress to the caller */
7917 progress.queryInterfaceTo (aProgress);
7918
7919 /* return the new state to the caller */
7920 *aMachineState = mData->mMachineState;
7921
7922 return S_OK;
7923}
7924
7925// public methods only for internal purposes
7926/////////////////////////////////////////////////////////////////////////////
7927
7928/**
7929 * Called from the client watcher thread to check for unexpected client
7930 * process death.
7931 *
7932 * @note On Win32, this method is called only when we've got the semaphore
7933 * (i.e. it has been signaled when we were waiting for it).
7934 *
7935 * On Win32, this method always returns true.
7936 *
7937 * On Linux, the method returns true if the client process has terminated
7938 * abnormally (and/or the session has been uninitialized) and false if it is
7939 * still alive.
7940 *
7941 * @note Locks this object for writing.
7942 */
7943bool SessionMachine::checkForDeath()
7944{
7945 Uninit::Reason reason;
7946 bool doUninit = false;
7947 bool rc = false;
7948
7949 /*
7950 * Enclose autoCaller with a block because calling uninit()
7951 * from under it will deadlock.
7952 */
7953 {
7954 AutoCaller autoCaller (this);
7955 if (!autoCaller.isOk())
7956 {
7957 /*
7958 * return true if not ready, to cause the client watcher to exclude
7959 * the corresponding session from watching
7960 */
7961 LogFlowThisFunc (("Already uninitialized!"));
7962 return true;
7963 }
7964
7965 AutoLock alock (this);
7966
7967 /*
7968 * Determine the reason of death: if the session state is Closing here,
7969 * everything is fine. Otherwise it means that the client did not call
7970 * OnSessionEnd() before it released the IPC semaphore.
7971 * This may happen either because the client process has abnormally
7972 * terminated, or because it simply forgot to call ISession::Close()
7973 * before exiting. We threat the latter also as an abnormal termination
7974 * (see Session::uninit() for details).
7975 */
7976 reason = mData->mSession.mState == SessionState_SessionClosing ?
7977 Uninit::Normal :
7978 Uninit::Abnormal;
7979
7980#if defined(__WIN__)
7981
7982 AssertMsg (mIPCSem, ("semaphore must be created"));
7983
7984 if (reason == Uninit::Abnormal)
7985 {
7986 LogWarningThisFunc (("ABNORMAL client termination! (wasRunning=%d)\n",
7987 mData->mMachineState >= MachineState_Running));
7988
7989 /* reset the state to Aborted */
7990 if (mData->mMachineState != MachineState_Aborted)
7991 setMachineState (MachineState_Aborted);
7992 }
7993
7994 /* release the IPC mutex */
7995 ::ReleaseMutex (mIPCSem);
7996
7997 doUninit = true;
7998
7999 rc = true;
8000
8001#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
8002
8003 AssertMsg (mIPCSem >= 0, ("semaphore must be created"));
8004
8005 int val = ::semctl (mIPCSem, 0, GETVAL);
8006 if (val > 0)
8007 {
8008 /* the semaphore is signaled, meaning the session is terminated */
8009
8010 if (reason == Uninit::Abnormal)
8011 {
8012 LogWarningThisFunc (("ABNORMAL client termination! (wasRunning=%d)\n",
8013 mData->mMachineState >= MachineState_Running));
8014
8015 /* reset the state to Aborted */
8016 if (mData->mMachineState != MachineState_Aborted)
8017 setMachineState (MachineState_Aborted);
8018 }
8019
8020 doUninit = true;
8021 }
8022
8023 rc = val > 0;
8024
8025#endif
8026
8027 } /* AutoCaller block */
8028
8029 if (doUninit)
8030 uninit (reason);
8031
8032 return rc;
8033}
8034
8035/**
8036 * @note Locks this object for reading.
8037 */
8038HRESULT SessionMachine::onDVDDriveChange()
8039{
8040 LogFlowThisFunc (("\n"));
8041
8042 AutoCaller autoCaller (this);
8043 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8044
8045 ComPtr <IInternalSessionControl> directControl;
8046 {
8047 AutoReaderLock alock (this);
8048 directControl = mData->mSession.mDirectControl;
8049 }
8050
8051 /* ignore notifications sent after #OnSessionEnd() is called */
8052 if (!directControl)
8053 return S_OK;
8054
8055 return directControl->OnDVDDriveChange();
8056}
8057
8058/**
8059 * @note Locks this object for reading.
8060 */
8061HRESULT SessionMachine::onFloppyDriveChange()
8062{
8063 LogFlowThisFunc (("\n"));
8064
8065 AutoCaller autoCaller (this);
8066 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8067
8068 ComPtr <IInternalSessionControl> directControl;
8069 {
8070 AutoReaderLock alock (this);
8071 directControl = mData->mSession.mDirectControl;
8072 }
8073
8074 /* ignore notifications sent after #OnSessionEnd() is called */
8075 if (!directControl)
8076 return S_OK;
8077
8078 return directControl->OnFloppyDriveChange();
8079}
8080
8081/**
8082 * @note Locks this object for reading.
8083 */
8084HRESULT SessionMachine::onNetworkAdapterChange(INetworkAdapter *networkAdapter)
8085{
8086 LogFlowThisFunc (("\n"));
8087
8088 AutoCaller autoCaller (this);
8089 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8090
8091 ComPtr <IInternalSessionControl> directControl;
8092 {
8093 AutoReaderLock alock (this);
8094 directControl = mData->mSession.mDirectControl;
8095 }
8096
8097 /* ignore notifications sent after #OnSessionEnd() is called */
8098 if (!directControl)
8099 return S_OK;
8100
8101 return directControl->OnNetworkAdapterChange(networkAdapter);
8102}
8103
8104/**
8105 * @note Locks this object for reading.
8106 */
8107HRESULT SessionMachine::onVRDPServerChange()
8108{
8109 LogFlowThisFunc (("\n"));
8110
8111 AutoCaller autoCaller (this);
8112 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8113
8114 ComPtr <IInternalSessionControl> directControl;
8115 {
8116 AutoReaderLock alock (this);
8117 directControl = mData->mSession.mDirectControl;
8118 }
8119
8120 /* ignore notifications sent after #OnSessionEnd() is called */
8121 if (!directControl)
8122 return S_OK;
8123
8124 return directControl->OnVRDPServerChange();
8125}
8126
8127/**
8128 * @note Locks this object for reading.
8129 */
8130HRESULT SessionMachine::onUSBControllerChange()
8131{
8132 LogFlowThisFunc (("\n"));
8133
8134 AutoCaller autoCaller (this);
8135 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8136
8137 ComPtr <IInternalSessionControl> directControl;
8138 {
8139 AutoReaderLock alock (this);
8140 directControl = mData->mSession.mDirectControl;
8141 }
8142
8143 /* ignore notifications sent after #OnSessionEnd() is called */
8144 if (!directControl)
8145 return S_OK;
8146
8147 return directControl->OnUSBControllerChange();
8148}
8149
8150/**
8151 * @note Locks this object for reading.
8152 */
8153HRESULT SessionMachine::onUSBDeviceAttach (IUSBDevice *aDevice)
8154{
8155 LogFlowThisFunc (("\n"));
8156
8157 AutoCaller autoCaller (this);
8158 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8159
8160 ComPtr <IInternalSessionControl> directControl;
8161 {
8162 AutoReaderLock alock (this);
8163 directControl = mData->mSession.mDirectControl;
8164 }
8165
8166 /* ignore notifications sent after #OnSessionEnd() is called */
8167 if (!directControl)
8168 return S_OK;
8169
8170 return directControl->OnUSBDeviceAttach (aDevice);
8171}
8172
8173/**
8174 * @note Locks this object for reading.
8175 */
8176HRESULT SessionMachine::onUSBDeviceDetach (INPTR GUIDPARAM aId)
8177{
8178 LogFlowThisFunc (("\n"));
8179
8180 AutoCaller autoCaller (this);
8181 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8182
8183 ComPtr <IInternalSessionControl> directControl;
8184 {
8185 AutoReaderLock alock (this);
8186 directControl = mData->mSession.mDirectControl;
8187 }
8188
8189 /* ignore notifications sent after #OnSessionEnd() is called */
8190 if (!directControl)
8191 return S_OK;
8192
8193 return directControl->OnUSBDeviceDetach (aId);
8194}
8195
8196// protected methods
8197/////////////////////////////////////////////////////////////////////////////
8198
8199/**
8200 * Helper method to finalize saving the state.
8201 *
8202 * @note Must be called from under this object's lock.
8203 *
8204 * @param aSuccess TRUE if the snapshot has been taken successfully
8205 *
8206 * @note Locks mParent + this objects for writing.
8207 */
8208HRESULT SessionMachine::endSavingState (BOOL aSuccess)
8209{
8210 LogFlowThisFuncEnter();
8211
8212 AutoCaller autoCaller (this);
8213 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8214
8215 /* mParent->removeProgress() needs mParent lock */
8216 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
8217
8218 HRESULT rc = S_OK;
8219
8220 if (aSuccess)
8221 {
8222 mSSData->mStateFilePath = mSnapshotData.mStateFilePath;
8223
8224 /* save all VM settings */
8225 rc = saveSettings();
8226 }
8227 else
8228 {
8229 /* delete the saved state file (it might have been already created) */
8230 RTFileDelete (Utf8Str (mSnapshotData.mStateFilePath));
8231 }
8232
8233 /* remove the completed progress object */
8234 mParent->removeProgress (mSnapshotData.mProgressId);
8235
8236 /* clear out the temporary saved state data */
8237 mSnapshotData.mLastState = MachineState_InvalidMachineState;
8238 mSnapshotData.mProgressId.clear();
8239 mSnapshotData.mStateFilePath.setNull();
8240
8241 LogFlowThisFuncLeave();
8242 return rc;
8243}
8244
8245/**
8246 * Helper method to finalize taking a snapshot.
8247 * Gets called only from #EndTakingSnapshot() that is expected to
8248 * be called by the VM process when it finishes *all* the tasks related to
8249 * taking a snapshot, either scucessfully or unsuccessfilly.
8250 *
8251 * @param aSuccess TRUE if the snapshot has been taken successfully
8252 *
8253 * @note Locks mParent + this objects for writing.
8254 */
8255HRESULT SessionMachine::endTakingSnapshot (BOOL aSuccess)
8256{
8257 LogFlowThisFuncEnter();
8258
8259 AutoCaller autoCaller (this);
8260 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8261
8262 /* Progress object uninitialization needs mParent lock */
8263 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
8264
8265 HRESULT rc = S_OK;
8266
8267 if (aSuccess)
8268 {
8269 /* the server progress must be completed on success */
8270 Assert (mSnapshotData.mServerProgress->completed());
8271
8272 mData->mCurrentSnapshot = mSnapshotData.mSnapshot;
8273 /* memorize the first snapshot if necessary */
8274 if (!mData->mFirstSnapshot)
8275 mData->mFirstSnapshot = mData->mCurrentSnapshot;
8276
8277 int opFlags = SaveSS_AddOp | SaveSS_UpdateCurrentId;
8278 if (mSnapshotData.mLastState != MachineState_Paused && !isModified())
8279 {
8280 /*
8281 * the machine was powered off or saved when taking a snapshot,
8282 * so reset the mCurrentStateModified flag
8283 */
8284 mData->mCurrentStateModified = FALSE;
8285 opFlags |= SaveSS_UpdateCurStateModified;
8286 }
8287
8288 rc = saveSnapshotSettings (mSnapshotData.mSnapshot, opFlags);
8289 }
8290
8291 if (!aSuccess || FAILED (rc))
8292 {
8293 if (mSnapshotData.mSnapshot)
8294 {
8295 /* wait for the completion of the server progress (diff VDI creation) */
8296 /// @todo (dmik) later, we will definitely want to cancel it instead
8297 // (when the cancel function is implemented)
8298 mSnapshotData.mServerProgress->WaitForCompletion (-1);
8299
8300 /*
8301 * delete all differencing VDIs created
8302 * (this will attach their parents back)
8303 */
8304 rc = deleteSnapshotDiffs (mSnapshotData.mSnapshot);
8305 /* continue cleanup on error */
8306
8307 /* delete the saved state file (it might have been already created) */
8308 if (mSnapshotData.mSnapshot->stateFilePath())
8309 RTFileDelete (Utf8Str (mSnapshotData.mSnapshot->stateFilePath()));
8310
8311 mSnapshotData.mSnapshot->uninit();
8312 }
8313 }
8314
8315 /* inform callbacks */
8316 if (aSuccess && SUCCEEDED (rc))
8317 mParent->onSnapshotTaken (mData->mUuid, mSnapshotData.mSnapshot->data().mId);
8318
8319 /* clear out the snapshot data */
8320 mSnapshotData.mLastState = MachineState_InvalidMachineState;
8321 mSnapshotData.mSnapshot.setNull();
8322 mSnapshotData.mServerProgress.setNull();
8323 /* uninitialize the combined progress (to remove it from the VBox collection) */
8324 if (!mSnapshotData.mCombinedProgress.isNull())
8325 {
8326 mSnapshotData.mCombinedProgress->uninit();
8327 mSnapshotData.mCombinedProgress.setNull();
8328 }
8329
8330 LogFlowThisFuncLeave();
8331 return rc;
8332}
8333
8334/**
8335 * Take snapshot task handler.
8336 * Must be called only by TakeSnapshotTask::handler()!
8337 *
8338 * The sole purpose of this task is to asynchronously create differencing VDIs
8339 * and copy the saved state file (when necessary). The VM process will wait
8340 * for this task to complete using the mSnapshotData.mServerProgress
8341 * returned to it.
8342 *
8343 * @note Locks mParent + this objects for writing.
8344 */
8345void SessionMachine::takeSnapshotHandler (TakeSnapshotTask &aTask)
8346{
8347 LogFlowThisFuncEnter();
8348
8349 AutoCaller autoCaller (this);
8350
8351 LogFlowThisFunc (("state=%d\n", autoCaller.state()));
8352 if (!autoCaller.isOk())
8353 {
8354 /*
8355 * we might have been uninitialized because the session was
8356 * accidentally closed by the client, so don't assert
8357 */
8358 LogFlowThisFuncLeave();
8359 return;
8360 }
8361
8362 /* endTakingSnapshot() needs mParent lock */
8363 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
8364
8365 HRESULT rc = S_OK;
8366
8367 LogFlowThisFunc (("Creating differencing VDIs...\n"));
8368
8369 /* create new differencing hard disks and attach them to this machine */
8370 rc = createSnapshotDiffs (&mSnapshotData.mSnapshot->data().mId,
8371 mUserData->mSnapshotFolderFull,
8372 mSnapshotData.mServerProgress,
8373 true /* aOnline */);
8374
8375 if (SUCCEEDED (rc) && mSnapshotData.mLastState == MachineState_Saved)
8376 {
8377 Utf8Str stateFrom = mSSData->mStateFilePath;
8378 Utf8Str stateTo = mSnapshotData.mSnapshot->stateFilePath();
8379
8380 LogFlowThisFunc (("Copying the execution state from '%s' to '%s'...\n",
8381 stateFrom.raw(), stateTo.raw()));
8382
8383 mSnapshotData.mServerProgress->advanceOperation (
8384 Bstr (tr ("Copying the execution state")));
8385
8386 /*
8387 * We can safely leave the lock here:
8388 * mMachineState is MachineState_Saving here
8389 */
8390 alock.leave();
8391
8392 /* copy the state file */
8393 int vrc = RTFileCopyEx (stateFrom, stateTo, progressCallback,
8394 static_cast <Progress *> (mSnapshotData.mServerProgress));
8395
8396 alock.enter();
8397
8398 if (VBOX_FAILURE (vrc))
8399 rc = setError (E_FAIL,
8400 tr ("Could not copy the state file '%ls' to '%ls' (%Vrc)"),
8401 stateFrom.raw(), stateTo.raw());
8402 }
8403
8404 /*
8405 * we have to call endTakingSnapshot() here if the snapshot was taken
8406 * offline, because the VM process will not do it in this case
8407 */
8408 if (mSnapshotData.mLastState != MachineState_Paused)
8409 {
8410 LogFlowThisFunc (("Finalizing the taken snapshot (rc=%08X)...\n", rc));
8411
8412 setMachineState (mSnapshotData.mLastState);
8413 updateMachineStateOnClient();
8414
8415 /* finalize the progress after setting the state, for consistency */
8416 mSnapshotData.mServerProgress->notifyComplete (rc);
8417
8418 endTakingSnapshot (SUCCEEDED (rc));
8419 }
8420 else
8421 {
8422 mSnapshotData.mServerProgress->notifyComplete (rc);
8423 }
8424
8425 LogFlowThisFuncLeave();
8426}
8427
8428/**
8429 * Discard snapshot task handler.
8430 * Must be called only by DiscardSnapshotTask::handler()!
8431 *
8432 * When aTask.subTask is true, the associated progress object is left
8433 * uncompleted on success. On failure, the progress is marked as completed
8434 * regardless of this parameter.
8435 *
8436 * @note Locks mParent + this + child objects for writing!
8437 */
8438void SessionMachine::discardSnapshotHandler (DiscardSnapshotTask &aTask)
8439{
8440 LogFlowThisFuncEnter();
8441
8442 AutoCaller autoCaller (this);
8443
8444 LogFlowThisFunc (("state=%d\n", autoCaller.state()));
8445 if (!autoCaller.isOk())
8446 {
8447 /*
8448 * we might have been uninitialized because the session was
8449 * accidentally closed by the client, so don't assert
8450 */
8451 aTask.progress->notifyComplete (
8452 E_FAIL, COM_IIDOF (IMachine), getComponentName(),
8453 tr ("The session has been accidentally closed"));
8454
8455 LogFlowThisFuncLeave();
8456 return;
8457 }
8458
8459 ComObjPtr <SnapshotMachine> sm = aTask.snapshot->data().mMachine;
8460
8461 /* mParent is locked because of Progress::notifyComplete(), etc. */
8462 AutoMultiLock <3> alock (mParent->wlock(), this->wlock(), sm->rlock());
8463
8464 /* Safe locking in the direction parent->child */
8465 AutoLock snapshotLock (aTask.snapshot);
8466 AutoLock snapshotChildrenLock (aTask.snapshot->childrenLock());
8467
8468 HRESULT rc = S_OK;
8469
8470 /* save the snapshot ID (for callbacks) */
8471 Guid snapshotId = aTask.snapshot->data().mId;
8472
8473 do
8474 {
8475 /* first pass: */
8476 LogFlowThisFunc (("Check hard disk accessibility and affected machines...\n"));
8477
8478 HDData::HDAttachmentList::const_iterator it;
8479 for (it = sm->mHDData->mHDAttachments.begin();
8480 it != sm->mHDData->mHDAttachments.end();
8481 ++ it)
8482 {
8483 ComObjPtr <HardDiskAttachment> hda = *it;
8484 ComObjPtr <HardDisk> hd = hda->hardDisk();
8485 ComObjPtr <HardDisk> parent = hd->parent();
8486
8487 AutoLock hdLock (hd);
8488
8489 if (hd->hasForeignChildren())
8490 {
8491 rc = setError (E_FAIL,
8492 tr ("One or more hard disks belonging to other machines are "
8493 "based on the hard disk '%ls' stored in the snapshot '%ls'"),
8494 hd->toString().raw(), aTask.snapshot->data().mName.raw());
8495 break;
8496 }
8497
8498 if (hd->type() == HardDiskType_NormalHardDisk)
8499 {
8500 AutoLock hdChildrenLock (hd->childrenLock());
8501 size_t childrenCount = hd->children().size();
8502 if (childrenCount > 1)
8503 {
8504 rc = setError (E_FAIL,
8505 tr ("Normal hard disk '%ls' stored in the snapshot '%ls' "
8506 "has more than one child hard disk (%d)"),
8507 hd->toString().raw(), aTask.snapshot->data().mName.raw(),
8508 childrenCount);
8509 break;
8510 }
8511 }
8512 else
8513 {
8514 ComAssertMsgFailedBreak (("Invalid hard disk type %d\n", hd->type()),
8515 rc = E_FAIL);
8516 }
8517
8518 Bstr accessError;
8519 rc = hd->getAccessibleWithChildren (accessError);
8520 CheckComRCBreakRC (rc);
8521
8522 if (!accessError.isNull())
8523 {
8524 rc = setError (E_FAIL,
8525 tr ("Hard disk '%ls' stored in the snapshot '%ls' is not "
8526 "accessible (%ls)"),
8527 hd->toString().raw(), aTask.snapshot->data().mName.raw(),
8528 accessError.raw());
8529 break;
8530 }
8531
8532 rc = hd->setBusyWithChildren();
8533 if (FAILED (rc))
8534 {
8535 /* reset the busy flag of all previous hard disks */
8536 while (it != sm->mHDData->mHDAttachments.begin())
8537 (*(-- it))->hardDisk()->clearBusyWithChildren();
8538 break;
8539 }
8540 }
8541
8542 CheckComRCBreakRC (rc);
8543
8544 /* second pass: */
8545 LogFlowThisFunc (("Performing actual vdi merging...\n"));
8546
8547 for (it = sm->mHDData->mHDAttachments.begin();
8548 it != sm->mHDData->mHDAttachments.end();
8549 ++ it)
8550 {
8551 ComObjPtr <HardDiskAttachment> hda = *it;
8552 ComObjPtr <HardDisk> hd = hda->hardDisk();
8553 ComObjPtr <HardDisk> parent = hd->parent();
8554
8555 AutoLock hdLock (hd);
8556
8557 Bstr hdRootString = hd->root()->toString (true /* aShort */);
8558
8559 if (parent)
8560 {
8561 if (hd->isParentImmutable())
8562 {
8563 aTask.progress->advanceOperation (Bstr (Utf8StrFmt (
8564 tr ("Discarding changes to immutable hard disk '%ls'"),
8565 hdRootString.raw())));
8566
8567 /* clear the busy flag before unregistering */
8568 hd->clearBusy();
8569
8570 /*
8571 * unregisterDiffHardDisk() is supposed to delete and uninit
8572 * the differencing hard disk
8573 */
8574 rc = mParent->unregisterDiffHardDisk (hd);
8575 CheckComRCBreakRC (rc);
8576 continue;
8577 }
8578 else
8579 {
8580 /*
8581 * differencing VDI:
8582 * merge this image to all its children
8583 */
8584
8585 aTask.progress->advanceOperation (Bstr (Utf8StrFmt (
8586 tr ("Merging changes to normal hard disk '%ls' to children"),
8587 hdRootString.raw())));
8588
8589 snapshotChildrenLock.unlock();
8590 snapshotLock.unlock();
8591 alock.leave();
8592
8593 rc = hd->asVDI()->mergeImageToChildren (aTask.progress);
8594
8595 alock.enter();
8596 snapshotLock.lock();
8597 snapshotChildrenLock.lock();
8598
8599 // debug code
8600 // if (it != sm->mHDData->mHDAttachments.begin())
8601 // {
8602 // rc = setError (E_FAIL, "Simulated failure");
8603 // break;
8604 //}
8605
8606 if (SUCCEEDED (rc))
8607 rc = mParent->unregisterDiffHardDisk (hd);
8608 else
8609 hd->clearBusyWithChildren();
8610
8611 CheckComRCBreakRC (rc);
8612 }
8613 }
8614 else if (hd->type() == HardDiskType_NormalHardDisk)
8615 {
8616 /*
8617 * normal vdi has the only child or none
8618 * (checked in the first pass)
8619 */
8620
8621 ComObjPtr <HardDisk> child;
8622 {
8623 AutoLock hdChildrenLock (hd->childrenLock());
8624 if (hd->children().size())
8625 child = hd->children().front();
8626 }
8627
8628 if (child.isNull())
8629 {
8630 aTask.progress->advanceOperation (Bstr (Utf8StrFmt (
8631 tr ("Detaching normal hard disk '%ls'"),
8632 hdRootString.raw())));
8633
8634 /* just deassociate the normal image from this machine */
8635 hd->setMachineId (Guid());
8636 hd->setSnapshotId (Guid());
8637
8638 /* clear the busy flag */
8639 hd->clearBusy();
8640 }
8641 else
8642 {
8643 AutoLock childLock (child);
8644
8645 aTask.progress->advanceOperation (Bstr (Utf8StrFmt (
8646 tr ("Preserving changes to normal hard disk '%ls'"),
8647 hdRootString.raw())));
8648
8649 ComObjPtr <Machine> cm;
8650 ComObjPtr <Snapshot> cs;
8651 ComObjPtr <HardDiskAttachment> childHda;
8652 rc = findHardDiskAttachment (child, &cm, &cs, &childHda);
8653 CheckComRCBreakRC (rc);
8654 /* must be the same machine (checked in the first pass) */
8655 ComAssertBreak (cm->mData->mUuid == mData->mUuid, rc = E_FAIL);
8656
8657 /* merge the child to this basic image */
8658
8659 snapshotChildrenLock.unlock();
8660 snapshotLock.unlock();
8661 alock.leave();
8662
8663 rc = child->asVDI()->mergeImageToParent (aTask.progress);
8664
8665 alock.enter();
8666 snapshotLock.lock();
8667 snapshotChildrenLock.lock();
8668
8669 if (SUCCEEDED (rc))
8670 rc = mParent->unregisterDiffHardDisk (child);
8671 else
8672 hd->clearBusyWithChildren();
8673
8674 CheckComRCBreakRC (rc);
8675
8676 /* replace the child image in the appropriate place */
8677 childHda->updateHardDisk (hd, FALSE /* aDirty */);
8678
8679 if (!cs)
8680 {
8681 aTask.settingsChanged = true;
8682 }
8683 else
8684 {
8685 rc = cm->saveSnapshotSettings (cs, SaveSS_UpdateAllOp);
8686 CheckComRCBreakRC (rc);
8687 }
8688 }
8689 }
8690 else
8691 {
8692 ComAssertMsgFailedBreak (("Invalid hard disk type %d\n", hd->type()),
8693 rc = E_FAIL);
8694 }
8695 }
8696
8697 /* fetch the current error info */
8698 ErrorInfo mergeEi;
8699 HRESULT mergeRc = rc;
8700
8701 if (FAILED (rc))
8702 {
8703 /* clear the busy flag on the rest of hard disks */
8704 for (++ it; it != sm->mHDData->mHDAttachments.end(); ++ it)
8705 (*it)->hardDisk()->clearBusyWithChildren();
8706 }
8707
8708 /*
8709 * we have to try to discard the snapshot even if merging failed
8710 * because some images might have been already merged (and deleted)
8711 */
8712
8713 do
8714 {
8715 LogFlowThisFunc (("Discarding the snapshot (reparenting children)...\n"));
8716
8717 ComObjPtr <Snapshot> parentSnapshot = aTask.snapshot->parent();
8718
8719 /// @todo (dmik):
8720 // when we introduce clones later, discarding the snapshot
8721 // will affect the current and first snapshots of clones, if they are
8722 // direct children of this snapshot. So we will need to lock machines
8723 // associated with child snapshots as well and update mCurrentSnapshot
8724 // and/or mFirstSnapshot fields.
8725
8726 if (aTask.snapshot == mData->mCurrentSnapshot)
8727 {
8728 /* currently, the parent snapshot must refer to the same machine */
8729 ComAssertBreak (
8730 !parentSnapshot ||
8731 parentSnapshot->data().mMachine->mData->mUuid == mData->mUuid,
8732 rc = E_FAIL);
8733 mData->mCurrentSnapshot = parentSnapshot;
8734 /* mark the current state as modified */
8735 mData->mCurrentStateModified = TRUE;
8736 }
8737
8738 if (aTask.snapshot == mData->mFirstSnapshot)
8739 {
8740 /*
8741 * the first snapshot must have only one child when discarded,
8742 * or no children at all
8743 */
8744 ComAssertBreak (aTask.snapshot->children().size() <= 1, rc = E_FAIL);
8745
8746 if (aTask.snapshot->children().size() == 1)
8747 {
8748 ComObjPtr <Snapshot> childSnapshot = aTask.snapshot->children().front();
8749 ComAssertBreak (
8750 childSnapshot->data().mMachine->mData->mUuid == mData->mUuid,
8751 rc = E_FAIL);
8752 mData->mFirstSnapshot = childSnapshot;
8753 }
8754 else
8755 mData->mFirstSnapshot.setNull();
8756 }
8757
8758 /// @todo (dmik)
8759 // if we implement some warning mechanism later, we'll have
8760 // to return a warning if the state file path cannot be deleted
8761 Bstr stateFilePath = aTask.snapshot->stateFilePath();
8762 if (stateFilePath)
8763 RTFileDelete (Utf8Str (stateFilePath));
8764
8765 aTask.snapshot->discard();
8766
8767 rc = saveSnapshotSettings (parentSnapshot,
8768 SaveSS_UpdateAllOp | SaveSS_UpdateCurrentId);
8769 }
8770 while (0);
8771
8772 /* restore the merge error if any */
8773 if (FAILED (mergeRc))
8774 {
8775 rc = mergeRc;
8776 setError (mergeEi);
8777 }
8778 }
8779 while (0);
8780
8781 if (!aTask.subTask || FAILED (rc))
8782 {
8783 if (!aTask.subTask)
8784 {
8785 /* save current error info */
8786 ErrorInfo ei;
8787
8788 /* restore the machine state */
8789 setMachineState (aTask.state);
8790 updateMachineStateOnClient();
8791
8792 /*
8793 * save settings anyway, since we've already changed the current
8794 * machine configuration
8795 */
8796 if (aTask.settingsChanged)
8797 {
8798 saveSettings (true /* aMarkCurStateAsModified */,
8799 true /* aInformCallbacksAnyway */);
8800 }
8801
8802 /* restore current error info */
8803 setError (ei);
8804 }
8805
8806 /* set the result (this will try to fetch current error info on failure) */
8807 aTask.progress->notifyComplete (rc);
8808 }
8809
8810 if (SUCCEEDED (rc))
8811 mParent->onSnapshotDiscarded (mData->mUuid, snapshotId);
8812
8813 LogFlowThisFunc (("Done discarding snapshot (rc=%08X)\n", rc));
8814 LogFlowThisFuncLeave();
8815}
8816
8817/**
8818 * Discard current state task handler.
8819 * Must be called only by DiscardCurrentStateTask::handler()!
8820 *
8821 * @note Locks mParent + this object for writing.
8822 */
8823void SessionMachine::discardCurrentStateHandler (DiscardCurrentStateTask &aTask)
8824{
8825 LogFlowThisFuncEnter();
8826
8827 AutoCaller autoCaller (this);
8828
8829 LogFlowThisFunc (("state=%d\n", autoCaller.state()));
8830 if (!autoCaller.isOk())
8831 {
8832 /*
8833 * we might have been uninitialized because the session was
8834 * accidentally closed by the client, so don't assert
8835 */
8836 aTask.progress->notifyComplete (
8837 E_FAIL, COM_IIDOF (IMachine), getComponentName(),
8838 tr ("The session has been accidentally closed"));
8839
8840 LogFlowThisFuncLeave();
8841 return;
8842 }
8843
8844 /* mParent is locked because of Progress::notifyComplete(), etc. */
8845 AutoMultiLock <2> alock (mParent->wlock(), this->wlock());
8846
8847 /*
8848 * discard all current changes to mUserData (name, OSType etc.)
8849 * (note that the machine is powered off, so there is no need
8850 * to inform the direct session)
8851 */
8852 if (isModified())
8853 rollback (false /* aNotify */);
8854
8855 HRESULT rc = S_OK;
8856
8857 bool errorInSubtask = false;
8858 bool stateRestored = false;
8859
8860 const bool isLastSnapshot = mData->mCurrentSnapshot->parent().isNull();
8861
8862 do
8863 {
8864 /*
8865 * discard the saved state file if the machine was Saved prior
8866 * to this operation
8867 */
8868 if (aTask.state == MachineState_Saved)
8869 {
8870 Assert (!mSSData->mStateFilePath.isEmpty());
8871 RTFileDelete (Utf8Str (mSSData->mStateFilePath));
8872 mSSData->mStateFilePath.setNull();
8873 aTask.modifyLastState (MachineState_PoweredOff);
8874 rc = saveStateSettings (SaveSTS_StateFilePath);
8875 CheckComRCBreakRC (rc);
8876 }
8877
8878 if (aTask.discardCurrentSnapshot && !isLastSnapshot)
8879 {
8880 /*
8881 * the "discard current snapshot and state" task is in action,
8882 * the current snapshot is not the last one.
8883 * Discard the current snapshot first.
8884 */
8885
8886 DiscardSnapshotTask subTask (aTask, mData->mCurrentSnapshot);
8887 subTask.subTask = true;
8888 discardSnapshotHandler (subTask);
8889 aTask.settingsChanged = subTask.settingsChanged;
8890 if (aTask.progress->completed())
8891 {
8892 /*
8893 * the progress can be completed by a subtask only if there was
8894 * a failure
8895 */
8896 Assert (FAILED (aTask.progress->resultCode()));
8897 errorInSubtask = true;
8898 rc = aTask.progress->resultCode();
8899 break;
8900 }
8901 }
8902
8903 LONG64 snapshotTimeStamp = 0;
8904
8905 {
8906 ComObjPtr <Snapshot> curSnapshot = mData->mCurrentSnapshot;
8907 AutoLock snapshotLock (curSnapshot);
8908
8909 /* remember the timestamp of the snapshot we're restoring from */
8910 snapshotTimeStamp = curSnapshot->data().mTimeStamp;
8911
8912 /* copy all hardware data from the current snapshot */
8913 copyFrom (curSnapshot->data().mMachine);
8914
8915 LogFlowThisFunc (("Restoring VDIs from the snapshot...\n"));
8916
8917 /* restore the attachmends from the snapshot */
8918 mHDData.backup();
8919 mHDData->mHDAttachments =
8920 curSnapshot->data().mMachine->mHDData->mHDAttachments;
8921
8922 snapshotLock.unlock();
8923 alock.leave();
8924 rc = createSnapshotDiffs (NULL, mUserData->mSnapshotFolderFull,
8925 aTask.progress,
8926 false /* aOnline */);
8927 alock.enter();
8928 snapshotLock.lock();
8929
8930 if (FAILED (rc))
8931 {
8932 /* here we can still safely rollback, so do it */
8933 ErrorInfo ei;
8934 /* undo all changes */
8935 rollback (false /* aNotify */);
8936 setError (ei);
8937 break;
8938 }
8939
8940 /*
8941 * note: old VDIs will be deassociated/deleted on #commit() called
8942 * either from #saveSettings() or directly at the end
8943 */
8944
8945 /* should not have a saved state file associated at this point */
8946 Assert (mSSData->mStateFilePath.isNull());
8947
8948 if (curSnapshot->stateFilePath())
8949 {
8950 Utf8Str snapStateFilePath = curSnapshot->stateFilePath();
8951
8952 Utf8Str stateFilePath = Utf8StrFmt ("%ls%c{%Vuuid}.sav",
8953 mUserData->mSnapshotFolderFull.raw(),
8954 RTPATH_DELIMITER, mData->mUuid.raw());
8955
8956 LogFlowThisFunc (("Copying saved state file from '%s' to '%s'...\n",
8957 snapStateFilePath.raw(), stateFilePath.raw()));
8958
8959 aTask.progress->advanceOperation (
8960 Bstr (tr ("Restoring the execution state")));
8961
8962 /* copy the state file */
8963 snapshotLock.unlock();
8964 alock.leave();
8965 int vrc = RTFileCopyEx (snapStateFilePath, stateFilePath,
8966 progressCallback, aTask.progress);
8967 alock.enter();
8968 snapshotLock.lock();
8969
8970 if (VBOX_SUCCESS (vrc))
8971 {
8972 mSSData->mStateFilePath = stateFilePath;
8973 }
8974 else
8975 {
8976 rc = setError (E_FAIL,
8977 tr ("Could not copy the state file '%s' to '%s' (%Vrc)"),
8978 snapStateFilePath.raw(), stateFilePath.raw(), vrc);
8979 break;
8980 }
8981 }
8982 }
8983
8984 bool informCallbacks = false;
8985
8986 if (aTask.discardCurrentSnapshot && isLastSnapshot)
8987 {
8988 /*
8989 * discard the current snapshot and state task is in action,
8990 * the current snapshot is the last one.
8991 * Discard the current snapshot after discarding the current state.
8992 */
8993
8994 /* commit changes to fixup hard disks before discarding */
8995 rc = commit();
8996 if (SUCCEEDED (rc))
8997 {
8998 DiscardSnapshotTask subTask (aTask, mData->mCurrentSnapshot);
8999 subTask.subTask = true;
9000 discardSnapshotHandler (subTask);
9001 aTask.settingsChanged = subTask.settingsChanged;
9002 if (aTask.progress->completed())
9003 {
9004 /*
9005 * the progress can be completed by a subtask only if there
9006 * was a failure
9007 */
9008 Assert (FAILED (aTask.progress->resultCode()));
9009 errorInSubtask = true;
9010 rc = aTask.progress->resultCode();
9011 }
9012 }
9013
9014 /*
9015 * we've committed already, so inform callbacks anyway to ensure
9016 * they don't miss some change
9017 */
9018 informCallbacks = true;
9019 }
9020
9021 /*
9022 * we have already discarded the current state, so set the
9023 * execution state accordingly no matter of the discard snapshot result
9024 */
9025 if (mSSData->mStateFilePath)
9026 setMachineState (MachineState_Saved);
9027 else
9028 setMachineState (MachineState_PoweredOff);
9029
9030 updateMachineStateOnClient();
9031 stateRestored = true;
9032
9033 if (errorInSubtask)
9034 break;
9035
9036 /* assign the timestamp from the snapshot */
9037 Assert (snapshotTimeStamp != 0);
9038 mData->mLastStateChange = snapshotTimeStamp;
9039
9040 /* mark the current state as not modified */
9041 mData->mCurrentStateModified = FALSE;
9042
9043 /* save all settings and commit */
9044 rc = saveSettings (false /* aMarkCurStateAsModified */,
9045 informCallbacks);
9046 aTask.settingsChanged = false;
9047 }
9048 while (0);
9049
9050 if (FAILED (rc))
9051 {
9052 ErrorInfo ei;
9053
9054 if (!stateRestored)
9055 {
9056 /* restore the machine state */
9057 setMachineState (aTask.state);
9058 updateMachineStateOnClient();
9059 }
9060
9061 /*
9062 * save all settings and commit if still modified (there is no way to
9063 * rollback properly). Note that isModified() will return true after
9064 * copyFrom(). Also save the settings if requested by the subtask.
9065 */
9066 if (isModified() || aTask.settingsChanged)
9067 {
9068 if (aTask.settingsChanged)
9069 saveSettings (true /* aMarkCurStateAsModified */,
9070 true /* aInformCallbacksAnyway */);
9071 else
9072 saveSettings();
9073 }
9074
9075 setError (ei);
9076 }
9077
9078 if (!errorInSubtask)
9079 {
9080 /* set the result (this will try to fetch current error info on failure) */
9081 aTask.progress->notifyComplete (rc);
9082 }
9083
9084 if (SUCCEEDED (rc))
9085 mParent->onSnapshotDiscarded (mData->mUuid, Guid());
9086
9087 LogFlowThisFunc (("Done discarding current state (rc=%08X)\n", rc));
9088
9089 LogFlowThisFuncLeave();
9090}
9091
9092/**
9093 * Helper to change the machine state (reimplementation).
9094 *
9095 * @note Locks this object for writing.
9096 */
9097HRESULT SessionMachine::setMachineState (MachineState_T aMachineState)
9098{
9099 LogFlowThisFuncEnter();
9100 LogFlowThisFunc (("aMachineState=%d\n", aMachineState));
9101
9102 AutoCaller autoCaller (this);
9103 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9104
9105 AutoLock alock (this);
9106
9107 MachineState_T oldMachineState = mData->mMachineState;
9108
9109 AssertMsgReturn (oldMachineState != aMachineState,
9110 ("oldMachineState=%d, aMachineState=%d\n",
9111 oldMachineState, aMachineState), E_FAIL);
9112
9113 HRESULT rc = S_OK;
9114
9115 int stsFlags = 0;
9116 bool deleteSavedState = false;
9117
9118 /* detect some state transitions */
9119
9120 if (oldMachineState < MachineState_Running &&
9121 aMachineState >= MachineState_Running &&
9122 aMachineState != MachineState_Discarding)
9123 {
9124 /*
9125 * the EMT thread is about to start, so mark attached HDDs as busy
9126 * and all its ancestors as being in use
9127 */
9128 for (HDData::HDAttachmentList::const_iterator it =
9129 mHDData->mHDAttachments.begin();
9130 it != mHDData->mHDAttachments.end();
9131 ++ it)
9132 {
9133 ComObjPtr <HardDisk> hd = (*it)->hardDisk();
9134 AutoLock hdLock (hd);
9135 hd->setBusy();
9136 hd->addReaderOnAncestors();
9137 }
9138 }
9139 else
9140 if (oldMachineState >= MachineState_Running &&
9141 oldMachineState != MachineState_Discarding &&
9142 aMachineState < MachineState_Running)
9143 {
9144 /*
9145 * the EMT thread stopped, so mark attached HDDs as no more busy
9146 * and remove the in-use flag from all its ancestors
9147 */
9148 for (HDData::HDAttachmentList::const_iterator it =
9149 mHDData->mHDAttachments.begin();
9150 it != mHDData->mHDAttachments.end();
9151 ++ it)
9152 {
9153 ComObjPtr <HardDisk> hd = (*it)->hardDisk();
9154 AutoLock hdLock (hd);
9155 hd->releaseReaderOnAncestors();
9156 hd->clearBusy();
9157 }
9158 }
9159
9160 if (oldMachineState == MachineState_Restoring)
9161 {
9162 if (aMachineState != MachineState_Saved)
9163 {
9164 /*
9165 * delete the saved state file once the machine has finished
9166 * restoring from it (note that Console sets the state from
9167 * Restoring to Saved if the VM couldn't restore successfully,
9168 * to give the user an ability to fix an error and retry --
9169 * we keep the saved state file in this case)
9170 */
9171 deleteSavedState = true;
9172 }
9173 }
9174 else
9175 if (oldMachineState == MachineState_Saved &&
9176 (aMachineState == MachineState_PoweredOff ||
9177 aMachineState == MachineState_Aborted))
9178 {
9179 /*
9180 * delete the saved state after Console::DiscardSavedState() is called
9181 * or if the VM process (owning a direct VM session) crashed while the
9182 * VM was Saved
9183 */
9184
9185 /// @todo (dmik)
9186 // Not sure that deleting the saved state file just because of the
9187 // client death before it attempted to restore the VM is a good
9188 // thing. But when it crashes we need to go to the Aborted state
9189 // which cannot have the saved state file associated... The only
9190 // way to fix this is to make the Aborted condition not a VM state
9191 // but a bool flag: i.e., when a crash occurs, set it to true and
9192 // change the state to PoweredOff or Saved depending on the
9193 // saved state presence.
9194
9195 deleteSavedState = true;
9196 mData->mCurrentStateModified = TRUE;
9197 stsFlags |= SaveSTS_CurStateModified;
9198 }
9199
9200 if (aMachineState == MachineState_Starting ||
9201 aMachineState == MachineState_Restoring)
9202 {
9203 /*
9204 * set the current state modified flag to indicate that the
9205 * current state is no more identical to the state in the
9206 * current snapshot
9207 */
9208 if (!mData->mCurrentSnapshot.isNull())
9209 {
9210 mData->mCurrentStateModified = TRUE;
9211 stsFlags |= SaveSTS_CurStateModified;
9212 }
9213 }
9214
9215 if (deleteSavedState == true)
9216 {
9217 Assert (!mSSData->mStateFilePath.isEmpty());
9218 RTFileDelete (Utf8Str (mSSData->mStateFilePath));
9219 mSSData->mStateFilePath.setNull();
9220 stsFlags |= SaveSTS_StateFilePath;
9221 }
9222
9223 /* redirect to the underlying peer machine */
9224 mPeer->setMachineState (aMachineState);
9225
9226 if (aMachineState == MachineState_PoweredOff ||
9227 aMachineState == MachineState_Aborted ||
9228 aMachineState == MachineState_Saved)
9229 {
9230 stsFlags |= SaveSTS_StateTimeStamp;
9231 }
9232
9233 rc = saveStateSettings (stsFlags);
9234
9235 if ((oldMachineState != MachineState_PoweredOff &&
9236 oldMachineState != MachineState_Aborted) &&
9237 (aMachineState == MachineState_PoweredOff ||
9238 aMachineState == MachineState_Aborted))
9239 {
9240 /*
9241 * clear differencing hard disks based on immutable hard disks
9242 * once we've been shut down for any reason
9243 */
9244 rc = wipeOutImmutableDiffs();
9245 }
9246
9247 LogFlowThisFunc (("rc=%08X\n", rc));
9248 LogFlowThisFuncLeave();
9249 return rc;
9250}
9251
9252/**
9253 * Sends the current machine state value to the VM process.
9254 *
9255 * @note Locks this object for reading, then calls a client process.
9256 */
9257HRESULT SessionMachine::updateMachineStateOnClient()
9258{
9259 AutoCaller autoCaller (this);
9260 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9261
9262 ComPtr <IInternalSessionControl> directControl;
9263 {
9264 AutoReaderLock alock (this);
9265 AssertReturn (!!mData, E_FAIL);
9266 directControl = mData->mSession.mDirectControl;
9267
9268 /* directControl may be already set to NULL here in #OnSessionEnd()
9269 * called too early by the direct session process while there is still
9270 * some operation (like discarding the snapshot) in progress. The client
9271 * process in this case is waiting inside Session::close() for the
9272 * "end session" process object to complete, while #uninit() called by
9273 * #checkForDeath() on the Watcher thread is waiting for the pending
9274 * operation to complete. For now, we accept this inconsitent behavior
9275 * and simply do nothing here. */
9276
9277 if (mData->mSession.mState == SessionState_SessionClosing)
9278 return S_OK;
9279
9280 AssertReturn (!directControl.isNull(), E_FAIL);
9281 }
9282
9283 return directControl->UpdateMachineState (mData->mMachineState);
9284}
9285
9286/* static */
9287DECLCALLBACK(int) SessionMachine::taskHandler (RTTHREAD thread, void *pvUser)
9288{
9289 AssertReturn (pvUser, VERR_INVALID_POINTER);
9290
9291 Task *task = static_cast <Task *> (pvUser);
9292 task->handler();
9293
9294 // it's our responsibility to delete the task
9295 delete task;
9296
9297 return 0;
9298}
9299
9300/////////////////////////////////////////////////////////////////////////////
9301// SnapshotMachine class
9302/////////////////////////////////////////////////////////////////////////////
9303
9304DEFINE_EMPTY_CTOR_DTOR (SnapshotMachine)
9305
9306HRESULT SnapshotMachine::FinalConstruct()
9307{
9308 LogFlowThisFunc (("\n"));
9309
9310 /* set the proper type to indicate we're the SnapshotMachine instance */
9311 unconst (mType) = IsSnapshotMachine;
9312
9313 return S_OK;
9314}
9315
9316void SnapshotMachine::FinalRelease()
9317{
9318 LogFlowThisFunc (("\n"));
9319
9320 uninit();
9321}
9322
9323/**
9324 * Initializes the SnapshotMachine object when taking a snapshot.
9325 *
9326 * @param aSessionMachine machine to take a snapshot from
9327 * @param aSnapshotId snapshot ID of this snapshot machine
9328 * @param aStateFilePath file where the execution state will be later saved
9329 * (or NULL for the offline snapshot)
9330 *
9331 * @note Locks aSessionMachine object for reading.
9332 */
9333HRESULT SnapshotMachine::init (SessionMachine *aSessionMachine,
9334 INPTR GUIDPARAM aSnapshotId,
9335 INPTR BSTR aStateFilePath)
9336{
9337 LogFlowThisFuncEnter();
9338 LogFlowThisFunc (("mName={%ls}\n", aSessionMachine->mUserData->mName.raw()));
9339
9340 AssertReturn (aSessionMachine && !Guid (aSnapshotId).isEmpty(), E_INVALIDARG);
9341
9342 /* Enclose the state transition NotReady->InInit->Ready */
9343 AutoInitSpan autoInitSpan (this);
9344 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
9345
9346 mSnapshotId = aSnapshotId;
9347
9348 AutoReaderLock alock (aSessionMachine);
9349
9350 /* memorize the primary Machine instance (i.e. not SessionMachine!) */
9351 unconst (mPeer) = aSessionMachine->mPeer;
9352 /* share the parent pointer */
9353 unconst (mParent) = mPeer->mParent;
9354
9355 /* take the pointer to Data to share */
9356 mData.share (mPeer->mData);
9357 /*
9358 * take the pointer to UserData to share
9359 * (our UserData must always be the same as Machine's data)
9360 */
9361 mUserData.share (mPeer->mUserData);
9362 /* make a private copy of all other data (recent changes from SessionMachine) */
9363 mHWData.attachCopy (aSessionMachine->mHWData);
9364 mHDData.attachCopy (aSessionMachine->mHDData);
9365
9366 /* SSData is always unique for SnapshotMachine */
9367 mSSData.allocate();
9368 mSSData->mStateFilePath = aStateFilePath;
9369
9370 /*
9371 * create copies of all shared folders (mHWData after attiching a copy
9372 * contains just references to original objects)
9373 */
9374 for (HWData::SharedFolderList::iterator it = mHWData->mSharedFolders.begin();
9375 it != mHWData->mSharedFolders.end();
9376 ++ it)
9377 {
9378 ComObjPtr <SharedFolder> folder;
9379 folder.createObject();
9380 HRESULT rc = folder->initCopy (this, *it);
9381 CheckComRCReturnRC (rc);
9382 *it = folder;
9383 }
9384
9385 /* create all other child objects that will be immutable private copies */
9386
9387 unconst (mBIOSSettings).createObject();
9388 mBIOSSettings->initCopy (this, mPeer->mBIOSSettings);
9389
9390#ifdef VBOX_VRDP
9391 unconst (mVRDPServer).createObject();
9392 mVRDPServer->initCopy (this, mPeer->mVRDPServer);
9393#endif
9394
9395 unconst (mDVDDrive).createObject();
9396 mDVDDrive->initCopy (this, mPeer->mDVDDrive);
9397
9398 unconst (mFloppyDrive).createObject();
9399 mFloppyDrive->initCopy (this, mPeer->mFloppyDrive);
9400
9401 unconst (mAudioAdapter).createObject();
9402 mAudioAdapter->initCopy (this, mPeer->mAudioAdapter);
9403
9404 unconst (mUSBController).createObject();
9405 mUSBController->initCopy (this, mPeer->mUSBController);
9406
9407 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
9408 {
9409 unconst (mNetworkAdapters [slot]).createObject();
9410 mNetworkAdapters [slot]->initCopy (this, mPeer->mNetworkAdapters [slot]);
9411 }
9412
9413 /* Confirm a successful initialization when it's the case */
9414 autoInitSpan.setSucceeded();
9415
9416 LogFlowThisFuncLeave();
9417 return S_OK;
9418}
9419
9420/**
9421 * Initializes the SnapshotMachine object when loading from the settings file.
9422 *
9423 * @param aMachine machine the snapshot belngs to
9424 * @param aHWNode <Hardware> node
9425 * @param aHDAsNode <HardDiskAttachments> node
9426 * @param aSnapshotId snapshot ID of this snapshot machine
9427 * @param aStateFilePath file where the execution state is saved
9428 * (or NULL for the offline snapshot)
9429 *
9430 * @note Locks aMachine object for reading.
9431 */
9432HRESULT SnapshotMachine::init (Machine *aMachine, CFGNODE aHWNode, CFGNODE aHDAsNode,
9433 INPTR GUIDPARAM aSnapshotId, INPTR BSTR aStateFilePath)
9434{
9435 LogFlowThisFuncEnter();
9436 LogFlowThisFunc (("mName={%ls}\n", aMachine->mUserData->mName.raw()));
9437
9438 AssertReturn (aMachine && aHWNode && aHDAsNode && !Guid (aSnapshotId).isEmpty(),
9439 E_INVALIDARG);
9440
9441 /* Enclose the state transition NotReady->InInit->Ready */
9442 AutoInitSpan autoInitSpan (this);
9443 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
9444
9445 mSnapshotId = aSnapshotId;
9446
9447 AutoReaderLock alock (aMachine);
9448
9449 /* memorize the primary Machine instance */
9450 unconst (mPeer) = aMachine;
9451 /* share the parent pointer */
9452 unconst (mParent) = mPeer->mParent;
9453
9454 /* take the pointer to Data to share */
9455 mData.share (mPeer->mData);
9456 /*
9457 * take the pointer to UserData to share
9458 * (our UserData must always be the same as Machine's data)
9459 */
9460 mUserData.share (mPeer->mUserData);
9461 /* allocate private copies of all other data (will be loaded from settings) */
9462 mHWData.allocate();
9463 mHDData.allocate();
9464
9465 /* SSData is always unique for SnapshotMachine */
9466 mSSData.allocate();
9467 mSSData->mStateFilePath = aStateFilePath;
9468
9469 /* create all other child objects that will be immutable private copies */
9470
9471 unconst (mBIOSSettings).createObject();
9472 mBIOSSettings->init (this);
9473
9474#ifdef VBOX_VRDP
9475 unconst (mVRDPServer).createObject();
9476 mVRDPServer->init (this);
9477#endif
9478
9479 unconst (mDVDDrive).createObject();
9480 mDVDDrive->init (this);
9481
9482 unconst (mFloppyDrive).createObject();
9483 mFloppyDrive->init (this);
9484
9485 unconst (mAudioAdapter).createObject();
9486 mAudioAdapter->init (this);
9487
9488 unconst (mUSBController).createObject();
9489 mUSBController->init (this);
9490
9491 for (ULONG slot = 0; slot < ELEMENTS (mNetworkAdapters); slot ++)
9492 {
9493 unconst (mNetworkAdapters [slot]).createObject();
9494 mNetworkAdapters [slot]->init (this, slot);
9495 }
9496
9497 /* load hardware and harddisk settings */
9498
9499 HRESULT rc = loadHardware (aHWNode);
9500 if (SUCCEEDED (rc))
9501 rc = loadHardDisks (aHDAsNode, true /* aRegistered */, &mSnapshotId);
9502
9503 if (SUCCEEDED (rc))
9504 {
9505 /* commit all changes made during the initialization */
9506 commit();
9507 }
9508
9509 /* Confirm a successful initialization when it's the case */
9510 if (SUCCEEDED (rc))
9511 autoInitSpan.setSucceeded();
9512
9513 LogFlowThisFuncLeave();
9514 return rc;
9515}
9516
9517/**
9518 * Uninitializes this SnapshotMachine object.
9519 */
9520void SnapshotMachine::uninit()
9521{
9522 LogFlowThisFuncEnter();
9523
9524 /* Enclose the state transition Ready->InUninit->NotReady */
9525 AutoUninitSpan autoUninitSpan (this);
9526 if (autoUninitSpan.uninitDone())
9527 return;
9528
9529 uninitDataAndChildObjects();
9530
9531 unconst (mParent).setNull();
9532 unconst (mPeer).setNull();
9533
9534 LogFlowThisFuncLeave();
9535}
9536
9537// AutoLock::Lockable interface
9538////////////////////////////////////////////////////////////////////////////////
9539
9540/**
9541 * Overrides VirtualBoxBase::lockHandle() in order to share the lock handle
9542 * with the primary Machine instance (mPeer).
9543 */
9544AutoLock::Handle *SnapshotMachine::lockHandle() const
9545{
9546 AssertReturn (!mPeer.isNull(), NULL);
9547 return mPeer->lockHandle();
9548}
9549
9550// public methods only for internal purposes
9551////////////////////////////////////////////////////////////////////////////////
9552
9553/**
9554 * Called by the snapshot object associated with this SnapshotMachine when
9555 * snapshot data such as name or description is changed.
9556 *
9557 * @note Locks this object for writing.
9558 */
9559HRESULT SnapshotMachine::onSnapshotChange (Snapshot *aSnapshot)
9560{
9561 AutoLock alock (this);
9562
9563 mPeer->saveSnapshotSettings (aSnapshot, SaveSS_UpdateAttrsOp);
9564
9565 /* inform callbacks */
9566 mParent->onSnapshotChange (mData->mUuid, aSnapshot->data().mId);
9567
9568 return S_OK;
9569}
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