VirtualBox

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

Last change on this file since 794 was 779, checked in by vboxsync, 18 years ago

FE/Qt: Fixed the "Failed to access USB controller" message box: Added a custom message box with the usual "suppress it" check box since an option to not install the USB proxy filter is valid and supported (to avoid annoying users that did so every time the VM settings page is opened).

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