VirtualBox

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

Last change on this file since 26176 was 26171, checked in by vboxsync, 15 years ago

Main: get rid of Backupable<>::hasActualChanges and the operator== in all the machine data structures which it required; nuke obsolete Shareable.h

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 340.1 KB
Line 
1/* $Id: MachineImpl.cpp 26171 2010-02-02 20:37:36Z vboxsync $ */
2
3/** @file
4 * Implementation of IMachine in VBoxSVC.
5 */
6
7/*
8 * Copyright (C) 2006-2010 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23/* Make sure all the stdint.h macros are included - must come first! */
24#ifndef __STDC_LIMIT_MACROS
25# define __STDC_LIMIT_MACROS
26#endif
27#ifndef __STDC_CONSTANT_MACROS
28# define __STDC_CONSTANT_MACROS
29#endif
30
31#ifdef VBOX_WITH_SYS_V_IPC_SESSION_WATCHER
32# include <errno.h>
33# include <sys/types.h>
34# include <sys/stat.h>
35# include <sys/ipc.h>
36# include <sys/sem.h>
37#endif
38
39#include "VirtualBoxImpl.h"
40#include "MachineImpl.h"
41#include "ProgressImpl.h"
42#include "MediumAttachmentImpl.h"
43#include "MediumImpl.h"
44#include "USBControllerImpl.h"
45#include "HostImpl.h"
46#include "SharedFolderImpl.h"
47#include "GuestOSTypeImpl.h"
48#include "VirtualBoxErrorInfoImpl.h"
49#include "GuestImpl.h"
50#include "StorageControllerImpl.h"
51
52#ifdef VBOX_WITH_USB
53# include "USBProxyService.h"
54#endif
55
56#include "AutoCaller.h"
57#include "Logging.h"
58#include "Performance.h"
59
60#include <iprt/asm.h>
61#include <iprt/path.h>
62#include <iprt/dir.h>
63#include <iprt/env.h>
64#include <iprt/lockvalidator.h>
65#include <iprt/process.h>
66#include <iprt/cpp/utils.h>
67#include <iprt/string.h>
68
69#include <VBox/com/array.h>
70
71#include <VBox/err.h>
72#include <VBox/param.h>
73#include <VBox/settings.h>
74#include <VBox/ssm.h>
75
76#ifdef VBOX_WITH_GUEST_PROPS
77# include <VBox/HostServices/GuestPropertySvc.h>
78# include <VBox/com/array.h>
79#endif
80
81#include <algorithm>
82
83#include <typeinfo>
84
85#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
86#define HOSTSUFF_EXE ".exe"
87#else /* !RT_OS_WINDOWS */
88#define HOSTSUFF_EXE ""
89#endif /* !RT_OS_WINDOWS */
90
91// defines / prototypes
92/////////////////////////////////////////////////////////////////////////////
93
94/////////////////////////////////////////////////////////////////////////////
95// Machine::Data structure
96/////////////////////////////////////////////////////////////////////////////
97
98Machine::Data::Data()
99{
100 mRegistered = FALSE;
101 mAccessible = FALSE;
102 /* mUuid is initialized in Machine::init() */
103
104 mMachineState = MachineState_PoweredOff;
105 RTTimeNow(&mLastStateChange);
106
107 mMachineStateDeps = 0;
108 mMachineStateDepsSem = NIL_RTSEMEVENTMULTI;
109 mMachineStateChangePending = 0;
110
111 mCurrentStateModified = TRUE;
112 mHandleCfgFile = NIL_RTFILE;
113
114 mSession.mPid = NIL_RTPROCESS;
115 mSession.mState = SessionState_Closed;
116}
117
118Machine::Data::~Data()
119{
120 if (mMachineStateDepsSem != NIL_RTSEMEVENTMULTI)
121 {
122 RTSemEventMultiDestroy(mMachineStateDepsSem);
123 mMachineStateDepsSem = NIL_RTSEMEVENTMULTI;
124 }
125}
126
127/////////////////////////////////////////////////////////////////////////////
128// Machine::UserData structure
129/////////////////////////////////////////////////////////////////////////////
130
131Machine::UserData::UserData()
132{
133 /* default values for a newly created machine */
134
135 mNameSync = TRUE;
136 mTeleporterEnabled = FALSE;
137 mTeleporterPort = 0;
138
139 /* mName, mOSTypeId, mSnapshotFolder, mSnapshotFolderFull are initialized in
140 * Machine::init() */
141}
142
143Machine::UserData::~UserData()
144{
145}
146
147/////////////////////////////////////////////////////////////////////////////
148// Machine::HWData structure
149/////////////////////////////////////////////////////////////////////////////
150
151Machine::HWData::HWData()
152{
153 /* default values for a newly created machine */
154 mHWVersion = "2"; /** @todo get the default from the schema if that is possible. */
155 mMemorySize = 128;
156 mCPUCount = 1;
157 mCPUHotPlugEnabled = false;
158 mMemoryBalloonSize = 0;
159 mStatisticsUpdateInterval = 0;
160 mVRAMSize = 8;
161 mAccelerate3DEnabled = false;
162 mAccelerate2DVideoEnabled = false;
163 mMonitorCount = 1;
164 mHWVirtExEnabled = true;
165 mHWVirtExNestedPagingEnabled = true;
166 mHWVirtExVPIDEnabled = true;
167#if defined(RT_OS_DARWIN) || defined(RT_OS_WINDOWS)
168 mHWVirtExExclusive = false;
169#else
170 mHWVirtExExclusive = true;
171#endif
172#if HC_ARCH_BITS == 64 || defined(RT_OS_WINDOWS) || defined(RT_OS_DARWIN)
173 mPAEEnabled = true;
174#else
175 mPAEEnabled = false;
176#endif
177 mSyntheticCpu = false;
178 mPropertyServiceActive = false;
179
180 /* default boot order: floppy - DVD - HDD */
181 mBootOrder [0] = DeviceType_Floppy;
182 mBootOrder [1] = DeviceType_DVD;
183 mBootOrder [2] = DeviceType_HardDisk;
184 for (size_t i = 3; i < RT_ELEMENTS (mBootOrder); ++i)
185 mBootOrder [i] = DeviceType_Null;
186
187 mClipboardMode = ClipboardMode_Bidirectional;
188 mGuestPropertyNotificationPatterns = "";
189
190 mFirmwareType = FirmwareType_BIOS;
191
192 for (size_t i = 0; i < RT_ELEMENTS(mCPUAttached); i++)
193 mCPUAttached[i] = false;
194}
195
196Machine::HWData::~HWData()
197{
198}
199
200/////////////////////////////////////////////////////////////////////////////
201// Machine::HDData structure
202/////////////////////////////////////////////////////////////////////////////
203
204Machine::MediaData::MediaData()
205{
206}
207
208Machine::MediaData::~MediaData()
209{
210}
211
212/////////////////////////////////////////////////////////////////////////////
213// Machine class
214/////////////////////////////////////////////////////////////////////////////
215
216// constructor / destructor
217/////////////////////////////////////////////////////////////////////////////
218
219Machine::Machine()
220{}
221
222Machine::~Machine()
223{}
224
225HRESULT Machine::FinalConstruct()
226{
227 LogFlowThisFunc(("\n"));
228 return S_OK;
229}
230
231void Machine::FinalRelease()
232{
233 LogFlowThisFunc(("\n"));
234 uninit();
235}
236
237/**
238 * Initializes the instance.
239 *
240 * @param aParent Associated parent object
241 * @param aConfigFile Local file system path to the VM settings file (can
242 * be relative to the VirtualBox config directory).
243 * @param aMode Init_New, Init_Existing or Init_Registered
244 * @param aName name for the machine when aMode is Init_New
245 * (ignored otherwise)
246 * @param aOsType OS Type of this machine
247 * @param aNameSync |TRUE| to automatically sync settings dir and file
248 * name with the machine name. |FALSE| is used for legacy
249 * machines where the file name is specified by the
250 * user and should never change. Used only in Init_New
251 * mode (ignored otherwise).
252 * @param aId UUID of the machine. Required for aMode==Init_Registered
253 * and optional for aMode==Init_New. Used for consistency
254 * check when aMode is Init_Registered; must match UUID
255 * stored in the settings file. Used for predefining the
256 * UUID of a VM when aMode is Init_New.
257 *
258 * @return Success indicator. if not S_OK, the machine object is invalid
259 */
260HRESULT Machine::init(VirtualBox *aParent,
261 const Utf8Str &strConfigFile,
262 InitMode aMode,
263 CBSTR aName /* = NULL */,
264 GuestOSType *aOsType /* = NULL */,
265 BOOL aNameSync /* = TRUE */,
266 const Guid *aId /* = NULL */)
267{
268 LogFlowThisFuncEnter();
269 LogFlowThisFunc (("aConfigFile='%s', aMode=%d\n", strConfigFile.raw(), aMode));
270
271 AssertReturn (aParent, E_INVALIDARG);
272 AssertReturn (!strConfigFile.isEmpty(), E_INVALIDARG);
273 AssertReturn(aMode != Init_New || (aName != NULL && *aName != '\0'),
274 E_INVALIDARG);
275 AssertReturn(aMode != Init_Registered || aId != NULL, E_FAIL);
276
277 /* Enclose the state transition NotReady->InInit->Ready */
278 AutoInitSpan autoInitSpan(this);
279 AssertReturn(autoInitSpan.isOk(), E_FAIL);
280
281 HRESULT rc = S_OK;
282
283 /* share the parent weakly */
284 unconst(mParent) = aParent;
285
286 m_flModifications = 0;
287
288 /* allocate the essential machine data structure (the rest will be
289 * allocated later by initDataAndChildObjects() */
290 mData.allocate();
291
292 mData->m_pMachineConfigFile = NULL;
293 m_flModifications = 0;
294
295 /* memorize the config file name (as provided) */
296 mData->m_strConfigFile = strConfigFile;
297
298 /* get the full file name */
299 int vrc1 = mParent->calculateFullPath(strConfigFile, mData->m_strConfigFileFull);
300 if (RT_FAILURE(vrc1))
301 return setError(VBOX_E_FILE_ERROR,
302 tr("Invalid machine settings file name '%s' (%Rrc)"),
303 strConfigFile.raw(),
304 vrc1);
305
306 if (aMode == Init_Registered)
307 {
308 mData->mRegistered = TRUE;
309
310 /* store the supplied UUID (will be used to check for UUID consistency
311 * in loadSettings() */
312 unconst(mData->mUuid) = *aId;
313
314 // now load the settings from XML:
315 rc = registeredInit();
316 }
317 else
318 {
319 if (aMode == Init_Import)
320 {
321 // we're reading the settings file below
322 }
323 else if (aMode == Init_New)
324 {
325 /* check for the file existence */
326 RTFILE f = NIL_RTFILE;
327 int vrc = RTFileOpen(&f, mData->m_strConfigFileFull.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
328 if ( RT_SUCCESS(vrc)
329 || vrc == VERR_SHARING_VIOLATION
330 )
331 {
332 rc = setError(VBOX_E_FILE_ERROR,
333 tr("Machine settings file '%s' already exists"),
334 mData->m_strConfigFileFull.raw());
335 if (RT_SUCCESS(vrc))
336 RTFileClose(f);
337 }
338 else
339 {
340 if ( vrc != VERR_FILE_NOT_FOUND
341 && vrc != VERR_PATH_NOT_FOUND
342 )
343 rc = setError(VBOX_E_FILE_ERROR,
344 tr("Invalid machine settings file name '%s' (%Rrc)"),
345 mData->m_strConfigFileFull.raw(),
346 vrc);
347 }
348
349 // create an empty machine config
350 mData->m_pMachineConfigFile = new settings::MachineConfigFile(NULL);
351 }
352 else
353 AssertFailed();
354
355 if (SUCCEEDED(rc))
356 rc = initDataAndChildObjects();
357
358 if (SUCCEEDED(rc))
359 {
360 /* set to true now to cause uninit() to call
361 * uninitDataAndChildObjects() on failure */
362 mData->mAccessible = TRUE;
363
364 if (aMode != Init_New)
365 {
366 rc = loadSettings(false /* aRegistered */);
367 }
368 else
369 {
370 /* create the machine UUID */
371 if (aId)
372 unconst(mData->mUuid) = *aId;
373 else
374 unconst(mData->mUuid).create();
375
376 /* memorize the provided new machine's name */
377 mUserData->mName = aName;
378 mUserData->mNameSync = aNameSync;
379
380 /* initialize the default snapshots folder
381 * (note: depends on the name value set above!) */
382 rc = COMSETTER(SnapshotFolder)(NULL);
383 AssertComRC(rc);
384
385 if (aOsType)
386 {
387 /* Store OS type */
388 mUserData->mOSTypeId = aOsType->id();
389
390 /* Apply BIOS defaults */
391 mBIOSSettings->applyDefaults (aOsType);
392
393 /* Apply network adapters defaults */
394 for (ULONG slot = 0; slot < RT_ELEMENTS(mNetworkAdapters); ++slot)
395 mNetworkAdapters[slot]->applyDefaults(aOsType);
396
397 /* Apply serial port defaults */
398 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); ++slot)
399 mSerialPorts[slot]->applyDefaults(aOsType);
400 }
401 }
402
403 /* commit all changes made during the initialization */
404 if (SUCCEEDED(rc))
405 commit();
406 }
407 }
408
409 /* Confirm a successful initialization when it's the case */
410 if (SUCCEEDED(rc))
411 {
412 if (mData->mAccessible)
413 autoInitSpan.setSucceeded();
414 else
415 autoInitSpan.setLimited();
416 }
417
418 LogFlowThisFunc(("mName='%ls', mRegistered=%RTbool, mAccessible=%RTbool "
419 "rc=%08X\n",
420 !!mUserData ? mUserData->mName.raw() : NULL,
421 mData->mRegistered, mData->mAccessible, rc));
422
423 LogFlowThisFuncLeave();
424
425 return rc;
426}
427
428/**
429 * Initializes the registered machine by loading the settings file.
430 * This method is separated from #init() in order to make it possible to
431 * retry the operation after VirtualBox startup instead of refusing to
432 * startup the whole VirtualBox server in case if the settings file of some
433 * registered VM is invalid or inaccessible.
434 *
435 * @note Must be always called from this object's write lock
436 * (unless called from #init() that doesn't need any locking).
437 * @note Locks the mUSBController method for writing.
438 * @note Subclasses must not call this method.
439 */
440HRESULT Machine::registeredInit()
441{
442 AssertReturn(getClassID() == clsidMachine, E_FAIL);
443 AssertReturn(!mData->mUuid.isEmpty(), E_FAIL);
444 AssertReturn(!mData->mAccessible, E_FAIL);
445
446 HRESULT rc = initDataAndChildObjects();
447
448 if (SUCCEEDED(rc))
449 {
450 /* Temporarily reset the registered flag in order to let setters
451 * potentially called from loadSettings() succeed (isMutable() used in
452 * all setters will return FALSE for a Machine instance if mRegistered
453 * is TRUE). */
454 mData->mRegistered = FALSE;
455
456 rc = loadSettings(true /* aRegistered */);
457
458 /* Restore the registered flag (even on failure) */
459 mData->mRegistered = TRUE;
460 }
461
462 if (SUCCEEDED(rc))
463 {
464 /* Set mAccessible to TRUE only if we successfully locked and loaded
465 * the settings file */
466 mData->mAccessible = TRUE;
467
468 /* commit all changes made during loading the settings file */
469 commit(); // @todo r=dj why do we need a commit during init?!? this is very expensive
470 }
471 else
472 {
473 /* If the machine is registered, then, instead of returning a
474 * failure, we mark it as inaccessible and set the result to
475 * success to give it a try later */
476
477 /* fetch the current error info */
478 mData->mAccessError = com::ErrorInfo();
479 LogWarning(("Machine {%RTuuid} is inaccessible! [%ls]\n",
480 mData->mUuid.raw(),
481 mData->mAccessError.getText().raw()));
482
483 /* rollback all changes */
484 rollback(false /* aNotify */);
485
486 /* uninitialize the common part to make sure all data is reset to
487 * default (null) values */
488 uninitDataAndChildObjects();
489
490 rc = S_OK;
491 }
492
493 return rc;
494}
495
496/**
497 * Uninitializes the instance.
498 * Called either from FinalRelease() or by the parent when it gets destroyed.
499 *
500 * @note The caller of this method must make sure that this object
501 * a) doesn't have active callers on the current thread and b) is not locked
502 * by the current thread; otherwise uninit() will hang either a) due to
503 * AutoUninitSpan waiting for a number of calls to drop to zero or b) due to
504 * a dead-lock caused by this thread waiting for all callers on the other
505 * threads are done but preventing them from doing so by holding a lock.
506 */
507void Machine::uninit()
508{
509 LogFlowThisFuncEnter();
510
511 Assert (!isWriteLockOnCurrentThread());
512
513 /* Enclose the state transition Ready->InUninit->NotReady */
514 AutoUninitSpan autoUninitSpan(this);
515 if (autoUninitSpan.uninitDone())
516 return;
517
518 Assert(getClassID() == clsidMachine);
519 Assert(!!mData);
520
521 LogFlowThisFunc(("initFailed()=%d\n", autoUninitSpan.initFailed()));
522 LogFlowThisFunc(("mRegistered=%d\n", mData->mRegistered));
523
524 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
525
526 if (!mData->mSession.mMachine.isNull())
527 {
528 /* Theoretically, this can only happen if the VirtualBox server has been
529 * terminated while there were clients running that owned open direct
530 * sessions. Since in this case we are definitely called by
531 * VirtualBox::uninit(), we may be sure that SessionMachine::uninit()
532 * won't happen on the client watcher thread (because it does
533 * VirtualBox::addCaller() for the duration of the
534 * SessionMachine::checkForDeath() call, so that VirtualBox::uninit()
535 * cannot happen until the VirtualBox caller is released). This is
536 * important, because SessionMachine::uninit() cannot correctly operate
537 * after we return from this method (it expects the Machine instance is
538 * still valid). We'll call it ourselves below.
539 */
540 LogWarningThisFunc(("Session machine is not NULL (%p), the direct session is still open!\n",
541 (SessionMachine*)mData->mSession.mMachine));
542
543 if (Global::IsOnlineOrTransient(mData->mMachineState))
544 {
545 LogWarningThisFunc(("Setting state to Aborted!\n"));
546 /* set machine state using SessionMachine reimplementation */
547 static_cast<Machine*>(mData->mSession.mMachine)->setMachineState (MachineState_Aborted);
548 }
549
550 /*
551 * Uninitialize SessionMachine using public uninit() to indicate
552 * an unexpected uninitialization.
553 */
554 mData->mSession.mMachine->uninit();
555 /* SessionMachine::uninit() must set mSession.mMachine to null */
556 Assert(mData->mSession.mMachine.isNull());
557 }
558
559 /* the lock is no more necessary (SessionMachine is uninitialized) */
560 alock.leave();
561
562 // has machine been modified?
563 if (m_flModifications)
564 {
565 LogWarningThisFunc(("Discarding unsaved settings changes!\n"));
566 rollback(false /* aNotify */);
567 }
568
569 if (mData->mAccessible)
570 uninitDataAndChildObjects();
571
572 /* free the essential data structure last */
573 mData.free();
574
575 LogFlowThisFuncLeave();
576}
577
578// IMachine properties
579/////////////////////////////////////////////////////////////////////////////
580
581STDMETHODIMP Machine::COMGETTER(Parent) (IVirtualBox **aParent)
582{
583 CheckComArgOutPointerValid(aParent);
584
585 AutoLimitedCaller autoCaller(this);
586 if (FAILED(autoCaller.rc())) return autoCaller.rc();
587
588 /* mParent is constant during life time, no need to lock */
589 mParent.queryInterfaceTo(aParent);
590
591 return S_OK;
592}
593
594STDMETHODIMP Machine::COMGETTER(Accessible) (BOOL *aAccessible)
595{
596 CheckComArgOutPointerValid(aAccessible);
597
598 AutoLimitedCaller autoCaller(this);
599 if (FAILED(autoCaller.rc())) return autoCaller.rc();
600
601 LogFlowThisFunc(("ENTER\n"));
602
603 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
604
605 HRESULT rc = S_OK;
606
607 if (!mData->mAccessible)
608 {
609 /* try to initialize the VM once more if not accessible */
610
611 AutoReinitSpan autoReinitSpan(this);
612 AssertReturn(autoReinitSpan.isOk(), E_FAIL);
613
614#ifdef DEBUG
615 LogFlowThisFunc(("Dumping media backreferences\n"));
616 mParent->dumpAllBackRefs();
617#endif
618
619 if (mData->m_pMachineConfigFile)
620 {
621 // reset the XML file to force loadSettings() (called from registeredInit())
622 // to parse it again; the file might have changed
623 delete mData->m_pMachineConfigFile;
624 mData->m_pMachineConfigFile = NULL;
625 }
626
627 rc = registeredInit();
628
629 if (SUCCEEDED(rc) && mData->mAccessible)
630 {
631 autoReinitSpan.setSucceeded();
632
633 /* make sure interesting parties will notice the accessibility
634 * state change */
635 mParent->onMachineStateChange(mData->mUuid, mData->mMachineState);
636 mParent->onMachineDataChange(mData->mUuid);
637 }
638 }
639
640 if (SUCCEEDED(rc))
641 *aAccessible = mData->mAccessible;
642
643 LogFlowThisFuncLeave();
644
645 return rc;
646}
647
648STDMETHODIMP Machine::COMGETTER(AccessError) (IVirtualBoxErrorInfo **aAccessError)
649{
650 CheckComArgOutPointerValid(aAccessError);
651
652 AutoLimitedCaller autoCaller(this);
653 if (FAILED(autoCaller.rc())) return autoCaller.rc();
654
655 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
656
657 if (mData->mAccessible || !mData->mAccessError.isBasicAvailable())
658 {
659 /* return shortly */
660 aAccessError = NULL;
661 return S_OK;
662 }
663
664 HRESULT rc = S_OK;
665
666 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
667 rc = errorInfo.createObject();
668 if (SUCCEEDED(rc))
669 {
670 errorInfo->init (mData->mAccessError.getResultCode(),
671 mData->mAccessError.getInterfaceID(),
672 mData->mAccessError.getComponent(),
673 mData->mAccessError.getText());
674 rc = errorInfo.queryInterfaceTo(aAccessError);
675 }
676
677 return rc;
678}
679
680STDMETHODIMP Machine::COMGETTER(Name) (BSTR *aName)
681{
682 CheckComArgOutPointerValid(aName);
683
684 AutoCaller autoCaller(this);
685 if (FAILED(autoCaller.rc())) return autoCaller.rc();
686
687 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
688
689 mUserData->mName.cloneTo(aName);
690
691 return S_OK;
692}
693
694STDMETHODIMP Machine::COMSETTER(Name) (IN_BSTR aName)
695{
696 CheckComArgNotNull (aName);
697
698 if (!*aName)
699 return setError(E_INVALIDARG,
700 tr("Machine name cannot be empty"));
701
702 AutoCaller autoCaller(this);
703 if (FAILED(autoCaller.rc())) return autoCaller.rc();
704
705 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
706
707 HRESULT rc = checkStateDependency(MutableStateDep);
708 if (FAILED(rc)) return rc;
709
710 setModified(IsModified_MachineData);
711 mUserData.backup();
712 mUserData->mName = aName;
713
714 return S_OK;
715}
716
717STDMETHODIMP Machine::COMGETTER(Description) (BSTR *aDescription)
718{
719 CheckComArgOutPointerValid(aDescription);
720
721 AutoCaller autoCaller(this);
722 if (FAILED(autoCaller.rc())) return autoCaller.rc();
723
724 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
725
726 mUserData->mDescription.cloneTo(aDescription);
727
728 return S_OK;
729}
730
731STDMETHODIMP Machine::COMSETTER(Description) (IN_BSTR aDescription)
732{
733 AutoCaller autoCaller(this);
734 if (FAILED(autoCaller.rc())) return autoCaller.rc();
735
736 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
737
738 HRESULT rc = checkStateDependency(MutableStateDep);
739 if (FAILED(rc)) return rc;
740
741 setModified(IsModified_MachineData);
742 mUserData.backup();
743 mUserData->mDescription = aDescription;
744
745 return S_OK;
746}
747
748STDMETHODIMP Machine::COMGETTER(Id) (BSTR *aId)
749{
750 CheckComArgOutPointerValid(aId);
751
752 AutoLimitedCaller autoCaller(this);
753 if (FAILED(autoCaller.rc())) return autoCaller.rc();
754
755 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
756
757 mData->mUuid.toUtf16().cloneTo(aId);
758
759 return S_OK;
760}
761
762STDMETHODIMP Machine::COMGETTER(OSTypeId) (BSTR *aOSTypeId)
763{
764 CheckComArgOutPointerValid(aOSTypeId);
765
766 AutoCaller autoCaller(this);
767 if (FAILED(autoCaller.rc())) return autoCaller.rc();
768
769 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
770
771 mUserData->mOSTypeId.cloneTo(aOSTypeId);
772
773 return S_OK;
774}
775
776STDMETHODIMP Machine::COMSETTER(OSTypeId) (IN_BSTR aOSTypeId)
777{
778 CheckComArgNotNull (aOSTypeId);
779
780 AutoCaller autoCaller(this);
781 if (FAILED(autoCaller.rc())) return autoCaller.rc();
782
783 /* look up the object by Id to check it is valid */
784 ComPtr<IGuestOSType> guestOSType;
785 HRESULT rc = mParent->GetGuestOSType (aOSTypeId,
786 guestOSType.asOutParam());
787 if (FAILED(rc)) return rc;
788
789 /* when setting, always use the "etalon" value for consistency -- lookup
790 * by ID is case-insensitive and the input value may have different case */
791 Bstr osTypeId;
792 rc = guestOSType->COMGETTER(Id) (osTypeId.asOutParam());
793 if (FAILED(rc)) return rc;
794
795 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
796
797 rc = checkStateDependency(MutableStateDep);
798 if (FAILED(rc)) return rc;
799
800 setModified(IsModified_MachineData);
801 mUserData.backup();
802 mUserData->mOSTypeId = osTypeId;
803
804 return S_OK;
805}
806
807
808STDMETHODIMP Machine::COMGETTER(FirmwareType) (FirmwareType_T *aFirmwareType)
809{
810 CheckComArgOutPointerValid(aFirmwareType);
811
812 AutoCaller autoCaller(this);
813 if (FAILED(autoCaller.rc())) return autoCaller.rc();
814
815 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
816
817 *aFirmwareType = mHWData->mFirmwareType;
818
819 return S_OK;
820}
821
822STDMETHODIMP Machine::COMSETTER(FirmwareType) (FirmwareType_T aFirmwareType)
823{
824 AutoCaller autoCaller(this);
825 if (FAILED(autoCaller.rc())) return autoCaller.rc();
826 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
827
828 int rc = checkStateDependency(MutableStateDep);
829 if (FAILED(rc)) return rc;
830
831 setModified(IsModified_MachineData);
832 mHWData.backup();
833 mHWData->mFirmwareType = aFirmwareType;
834
835 return S_OK;
836}
837
838STDMETHODIMP Machine::COMGETTER(HardwareVersion) (BSTR *aHWVersion)
839{
840 if (!aHWVersion)
841 return E_POINTER;
842
843 AutoCaller autoCaller(this);
844 if (FAILED(autoCaller.rc())) return autoCaller.rc();
845
846 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
847
848 mHWData->mHWVersion.cloneTo(aHWVersion);
849
850 return S_OK;
851}
852
853STDMETHODIMP Machine::COMSETTER(HardwareVersion) (IN_BSTR aHWVersion)
854{
855 /* check known version */
856 Utf8Str hwVersion = aHWVersion;
857 if ( hwVersion.compare ("1") != 0
858 && hwVersion.compare ("2") != 0)
859 return setError(E_INVALIDARG,
860 tr("Invalid hardware version: %ls\n"), aHWVersion);
861
862 AutoCaller autoCaller(this);
863 if (FAILED(autoCaller.rc())) return autoCaller.rc();
864
865 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
866
867 HRESULT rc = checkStateDependency(MutableStateDep);
868 if (FAILED(rc)) return rc;
869
870 setModified(IsModified_MachineData);
871 mHWData.backup();
872 mHWData->mHWVersion = hwVersion;
873
874 return S_OK;
875}
876
877STDMETHODIMP Machine::COMGETTER(HardwareUUID)(BSTR *aUUID)
878{
879 CheckComArgOutPointerValid(aUUID);
880
881 AutoCaller autoCaller(this);
882 if (FAILED(autoCaller.rc())) return autoCaller.rc();
883
884 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
885
886 if (!mHWData->mHardwareUUID.isEmpty())
887 mHWData->mHardwareUUID.toUtf16().cloneTo(aUUID);
888 else
889 mData->mUuid.toUtf16().cloneTo(aUUID);
890
891 return S_OK;
892}
893
894STDMETHODIMP Machine::COMSETTER(HardwareUUID) (IN_BSTR aUUID)
895{
896 Guid hardwareUUID(aUUID);
897 if (hardwareUUID.isEmpty())
898 return E_INVALIDARG;
899
900 AutoCaller autoCaller(this);
901 if (FAILED(autoCaller.rc())) return autoCaller.rc();
902
903 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
904
905 HRESULT rc = checkStateDependency(MutableStateDep);
906 if (FAILED(rc)) return rc;
907
908 setModified(IsModified_MachineData);
909 mHWData.backup();
910 if (hardwareUUID == mData->mUuid)
911 mHWData->mHardwareUUID.clear();
912 else
913 mHWData->mHardwareUUID = hardwareUUID;
914
915 return S_OK;
916}
917
918STDMETHODIMP Machine::COMGETTER(MemorySize) (ULONG *memorySize)
919{
920 if (!memorySize)
921 return E_POINTER;
922
923 AutoCaller autoCaller(this);
924 if (FAILED(autoCaller.rc())) return autoCaller.rc();
925
926 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
927
928 *memorySize = mHWData->mMemorySize;
929
930 return S_OK;
931}
932
933STDMETHODIMP Machine::COMSETTER(MemorySize) (ULONG memorySize)
934{
935 /* check RAM limits */
936 if ( memorySize < MM_RAM_MIN_IN_MB
937 || memorySize > MM_RAM_MAX_IN_MB
938 )
939 return setError(E_INVALIDARG,
940 tr("Invalid RAM size: %lu MB (must be in range [%lu, %lu] MB)"),
941 memorySize, MM_RAM_MIN_IN_MB, MM_RAM_MAX_IN_MB);
942
943 AutoCaller autoCaller(this);
944 if (FAILED(autoCaller.rc())) return autoCaller.rc();
945
946 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
947
948 HRESULT rc = checkStateDependency(MutableStateDep);
949 if (FAILED(rc)) return rc;
950
951 setModified(IsModified_MachineData);
952 mHWData.backup();
953 mHWData->mMemorySize = memorySize;
954
955 return S_OK;
956}
957
958STDMETHODIMP Machine::COMGETTER(CPUCount) (ULONG *CPUCount)
959{
960 if (!CPUCount)
961 return E_POINTER;
962
963 AutoCaller autoCaller(this);
964 if (FAILED(autoCaller.rc())) return autoCaller.rc();
965
966 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
967
968 *CPUCount = mHWData->mCPUCount;
969
970 return S_OK;
971}
972
973STDMETHODIMP Machine::COMSETTER(CPUCount) (ULONG CPUCount)
974{
975 /* check CPU limits */
976 if ( CPUCount < SchemaDefs::MinCPUCount
977 || CPUCount > SchemaDefs::MaxCPUCount
978 )
979 return setError(E_INVALIDARG,
980 tr("Invalid virtual CPU count: %lu (must be in range [%lu, %lu])"),
981 CPUCount, SchemaDefs::MinCPUCount, SchemaDefs::MaxCPUCount);
982
983 AutoCaller autoCaller(this);
984 if (FAILED(autoCaller.rc())) return autoCaller.rc();
985
986 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
987
988 /* We cant go below the current number of CPUs if hotplug is enabled*/
989 if (mHWData->mCPUHotPlugEnabled)
990 {
991 for (unsigned idx = CPUCount; idx < SchemaDefs::MaxCPUCount; idx++)
992 {
993 if (mHWData->mCPUAttached[idx])
994 return setError(E_INVALIDARG,
995 tr(": %lu (must be higher than or equal to %lu)"),
996 CPUCount, idx+1);
997 }
998 }
999
1000 HRESULT rc = checkStateDependency(MutableStateDep);
1001 if (FAILED(rc)) return rc;
1002
1003 setModified(IsModified_MachineData);
1004 mHWData.backup();
1005 mHWData->mCPUCount = CPUCount;
1006
1007 return S_OK;
1008}
1009
1010STDMETHODIMP Machine::COMGETTER(CPUHotPlugEnabled) (BOOL *enabled)
1011{
1012 if (!enabled)
1013 return E_POINTER;
1014
1015 AutoCaller autoCaller(this);
1016 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1017
1018 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1019
1020 *enabled = mHWData->mCPUHotPlugEnabled;
1021
1022 return S_OK;
1023}
1024
1025STDMETHODIMP Machine::COMSETTER(CPUHotPlugEnabled) (BOOL enabled)
1026{
1027 HRESULT rc = S_OK;
1028
1029 AutoCaller autoCaller(this);
1030 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1031
1032 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1033
1034 rc = checkStateDependency(MutableStateDep);
1035 if (FAILED(rc)) return rc;
1036
1037 if (mHWData->mCPUHotPlugEnabled != enabled)
1038 {
1039 if (enabled)
1040 {
1041 setModified(IsModified_MachineData);
1042 mHWData.backup();
1043
1044 /* Add the amount of CPUs currently attached */
1045 for (unsigned i = 0; i < mHWData->mCPUCount; i++)
1046 {
1047 mHWData->mCPUAttached[i] = true;
1048 }
1049 }
1050 else
1051 {
1052 /*
1053 * We can disable hotplug only if the amount of maximum CPUs is equal
1054 * to the amount of attached CPUs
1055 */
1056 unsigned cCpusAttached = 0;
1057 unsigned iHighestId = 0;
1058
1059 for (unsigned i = 0; i < SchemaDefs::MaxCPUCount; i++)
1060 {
1061 if (mHWData->mCPUAttached[i])
1062 {
1063 cCpusAttached++;
1064 iHighestId = i;
1065 }
1066 }
1067
1068 if ( (cCpusAttached != mHWData->mCPUCount)
1069 || (iHighestId >= mHWData->mCPUCount))
1070 return setError(E_INVALIDARG,
1071 tr("CPU hotplugging can't be disabled because the maximum number of CPUs is not equal to the amount of CPUs attached\n"));
1072
1073 setModified(IsModified_MachineData);
1074 mHWData.backup();
1075 }
1076 }
1077
1078 mHWData->mCPUHotPlugEnabled = enabled;
1079
1080 return rc;
1081}
1082
1083STDMETHODIMP Machine::COMGETTER(VRAMSize) (ULONG *memorySize)
1084{
1085 if (!memorySize)
1086 return E_POINTER;
1087
1088 AutoCaller autoCaller(this);
1089 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1090
1091 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1092
1093 *memorySize = mHWData->mVRAMSize;
1094
1095 return S_OK;
1096}
1097
1098STDMETHODIMP Machine::COMSETTER(VRAMSize) (ULONG memorySize)
1099{
1100 /* check VRAM limits */
1101 if (memorySize < SchemaDefs::MinGuestVRAM ||
1102 memorySize > SchemaDefs::MaxGuestVRAM)
1103 return setError(E_INVALIDARG,
1104 tr("Invalid VRAM size: %lu MB (must be in range [%lu, %lu] MB)"),
1105 memorySize, SchemaDefs::MinGuestVRAM, SchemaDefs::MaxGuestVRAM);
1106
1107 AutoCaller autoCaller(this);
1108 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1109
1110 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1111
1112 HRESULT rc = checkStateDependency(MutableStateDep);
1113 if (FAILED(rc)) return rc;
1114
1115 setModified(IsModified_MachineData);
1116 mHWData.backup();
1117 mHWData->mVRAMSize = memorySize;
1118
1119 return S_OK;
1120}
1121
1122/** @todo this method should not be public */
1123STDMETHODIMP Machine::COMGETTER(MemoryBalloonSize) (ULONG *memoryBalloonSize)
1124{
1125 if (!memoryBalloonSize)
1126 return E_POINTER;
1127
1128 AutoCaller autoCaller(this);
1129 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1130
1131 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1132
1133 *memoryBalloonSize = mHWData->mMemoryBalloonSize;
1134
1135 return S_OK;
1136}
1137
1138/** @todo this method should not be public */
1139STDMETHODIMP Machine::COMSETTER(MemoryBalloonSize) (ULONG memoryBalloonSize)
1140{
1141 /* check limits */
1142 if (memoryBalloonSize >= VMMDEV_MAX_MEMORY_BALLOON (mHWData->mMemorySize))
1143 return setError(E_INVALIDARG,
1144 tr("Invalid memory balloon size: %lu MB (must be in range [%lu, %lu] MB)"),
1145 memoryBalloonSize, 0, VMMDEV_MAX_MEMORY_BALLOON (mHWData->mMemorySize));
1146
1147 AutoCaller autoCaller(this);
1148 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1149
1150 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1151
1152 HRESULT rc = checkStateDependency(MutableStateDep);
1153 if (FAILED(rc)) return rc;
1154
1155 setModified(IsModified_MachineData);
1156 mHWData.backup();
1157 mHWData->mMemoryBalloonSize = memoryBalloonSize;
1158
1159 return S_OK;
1160}
1161
1162/** @todo this method should not be public */
1163STDMETHODIMP Machine::COMGETTER(StatisticsUpdateInterval) (ULONG *statisticsUpdateInterval)
1164{
1165 if (!statisticsUpdateInterval)
1166 return E_POINTER;
1167
1168 AutoCaller autoCaller(this);
1169 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1170
1171 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1172
1173 *statisticsUpdateInterval = mHWData->mStatisticsUpdateInterval;
1174
1175 return S_OK;
1176}
1177
1178/** @todo this method should not be public */
1179STDMETHODIMP Machine::COMSETTER(StatisticsUpdateInterval) (ULONG statisticsUpdateInterval)
1180{
1181 AutoCaller autoCaller(this);
1182 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1183
1184 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1185
1186 HRESULT rc = checkStateDependency(MutableStateDep);
1187 if (FAILED(rc)) return rc;
1188
1189 setModified(IsModified_MachineData);
1190 mHWData.backup();
1191 mHWData->mStatisticsUpdateInterval = statisticsUpdateInterval;
1192
1193 return S_OK;
1194}
1195
1196
1197STDMETHODIMP Machine::COMGETTER(Accelerate3DEnabled)(BOOL *enabled)
1198{
1199 if (!enabled)
1200 return E_POINTER;
1201
1202 AutoCaller autoCaller(this);
1203 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1204
1205 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1206
1207 *enabled = mHWData->mAccelerate3DEnabled;
1208
1209 return S_OK;
1210}
1211
1212STDMETHODIMP Machine::COMSETTER(Accelerate3DEnabled)(BOOL enable)
1213{
1214 AutoCaller autoCaller(this);
1215 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1216
1217 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1218
1219 HRESULT rc = checkStateDependency(MutableStateDep);
1220 if (FAILED(rc)) return rc;
1221
1222 /** @todo check validity! */
1223
1224 setModified(IsModified_MachineData);
1225 mHWData.backup();
1226 mHWData->mAccelerate3DEnabled = enable;
1227
1228 return S_OK;
1229}
1230
1231
1232STDMETHODIMP Machine::COMGETTER(Accelerate2DVideoEnabled)(BOOL *enabled)
1233{
1234 if (!enabled)
1235 return E_POINTER;
1236
1237 AutoCaller autoCaller(this);
1238 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1239
1240 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1241
1242 *enabled = mHWData->mAccelerate2DVideoEnabled;
1243
1244 return S_OK;
1245}
1246
1247STDMETHODIMP Machine::COMSETTER(Accelerate2DVideoEnabled)(BOOL enable)
1248{
1249 AutoCaller autoCaller(this);
1250 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1251
1252 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1253
1254 HRESULT rc = checkStateDependency(MutableStateDep);
1255 if (FAILED(rc)) return rc;
1256
1257 /** @todo check validity! */
1258
1259 setModified(IsModified_MachineData);
1260 mHWData.backup();
1261 mHWData->mAccelerate2DVideoEnabled = enable;
1262
1263 return S_OK;
1264}
1265
1266STDMETHODIMP Machine::COMGETTER(MonitorCount) (ULONG *monitorCount)
1267{
1268 if (!monitorCount)
1269 return E_POINTER;
1270
1271 AutoCaller autoCaller(this);
1272 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1273
1274 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1275
1276 *monitorCount = mHWData->mMonitorCount;
1277
1278 return S_OK;
1279}
1280
1281STDMETHODIMP Machine::COMSETTER(MonitorCount) (ULONG monitorCount)
1282{
1283 /* make sure monitor count is a sensible number */
1284 if (monitorCount < 1 || monitorCount > SchemaDefs::MaxGuestMonitors)
1285 return setError(E_INVALIDARG,
1286 tr("Invalid monitor count: %lu (must be in range [%lu, %lu])"),
1287 monitorCount, 1, SchemaDefs::MaxGuestMonitors);
1288
1289 AutoCaller autoCaller(this);
1290 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1291
1292 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1293
1294 HRESULT rc = checkStateDependency(MutableStateDep);
1295 if (FAILED(rc)) return rc;
1296
1297 setModified(IsModified_MachineData);
1298 mHWData.backup();
1299 mHWData->mMonitorCount = monitorCount;
1300
1301 return S_OK;
1302}
1303
1304STDMETHODIMP Machine::COMGETTER(BIOSSettings)(IBIOSSettings **biosSettings)
1305{
1306 if (!biosSettings)
1307 return E_POINTER;
1308
1309 AutoCaller autoCaller(this);
1310 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1311
1312 /* mBIOSSettings is constant during life time, no need to lock */
1313 mBIOSSettings.queryInterfaceTo(biosSettings);
1314
1315 return S_OK;
1316}
1317
1318STDMETHODIMP Machine::GetCpuProperty(CpuPropertyType_T property, BOOL *aVal)
1319{
1320 if (!aVal)
1321 return E_POINTER;
1322
1323 AutoCaller autoCaller(this);
1324 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1325
1326 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1327
1328 switch(property)
1329 {
1330 case CpuPropertyType_PAE:
1331 *aVal = mHWData->mPAEEnabled;
1332 break;
1333
1334 case CpuPropertyType_Synthetic:
1335 *aVal = mHWData->mSyntheticCpu;
1336 break;
1337
1338 default:
1339 return E_INVALIDARG;
1340 }
1341 return S_OK;
1342}
1343
1344STDMETHODIMP Machine::SetCpuProperty(CpuPropertyType_T property, BOOL aVal)
1345{
1346 AutoCaller autoCaller(this);
1347 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1348
1349 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1350
1351 HRESULT rc = checkStateDependency(MutableStateDep);
1352 if (FAILED(rc)) return rc;
1353
1354 switch(property)
1355 {
1356 case CpuPropertyType_PAE:
1357 mHWData->mPAEEnabled = !!aVal;
1358 break;
1359
1360 case CpuPropertyType_Synthetic:
1361 mHWData->mSyntheticCpu = !!aVal;
1362 break;
1363
1364 default:
1365 return E_INVALIDARG;
1366 }
1367 return S_OK;
1368}
1369
1370STDMETHODIMP Machine::GetCpuIdLeaf(ULONG aId, ULONG *aValEax, ULONG *aValEbx, ULONG *aValEcx, ULONG *aValEdx)
1371{
1372 CheckComArgOutPointerValid(aValEax);
1373 CheckComArgOutPointerValid(aValEbx);
1374 CheckComArgOutPointerValid(aValEcx);
1375 CheckComArgOutPointerValid(aValEdx);
1376
1377 AutoCaller autoCaller(this);
1378 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1379
1380 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1381
1382 switch(aId)
1383 {
1384 case 0x0:
1385 case 0x1:
1386 case 0x2:
1387 case 0x3:
1388 case 0x4:
1389 case 0x5:
1390 case 0x6:
1391 case 0x7:
1392 case 0x8:
1393 case 0x9:
1394 case 0xA:
1395 if (mHWData->mCpuIdStdLeafs[aId].ulId != aId)
1396 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is not set"), aId);
1397
1398 *aValEax = mHWData->mCpuIdStdLeafs[aId].ulEax;
1399 *aValEbx = mHWData->mCpuIdStdLeafs[aId].ulEbx;
1400 *aValEcx = mHWData->mCpuIdStdLeafs[aId].ulEcx;
1401 *aValEdx = mHWData->mCpuIdStdLeafs[aId].ulEdx;
1402 break;
1403
1404 case 0x80000000:
1405 case 0x80000001:
1406 case 0x80000002:
1407 case 0x80000003:
1408 case 0x80000004:
1409 case 0x80000005:
1410 case 0x80000006:
1411 case 0x80000007:
1412 case 0x80000008:
1413 case 0x80000009:
1414 case 0x8000000A:
1415 if (mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId != aId)
1416 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is not set"), aId);
1417
1418 *aValEax = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEax;
1419 *aValEbx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEbx;
1420 *aValEcx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEcx;
1421 *aValEdx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEdx;
1422 break;
1423
1424 default:
1425 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
1426 }
1427 return S_OK;
1428}
1429
1430STDMETHODIMP Machine::SetCpuIdLeaf(ULONG aId, ULONG aValEax, ULONG aValEbx, ULONG aValEcx, ULONG aValEdx)
1431{
1432 AutoCaller autoCaller(this);
1433 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1434
1435 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1436
1437 HRESULT rc = checkStateDependency(MutableStateDep);
1438 if (FAILED(rc)) return rc;
1439
1440 switch(aId)
1441 {
1442 case 0x0:
1443 case 0x1:
1444 case 0x2:
1445 case 0x3:
1446 case 0x4:
1447 case 0x5:
1448 case 0x6:
1449 case 0x7:
1450 case 0x8:
1451 case 0x9:
1452 case 0xA:
1453 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdStdLeafs) == 0xA);
1454 AssertRelease(aId < RT_ELEMENTS(mHWData->mCpuIdStdLeafs));
1455 mHWData->mCpuIdStdLeafs[aId].ulId = aId;
1456 mHWData->mCpuIdStdLeafs[aId].ulEax = aValEax;
1457 mHWData->mCpuIdStdLeafs[aId].ulEbx = aValEbx;
1458 mHWData->mCpuIdStdLeafs[aId].ulEcx = aValEcx;
1459 mHWData->mCpuIdStdLeafs[aId].ulEdx = aValEdx;
1460 break;
1461
1462 case 0x80000000:
1463 case 0x80000001:
1464 case 0x80000002:
1465 case 0x80000003:
1466 case 0x80000004:
1467 case 0x80000005:
1468 case 0x80000006:
1469 case 0x80000007:
1470 case 0x80000008:
1471 case 0x80000009:
1472 case 0x8000000A:
1473 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdExtLeafs) == 0xA);
1474 AssertRelease(aId - 0x80000000 < RT_ELEMENTS(mHWData->mCpuIdExtLeafs));
1475 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId = aId;
1476 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEax = aValEax;
1477 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEbx = aValEbx;
1478 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEcx = aValEcx;
1479 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEdx = aValEdx;
1480 break;
1481
1482 default:
1483 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
1484 }
1485 return S_OK;
1486}
1487
1488STDMETHODIMP Machine::RemoveCpuIdLeaf(ULONG aId)
1489{
1490 AutoCaller autoCaller(this);
1491 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1492
1493 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1494
1495 HRESULT rc = checkStateDependency(MutableStateDep);
1496 if (FAILED(rc)) return rc;
1497
1498 switch(aId)
1499 {
1500 case 0x0:
1501 case 0x1:
1502 case 0x2:
1503 case 0x3:
1504 case 0x4:
1505 case 0x5:
1506 case 0x6:
1507 case 0x7:
1508 case 0x8:
1509 case 0x9:
1510 case 0xA:
1511 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdStdLeafs) == 0xA);
1512 AssertRelease(aId < RT_ELEMENTS(mHWData->mCpuIdStdLeafs));
1513 /* Invalidate leaf. */
1514 mHWData->mCpuIdStdLeafs[aId].ulId = UINT32_MAX;
1515 break;
1516
1517 case 0x80000000:
1518 case 0x80000001:
1519 case 0x80000002:
1520 case 0x80000003:
1521 case 0x80000004:
1522 case 0x80000005:
1523 case 0x80000006:
1524 case 0x80000007:
1525 case 0x80000008:
1526 case 0x80000009:
1527 case 0x8000000A:
1528 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdExtLeafs) == 0xA);
1529 AssertRelease(aId - 0x80000000 < RT_ELEMENTS(mHWData->mCpuIdExtLeafs));
1530 /* Invalidate leaf. */
1531 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId = UINT32_MAX;
1532 break;
1533
1534 default:
1535 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
1536 }
1537 return S_OK;
1538}
1539
1540STDMETHODIMP Machine::RemoveAllCpuIdLeafs()
1541{
1542 AutoCaller autoCaller(this);
1543 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1544
1545 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1546
1547 HRESULT rc = checkStateDependency(MutableStateDep);
1548 if (FAILED(rc)) return rc;
1549
1550 /* Invalidate all standard leafs. */
1551 for (unsigned i = 0; i < RT_ELEMENTS(mHWData->mCpuIdStdLeafs); i++)
1552 mHWData->mCpuIdStdLeafs[i].ulId = UINT32_MAX;
1553
1554 /* Invalidate all extended leafs. */
1555 for (unsigned i = 0; i < RT_ELEMENTS(mHWData->mCpuIdExtLeafs); i++)
1556 mHWData->mCpuIdExtLeafs[i].ulId = UINT32_MAX;
1557
1558 return S_OK;
1559}
1560
1561STDMETHODIMP Machine::GetHWVirtExProperty(HWVirtExPropertyType_T property, BOOL *aVal)
1562{
1563 if (!aVal)
1564 return E_POINTER;
1565
1566 AutoCaller autoCaller(this);
1567 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1568
1569 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1570
1571 switch(property)
1572 {
1573 case HWVirtExPropertyType_Enabled:
1574 *aVal = mHWData->mHWVirtExEnabled;
1575 break;
1576
1577 case HWVirtExPropertyType_Exclusive:
1578 *aVal = mHWData->mHWVirtExExclusive;
1579 break;
1580
1581 case HWVirtExPropertyType_VPID:
1582 *aVal = mHWData->mHWVirtExVPIDEnabled;
1583 break;
1584
1585 case HWVirtExPropertyType_NestedPaging:
1586 *aVal = mHWData->mHWVirtExNestedPagingEnabled;
1587 break;
1588
1589 default:
1590 return E_INVALIDARG;
1591 }
1592 return S_OK;
1593}
1594
1595STDMETHODIMP Machine::SetHWVirtExProperty(HWVirtExPropertyType_T property, BOOL aVal)
1596{
1597 AutoCaller autoCaller(this);
1598 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1599
1600 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1601
1602 HRESULT rc = checkStateDependency(MutableStateDep);
1603 if (FAILED(rc)) return rc;
1604
1605 BOOL *pb;
1606
1607 switch(property)
1608 {
1609 case HWVirtExPropertyType_Enabled:
1610 pb = &mHWData->mHWVirtExEnabled;
1611 break;
1612
1613 case HWVirtExPropertyType_Exclusive:
1614 pb = &mHWData->mHWVirtExExclusive;
1615 break;
1616
1617 case HWVirtExPropertyType_VPID:
1618 pb = &mHWData->mHWVirtExVPIDEnabled;
1619 break;
1620
1621 case HWVirtExPropertyType_NestedPaging:
1622 pb = &mHWData->mHWVirtExNestedPagingEnabled;
1623 break;
1624
1625 default:
1626 return E_INVALIDARG;
1627 }
1628
1629 if (*pb != !!aVal)
1630 {
1631 setModified(IsModified_MachineData);
1632 mHWData.backup();
1633 *pb = !!aVal;
1634 }
1635 return S_OK;
1636}
1637
1638STDMETHODIMP Machine::COMGETTER(SnapshotFolder) (BSTR *aSnapshotFolder)
1639{
1640 CheckComArgOutPointerValid(aSnapshotFolder);
1641
1642 AutoCaller autoCaller(this);
1643 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1644
1645 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1646
1647 mUserData->mSnapshotFolderFull.cloneTo(aSnapshotFolder);
1648
1649 return S_OK;
1650}
1651
1652STDMETHODIMP Machine::COMSETTER(SnapshotFolder) (IN_BSTR aSnapshotFolder)
1653{
1654 /* @todo (r=dmik):
1655 * 1. Allow to change the name of the snapshot folder containing snapshots
1656 * 2. Rename the folder on disk instead of just changing the property
1657 * value (to be smart and not to leave garbage). Note that it cannot be
1658 * done here because the change may be rolled back. Thus, the right
1659 * place is #saveSettings().
1660 */
1661
1662 AutoCaller autoCaller(this);
1663 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1664
1665 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1666
1667 HRESULT rc = checkStateDependency(MutableStateDep);
1668 if (FAILED(rc)) return rc;
1669
1670 if (!mData->mCurrentSnapshot.isNull())
1671 return setError(E_FAIL,
1672 tr("The snapshot folder of a machine with snapshots cannot be changed (please delete all snapshots first)"));
1673
1674 Utf8Str snapshotFolder = aSnapshotFolder;
1675
1676 if (snapshotFolder.isEmpty())
1677 {
1678 if (isInOwnDir())
1679 {
1680 /* the default snapshots folder is 'Snapshots' in the machine dir */
1681 snapshotFolder = "Snapshots";
1682 }
1683 else
1684 {
1685 /* the default snapshots folder is {UUID}, for backwards
1686 * compatibility and to resolve conflicts */
1687 snapshotFolder = Utf8StrFmt("{%RTuuid}", mData->mUuid.raw());
1688 }
1689 }
1690
1691 int vrc = calculateFullPath(snapshotFolder, snapshotFolder);
1692 if (RT_FAILURE(vrc))
1693 return setError(E_FAIL,
1694 tr("Invalid snapshot folder '%ls' (%Rrc)"),
1695 aSnapshotFolder, vrc);
1696
1697 setModified(IsModified_MachineData);
1698 mUserData.backup();
1699 mUserData->mSnapshotFolder = aSnapshotFolder;
1700 mUserData->mSnapshotFolderFull = snapshotFolder;
1701
1702 return S_OK;
1703}
1704
1705STDMETHODIMP Machine::COMGETTER(MediumAttachments)(ComSafeArrayOut(IMediumAttachment*, aAttachments))
1706{
1707 if (ComSafeArrayOutIsNull(aAttachments))
1708 return E_POINTER;
1709
1710 AutoCaller autoCaller(this);
1711 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1712
1713 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1714
1715 SafeIfaceArray<IMediumAttachment> attachments(mMediaData->mAttachments);
1716 attachments.detachTo(ComSafeArrayOutArg(aAttachments));
1717
1718 return S_OK;
1719}
1720
1721STDMETHODIMP Machine::COMGETTER(VRDPServer)(IVRDPServer **vrdpServer)
1722{
1723#ifdef VBOX_WITH_VRDP
1724 if (!vrdpServer)
1725 return E_POINTER;
1726
1727 AutoCaller autoCaller(this);
1728 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1729
1730 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1731
1732 Assert (!!mVRDPServer);
1733 mVRDPServer.queryInterfaceTo(vrdpServer);
1734
1735 return S_OK;
1736#else
1737 NOREF(vrdpServer);
1738 ReturnComNotImplemented();
1739#endif
1740}
1741
1742STDMETHODIMP Machine::COMGETTER(AudioAdapter)(IAudioAdapter **audioAdapter)
1743{
1744 if (!audioAdapter)
1745 return E_POINTER;
1746
1747 AutoCaller autoCaller(this);
1748 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1749
1750 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1751
1752 mAudioAdapter.queryInterfaceTo(audioAdapter);
1753 return S_OK;
1754}
1755
1756STDMETHODIMP Machine::COMGETTER(USBController) (IUSBController **aUSBController)
1757{
1758#ifdef VBOX_WITH_USB
1759 CheckComArgOutPointerValid(aUSBController);
1760
1761 AutoCaller autoCaller(this);
1762 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1763
1764 MultiResult rc = mParent->host()->checkUSBProxyService();
1765 if (FAILED(rc)) return rc;
1766
1767 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1768
1769 return rc = mUSBController.queryInterfaceTo(aUSBController);
1770#else
1771 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1772 * extended error info to indicate that USB is simply not available
1773 * (w/o treting it as a failure), for example, as in OSE */
1774 NOREF(aUSBController);
1775 ReturnComNotImplemented();
1776#endif
1777}
1778
1779STDMETHODIMP Machine::COMGETTER(SettingsFilePath) (BSTR *aFilePath)
1780{
1781 CheckComArgOutPointerValid(aFilePath);
1782
1783 AutoLimitedCaller autoCaller(this);
1784 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1785
1786 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1787
1788 mData->m_strConfigFileFull.cloneTo(aFilePath);
1789 return S_OK;
1790}
1791
1792STDMETHODIMP Machine::COMGETTER(SettingsModified) (BOOL *aModified)
1793{
1794 CheckComArgOutPointerValid(aModified);
1795
1796 AutoCaller autoCaller(this);
1797 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1798
1799 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1800
1801 HRESULT rc = checkStateDependency(MutableStateDep);
1802 if (FAILED(rc)) return rc;
1803
1804 if (mData->mInitMode == Init_New)
1805 /*
1806 * if this is a new machine then no config file exists yet, so always return TRUE
1807 */
1808 *aModified = TRUE;
1809 else
1810 *aModified = (m_flModifications != 0);
1811
1812 return S_OK;
1813}
1814
1815STDMETHODIMP Machine::COMGETTER(SessionState) (SessionState_T *aSessionState)
1816{
1817 CheckComArgOutPointerValid(aSessionState);
1818
1819 AutoCaller autoCaller(this);
1820 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1821
1822 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1823
1824 *aSessionState = mData->mSession.mState;
1825
1826 return S_OK;
1827}
1828
1829STDMETHODIMP Machine::COMGETTER(SessionType) (BSTR *aSessionType)
1830{
1831 CheckComArgOutPointerValid(aSessionType);
1832
1833 AutoCaller autoCaller(this);
1834 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1835
1836 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1837
1838 if (mData->mSession.mType.isNull())
1839 Bstr("").cloneTo(aSessionType);
1840 else
1841 mData->mSession.mType.cloneTo(aSessionType);
1842
1843 return S_OK;
1844}
1845
1846STDMETHODIMP Machine::COMGETTER(SessionPid) (ULONG *aSessionPid)
1847{
1848 CheckComArgOutPointerValid(aSessionPid);
1849
1850 AutoCaller autoCaller(this);
1851 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1852
1853 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1854
1855 *aSessionPid = mData->mSession.mPid;
1856
1857 return S_OK;
1858}
1859
1860STDMETHODIMP Machine::COMGETTER(State) (MachineState_T *machineState)
1861{
1862 if (!machineState)
1863 return E_POINTER;
1864
1865 AutoCaller autoCaller(this);
1866 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1867
1868 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1869
1870 *machineState = mData->mMachineState;
1871
1872 return S_OK;
1873}
1874
1875STDMETHODIMP Machine::COMGETTER(LastStateChange) (LONG64 *aLastStateChange)
1876{
1877 CheckComArgOutPointerValid(aLastStateChange);
1878
1879 AutoCaller autoCaller(this);
1880 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1881
1882 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1883
1884 *aLastStateChange = RTTimeSpecGetMilli (&mData->mLastStateChange);
1885
1886 return S_OK;
1887}
1888
1889STDMETHODIMP Machine::COMGETTER(StateFilePath) (BSTR *aStateFilePath)
1890{
1891 CheckComArgOutPointerValid(aStateFilePath);
1892
1893 AutoCaller autoCaller(this);
1894 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1895
1896 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1897
1898 if (mSSData->mStateFilePath.isEmpty())
1899 Bstr("").cloneTo(aStateFilePath);
1900 else
1901 mSSData->mStateFilePath.cloneTo(aStateFilePath);
1902
1903 return S_OK;
1904}
1905
1906STDMETHODIMP Machine::COMGETTER(LogFolder) (BSTR *aLogFolder)
1907{
1908 CheckComArgOutPointerValid(aLogFolder);
1909
1910 AutoCaller autoCaller(this);
1911 AssertComRCReturnRC(autoCaller.rc());
1912
1913 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1914
1915 Utf8Str logFolder;
1916 getLogFolder (logFolder);
1917
1918 Bstr (logFolder).cloneTo(aLogFolder);
1919
1920 return S_OK;
1921}
1922
1923STDMETHODIMP Machine::COMGETTER(CurrentSnapshot) (ISnapshot **aCurrentSnapshot)
1924{
1925 CheckComArgOutPointerValid(aCurrentSnapshot);
1926
1927 AutoCaller autoCaller(this);
1928 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1929
1930 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1931
1932 mData->mCurrentSnapshot.queryInterfaceTo(aCurrentSnapshot);
1933
1934 return S_OK;
1935}
1936
1937STDMETHODIMP Machine::COMGETTER(SnapshotCount)(ULONG *aSnapshotCount)
1938{
1939 CheckComArgOutPointerValid(aSnapshotCount);
1940
1941 AutoCaller autoCaller(this);
1942 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1943
1944 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1945
1946 *aSnapshotCount = mData->mFirstSnapshot.isNull()
1947 ? 0
1948 : mData->mFirstSnapshot->getAllChildrenCount() + 1;
1949
1950 return S_OK;
1951}
1952
1953STDMETHODIMP Machine::COMGETTER(CurrentStateModified) (BOOL *aCurrentStateModified)
1954{
1955 CheckComArgOutPointerValid(aCurrentStateModified);
1956
1957 AutoCaller autoCaller(this);
1958 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1959
1960 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1961
1962 /* Note: for machines with no snapshots, we always return FALSE
1963 * (mData->mCurrentStateModified will be TRUE in this case, for historical
1964 * reasons :) */
1965
1966 *aCurrentStateModified = mData->mFirstSnapshot.isNull()
1967 ? FALSE
1968 : mData->mCurrentStateModified;
1969
1970 return S_OK;
1971}
1972
1973STDMETHODIMP Machine::COMGETTER(SharedFolders) (ComSafeArrayOut(ISharedFolder *, aSharedFolders))
1974{
1975 CheckComArgOutSafeArrayPointerValid(aSharedFolders);
1976
1977 AutoCaller autoCaller(this);
1978 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1979
1980 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1981
1982 SafeIfaceArray<ISharedFolder> folders(mHWData->mSharedFolders);
1983 folders.detachTo(ComSafeArrayOutArg(aSharedFolders));
1984
1985 return S_OK;
1986}
1987
1988STDMETHODIMP Machine::COMGETTER(ClipboardMode) (ClipboardMode_T *aClipboardMode)
1989{
1990 CheckComArgOutPointerValid(aClipboardMode);
1991
1992 AutoCaller autoCaller(this);
1993 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1994
1995 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1996
1997 *aClipboardMode = mHWData->mClipboardMode;
1998
1999 return S_OK;
2000}
2001
2002STDMETHODIMP
2003Machine::COMSETTER(ClipboardMode) (ClipboardMode_T aClipboardMode)
2004{
2005 AutoCaller autoCaller(this);
2006 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2007
2008 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2009
2010 HRESULT rc = checkStateDependency(MutableStateDep);
2011 if (FAILED(rc)) return rc;
2012
2013 setModified(IsModified_MachineData);
2014 mHWData.backup();
2015 mHWData->mClipboardMode = aClipboardMode;
2016
2017 return S_OK;
2018}
2019
2020STDMETHODIMP
2021Machine::COMGETTER(GuestPropertyNotificationPatterns)(BSTR *aPatterns)
2022{
2023 CheckComArgOutPointerValid(aPatterns);
2024
2025 AutoCaller autoCaller(this);
2026 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2027
2028 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2029
2030 try
2031 {
2032 mHWData->mGuestPropertyNotificationPatterns.cloneTo(aPatterns);
2033 }
2034 catch (...)
2035 {
2036 return VirtualBox::handleUnexpectedExceptions(RT_SRC_POS);
2037 }
2038
2039 return S_OK;
2040}
2041
2042STDMETHODIMP
2043Machine::COMSETTER(GuestPropertyNotificationPatterns)(IN_BSTR aPatterns)
2044{
2045 CheckComArgNotNull(aPatterns);
2046 AutoCaller autoCaller(this);
2047 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2048
2049 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2050
2051 HRESULT rc = checkStateDependency(MutableStateDep);
2052 if (FAILED(rc)) return rc;
2053
2054 setModified(IsModified_MachineData);
2055 mHWData.backup();
2056 mHWData->mGuestPropertyNotificationPatterns = aPatterns;
2057 return rc;
2058}
2059
2060STDMETHODIMP
2061Machine::COMGETTER(StorageControllers) (ComSafeArrayOut(IStorageController *, aStorageControllers))
2062{
2063 CheckComArgOutSafeArrayPointerValid(aStorageControllers);
2064
2065 AutoCaller autoCaller(this);
2066 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2067
2068 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2069
2070 SafeIfaceArray<IStorageController> ctrls (*mStorageControllers.data());
2071 ctrls.detachTo(ComSafeArrayOutArg(aStorageControllers));
2072
2073 return S_OK;
2074}
2075
2076STDMETHODIMP
2077Machine::COMGETTER(TeleporterEnabled)(BOOL *aEnabled)
2078{
2079 CheckComArgOutPointerValid(aEnabled);
2080
2081 AutoCaller autoCaller(this);
2082 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2083
2084 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2085
2086 *aEnabled = mUserData->mTeleporterEnabled;
2087
2088 return S_OK;
2089}
2090
2091STDMETHODIMP Machine::COMSETTER(TeleporterEnabled)(BOOL aEnabled)
2092{
2093 AutoCaller autoCaller(this);
2094 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2095
2096 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2097
2098 /* Only allow it to be set to true when PoweredOff or Aborted.
2099 (Clearing it is always permitted.) */
2100 if ( aEnabled
2101 && mData->mRegistered
2102 && ( getClassID() != clsidSessionMachine
2103 || ( mData->mMachineState != MachineState_PoweredOff
2104 && mData->mMachineState != MachineState_Teleported
2105 && mData->mMachineState != MachineState_Aborted
2106 )
2107 )
2108 )
2109 return setError(VBOX_E_INVALID_VM_STATE,
2110 tr("The machine is not powered off (state is %s)"),
2111 Global::stringifyMachineState(mData->mMachineState));
2112
2113 setModified(IsModified_MachineData);
2114 mUserData.backup();
2115 mUserData->mTeleporterEnabled = aEnabled;
2116
2117 return S_OK;
2118}
2119
2120STDMETHODIMP Machine::COMGETTER(TeleporterPort)(ULONG *aPort)
2121{
2122 CheckComArgOutPointerValid(aPort);
2123
2124 AutoCaller autoCaller(this);
2125 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2126
2127 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2128
2129 *aPort = mUserData->mTeleporterPort;
2130
2131 return S_OK;
2132}
2133
2134STDMETHODIMP Machine::COMSETTER(TeleporterPort)(ULONG aPort)
2135{
2136 if (aPort >= _64K)
2137 return setError(E_INVALIDARG, tr("Invalid port number %d"), aPort);
2138
2139 AutoCaller autoCaller(this);
2140 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2141
2142 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2143
2144 HRESULT rc = checkStateDependency(MutableStateDep);
2145 if (FAILED(rc)) return rc;
2146
2147 setModified(IsModified_MachineData);
2148 mUserData.backup();
2149 mUserData->mTeleporterPort = aPort;
2150
2151 return S_OK;
2152}
2153
2154STDMETHODIMP Machine::COMGETTER(TeleporterAddress)(BSTR *aAddress)
2155{
2156 CheckComArgOutPointerValid(aAddress);
2157
2158 AutoCaller autoCaller(this);
2159 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2160
2161 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2162
2163 mUserData->mTeleporterAddress.cloneTo(aAddress);
2164
2165 return S_OK;
2166}
2167
2168STDMETHODIMP Machine::COMSETTER(TeleporterAddress)(IN_BSTR aAddress)
2169{
2170 AutoCaller autoCaller(this);
2171 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2172
2173 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2174
2175 HRESULT rc = checkStateDependency(MutableStateDep);
2176 if (FAILED(rc)) return rc;
2177
2178 setModified(IsModified_MachineData);
2179 mUserData.backup();
2180 mUserData->mTeleporterAddress = aAddress;
2181
2182 return S_OK;
2183}
2184
2185STDMETHODIMP Machine::COMGETTER(TeleporterPassword)(BSTR *aPassword)
2186{
2187 CheckComArgOutPointerValid(aPassword);
2188
2189 AutoCaller autoCaller(this);
2190 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2191
2192 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2193
2194 mUserData->mTeleporterPassword.cloneTo(aPassword);
2195
2196 return S_OK;
2197}
2198
2199STDMETHODIMP Machine::COMSETTER(TeleporterPassword)(IN_BSTR aPassword)
2200{
2201 AutoCaller autoCaller(this);
2202 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2203
2204 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2205
2206 HRESULT rc = checkStateDependency(MutableStateDep);
2207 if (FAILED(rc)) return rc;
2208
2209 setModified(IsModified_MachineData);
2210 mUserData.backup();
2211 mUserData->mTeleporterPassword = aPassword;
2212
2213 return S_OK;
2214}
2215
2216STDMETHODIMP Machine::COMGETTER(RTCUseUTC)(BOOL *aEnabled)
2217{
2218 CheckComArgOutPointerValid(aEnabled);
2219
2220 AutoCaller autoCaller(this);
2221 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2222
2223 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2224
2225 *aEnabled = mUserData->mRTCUseUTC;
2226
2227 return S_OK;
2228}
2229
2230STDMETHODIMP Machine::COMSETTER(RTCUseUTC)(BOOL aEnabled)
2231{
2232 AutoCaller autoCaller(this);
2233 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2234
2235 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2236
2237 /* Only allow it to be set to true when PoweredOff or Aborted.
2238 (Clearing it is always permitted.) */
2239 if ( aEnabled
2240 && mData->mRegistered
2241 && ( getClassID() != clsidSessionMachine
2242 || ( mData->mMachineState != MachineState_PoweredOff
2243 && mData->mMachineState != MachineState_Teleported
2244 && mData->mMachineState != MachineState_Aborted
2245 )
2246 )
2247 )
2248 return setError(VBOX_E_INVALID_VM_STATE,
2249 tr("The machine is not powered off (state is %s)"),
2250 Global::stringifyMachineState(mData->mMachineState));
2251
2252 setModified(IsModified_MachineData);
2253 mUserData.backup();
2254 mUserData->mRTCUseUTC = aEnabled;
2255
2256 return S_OK;
2257}
2258
2259STDMETHODIMP Machine::SetBootOrder(ULONG aPosition, DeviceType_T aDevice)
2260{
2261 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
2262 return setError(E_INVALIDARG,
2263 tr ("Invalid boot position: %lu (must be in range [1, %lu])"),
2264 aPosition, SchemaDefs::MaxBootPosition);
2265
2266 if (aDevice == DeviceType_USB)
2267 return setError(E_NOTIMPL,
2268 tr("Booting from USB device is currently not supported"));
2269
2270 AutoCaller autoCaller(this);
2271 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2272
2273 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2274
2275 HRESULT rc = checkStateDependency(MutableStateDep);
2276 if (FAILED(rc)) return rc;
2277
2278 setModified(IsModified_MachineData);
2279 mHWData.backup();
2280 mHWData->mBootOrder [aPosition - 1] = aDevice;
2281
2282 return S_OK;
2283}
2284
2285STDMETHODIMP Machine::GetBootOrder (ULONG aPosition, DeviceType_T *aDevice)
2286{
2287 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
2288 return setError(E_INVALIDARG,
2289 tr("Invalid boot position: %lu (must be in range [1, %lu])"),
2290 aPosition, SchemaDefs::MaxBootPosition);
2291
2292 AutoCaller autoCaller(this);
2293 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2294
2295 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2296
2297 *aDevice = mHWData->mBootOrder [aPosition - 1];
2298
2299 return S_OK;
2300}
2301
2302STDMETHODIMP Machine::AttachDevice(IN_BSTR aControllerName,
2303 LONG aControllerPort,
2304 LONG aDevice,
2305 DeviceType_T aType,
2306 IN_BSTR aId)
2307{
2308 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aType=%d aId=\"%ls\"\n",
2309 aControllerName, aControllerPort, aDevice, aType, aId));
2310
2311 CheckComArgNotNull(aControllerName);
2312 CheckComArgNotNull(aId);
2313
2314 AutoCaller autoCaller(this);
2315 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2316
2317 // if this becomes true then we need to call saveSettings in the end
2318 // @todo r=dj there is no error handling so far...
2319 bool fNeedsSaveSettings = false;
2320
2321 /* protect the media tree all the while we're in here, as well as our member variables */
2322 AutoMultiWriteLock2 alock(this->lockHandle(), &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2323
2324 HRESULT rc = checkStateDependency(MutableStateDep);
2325 if (FAILED(rc)) return rc;
2326
2327 /// @todo NEWMEDIA implicit machine registration
2328 if (!mData->mRegistered)
2329 return setError(VBOX_E_INVALID_OBJECT_STATE,
2330 tr("Cannot attach storage devices to an unregistered machine"));
2331
2332 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
2333
2334 if (Global::IsOnlineOrTransient(mData->mMachineState))
2335 return setError(VBOX_E_INVALID_VM_STATE,
2336 tr("Invalid machine state: %s"),
2337 Global::stringifyMachineState(mData->mMachineState));
2338
2339 /* Check for an existing controller. */
2340 ComObjPtr<StorageController> ctl;
2341 rc = getStorageControllerByName(aControllerName, ctl, true /* aSetError */);
2342 if (FAILED(rc)) return rc;
2343
2344 /* check that the port and device are not out of range. */
2345 ULONG portCount;
2346 ULONG devicesPerPort;
2347 rc = ctl->COMGETTER(PortCount)(&portCount);
2348 if (FAILED(rc)) return rc;
2349 rc = ctl->COMGETTER(MaxDevicesPerPortCount)(&devicesPerPort);
2350 if (FAILED(rc)) return rc;
2351
2352 if ( (aControllerPort < 0)
2353 || (aControllerPort >= (LONG)portCount)
2354 || (aDevice < 0)
2355 || (aDevice >= (LONG)devicesPerPort)
2356 )
2357 return setError(E_INVALIDARG,
2358 tr("The port and/or count parameter are out of range [%lu:%lu]"),
2359 portCount,
2360 devicesPerPort);
2361
2362 /* check if the device slot is already busy */
2363 MediumAttachment *pAttachTemp;
2364 if ((pAttachTemp = findAttachment(mMediaData->mAttachments,
2365 aControllerName,
2366 aControllerPort,
2367 aDevice)))
2368 {
2369 Medium *pMedium = pAttachTemp->getMedium();
2370 if (pMedium)
2371 {
2372 AutoReadLock mediumLock(pMedium COMMA_LOCKVAL_SRC_POS);
2373 return setError(VBOX_E_OBJECT_IN_USE,
2374 tr("Medium '%s' is already attached to device slot %d on port %d of controller '%ls' of this virtual machine"),
2375 pMedium->getLocationFull().raw(),
2376 aDevice,
2377 aControllerPort,
2378 aControllerName);
2379 }
2380 else
2381 return setError(VBOX_E_OBJECT_IN_USE,
2382 tr("Device is already attached to slot %d on port %d of controller '%ls' of this virtual machine"),
2383 aDevice, aControllerPort, aControllerName);
2384 }
2385
2386 Guid uuid(aId);
2387
2388 ComObjPtr<Medium> medium;
2389 switch (aType)
2390 {
2391 case DeviceType_HardDisk:
2392 /* find a hard disk by UUID */
2393 rc = mParent->findHardDisk(&uuid, NULL, true /* aSetError */, &medium);
2394 if (FAILED(rc)) return rc;
2395 break;
2396
2397 case DeviceType_DVD: // @todo r=dj eliminate this, replace with findDVDImage
2398 if (!uuid.isEmpty())
2399 {
2400 /* first search for host drive */
2401 SafeIfaceArray<IMedium> drivevec;
2402 rc = mParent->host()->COMGETTER(DVDDrives)(ComSafeArrayAsOutParam(drivevec));
2403 if (SUCCEEDED(rc))
2404 {
2405 for (size_t i = 0; i < drivevec.size(); ++i)
2406 {
2407 /// @todo eliminate this conversion
2408 ComObjPtr<Medium> med = (Medium *)drivevec[i];
2409 if (med->getId() == uuid)
2410 {
2411 medium = med;
2412 break;
2413 }
2414 }
2415 }
2416
2417 if (medium.isNull())
2418 {
2419 /* find a DVD image by UUID */
2420 rc = mParent->findDVDImage(&uuid, NULL, true /* aSetError */, &medium);
2421 if (FAILED(rc)) return rc;
2422 }
2423 }
2424 else
2425 {
2426 /* null UUID means null medium, which needs no code */
2427 }
2428 break;
2429
2430 case DeviceType_Floppy: // @todo r=dj eliminate this, replace with findFloppyImage
2431 if (!uuid.isEmpty())
2432 {
2433 /* first search for host drive */
2434 SafeIfaceArray<IMedium> drivevec;
2435 rc = mParent->host()->COMGETTER(FloppyDrives)(ComSafeArrayAsOutParam(drivevec));
2436 if (SUCCEEDED(rc))
2437 {
2438 for (size_t i = 0; i < drivevec.size(); ++i)
2439 {
2440 /// @todo eliminate this conversion
2441 ComObjPtr<Medium> med = (Medium *)drivevec[i];
2442 if (med->getId() == uuid)
2443 {
2444 medium = med;
2445 break;
2446 }
2447 }
2448 }
2449
2450 if (medium.isNull())
2451 {
2452 /* find a floppy image by UUID */
2453 rc = mParent->findFloppyImage(&uuid, NULL, true /* aSetError */, &medium);
2454 if (FAILED(rc)) return rc;
2455 }
2456 }
2457 else
2458 {
2459 /* null UUID means null medium, which needs no code */
2460 }
2461 break;
2462
2463 default:
2464 return setError(E_INVALIDARG,
2465 tr("The device type %d is not recognized"),
2466 (int)aType);
2467 }
2468
2469 AutoCaller mediumCaller(medium);
2470 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
2471
2472 AutoWriteLock mediumLock(medium COMMA_LOCKVAL_SRC_POS);
2473
2474 if ( (pAttachTemp = findAttachment(mMediaData->mAttachments, medium))
2475 && !medium.isNull()
2476 )
2477 return setError(VBOX_E_OBJECT_IN_USE,
2478 tr("Medium '%s' is already attached to this virtual machine"),
2479 medium->getLocationFull().raw());
2480
2481 bool indirect = false;
2482 if (!medium.isNull())
2483 indirect = medium->isReadOnly();
2484 bool associate = true;
2485
2486 do
2487 {
2488 if (aType == DeviceType_HardDisk && mMediaData.isBackedUp())
2489 {
2490 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
2491
2492 /* check if the medium was attached to the VM before we started
2493 * changing attachments in which case the attachment just needs to
2494 * be restored */
2495 if ((pAttachTemp = findAttachment(oldAtts, medium)))
2496 {
2497 AssertReturn(!indirect, E_FAIL);
2498
2499 /* see if it's the same bus/channel/device */
2500 if (pAttachTemp->matches(aControllerName, aControllerPort, aDevice))
2501 {
2502 /* the simplest case: restore the whole attachment
2503 * and return, nothing else to do */
2504 mMediaData->mAttachments.push_back(pAttachTemp);
2505 return S_OK;
2506 }
2507
2508 /* bus/channel/device differ; we need a new attachment object,
2509 * but don't try to associate it again */
2510 associate = false;
2511 break;
2512 }
2513 }
2514
2515 /* go further only if the attachment is to be indirect */
2516 if (!indirect)
2517 break;
2518
2519 /* perform the so called smart attachment logic for indirect
2520 * attachments. Note that smart attachment is only applicable to base
2521 * hard disks. */
2522
2523 if (medium->getParent().isNull())
2524 {
2525 /* first, investigate the backup copy of the current hard disk
2526 * attachments to make it possible to re-attach existing diffs to
2527 * another device slot w/o losing their contents */
2528 if (mMediaData.isBackedUp())
2529 {
2530 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
2531
2532 MediaData::AttachmentList::const_iterator foundIt = oldAtts.end();
2533 uint32_t foundLevel = 0;
2534
2535 for (MediaData::AttachmentList::const_iterator it = oldAtts.begin();
2536 it != oldAtts.end();
2537 ++it)
2538 {
2539 uint32_t level = 0;
2540 MediumAttachment *pAttach = *it;
2541 ComObjPtr<Medium> pMedium = pAttach->getMedium();
2542 Assert(!pMedium.isNull() || pAttach->getType() != DeviceType_HardDisk);
2543 if (pMedium.isNull())
2544 continue;
2545
2546 if (pMedium->getBase(&level).equalsTo(medium))
2547 {
2548 /* skip the hard disk if its currently attached (we
2549 * cannot attach the same hard disk twice) */
2550 if (findAttachment(mMediaData->mAttachments,
2551 pMedium))
2552 continue;
2553
2554 /* matched device, channel and bus (i.e. attached to the
2555 * same place) will win and immediately stop the search;
2556 * otherwise the attachment that has the youngest
2557 * descendant of medium will be used
2558 */
2559 if (pAttach->matches(aControllerName, aControllerPort, aDevice))
2560 {
2561 /* the simplest case: restore the whole attachment
2562 * and return, nothing else to do */
2563 mMediaData->mAttachments.push_back(*it);
2564 return S_OK;
2565 }
2566 else if ( foundIt == oldAtts.end()
2567 || level > foundLevel /* prefer younger */
2568 )
2569 {
2570 foundIt = it;
2571 foundLevel = level;
2572 }
2573 }
2574 }
2575
2576 if (foundIt != oldAtts.end())
2577 {
2578 /* use the previously attached hard disk */
2579 medium = (*foundIt)->getMedium();
2580 mediumCaller.attach(medium);
2581 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
2582 mediumLock.attach(medium);
2583 /* not implicit, doesn't require association with this VM */
2584 indirect = false;
2585 associate = false;
2586 /* go right to the MediumAttachment creation */
2587 break;
2588 }
2589 }
2590
2591 /* then, search through snapshots for the best diff in the given
2592 * hard disk's chain to base the new diff on */
2593
2594 ComObjPtr<Medium> base;
2595 ComObjPtr<Snapshot> snap = mData->mCurrentSnapshot;
2596 while (snap)
2597 {
2598 AutoReadLock snapLock(snap COMMA_LOCKVAL_SRC_POS);
2599
2600 const MediaData::AttachmentList &snapAtts = snap->getSnapshotMachine()->mMediaData->mAttachments;
2601
2602 MediaData::AttachmentList::const_iterator foundIt = snapAtts.end();
2603 uint32_t foundLevel = 0;
2604
2605 for (MediaData::AttachmentList::const_iterator it = snapAtts.begin();
2606 it != snapAtts.end();
2607 ++it)
2608 {
2609 MediumAttachment *pAttach = *it;
2610 ComObjPtr<Medium> pMedium = pAttach->getMedium();
2611 Assert(!pMedium.isNull() || pAttach->getType() != DeviceType_HardDisk);
2612 if (pMedium.isNull())
2613 continue;
2614
2615 uint32_t level = 0;
2616 if (pMedium->getBase(&level).equalsTo(medium))
2617 {
2618 /* matched device, channel and bus (i.e. attached to the
2619 * same place) will win and immediately stop the search;
2620 * otherwise the attachment that has the youngest
2621 * descendant of medium will be used
2622 */
2623 if ( (*it)->getDevice() == aDevice
2624 && (*it)->getPort() == aControllerPort
2625 && (*it)->getControllerName() == aControllerName
2626 )
2627 {
2628 foundIt = it;
2629 break;
2630 }
2631 else if ( foundIt == snapAtts.end()
2632 || level > foundLevel /* prefer younger */
2633 )
2634 {
2635 foundIt = it;
2636 foundLevel = level;
2637 }
2638 }
2639 }
2640
2641 if (foundIt != snapAtts.end())
2642 {
2643 base = (*foundIt)->getMedium();
2644 break;
2645 }
2646
2647 snap = snap->getParent();
2648 }
2649
2650 /* found a suitable diff, use it as a base */
2651 if (!base.isNull())
2652 {
2653 medium = base;
2654 mediumCaller.attach(medium);
2655 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
2656 mediumLock.attach(medium);
2657 }
2658 }
2659
2660 ComObjPtr<Medium> diff;
2661 diff.createObject();
2662 rc = diff->init(mParent,
2663 medium->preferredDiffFormat().raw(),
2664 BstrFmt("%ls"RTPATH_SLASH_STR,
2665 mUserData->mSnapshotFolderFull.raw()).raw(),
2666 &fNeedsSaveSettings);
2667 if (FAILED(rc)) return rc;
2668
2669 /* make sure the hard disk is not modified before createDiffStorage() */
2670 rc = medium->LockRead(NULL);
2671 if (FAILED(rc)) return rc;
2672
2673 /* will leave the lock before the potentially lengthy operation, so
2674 * protect with the special state */
2675 MachineState_T oldState = mData->mMachineState;
2676 setMachineState(MachineState_SettingUp);
2677
2678 mediumLock.leave();
2679 alock.leave();
2680
2681 rc = medium->createDiffStorageAndWait(diff, MediumVariant_Standard, &fNeedsSaveSettings);
2682
2683 alock.enter();
2684 mediumLock.enter();
2685
2686 setMachineState(oldState);
2687
2688 medium->UnlockRead(NULL);
2689
2690 if (FAILED(rc)) return rc;
2691
2692 /* use the created diff for the actual attachment */
2693 medium = diff;
2694 mediumCaller.attach(medium);
2695 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
2696 mediumLock.attach(medium);
2697 }
2698 while (0);
2699
2700 ComObjPtr<MediumAttachment> attachment;
2701 attachment.createObject();
2702 rc = attachment->init(this, medium, aControllerName, aControllerPort, aDevice, aType, indirect);
2703 if (FAILED(rc)) return rc;
2704
2705 if (associate && !medium.isNull())
2706 {
2707 /* as the last step, associate the medium to the VM */
2708 rc = medium->attachTo(mData->mUuid);
2709 /* here we can fail because of Deleting, or being in process of
2710 * creating a Diff */
2711 if (FAILED(rc)) return rc;
2712 }
2713
2714 /* success: finally remember the attachment */
2715 setModified(IsModified_Storage);
2716 mMediaData.backup();
2717 mMediaData->mAttachments.push_back(attachment);
2718
2719 if (fNeedsSaveSettings)
2720 {
2721 mediumLock.release();
2722 alock.release();
2723
2724 AutoWriteLock vboxLock(mParent COMMA_LOCKVAL_SRC_POS);
2725 mParent->saveSettings();
2726 }
2727
2728 return rc;
2729}
2730
2731STDMETHODIMP Machine::DetachDevice(IN_BSTR aControllerName, LONG aControllerPort,
2732 LONG aDevice)
2733{
2734 CheckComArgNotNull(aControllerName);
2735
2736 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%ld aDevice=%ld\n",
2737 aControllerName, aControllerPort, aDevice));
2738
2739 AutoCaller autoCaller(this);
2740 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2741
2742 bool fNeedsSaveSettings = false;
2743
2744 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2745
2746 HRESULT rc = checkStateDependency(MutableStateDep);
2747 if (FAILED(rc)) return rc;
2748
2749 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
2750
2751 if (Global::IsOnlineOrTransient(mData->mMachineState))
2752 return setError(VBOX_E_INVALID_VM_STATE,
2753 tr("Invalid machine state: %s"),
2754 Global::stringifyMachineState(mData->mMachineState));
2755
2756 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
2757 aControllerName,
2758 aControllerPort,
2759 aDevice);
2760 if (!pAttach)
2761 return setError(VBOX_E_OBJECT_NOT_FOUND,
2762 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
2763 aDevice, aControllerPort, aControllerName);
2764
2765 ComObjPtr<Medium> oldmedium = pAttach->getMedium();
2766 DeviceType_T mediumType = pAttach->getType();
2767
2768 if (pAttach->isImplicit())
2769 {
2770 /* attempt to implicitly delete the implicitly created diff */
2771
2772 /// @todo move the implicit flag from MediumAttachment to Medium
2773 /// and forbid any hard disk operation when it is implicit. Or maybe
2774 /// a special media state for it to make it even more simple.
2775
2776 Assert(mMediaData.isBackedUp());
2777
2778 /* will leave the lock before the potentially lengthy operation, so
2779 * protect with the special state */
2780 MachineState_T oldState = mData->mMachineState;
2781 setMachineState(MachineState_SettingUp);
2782
2783 alock.leave();
2784
2785 rc = oldmedium->deleteStorageAndWait(NULL /*aProgress*/, &fNeedsSaveSettings);
2786
2787 alock.enter();
2788
2789 setMachineState(oldState);
2790
2791 if (FAILED(rc)) return rc;
2792 }
2793
2794 setModified(IsModified_Storage);
2795 mMediaData.backup();
2796
2797 /* we cannot use erase (it) below because backup() above will create
2798 * a copy of the list and make this copy active, but the iterator
2799 * still refers to the original and is not valid for the copy */
2800 mMediaData->mAttachments.remove(pAttach);
2801
2802 /* For non-hard disk media, detach straight away. */
2803 if (mediumType != DeviceType_HardDisk && !oldmedium.isNull())
2804 oldmedium->detachFrom(mData->mUuid);
2805
2806 if (fNeedsSaveSettings)
2807 {
2808 alock.release();
2809 AutoWriteLock vboxlock(this COMMA_LOCKVAL_SRC_POS);
2810 saveSettings();
2811 }
2812
2813 return S_OK;
2814}
2815
2816STDMETHODIMP Machine::PassthroughDevice(IN_BSTR aControllerName, LONG aControllerPort,
2817 LONG aDevice, BOOL aPassthrough)
2818{
2819 CheckComArgNotNull(aControllerName);
2820
2821 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%ld aDevice=%ld aPassthrough=%d\n",
2822 aControllerName, aControllerPort, aDevice, aPassthrough));
2823
2824 AutoCaller autoCaller(this);
2825 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2826
2827 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2828
2829 HRESULT rc = checkStateDependency(MutableStateDep);
2830 if (FAILED(rc)) return rc;
2831
2832 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
2833
2834 if (Global::IsOnlineOrTransient(mData->mMachineState))
2835 return setError(VBOX_E_INVALID_VM_STATE,
2836 tr("Invalid machine state: %s"),
2837 Global::stringifyMachineState(mData->mMachineState));
2838
2839 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
2840 aControllerName,
2841 aControllerPort,
2842 aDevice);
2843 if (!pAttach)
2844 return setError(VBOX_E_OBJECT_NOT_FOUND,
2845 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
2846 aDevice, aControllerPort, aControllerName);
2847
2848
2849 setModified(IsModified_Storage);
2850 mMediaData.backup();
2851
2852 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
2853
2854 if (pAttach->getType() != DeviceType_DVD)
2855 return setError(E_INVALIDARG,
2856 tr("Setting passthrough rejected as the device attached to device slot %d on port %d of controller '%ls' is not a DVD"),
2857 aDevice, aControllerPort, aControllerName);
2858 pAttach->updatePassthrough(!!aPassthrough);
2859
2860 return S_OK;
2861}
2862
2863STDMETHODIMP Machine::MountMedium(IN_BSTR aControllerName,
2864 LONG aControllerPort,
2865 LONG aDevice,
2866 IN_BSTR aId,
2867 BOOL aForce)
2868{
2869 int rc = S_OK;
2870 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%ld aDevice=%ld aForce=%d\n",
2871 aControllerName, aControllerPort, aDevice, aForce));
2872
2873 CheckComArgNotNull(aControllerName);
2874 CheckComArgNotNull(aId);
2875
2876 AutoCaller autoCaller(this);
2877 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2878
2879 // we're calling host methods for getting DVD and floppy drives so lock host first
2880 AutoMultiWriteLock2 alock(mParent->host(), this COMMA_LOCKVAL_SRC_POS);
2881
2882 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
2883 aControllerName,
2884 aControllerPort,
2885 aDevice);
2886 if (pAttach.isNull())
2887 return setError(VBOX_E_OBJECT_NOT_FOUND,
2888 tr("No drive attached to device slot %d on port %d of controller '%ls'"),
2889 aDevice, aControllerPort, aControllerName);
2890
2891 /* Remember previously mounted medium. The medium before taking the
2892 * backup is not necessarily the same thing. */
2893 ComObjPtr<Medium> oldmedium;
2894 oldmedium = pAttach->getMedium();
2895
2896 Guid uuid(aId);
2897 ComObjPtr<Medium> medium;
2898 DeviceType_T mediumType = pAttach->getType();
2899 switch (mediumType)
2900 {
2901 case DeviceType_DVD:
2902 if (!uuid.isEmpty())
2903 {
2904 /* find a DVD by host device UUID */
2905 MediaList llHostDVDDrives;
2906 rc = mParent->host()->getDVDDrives(llHostDVDDrives);
2907 if (SUCCEEDED(rc))
2908 {
2909 for (MediaList::iterator it = llHostDVDDrives.begin();
2910 it != llHostDVDDrives.end();
2911 ++it)
2912 {
2913 ComObjPtr<Medium> &p = *it;
2914 if (uuid == p->getId())
2915 {
2916 medium = p;
2917 break;
2918 }
2919 }
2920 }
2921 /* find a DVD by UUID */
2922 if (medium.isNull())
2923 rc = mParent->findDVDImage(&uuid, NULL, true /* aDoSetError */, &medium);
2924 }
2925 if (FAILED(rc)) return rc;
2926 break;
2927 case DeviceType_Floppy:
2928 if (!uuid.isEmpty())
2929 {
2930 /* find a Floppy by host device UUID */
2931 MediaList llHostFloppyDrives;
2932 rc = mParent->host()->getFloppyDrives(llHostFloppyDrives);
2933 if (SUCCEEDED(rc))
2934 {
2935 for (MediaList::iterator it = llHostFloppyDrives.begin();
2936 it != llHostFloppyDrives.end();
2937 ++it)
2938 {
2939 ComObjPtr<Medium> &p = *it;
2940 if (uuid == p->getId())
2941 {
2942 medium = p;
2943 break;
2944 }
2945 }
2946 }
2947 /* find a Floppy by UUID */
2948 if (medium.isNull())
2949 rc = mParent->findFloppyImage(&uuid, NULL, true /* aDoSetError */, &medium);
2950 }
2951 if (FAILED(rc)) return rc;
2952 break;
2953 default:
2954 return setError(VBOX_E_INVALID_OBJECT_STATE,
2955 tr("Cannot change medium attached to device slot %d on port %d of controller '%ls'"),
2956 aDevice, aControllerPort, aControllerName);
2957 }
2958
2959 if (SUCCEEDED(rc))
2960 {
2961 setModified(IsModified_Storage);
2962 mMediaData.backup();
2963
2964 /* The backup operation makes the pAttach reference point to the
2965 * old settings. Re-get the correct reference. */
2966 pAttach = findAttachment(mMediaData->mAttachments,
2967 aControllerName,
2968 aControllerPort,
2969 aDevice);
2970 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
2971 /* For non-hard disk media, detach straight away. */
2972 if (mediumType != DeviceType_HardDisk && !oldmedium.isNull())
2973 oldmedium->detachFrom(mData->mUuid);
2974 if (!medium.isNull())
2975 medium->attachTo(mData->mUuid);
2976 pAttach->updateMedium(medium, false /* aImplicit */);
2977 }
2978
2979 alock.leave();
2980 rc = onMediumChange(pAttach, aForce);
2981 alock.enter();
2982
2983 /* On error roll back this change only. */
2984 if (FAILED(rc))
2985 {
2986 if (!medium.isNull())
2987 medium->detachFrom(mData->mUuid);
2988 pAttach = findAttachment(mMediaData->mAttachments,
2989 aControllerName,
2990 aControllerPort,
2991 aDevice);
2992 /* If the attachment is gone in the mean time, bail out. */
2993 if (pAttach.isNull())
2994 return rc;
2995 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
2996 /* For non-hard disk media, re-attach straight away. */
2997 if (mediumType != DeviceType_HardDisk && !oldmedium.isNull())
2998 oldmedium->attachTo(mData->mUuid);
2999 pAttach->updateMedium(oldmedium, false /* aImplicit */);
3000 }
3001
3002 return rc;
3003}
3004
3005STDMETHODIMP Machine::GetMedium(IN_BSTR aControllerName,
3006 LONG aControllerPort,
3007 LONG aDevice,
3008 IMedium **aMedium)
3009{
3010 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%ld aDevice=%ld\n",
3011 aControllerName, aControllerPort, aDevice));
3012
3013 CheckComArgNotNull(aControllerName);
3014 CheckComArgOutPointerValid(aMedium);
3015
3016 AutoCaller autoCaller(this);
3017 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3018
3019 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3020
3021 *aMedium = NULL;
3022
3023 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
3024 aControllerName,
3025 aControllerPort,
3026 aDevice);
3027 if (pAttach.isNull())
3028 return setError(VBOX_E_OBJECT_NOT_FOUND,
3029 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
3030 aDevice, aControllerPort, aControllerName);
3031
3032 pAttach->getMedium().queryInterfaceTo(aMedium);
3033
3034 return S_OK;
3035}
3036
3037STDMETHODIMP Machine::GetSerialPort (ULONG slot, ISerialPort **port)
3038{
3039 CheckComArgOutPointerValid(port);
3040 CheckComArgExpr (slot, slot < RT_ELEMENTS (mSerialPorts));
3041
3042 AutoCaller autoCaller(this);
3043 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3044
3045 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3046
3047 mSerialPorts [slot].queryInterfaceTo(port);
3048
3049 return S_OK;
3050}
3051
3052STDMETHODIMP Machine::GetParallelPort (ULONG slot, IParallelPort **port)
3053{
3054 CheckComArgOutPointerValid(port);
3055 CheckComArgExpr (slot, slot < RT_ELEMENTS (mParallelPorts));
3056
3057 AutoCaller autoCaller(this);
3058 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3059
3060 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3061
3062 mParallelPorts [slot].queryInterfaceTo(port);
3063
3064 return S_OK;
3065}
3066
3067STDMETHODIMP Machine::GetNetworkAdapter (ULONG slot, INetworkAdapter **adapter)
3068{
3069 CheckComArgOutPointerValid(adapter);
3070 CheckComArgExpr (slot, slot < RT_ELEMENTS (mNetworkAdapters));
3071
3072 AutoCaller autoCaller(this);
3073 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3074
3075 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3076
3077 mNetworkAdapters[slot].queryInterfaceTo(adapter);
3078
3079 return S_OK;
3080}
3081
3082STDMETHODIMP Machine::GetExtraDataKeys(ComSafeArrayOut(BSTR, aKeys))
3083{
3084 if (ComSafeArrayOutIsNull(aKeys))
3085 return E_POINTER;
3086
3087 AutoCaller autoCaller(this);
3088 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3089
3090 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3091
3092 com::SafeArray<BSTR> saKeys(mData->m_pMachineConfigFile->mapExtraDataItems.size());
3093 int i = 0;
3094 for (settings::ExtraDataItemsMap::const_iterator it = mData->m_pMachineConfigFile->mapExtraDataItems.begin();
3095 it != mData->m_pMachineConfigFile->mapExtraDataItems.end();
3096 ++it, ++i)
3097 {
3098 const Utf8Str &strKey = it->first;
3099 strKey.cloneTo(&saKeys[i]);
3100 }
3101 saKeys.detachTo(ComSafeArrayOutArg(aKeys));
3102
3103 return S_OK;
3104 }
3105
3106 /**
3107 * @note Locks this object for reading.
3108 */
3109STDMETHODIMP Machine::GetExtraData(IN_BSTR aKey,
3110 BSTR *aValue)
3111{
3112 CheckComArgNotNull(aKey);
3113 CheckComArgOutPointerValid(aValue);
3114
3115 AutoCaller autoCaller(this);
3116 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3117
3118 /* start with nothing found */
3119 Bstr bstrResult("");
3120
3121 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3122
3123 settings::ExtraDataItemsMap::const_iterator it = mData->m_pMachineConfigFile->mapExtraDataItems.find(Utf8Str(aKey));
3124 if (it != mData->m_pMachineConfigFile->mapExtraDataItems.end())
3125 // found:
3126 bstrResult = it->second; // source is a Utf8Str
3127
3128 /* return the result to caller (may be empty) */
3129 bstrResult.cloneTo(aValue);
3130
3131 return S_OK;
3132}
3133
3134 /**
3135 * @note Locks mParent for writing + this object for writing.
3136 */
3137STDMETHODIMP Machine::SetExtraData(IN_BSTR aKey, IN_BSTR aValue)
3138{
3139 CheckComArgNotNull(aKey);
3140
3141 AutoCaller autoCaller(this);
3142 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3143
3144 Utf8Str strKey(aKey);
3145 Utf8Str strValue(aValue);
3146 Utf8Str strOldValue; // empty
3147
3148 // locking note: we only hold the read lock briefly to look up the old value,
3149 // then release it and call the onExtraCanChange callbacks. There is a small
3150 // chance of a race insofar as the callback might be called twice if two callers
3151 // change the same key at the same time, but that's a much better solution
3152 // than the deadlock we had here before. The actual changing of the extradata
3153 // is then performed under the write lock and race-free.
3154
3155 // look up the old value first; if nothing's changed then we need not do anything
3156 {
3157 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); // hold read lock only while looking up
3158 settings::ExtraDataItemsMap::const_iterator it = mData->m_pMachineConfigFile->mapExtraDataItems.find(strKey);
3159 if (it != mData->m_pMachineConfigFile->mapExtraDataItems.end())
3160 strOldValue = it->second;
3161 }
3162
3163 bool fChanged;
3164 if ((fChanged = (strOldValue != strValue)))
3165 {
3166 // ask for permission from all listeners outside the locks;
3167 // onExtraDataCanChange() only briefly requests the VirtualBox
3168 // lock to copy the list of callbacks to invoke
3169 Bstr error;
3170 Bstr bstrValue;
3171 if (aValue)
3172 bstrValue = aValue;
3173 else
3174 bstrValue = (const char *)"";
3175
3176 if (!mParent->onExtraDataCanChange(mData->mUuid, aKey, bstrValue, error))
3177 {
3178 const char *sep = error.isEmpty() ? "" : ": ";
3179 CBSTR err = error.isNull() ? (CBSTR) L"" : error.raw();
3180 LogWarningFunc(("Someone vetoed! Change refused%s%ls\n",
3181 sep, err));
3182 return setError(E_ACCESSDENIED,
3183 tr("Could not set extra data because someone refused the requested change of '%ls' to '%ls'%s%ls"),
3184 aKey,
3185 bstrValue.raw(),
3186 sep,
3187 err);
3188 }
3189
3190 // data is changing and change not vetoed: then write it out under the locks
3191
3192 // saveSettings() needs VirtualBox write lock
3193 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
3194
3195 if (getClassID() == clsidSnapshotMachine)
3196 {
3197 HRESULT rc = checkStateDependency(MutableStateDep);
3198 if (FAILED(rc)) return rc;
3199 }
3200
3201 if (strValue.isEmpty())
3202 mData->m_pMachineConfigFile->mapExtraDataItems.erase(strKey);
3203 else
3204 mData->m_pMachineConfigFile->mapExtraDataItems[strKey] = strValue;
3205 // creates a new key if needed
3206
3207 /* save settings on success */
3208 HRESULT rc = saveSettings();
3209 if (FAILED(rc)) return rc;
3210 }
3211
3212 // fire notification outside the lock
3213 if (fChanged)
3214 mParent->onExtraDataChange(mData->mUuid, aKey, aValue);
3215
3216 return S_OK;
3217}
3218
3219STDMETHODIMP Machine::SaveSettings()
3220{
3221 AutoCaller autoCaller(this);
3222 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3223
3224 /* saveSettings() needs mParent lock */
3225 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
3226
3227 /* when there was auto-conversion, we want to save the file even if
3228 * the VM is saved */
3229 HRESULT rc = checkStateDependency(MutableStateDep);
3230 if (FAILED(rc)) return rc;
3231
3232 /* the settings file path may never be null */
3233 ComAssertRet(!mData->m_strConfigFileFull.isEmpty(), E_FAIL);
3234
3235 /* save all VM data excluding snapshots */
3236 return saveSettings();
3237}
3238
3239STDMETHODIMP Machine::DiscardSettings()
3240{
3241 AutoCaller autoCaller(this);
3242 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3243
3244 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3245
3246 HRESULT rc = checkStateDependency(MutableStateDep);
3247 if (FAILED(rc)) return rc;
3248
3249 /*
3250 * during this rollback, the session will be notified if data has
3251 * been actually changed
3252 */
3253 rollback (true /* aNotify */);
3254
3255 return S_OK;
3256}
3257
3258STDMETHODIMP Machine::DeleteSettings()
3259{
3260 AutoCaller autoCaller(this);
3261 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3262
3263 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3264
3265 HRESULT rc = checkStateDependency(MutableStateDep);
3266 if (FAILED(rc)) return rc;
3267
3268 if (mData->mRegistered)
3269 return setError(VBOX_E_INVALID_VM_STATE,
3270 tr("Cannot delete settings of a registered machine"));
3271
3272 /* delete the settings only when the file actually exists */
3273 if (mData->m_pMachineConfigFile->fileExists())
3274 {
3275 int vrc = RTFileDelete(mData->m_strConfigFileFull.c_str());
3276 if (RT_FAILURE(vrc))
3277 return setError(VBOX_E_IPRT_ERROR,
3278 tr("Could not delete the settings file '%s' (%Rrc)"),
3279 mData->m_strConfigFileFull.raw(),
3280 vrc);
3281
3282 /* delete the Logs folder, nothing important should be left
3283 * there (we don't check for errors because the user might have
3284 * some private files there that we don't want to delete) */
3285 Utf8Str logFolder;
3286 getLogFolder(logFolder);
3287 Assert(logFolder.length());
3288 if (RTDirExists(logFolder.c_str()))
3289 {
3290 /* Delete all VBox.log[.N] files from the Logs folder
3291 * (this must be in sync with the rotation logic in
3292 * Console::powerUpThread()). Also, delete the VBox.png[.N]
3293 * files that may have been created by the GUI. */
3294 Utf8Str log = Utf8StrFmt("%s/VBox.log", logFolder.raw());
3295 RTFileDelete(log.c_str());
3296 log = Utf8StrFmt("%s/VBox.png", logFolder.raw());
3297 RTFileDelete(log.c_str());
3298 for (int i = 3; i >= 0; i--)
3299 {
3300 log = Utf8StrFmt("%s/VBox.log.%d", logFolder.raw(), i);
3301 RTFileDelete(log.c_str());
3302 log = Utf8StrFmt("%s/VBox.png.%d", logFolder.raw(), i);
3303 RTFileDelete(log.c_str());
3304 }
3305
3306 RTDirRemove(logFolder.c_str());
3307 }
3308
3309 /* delete the Snapshots folder, nothing important should be left
3310 * there (we don't check for errors because the user might have
3311 * some private files there that we don't want to delete) */
3312 Utf8Str snapshotFolder(mUserData->mSnapshotFolderFull);
3313 Assert(snapshotFolder.length());
3314 if (RTDirExists(snapshotFolder.c_str()))
3315 RTDirRemove(snapshotFolder.c_str());
3316
3317 /* delete the directory that contains the settings file, but only
3318 * if it matches the VM name (i.e. a structure created by default in
3319 * prepareSaveSettings()) */
3320 {
3321 Utf8Str settingsDir;
3322 if (isInOwnDir(&settingsDir))
3323 RTDirRemove(settingsDir.c_str());
3324 }
3325 }
3326
3327 return S_OK;
3328}
3329
3330STDMETHODIMP Machine::GetSnapshot (IN_BSTR aId, ISnapshot **aSnapshot)
3331{
3332 CheckComArgOutPointerValid(aSnapshot);
3333
3334 AutoCaller autoCaller(this);
3335 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3336
3337 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3338
3339 Guid uuid(aId);
3340 /* Todo: fix this properly by perhaps introducing an isValid method for the Guid class */
3341 if ( aId
3342 && uuid.isEmpty())
3343 {
3344 RTUUID uuidTemp;
3345 /* Either it's a null UUID or the conversion failed. (null uuid has a special meaning in findSnapshot) */
3346 if (RT_FAILURE(RTUuidFromUtf16(&uuidTemp, aId)))
3347 return setError(E_FAIL,
3348 tr("Could not find a snapshot with UUID {%ls}"),
3349 aId);
3350 }
3351
3352 ComObjPtr<Snapshot> snapshot;
3353
3354 HRESULT rc = findSnapshot(uuid, snapshot, true /* aSetError */);
3355 snapshot.queryInterfaceTo(aSnapshot);
3356
3357 return rc;
3358}
3359
3360STDMETHODIMP Machine::FindSnapshot (IN_BSTR aName, ISnapshot **aSnapshot)
3361{
3362 CheckComArgNotNull (aName);
3363 CheckComArgOutPointerValid(aSnapshot);
3364
3365 AutoCaller autoCaller(this);
3366 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3367
3368 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3369
3370 ComObjPtr<Snapshot> snapshot;
3371
3372 HRESULT rc = findSnapshot(aName, snapshot, true /* aSetError */);
3373 snapshot.queryInterfaceTo(aSnapshot);
3374
3375 return rc;
3376}
3377
3378STDMETHODIMP Machine::SetCurrentSnapshot (IN_BSTR /* aId */)
3379{
3380 /// @todo (dmik) don't forget to set
3381 // mData->mCurrentStateModified to FALSE
3382
3383 return setError (E_NOTIMPL, "Not implemented");
3384}
3385
3386STDMETHODIMP Machine::CreateSharedFolder (IN_BSTR aName, IN_BSTR aHostPath, BOOL aWritable)
3387{
3388 CheckComArgNotNull(aName);
3389 CheckComArgNotNull(aHostPath);
3390
3391 AutoCaller autoCaller(this);
3392 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3393
3394 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3395
3396 HRESULT rc = checkStateDependency(MutableStateDep);
3397 if (FAILED(rc)) return rc;
3398
3399 ComObjPtr<SharedFolder> sharedFolder;
3400 rc = findSharedFolder (aName, sharedFolder, false /* aSetError */);
3401 if (SUCCEEDED(rc))
3402 return setError(VBOX_E_OBJECT_IN_USE,
3403 tr("Shared folder named '%ls' already exists"),
3404 aName);
3405
3406 sharedFolder.createObject();
3407 rc = sharedFolder->init(getMachine(), aName, aHostPath, aWritable);
3408 if (FAILED(rc)) return rc;
3409
3410 setModified(IsModified_SharedFolders);
3411 mHWData.backup();
3412 mHWData->mSharedFolders.push_back (sharedFolder);
3413
3414 /* inform the direct session if any */
3415 alock.leave();
3416 onSharedFolderChange();
3417
3418 return S_OK;
3419}
3420
3421STDMETHODIMP Machine::RemoveSharedFolder (IN_BSTR aName)
3422{
3423 CheckComArgNotNull (aName);
3424
3425 AutoCaller autoCaller(this);
3426 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3427
3428 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3429
3430 HRESULT rc = checkStateDependency(MutableStateDep);
3431 if (FAILED(rc)) return rc;
3432
3433 ComObjPtr<SharedFolder> sharedFolder;
3434 rc = findSharedFolder (aName, sharedFolder, true /* aSetError */);
3435 if (FAILED(rc)) return rc;
3436
3437 setModified(IsModified_SharedFolders);
3438 mHWData.backup();
3439 mHWData->mSharedFolders.remove (sharedFolder);
3440
3441 /* inform the direct session if any */
3442 alock.leave();
3443 onSharedFolderChange();
3444
3445 return S_OK;
3446}
3447
3448STDMETHODIMP Machine::CanShowConsoleWindow (BOOL *aCanShow)
3449{
3450 CheckComArgOutPointerValid(aCanShow);
3451
3452 /* start with No */
3453 *aCanShow = FALSE;
3454
3455 AutoCaller autoCaller(this);
3456 AssertComRCReturnRC(autoCaller.rc());
3457
3458 ComPtr<IInternalSessionControl> directControl;
3459 {
3460 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3461
3462 if (mData->mSession.mState != SessionState_Open)
3463 return setError(VBOX_E_INVALID_VM_STATE,
3464 tr("Machine session is not open (session state: %s)"),
3465 Global::stringifySessionState(mData->mSession.mState));
3466
3467 directControl = mData->mSession.mDirectControl;
3468 }
3469
3470 /* ignore calls made after #OnSessionEnd() is called */
3471 if (!directControl)
3472 return S_OK;
3473
3474 ULONG64 dummy;
3475 return directControl->OnShowWindow (TRUE /* aCheck */, aCanShow, &dummy);
3476}
3477
3478STDMETHODIMP Machine::ShowConsoleWindow (ULONG64 *aWinId)
3479{
3480 CheckComArgOutPointerValid(aWinId);
3481
3482 AutoCaller autoCaller(this);
3483 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
3484
3485 ComPtr<IInternalSessionControl> directControl;
3486 {
3487 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3488
3489 if (mData->mSession.mState != SessionState_Open)
3490 return setError(E_FAIL,
3491 tr("Machine session is not open (session state: %s)"),
3492 Global::stringifySessionState(mData->mSession.mState));
3493
3494 directControl = mData->mSession.mDirectControl;
3495 }
3496
3497 /* ignore calls made after #OnSessionEnd() is called */
3498 if (!directControl)
3499 return S_OK;
3500
3501 BOOL dummy;
3502 return directControl->OnShowWindow (FALSE /* aCheck */, &dummy, aWinId);
3503}
3504
3505STDMETHODIMP Machine::GetGuestProperty(IN_BSTR aName,
3506 BSTR *aValue,
3507 ULONG64 *aTimestamp,
3508 BSTR *aFlags)
3509{
3510#if !defined (VBOX_WITH_GUEST_PROPS)
3511 ReturnComNotImplemented();
3512#else
3513 CheckComArgNotNull(aName);
3514 CheckComArgOutPointerValid(aValue);
3515 CheckComArgOutPointerValid(aTimestamp);
3516 CheckComArgOutPointerValid(aFlags);
3517
3518 AutoCaller autoCaller(this);
3519 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3520
3521 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3522
3523 using namespace guestProp;
3524 HRESULT rc = E_FAIL;
3525
3526 Utf8Str strName(aName);
3527
3528 if (!mHWData->mPropertyServiceActive)
3529 {
3530 bool found = false;
3531 for (HWData::GuestPropertyList::const_iterator it = mHWData->mGuestProperties.begin();
3532 (it != mHWData->mGuestProperties.end()) && !found;
3533 ++it)
3534 {
3535 if (it->strName == strName)
3536 {
3537 char szFlags[MAX_FLAGS_LEN + 1];
3538 it->strValue.cloneTo(aValue);
3539 *aTimestamp = it->mTimestamp;
3540 writeFlags(it->mFlags, szFlags);
3541 Bstr(szFlags).cloneTo(aFlags);
3542 found = true;
3543 }
3544 }
3545 rc = S_OK;
3546 }
3547 else
3548 {
3549 ComPtr<IInternalSessionControl> directControl =
3550 mData->mSession.mDirectControl;
3551
3552 /* just be on the safe side when calling another process */
3553 alock.release();
3554
3555 /* fail if we were called after #OnSessionEnd() is called. This is a
3556 * silly race condition. */
3557
3558 if (!directControl)
3559 rc = E_FAIL;
3560 else
3561 rc = directControl->AccessGuestProperty (aName, NULL, NULL,
3562 false /* isSetter */,
3563 aValue, aTimestamp, aFlags);
3564 }
3565 return rc;
3566#endif /* else !defined (VBOX_WITH_GUEST_PROPS) */
3567}
3568
3569STDMETHODIMP Machine::GetGuestPropertyValue (IN_BSTR aName, BSTR *aValue)
3570{
3571 ULONG64 dummyTimestamp;
3572 BSTR dummyFlags;
3573 return GetGuestProperty (aName, aValue, &dummyTimestamp, &dummyFlags);
3574}
3575
3576STDMETHODIMP Machine::GetGuestPropertyTimestamp (IN_BSTR aName, ULONG64 *aTimestamp)
3577{
3578 BSTR dummyValue;
3579 BSTR dummyFlags;
3580 return GetGuestProperty (aName, &dummyValue, aTimestamp, &dummyFlags);
3581}
3582
3583STDMETHODIMP Machine::SetGuestProperty(IN_BSTR aName,
3584 IN_BSTR aValue,
3585 IN_BSTR aFlags)
3586{
3587#if !defined (VBOX_WITH_GUEST_PROPS)
3588 ReturnComNotImplemented();
3589#else
3590 using namespace guestProp;
3591
3592 CheckComArgNotNull(aName);
3593 CheckComArgNotNull(aValue);
3594 if ((aFlags != NULL) && !VALID_PTR (aFlags))
3595 return E_INVALIDARG;
3596
3597 HRESULT rc = S_OK;
3598
3599 try
3600 {
3601 Utf8Str utf8Name(aName);
3602 Utf8Str utf8Flags(aFlags);
3603
3604 AutoCaller autoCaller(this);
3605 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3606
3607 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3608
3609 rc = checkStateDependency(MutableStateDep);
3610 if (FAILED(rc)) return rc;
3611
3612 rc = S_OK;
3613 uint32_t fFlags = NILFLAG;
3614 if ( (aFlags != NULL)
3615 && RT_FAILURE(validateFlags (utf8Flags.raw(), &fFlags))
3616 )
3617 return setError(E_INVALIDARG,
3618 tr("Invalid flag values: '%ls'"),
3619 aFlags);
3620
3621 if (!mHWData->mPropertyServiceActive)
3622 {
3623 bool found = false;
3624 HWData::GuestProperty property;
3625 property.mFlags = NILFLAG;
3626
3627 /** @todo r=bird: see efficiency rant in PushGuestProperty. (Yeah, I know,
3628 * this is simple and do an OK job atm.) */
3629 for (HWData::GuestPropertyList::iterator it = mHWData->mGuestProperties.begin();
3630 it != mHWData->mGuestProperties.end();
3631 ++it)
3632 if (it->strName == utf8Name)
3633 {
3634 property = *it;
3635 if (it->mFlags & (RDONLYHOST))
3636 rc = setError(E_ACCESSDENIED,
3637 tr("The property '%ls' cannot be changed by the host"),
3638 aName);
3639 else
3640 {
3641 setModified(IsModified_MachineData);
3642 mHWData.backup(); // @todo r=dj backup in a loop?!?
3643
3644 /* The backup() operation invalidates our iterator, so
3645 * get a new one. */
3646 for (it = mHWData->mGuestProperties.begin();
3647 it->strName != utf8Name;
3648 ++it)
3649 ;
3650 mHWData->mGuestProperties.erase(it);
3651 }
3652 found = true;
3653 break;
3654 }
3655 if (found && SUCCEEDED(rc))
3656 {
3657 if (*aValue)
3658 {
3659 RTTIMESPEC time;
3660 property.strValue = aValue;
3661 property.mTimestamp = RTTimeSpecGetNano(RTTimeNow(&time));
3662 if (aFlags != NULL)
3663 property.mFlags = fFlags;
3664 mHWData->mGuestProperties.push_back (property);
3665 }
3666 }
3667 else if (SUCCEEDED(rc) && *aValue)
3668 {
3669 RTTIMESPEC time;
3670 setModified(IsModified_MachineData);
3671 mHWData.backup();
3672 property.strName = aName;
3673 property.strValue = aValue;
3674 property.mTimestamp = RTTimeSpecGetNano(RTTimeNow(&time));
3675 property.mFlags = fFlags;
3676 mHWData->mGuestProperties.push_back (property);
3677 }
3678 if ( SUCCEEDED(rc)
3679 && ( mHWData->mGuestPropertyNotificationPatterns.isEmpty()
3680 || RTStrSimplePatternMultiMatch(mHWData->mGuestPropertyNotificationPatterns.raw(), RTSTR_MAX,
3681 utf8Name.raw(), RTSTR_MAX, NULL) )
3682 )
3683 {
3684 /** @todo r=bird: Why aren't we leaving the lock here? The
3685 * same code in PushGuestProperty does... */
3686 mParent->onGuestPropertyChange(mData->mUuid, aName, aValue, aFlags);
3687 }
3688 }
3689 else
3690 {
3691 ComPtr<IInternalSessionControl> directControl =
3692 mData->mSession.mDirectControl;
3693
3694 /* just be on the safe side when calling another process */
3695 alock.leave();
3696
3697 BSTR dummy = NULL;
3698 ULONG64 dummy64;
3699 if (!directControl)
3700 rc = E_FAIL;
3701 else
3702 rc = directControl->AccessGuestProperty(aName,
3703 *aValue ? aValue : NULL, /** @todo Fix when adding DeleteGuestProperty(), see defect. */
3704 aFlags,
3705 true /* isSetter */,
3706 &dummy, &dummy64, &dummy);
3707 }
3708 }
3709 catch (std::bad_alloc &)
3710 {
3711 rc = E_OUTOFMEMORY;
3712 }
3713
3714 return rc;
3715#endif /* else !defined (VBOX_WITH_GUEST_PROPS) */
3716}
3717
3718STDMETHODIMP Machine::SetGuestPropertyValue (IN_BSTR aName, IN_BSTR aValue)
3719{
3720 return SetGuestProperty (aName, aValue, NULL);
3721}
3722
3723STDMETHODIMP Machine::EnumerateGuestProperties(IN_BSTR aPatterns,
3724 ComSafeArrayOut(BSTR, aNames),
3725 ComSafeArrayOut(BSTR, aValues),
3726 ComSafeArrayOut(ULONG64, aTimestamps),
3727 ComSafeArrayOut(BSTR, aFlags))
3728{
3729#if !defined (VBOX_WITH_GUEST_PROPS)
3730 ReturnComNotImplemented();
3731#else
3732 if (!VALID_PTR (aPatterns) && (aPatterns != NULL))
3733 return E_POINTER;
3734
3735 CheckComArgOutSafeArrayPointerValid(aNames);
3736 CheckComArgOutSafeArrayPointerValid(aValues);
3737 CheckComArgOutSafeArrayPointerValid(aTimestamps);
3738 CheckComArgOutSafeArrayPointerValid(aFlags);
3739
3740 AutoCaller autoCaller(this);
3741 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3742
3743 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3744
3745 using namespace guestProp;
3746 HRESULT rc = E_FAIL;
3747
3748 Utf8Str strPatterns(aPatterns);
3749
3750 if (!mHWData->mPropertyServiceActive)
3751 {
3752
3753 /*
3754 * Look for matching patterns and build up a list.
3755 */
3756 HWData::GuestPropertyList propList;
3757 for (HWData::GuestPropertyList::iterator it = mHWData->mGuestProperties.begin();
3758 it != mHWData->mGuestProperties.end();
3759 ++it)
3760 if ( strPatterns.isEmpty()
3761 || RTStrSimplePatternMultiMatch(strPatterns.raw(),
3762 RTSTR_MAX,
3763 it->strName.raw(),
3764 RTSTR_MAX, NULL)
3765 )
3766 propList.push_back(*it);
3767
3768 /*
3769 * And build up the arrays for returning the property information.
3770 */
3771 size_t cEntries = propList.size();
3772 SafeArray<BSTR> names (cEntries);
3773 SafeArray<BSTR> values (cEntries);
3774 SafeArray<ULONG64> timestamps (cEntries);
3775 SafeArray<BSTR> flags (cEntries);
3776 size_t iProp = 0;
3777 for (HWData::GuestPropertyList::iterator it = propList.begin();
3778 it != propList.end();
3779 ++it)
3780 {
3781 char szFlags[MAX_FLAGS_LEN + 1];
3782 it->strName.cloneTo(&names[iProp]);
3783 it->strValue.cloneTo(&values[iProp]);
3784 timestamps[iProp] = it->mTimestamp;
3785 writeFlags(it->mFlags, szFlags);
3786 Bstr(szFlags).cloneTo(&flags[iProp]);
3787 ++iProp;
3788 }
3789 names.detachTo(ComSafeArrayOutArg(aNames));
3790 values.detachTo(ComSafeArrayOutArg(aValues));
3791 timestamps.detachTo(ComSafeArrayOutArg(aTimestamps));
3792 flags.detachTo(ComSafeArrayOutArg(aFlags));
3793 rc = S_OK;
3794 }
3795 else
3796 {
3797 ComPtr<IInternalSessionControl> directControl = mData->mSession.mDirectControl;
3798
3799 /* just be on the safe side when calling another process */
3800 alock.release();
3801
3802 if (!directControl)
3803 rc = E_FAIL;
3804 else
3805 rc = directControl->EnumerateGuestProperties(aPatterns,
3806 ComSafeArrayOutArg(aNames),
3807 ComSafeArrayOutArg(aValues),
3808 ComSafeArrayOutArg(aTimestamps),
3809 ComSafeArrayOutArg(aFlags));
3810 }
3811 return rc;
3812#endif /* else !defined (VBOX_WITH_GUEST_PROPS) */
3813}
3814
3815STDMETHODIMP Machine::GetMediumAttachmentsOfController(IN_BSTR aName,
3816 ComSafeArrayOut(IMediumAttachment*, aAttachments))
3817{
3818 MediaData::AttachmentList atts;
3819
3820 HRESULT rc = getMediumAttachmentsOfController(aName, atts);
3821 if (FAILED(rc)) return rc;
3822
3823 SafeIfaceArray<IMediumAttachment> attachments(atts);
3824 attachments.detachTo(ComSafeArrayOutArg(aAttachments));
3825
3826 return S_OK;
3827}
3828
3829STDMETHODIMP Machine::GetMediumAttachment(IN_BSTR aControllerName,
3830 LONG aControllerPort,
3831 LONG aDevice,
3832 IMediumAttachment **aAttachment)
3833{
3834 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
3835 aControllerName, aControllerPort, aDevice));
3836
3837 CheckComArgNotNull(aControllerName);
3838 CheckComArgOutPointerValid(aAttachment);
3839
3840 AutoCaller autoCaller(this);
3841 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3842
3843 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3844
3845 *aAttachment = NULL;
3846
3847 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
3848 aControllerName,
3849 aControllerPort,
3850 aDevice);
3851 if (pAttach.isNull())
3852 return setError(VBOX_E_OBJECT_NOT_FOUND,
3853 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
3854 aDevice, aControllerPort, aControllerName);
3855
3856 pAttach.queryInterfaceTo(aAttachment);
3857
3858 return S_OK;
3859}
3860
3861STDMETHODIMP Machine::AddStorageController(IN_BSTR aName,
3862 StorageBus_T aConnectionType,
3863 IStorageController **controller)
3864{
3865 CheckComArgStrNotEmptyOrNull(aName);
3866
3867 if ( (aConnectionType <= StorageBus_Null)
3868 || (aConnectionType > StorageBus_SAS))
3869 return setError (E_INVALIDARG,
3870 tr ("Invalid connection type: %d"),
3871 aConnectionType);
3872
3873 AutoCaller autoCaller(this);
3874 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3875
3876 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3877
3878 HRESULT rc = checkStateDependency(MutableStateDep);
3879 if (FAILED(rc)) return rc;
3880
3881 /* try to find one with the name first. */
3882 ComObjPtr<StorageController> ctrl;
3883
3884 rc = getStorageControllerByName (aName, ctrl, false /* aSetError */);
3885 if (SUCCEEDED(rc))
3886 return setError (VBOX_E_OBJECT_IN_USE,
3887 tr ("Storage controller named '%ls' already exists"), aName);
3888
3889 ctrl.createObject();
3890
3891 /* get a new instance number for the storage controller */
3892 ULONG ulInstance = 0;
3893 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
3894 it != mStorageControllers->end();
3895 ++it)
3896 {
3897 if ((*it)->getStorageBus() == aConnectionType)
3898 {
3899 ULONG ulCurInst = (*it)->getInstance();
3900
3901 if (ulCurInst >= ulInstance)
3902 ulInstance = ulCurInst + 1;
3903 }
3904 }
3905
3906 rc = ctrl->init(this, aName, aConnectionType, ulInstance);
3907 if (FAILED(rc)) return rc;
3908
3909 setModified(IsModified_Storage);
3910 mStorageControllers.backup();
3911 mStorageControllers->push_back (ctrl);
3912
3913 ctrl.queryInterfaceTo(controller);
3914
3915 /* inform the direct session if any */
3916 alock.leave();
3917 onStorageControllerChange();
3918
3919 return S_OK;
3920}
3921
3922STDMETHODIMP Machine::GetStorageControllerByName(IN_BSTR aName,
3923 IStorageController **aStorageController)
3924{
3925 CheckComArgStrNotEmptyOrNull(aName);
3926
3927 AutoCaller autoCaller(this);
3928 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3929
3930 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3931
3932 ComObjPtr<StorageController> ctrl;
3933
3934 HRESULT rc = getStorageControllerByName (aName, ctrl, true /* aSetError */);
3935 if (SUCCEEDED(rc))
3936 ctrl.queryInterfaceTo(aStorageController);
3937
3938 return rc;
3939}
3940
3941STDMETHODIMP Machine::GetStorageControllerByInstance(ULONG aInstance,
3942 IStorageController **aStorageController)
3943{
3944 AutoCaller autoCaller(this);
3945 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3946
3947 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3948
3949 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
3950 it != mStorageControllers->end();
3951 ++it)
3952 {
3953 if ((*it)->getInstance() == aInstance)
3954 {
3955 (*it).queryInterfaceTo(aStorageController);
3956 return S_OK;
3957 }
3958 }
3959
3960 return setError(VBOX_E_OBJECT_NOT_FOUND,
3961 tr("Could not find a storage controller with instance number '%lu'"),
3962 aInstance);
3963}
3964
3965STDMETHODIMP Machine::RemoveStorageController(IN_BSTR aName)
3966{
3967 CheckComArgStrNotEmptyOrNull(aName);
3968
3969 AutoCaller autoCaller(this);
3970 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3971
3972 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3973
3974 HRESULT rc = checkStateDependency(MutableStateDep);
3975 if (FAILED(rc)) return rc;
3976
3977 ComObjPtr<StorageController> ctrl;
3978 rc = getStorageControllerByName (aName, ctrl, true /* aSetError */);
3979 if (FAILED(rc)) return rc;
3980
3981 /* We can remove the controller only if there is no device attached. */
3982 /* check if the device slot is already busy */
3983 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
3984 it != mMediaData->mAttachments.end();
3985 ++it)
3986 {
3987 if ((*it)->getControllerName() == aName)
3988 return setError(VBOX_E_OBJECT_IN_USE,
3989 tr("Storage controller named '%ls' has still devices attached"),
3990 aName);
3991 }
3992
3993 /* We can remove it now. */
3994 setModified(IsModified_Storage);
3995 mStorageControllers.backup();
3996
3997 ctrl->unshare();
3998
3999 mStorageControllers->remove (ctrl);
4000
4001 /* inform the direct session if any */
4002 alock.leave();
4003 onStorageControllerChange();
4004
4005 return S_OK;
4006}
4007
4008/* @todo where is the right place for this? */
4009#define sSSMDisplayScreenshotVer 0x00010001
4010
4011static int readSavedDisplayScreenshot(Utf8Str *pStateFilePath, uint32_t u32Type, uint8_t **ppu8Data, uint32_t *pcbData, uint32_t *pu32Width, uint32_t *pu32Height)
4012{
4013 LogFlowFunc(("u32Type = %d [%s]\n", u32Type, pStateFilePath->raw()));
4014
4015 /* @todo cache read data */
4016 if (pStateFilePath->isEmpty())
4017 {
4018 /* No saved state data. */
4019 return VERR_NOT_SUPPORTED;
4020 }
4021
4022 uint8_t *pu8Data = NULL;
4023 uint32_t cbData = 0;
4024 uint32_t u32Width = 0;
4025 uint32_t u32Height = 0;
4026
4027 PSSMHANDLE pSSM;
4028 int rc = SSMR3Open(pStateFilePath->raw(), 0 /*fFlags*/, &pSSM);
4029 if (RT_SUCCESS(rc))
4030 {
4031 uint32_t uVersion;
4032 rc = SSMR3Seek(pSSM, "DisplayScreenshot", 1100 /*iInstance*/, &uVersion);
4033 if (RT_SUCCESS(rc))
4034 {
4035 if (uVersion == sSSMDisplayScreenshotVer)
4036 {
4037 uint32_t cBlocks;
4038 rc = SSMR3GetU32(pSSM, &cBlocks);
4039 AssertRCReturn(rc, rc);
4040
4041 for (uint32_t i = 0; i < cBlocks; i++)
4042 {
4043 uint32_t cbBlock;
4044 rc = SSMR3GetU32(pSSM, &cbBlock);
4045 AssertRCBreak(rc);
4046
4047 uint32_t typeOfBlock;
4048 rc = SSMR3GetU32(pSSM, &typeOfBlock);
4049 AssertRCBreak(rc);
4050
4051 LogFlowFunc(("[%d] type %d, size %d bytes\n", i, typeOfBlock, cbBlock));
4052
4053 if (typeOfBlock == u32Type)
4054 {
4055 if (cbBlock > 2 * sizeof (uint32_t))
4056 {
4057 cbData = cbBlock - 2 * sizeof (uint32_t);
4058 pu8Data = (uint8_t *)RTMemAlloc(cbData);
4059 if (pu8Data == NULL)
4060 {
4061 rc = VERR_NO_MEMORY;
4062 break;
4063 }
4064
4065 rc = SSMR3GetU32(pSSM, &u32Width);
4066 AssertRCBreak(rc);
4067 rc = SSMR3GetU32(pSSM, &u32Height);
4068 AssertRCBreak(rc);
4069 rc = SSMR3GetMem(pSSM, pu8Data, cbData);
4070 AssertRCBreak(rc);
4071 }
4072 else
4073 {
4074 /* No saved state data. */
4075 rc = VERR_NOT_SUPPORTED;
4076 }
4077
4078 break;
4079 }
4080 else
4081 {
4082 if (cbBlock != 0)
4083 {
4084 rc = SSMR3Skip(pSSM, cbBlock);
4085 AssertRCBreak(rc);
4086 }
4087 }
4088 }
4089 }
4090 else
4091 {
4092 rc = VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
4093 }
4094 }
4095
4096 SSMR3Close(pSSM);
4097 }
4098
4099 if (RT_SUCCESS(rc))
4100 {
4101 if (u32Type == 0 && cbData % 4 != 0)
4102 {
4103 /* Bitmap is 32bpp, so data is invalid. */
4104 rc = VERR_SSM_UNEXPECTED_DATA;
4105 }
4106 }
4107
4108 if (RT_SUCCESS(rc))
4109 {
4110 *ppu8Data = pu8Data;
4111 *pcbData = cbData;
4112 *pu32Width = u32Width;
4113 *pu32Height = u32Height;
4114 LogFlowFunc(("cbData %d, u32Width %d, u32Height %d\n", cbData, u32Width, u32Height));
4115 }
4116
4117 LogFlowFunc(("rc %Rrc\n", rc));
4118 return rc;
4119}
4120
4121static void freeSavedDisplayScreenshot(uint8_t *pu8Data)
4122{
4123 /* @todo not necessary when caching is implemented. */
4124 RTMemFree(pu8Data);
4125}
4126
4127STDMETHODIMP Machine::QuerySavedThumbnailSize(ULONG *aSize, ULONG *aWidth, ULONG *aHeight)
4128{
4129 LogFlowThisFunc(("\n"));
4130
4131 CheckComArgNotNull(aSize);
4132 CheckComArgNotNull(aWidth);
4133 CheckComArgNotNull(aHeight);
4134
4135 AutoCaller autoCaller(this);
4136 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4137
4138 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4139
4140 uint8_t *pu8Data = NULL;
4141 uint32_t cbData = 0;
4142 uint32_t u32Width = 0;
4143 uint32_t u32Height = 0;
4144
4145 int vrc = readSavedDisplayScreenshot(&mSSData->mStateFilePath, 0 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
4146
4147 if (RT_FAILURE(vrc))
4148 return setError (VBOX_E_IPRT_ERROR,
4149 tr("Saved screenshot data is not available (%Rrc)"), vrc);
4150
4151 *aSize = cbData;
4152 *aWidth = u32Width;
4153 *aHeight = u32Height;
4154
4155 freeSavedDisplayScreenshot(pu8Data);
4156
4157 return S_OK;
4158}
4159
4160STDMETHODIMP Machine::ReadSavedThumbnailToArray(BOOL aBGR, ULONG *aWidth, ULONG *aHeight, ComSafeArrayOut(BYTE, aData))
4161{
4162 LogFlowThisFunc(("\n"));
4163
4164 CheckComArgNotNull(aWidth);
4165 CheckComArgNotNull(aHeight);
4166 CheckComArgExpr(aData, !ComSafeArrayOutIsNull(aData));
4167
4168 AutoCaller autoCaller(this);
4169 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4170
4171 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4172
4173 uint8_t *pu8Data = NULL;
4174 uint32_t cbData = 0;
4175 uint32_t u32Width = 0;
4176 uint32_t u32Height = 0;
4177
4178 int vrc = readSavedDisplayScreenshot(&mSSData->mStateFilePath, 0 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
4179
4180 if (RT_FAILURE(vrc))
4181 return setError (VBOX_E_IPRT_ERROR,
4182 tr("Saved screenshot data is not available (%Rrc)"), vrc);
4183
4184 *aWidth = u32Width;
4185 *aHeight = u32Height;
4186
4187 com::SafeArray<BYTE> bitmap(cbData);
4188 /* Convert pixels to format expected by the API caller. */
4189 if (aBGR)
4190 {
4191 /* [0] B, [1] G, [2] R, [3] A. */
4192 for (unsigned i = 0; i < cbData; i += 4)
4193 {
4194 bitmap[i] = pu8Data[i];
4195 bitmap[i + 1] = pu8Data[i + 1];
4196 bitmap[i + 2] = pu8Data[i + 2];
4197 bitmap[i + 3] = 0xff;
4198 }
4199 }
4200 else
4201 {
4202 /* [0] R, [1] G, [2] B, [3] A. */
4203 for (unsigned i = 0; i < cbData; i += 4)
4204 {
4205 bitmap[i] = pu8Data[i + 2];
4206 bitmap[i + 1] = pu8Data[i + 1];
4207 bitmap[i + 2] = pu8Data[i];
4208 bitmap[i + 3] = 0xff;
4209 }
4210 }
4211 bitmap.detachTo(ComSafeArrayOutArg(aData));
4212
4213 freeSavedDisplayScreenshot(pu8Data);
4214
4215 return S_OK;
4216}
4217
4218STDMETHODIMP Machine::QuerySavedScreenshotPNGSize(ULONG *aSize, ULONG *aWidth, ULONG *aHeight)
4219{
4220 LogFlowThisFunc(("\n"));
4221
4222 CheckComArgNotNull(aSize);
4223 CheckComArgNotNull(aWidth);
4224 CheckComArgNotNull(aHeight);
4225
4226 AutoCaller autoCaller(this);
4227 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4228
4229 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4230
4231 uint8_t *pu8Data = NULL;
4232 uint32_t cbData = 0;
4233 uint32_t u32Width = 0;
4234 uint32_t u32Height = 0;
4235
4236 int vrc = readSavedDisplayScreenshot(&mSSData->mStateFilePath, 1 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
4237
4238 if (RT_FAILURE(vrc))
4239 return setError (VBOX_E_IPRT_ERROR,
4240 tr("Saved screenshot data is not available (%Rrc)"), vrc);
4241
4242 *aSize = cbData;
4243 *aWidth = u32Width;
4244 *aHeight = u32Height;
4245
4246 freeSavedDisplayScreenshot(pu8Data);
4247
4248 return S_OK;
4249}
4250
4251STDMETHODIMP Machine::ReadSavedScreenshotPNGToArray(ULONG *aWidth, ULONG *aHeight, ComSafeArrayOut(BYTE, aData))
4252{
4253 LogFlowThisFunc(("\n"));
4254
4255 CheckComArgNotNull(aWidth);
4256 CheckComArgNotNull(aHeight);
4257 CheckComArgExpr(aData, !ComSafeArrayOutIsNull(aData));
4258
4259 AutoCaller autoCaller(this);
4260 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4261
4262 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4263
4264 uint8_t *pu8Data = NULL;
4265 uint32_t cbData = 0;
4266 uint32_t u32Width = 0;
4267 uint32_t u32Height = 0;
4268
4269 int vrc = readSavedDisplayScreenshot(&mSSData->mStateFilePath, 1 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
4270
4271 if (RT_FAILURE(vrc))
4272 return setError (VBOX_E_IPRT_ERROR,
4273 tr("Saved screenshot data is not available (%Rrc)"), vrc);
4274
4275 *aWidth = u32Width;
4276 *aHeight = u32Height;
4277
4278 com::SafeArray<BYTE> png(cbData);
4279 for (unsigned i = 0; i < cbData; i++)
4280 png[i] = pu8Data[i];
4281 png.detachTo(ComSafeArrayOutArg(aData));
4282
4283 freeSavedDisplayScreenshot(pu8Data);
4284
4285 return S_OK;
4286}
4287
4288STDMETHODIMP Machine::HotPlugCPU(ULONG aCpu)
4289{
4290 HRESULT rc = S_OK;
4291 LogFlowThisFunc(("\n"));
4292
4293 AutoCaller autoCaller(this);
4294 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4295
4296 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4297
4298 if (!mHWData->mCPUHotPlugEnabled)
4299 return setError(E_INVALIDARG, tr("CPU hotplug is not enabled"));
4300
4301 if (aCpu >= mHWData->mCPUCount)
4302 return setError(E_INVALIDARG, tr("CPU id exceeds number of possible CPUs [0:%lu]"), mHWData->mCPUCount-1);
4303
4304 if (mHWData->mCPUAttached[aCpu])
4305 return setError(VBOX_E_OBJECT_IN_USE, tr("CPU %lu is already attached"), aCpu);
4306
4307 alock.leave();
4308 rc = onCPUChange(aCpu, false);
4309 alock.enter();
4310 if (FAILED(rc)) return rc;
4311
4312 setModified(IsModified_MachineData);
4313 mHWData.backup();
4314 mHWData->mCPUAttached[aCpu] = true;
4315
4316 /* Save settings if online */
4317 if (Global::IsOnline(mData->mMachineState))
4318 SaveSettings();
4319
4320 return S_OK;
4321}
4322
4323STDMETHODIMP Machine::HotUnplugCPU(ULONG aCpu)
4324{
4325 HRESULT rc = S_OK;
4326 LogFlowThisFunc(("\n"));
4327
4328 AutoCaller autoCaller(this);
4329 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4330
4331 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4332
4333 if (!mHWData->mCPUHotPlugEnabled)
4334 return setError(E_INVALIDARG, tr("CPU hotplug is not enabled"));
4335
4336 if (aCpu >= SchemaDefs::MaxCPUCount)
4337 return setError(E_INVALIDARG,
4338 tr("CPU index exceeds maximum CPU count (must be in range [0:%lu])"),
4339 SchemaDefs::MaxCPUCount);
4340
4341 if (!mHWData->mCPUAttached[aCpu])
4342 return setError(VBOX_E_OBJECT_NOT_FOUND, tr("CPU %lu is not attached"), aCpu);
4343
4344 /* CPU 0 can't be detached */
4345 if (aCpu == 0)
4346 return setError(E_INVALIDARG, tr("It is not possible to detach CPU 0"));
4347
4348 alock.leave();
4349 rc = onCPUChange(aCpu, true);
4350 alock.enter();
4351 if (FAILED(rc)) return rc;
4352
4353 setModified(IsModified_MachineData);
4354 mHWData.backup();
4355 mHWData->mCPUAttached[aCpu] = false;
4356
4357 /* Save settings if online */
4358 if (Global::IsOnline(mData->mMachineState))
4359 SaveSettings();
4360
4361 return S_OK;
4362}
4363
4364STDMETHODIMP Machine::GetCPUStatus(ULONG aCpu, BOOL *aCpuAttached)
4365{
4366 LogFlowThisFunc(("\n"));
4367
4368 CheckComArgNotNull(aCpuAttached);
4369
4370 *aCpuAttached = false;
4371
4372 AutoCaller autoCaller(this);
4373 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4374
4375 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4376
4377 /* If hotplug is enabled the CPU is always enabled. */
4378 if (!mHWData->mCPUHotPlugEnabled)
4379 {
4380 if (aCpu < mHWData->mCPUCount)
4381 *aCpuAttached = true;
4382 }
4383 else
4384 {
4385 if (aCpu < SchemaDefs::MaxCPUCount)
4386 *aCpuAttached = mHWData->mCPUAttached[aCpu];
4387 }
4388
4389 return S_OK;
4390}
4391
4392// public methods for internal purposes
4393/////////////////////////////////////////////////////////////////////////////
4394
4395/**
4396 * Adds the given IsModified_* flag to the dirty flags of the machine.
4397 * This must be called either during loadSettings or under the machine write lock.
4398 * @param fl
4399 */
4400void Machine::setModified(uint32_t fl)
4401{
4402 m_flModifications |= fl;
4403}
4404
4405/**
4406 * Saves the registry entry of this machine to the given configuration node.
4407 *
4408 * @param aEntryNode Node to save the registry entry to.
4409 *
4410 * @note locks this object for reading.
4411 */
4412HRESULT Machine::saveRegistryEntry(settings::MachineRegistryEntry &data)
4413{
4414 AutoLimitedCaller autoCaller(this);
4415 AssertComRCReturnRC(autoCaller.rc());
4416
4417 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4418
4419 data.uuid = mData->mUuid;
4420 data.strSettingsFile = mData->m_strConfigFile;
4421
4422 return S_OK;
4423}
4424
4425/**
4426 * Calculates the absolute path of the given path taking the directory of the
4427 * machine settings file as the current directory.
4428 *
4429 * @param aPath Path to calculate the absolute path for.
4430 * @param aResult Where to put the result (used only on success, can be the
4431 * same Utf8Str instance as passed in @a aPath).
4432 * @return IPRT result.
4433 *
4434 * @note Locks this object for reading.
4435 */
4436int Machine::calculateFullPath(const Utf8Str &strPath, Utf8Str &aResult)
4437{
4438 AutoCaller autoCaller(this);
4439 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4440
4441 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4442
4443 AssertReturn (!mData->m_strConfigFileFull.isEmpty(), VERR_GENERAL_FAILURE);
4444
4445 Utf8Str strSettingsDir = mData->m_strConfigFileFull;
4446
4447 strSettingsDir.stripFilename();
4448 char folder[RTPATH_MAX];
4449 int vrc = RTPathAbsEx(strSettingsDir.c_str(), strPath.c_str(), folder, sizeof(folder));
4450 if (RT_SUCCESS(vrc))
4451 aResult = folder;
4452
4453 return vrc;
4454}
4455
4456/**
4457 * Tries to calculate the relative path of the given absolute path using the
4458 * directory of the machine settings file as the base directory.
4459 *
4460 * @param aPath Absolute path to calculate the relative path for.
4461 * @param aResult Where to put the result (used only when it's possible to
4462 * make a relative path from the given absolute path; otherwise
4463 * left untouched).
4464 *
4465 * @note Locks this object for reading.
4466 */
4467void Machine::calculateRelativePath(const Utf8Str &strPath, Utf8Str &aResult)
4468{
4469 AutoCaller autoCaller(this);
4470 AssertComRCReturn (autoCaller.rc(), (void) 0);
4471
4472 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4473
4474 AssertReturnVoid (!mData->m_strConfigFileFull.isEmpty());
4475
4476 Utf8Str settingsDir = mData->m_strConfigFileFull;
4477
4478 settingsDir.stripFilename();
4479 if (RTPathStartsWith(strPath.c_str(), settingsDir.c_str()))
4480 {
4481 /* when assigning, we create a separate Utf8Str instance because both
4482 * aPath and aResult can point to the same memory location when this
4483 * func is called (if we just do aResult = aPath, aResult will be freed
4484 * first, and since its the same as aPath, an attempt to copy garbage
4485 * will be made. */
4486 aResult = Utf8Str(strPath.c_str() + settingsDir.length() + 1);
4487 }
4488}
4489
4490/**
4491 * Returns the full path to the machine's log folder in the
4492 * \a aLogFolder argument.
4493 */
4494void Machine::getLogFolder (Utf8Str &aLogFolder)
4495{
4496 AutoCaller autoCaller(this);
4497 AssertComRCReturnVoid (autoCaller.rc());
4498
4499 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4500
4501 Utf8Str settingsDir;
4502 if (isInOwnDir (&settingsDir))
4503 {
4504 /* Log folder is <Machines>/<VM_Name>/Logs */
4505 aLogFolder = Utf8StrFmt ("%s%cLogs", settingsDir.raw(), RTPATH_DELIMITER);
4506 }
4507 else
4508 {
4509 /* Log folder is <Machines>/<VM_SnapshotFolder>/Logs */
4510 Assert (!mUserData->mSnapshotFolderFull.isEmpty());
4511 aLogFolder = Utf8StrFmt ("%ls%cLogs", mUserData->mSnapshotFolderFull.raw(),
4512 RTPATH_DELIMITER);
4513 }
4514}
4515
4516/**
4517 * @note Locks this object for writing, calls the client process (outside the
4518 * lock).
4519 */
4520HRESULT Machine::openSession(IInternalSessionControl *aControl)
4521{
4522 LogFlowThisFuncEnter();
4523
4524 AssertReturn(aControl, E_FAIL);
4525
4526 AutoCaller autoCaller(this);
4527 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4528
4529 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4530
4531 if (!mData->mRegistered)
4532 return setError(E_UNEXPECTED,
4533 tr("The machine '%ls' is not registered"),
4534 mUserData->mName.raw());
4535
4536 LogFlowThisFunc(("mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
4537
4538 /* Hack: in case the session is closing and there is a progress object
4539 * which allows waiting for the session to be closed, take the opportunity
4540 * and do a limited wait (max. 1 second). This helps a lot when the system
4541 * is busy and thus session closing can take a little while. */
4542 if ( mData->mSession.mState == SessionState_Closing
4543 && mData->mSession.mProgress)
4544 {
4545 alock.leave();
4546 mData->mSession.mProgress->WaitForCompletion(1000);
4547 alock.enter();
4548 LogFlowThisFunc(("after waiting: mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
4549 }
4550
4551 if (mData->mSession.mState == SessionState_Open ||
4552 mData->mSession.mState == SessionState_Closing)
4553 return setError(VBOX_E_INVALID_OBJECT_STATE,
4554 tr("A session for the machine '%ls' is currently open (or being closed)"),
4555 mUserData->mName.raw());
4556
4557 /* may not be busy */
4558 AssertReturn(!Global::IsOnlineOrTransient(mData->mMachineState), E_FAIL);
4559
4560 /* get the session PID */
4561 RTPROCESS pid = NIL_RTPROCESS;
4562 AssertCompile(sizeof(ULONG) == sizeof(RTPROCESS));
4563 aControl->GetPID((ULONG *) &pid);
4564 Assert(pid != NIL_RTPROCESS);
4565
4566 if (mData->mSession.mState == SessionState_Spawning)
4567 {
4568 /* This machine is awaiting for a spawning session to be opened, so
4569 * reject any other open attempts from processes other than one
4570 * started by #openRemoteSession(). */
4571
4572 LogFlowThisFunc(("mSession.mPid=%d(0x%x)\n",
4573 mData->mSession.mPid, mData->mSession.mPid));
4574 LogFlowThisFunc(("session.pid=%d(0x%x)\n", pid, pid));
4575
4576 if (mData->mSession.mPid != pid)
4577 return setError(E_ACCESSDENIED,
4578 tr("An unexpected process (PID=0x%08X) has tried to open a direct "
4579 "session with the machine named '%ls', while only a process "
4580 "started by OpenRemoteSession (PID=0x%08X) is allowed"),
4581 pid, mUserData->mName.raw(), mData->mSession.mPid);
4582 }
4583
4584 /* create a SessionMachine object */
4585 ComObjPtr<SessionMachine> sessionMachine;
4586 sessionMachine.createObject();
4587 HRESULT rc = sessionMachine->init(this);
4588 AssertComRC(rc);
4589
4590 /* NOTE: doing return from this function after this point but
4591 * before the end is forbidden since it may call SessionMachine::uninit()
4592 * (through the ComObjPtr's destructor) which requests the VirtualBox write
4593 * lock while still holding the Machine lock in alock so that a deadlock
4594 * is possible due to the wrong lock order. */
4595
4596 if (SUCCEEDED(rc))
4597 {
4598#ifdef VBOX_WITH_RESOURCE_USAGE_API
4599 registerMetrics(mParent->performanceCollector(), this, pid);
4600#endif /* VBOX_WITH_RESOURCE_USAGE_API */
4601
4602 /*
4603 * Set the session state to Spawning to protect against subsequent
4604 * attempts to open a session and to unregister the machine after
4605 * we leave the lock.
4606 */
4607 SessionState_T origState = mData->mSession.mState;
4608 mData->mSession.mState = SessionState_Spawning;
4609
4610 /*
4611 * Leave the lock before calling the client process -- it will call
4612 * Machine/SessionMachine methods. Leaving the lock here is quite safe
4613 * because the state is Spawning, so that openRemotesession() and
4614 * openExistingSession() calls will fail. This method, called before we
4615 * enter the lock again, will fail because of the wrong PID.
4616 *
4617 * Note that mData->mSession.mRemoteControls accessed outside
4618 * the lock may not be modified when state is Spawning, so it's safe.
4619 */
4620 alock.leave();
4621
4622 LogFlowThisFunc(("Calling AssignMachine()...\n"));
4623 rc = aControl->AssignMachine(sessionMachine);
4624 LogFlowThisFunc(("AssignMachine() returned %08X\n", rc));
4625
4626 /* The failure may occur w/o any error info (from RPC), so provide one */
4627 if (FAILED(rc))
4628 setError(VBOX_E_VM_ERROR,
4629 tr("Failed to assign the machine to the session (%Rrc)"), rc);
4630
4631 if (SUCCEEDED(rc) && origState == SessionState_Spawning)
4632 {
4633 /* complete the remote session initialization */
4634
4635 /* get the console from the direct session */
4636 ComPtr<IConsole> console;
4637 rc = aControl->GetRemoteConsole(console.asOutParam());
4638 ComAssertComRC(rc);
4639
4640 if (SUCCEEDED(rc) && !console)
4641 {
4642 ComAssert(!!console);
4643 rc = E_FAIL;
4644 }
4645
4646 /* assign machine & console to the remote session */
4647 if (SUCCEEDED(rc))
4648 {
4649 /*
4650 * after openRemoteSession(), the first and the only
4651 * entry in remoteControls is that remote session
4652 */
4653 LogFlowThisFunc(("Calling AssignRemoteMachine()...\n"));
4654 rc = mData->mSession.mRemoteControls.front()->
4655 AssignRemoteMachine(sessionMachine, console);
4656 LogFlowThisFunc(("AssignRemoteMachine() returned %08X\n", rc));
4657
4658 /* The failure may occur w/o any error info (from RPC), so provide one */
4659 if (FAILED(rc))
4660 setError(VBOX_E_VM_ERROR,
4661 tr("Failed to assign the machine to the remote session (%Rrc)"), rc);
4662 }
4663
4664 if (FAILED(rc))
4665 aControl->Uninitialize();
4666 }
4667
4668 /* enter the lock again */
4669 alock.enter();
4670
4671 /* Restore the session state */
4672 mData->mSession.mState = origState;
4673 }
4674
4675 /* finalize spawning anyway (this is why we don't return on errors above) */
4676 if (mData->mSession.mState == SessionState_Spawning)
4677 {
4678 /* Note that the progress object is finalized later */
4679
4680 /* We don't reset mSession.mPid here because it is necessary for
4681 * SessionMachine::uninit() to reap the child process later. */
4682
4683 if (FAILED(rc))
4684 {
4685 /* Close the remote session, remove the remote control from the list
4686 * and reset session state to Closed (@note keep the code in sync
4687 * with the relevant part in openSession()). */
4688
4689 Assert (mData->mSession.mRemoteControls.size() == 1);
4690 if (mData->mSession.mRemoteControls.size() == 1)
4691 {
4692 ErrorInfoKeeper eik;
4693 mData->mSession.mRemoteControls.front()->Uninitialize();
4694 }
4695
4696 mData->mSession.mRemoteControls.clear();
4697 mData->mSession.mState = SessionState_Closed;
4698 }
4699 }
4700 else
4701 {
4702 /* memorize PID of the directly opened session */
4703 if (SUCCEEDED(rc))
4704 mData->mSession.mPid = pid;
4705 }
4706
4707 if (SUCCEEDED(rc))
4708 {
4709 /* memorize the direct session control and cache IUnknown for it */
4710 mData->mSession.mDirectControl = aControl;
4711 mData->mSession.mState = SessionState_Open;
4712 /* associate the SessionMachine with this Machine */
4713 mData->mSession.mMachine = sessionMachine;
4714
4715 /* request an IUnknown pointer early from the remote party for later
4716 * identity checks (it will be internally cached within mDirectControl
4717 * at least on XPCOM) */
4718 ComPtr<IUnknown> unk = mData->mSession.mDirectControl;
4719 NOREF(unk);
4720 }
4721
4722 if (mData->mSession.mProgress)
4723 {
4724 /* finalize the progress after setting the state, for consistency */
4725 mData->mSession.mProgress->notifyComplete(rc);
4726 mData->mSession.mProgress.setNull();
4727 }
4728
4729 /* Leave the lock since SessionMachine::uninit() locks VirtualBox which
4730 * would break the lock order */
4731 alock.leave();
4732
4733 /* uninitialize the created session machine on failure */
4734 if (FAILED(rc))
4735 sessionMachine->uninit();
4736
4737 LogFlowThisFunc(("rc=%08X\n", rc));
4738 LogFlowThisFuncLeave();
4739 return rc;
4740}
4741
4742/**
4743 * @note Locks this object for writing, calls the client process
4744 * (inside the lock).
4745 */
4746HRESULT Machine::openRemoteSession(IInternalSessionControl *aControl,
4747 IN_BSTR aType,
4748 IN_BSTR aEnvironment,
4749 Progress *aProgress)
4750{
4751 LogFlowThisFuncEnter();
4752
4753 AssertReturn(aControl, E_FAIL);
4754 AssertReturn(aProgress, E_FAIL);
4755
4756 AutoCaller autoCaller(this);
4757 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4758
4759 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4760
4761 if (!mData->mRegistered)
4762 return setError(E_UNEXPECTED,
4763 tr("The machine '%ls' is not registered"),
4764 mUserData->mName.raw());
4765
4766 LogFlowThisFunc(("mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
4767
4768 if (mData->mSession.mState == SessionState_Open ||
4769 mData->mSession.mState == SessionState_Spawning ||
4770 mData->mSession.mState == SessionState_Closing)
4771 return setError(VBOX_E_INVALID_OBJECT_STATE,
4772 tr("A session for the machine '%ls' is currently open (or being opened or closed)"),
4773 mUserData->mName.raw());
4774
4775 /* may not be busy */
4776 AssertReturn(!Global::IsOnlineOrTransient (mData->mMachineState), E_FAIL);
4777
4778 /* get the path to the executable */
4779 char szPath[RTPATH_MAX];
4780 RTPathAppPrivateArch(szPath, RTPATH_MAX);
4781 size_t sz = strlen(szPath);
4782 szPath[sz++] = RTPATH_DELIMITER;
4783 szPath[sz] = 0;
4784 char *cmd = szPath + sz;
4785 sz = RTPATH_MAX - sz;
4786
4787 int vrc = VINF_SUCCESS;
4788 RTPROCESS pid = NIL_RTPROCESS;
4789
4790 RTENV env = RTENV_DEFAULT;
4791
4792 if (aEnvironment != NULL && *aEnvironment)
4793 {
4794 char *newEnvStr = NULL;
4795
4796 do
4797 {
4798 /* clone the current environment */
4799 int vrc2 = RTEnvClone(&env, RTENV_DEFAULT);
4800 AssertRCBreakStmt(vrc2, vrc = vrc2);
4801
4802 newEnvStr = RTStrDup(Utf8Str(aEnvironment).c_str());
4803 AssertPtrBreakStmt(newEnvStr, vrc = vrc2);
4804
4805 /* put new variables to the environment
4806 * (ignore empty variable names here since RTEnv API
4807 * intentionally doesn't do that) */
4808 char *var = newEnvStr;
4809 for (char *p = newEnvStr; *p; ++p)
4810 {
4811 if (*p == '\n' && (p == newEnvStr || *(p - 1) != '\\'))
4812 {
4813 *p = '\0';
4814 if (*var)
4815 {
4816 char *val = strchr (var, '=');
4817 if (val)
4818 {
4819 *val++ = '\0';
4820 vrc2 = RTEnvSetEx (env, var, val);
4821 }
4822 else
4823 vrc2 = RTEnvUnsetEx (env, var);
4824 if (RT_FAILURE(vrc2))
4825 break;
4826 }
4827 var = p + 1;
4828 }
4829 }
4830 if (RT_SUCCESS(vrc2) && *var)
4831 vrc2 = RTEnvPutEx (env, var);
4832
4833 AssertRCBreakStmt (vrc2, vrc = vrc2);
4834 }
4835 while (0);
4836
4837 if (newEnvStr != NULL)
4838 RTStrFree(newEnvStr);
4839 }
4840
4841 Utf8Str strType(aType);
4842
4843 /* Qt is default */
4844#ifdef VBOX_WITH_QTGUI
4845 if (strType == "gui" || strType == "GUI/Qt")
4846 {
4847# ifdef RT_OS_DARWIN /* Avoid Launch Services confusing this with the selector by using a helper app. */
4848 const char VirtualBox_exe[] = "../Resources/VirtualBoxVM.app/Contents/MacOS/VirtualBoxVM";
4849# else
4850 const char VirtualBox_exe[] = "VirtualBox" HOSTSUFF_EXE;
4851# endif
4852 Assert (sz >= sizeof (VirtualBox_exe));
4853 strcpy (cmd, VirtualBox_exe);
4854
4855 Utf8Str idStr = mData->mUuid.toString();
4856# ifdef RT_OS_WINDOWS /** @todo drop this once the RTProcCreate bug has been fixed */
4857 const char * args[] = {szPath, "--startvm", idStr.c_str(), 0 };
4858# else
4859 Utf8Str strName = mUserData->mName;
4860 const char * args[] = {szPath, "--comment", strName.c_str(), "--startvm", idStr.c_str(), 0 };
4861# endif
4862 vrc = RTProcCreate(szPath, args, env, 0, &pid);
4863 }
4864#else /* !VBOX_WITH_QTGUI */
4865 if (0)
4866 ;
4867#endif /* VBOX_WITH_QTGUI */
4868
4869 else
4870
4871#ifdef VBOX_WITH_VBOXSDL
4872 if (strType == "sdl" || strType == "GUI/SDL")
4873 {
4874 const char VBoxSDL_exe[] = "VBoxSDL" HOSTSUFF_EXE;
4875 Assert (sz >= sizeof (VBoxSDL_exe));
4876 strcpy (cmd, VBoxSDL_exe);
4877
4878 Utf8Str idStr = mData->mUuid.toString();
4879# ifdef RT_OS_WINDOWS
4880 const char * args[] = {szPath, "--startvm", idStr.c_str(), 0 };
4881# else
4882 Utf8Str strName = mUserData->mName;
4883 const char * args[] = {szPath, "--comment", strName.c_str(), "--startvm", idStr.c_str(), 0 };
4884# endif
4885 vrc = RTProcCreate(szPath, args, env, 0, &pid);
4886 }
4887#else /* !VBOX_WITH_VBOXSDL */
4888 if (0)
4889 ;
4890#endif /* !VBOX_WITH_VBOXSDL */
4891
4892 else
4893
4894#ifdef VBOX_WITH_HEADLESS
4895 if ( strType == "headless"
4896 || strType == "capture"
4897#ifdef VBOX_WITH_VRDP
4898 || strType == "vrdp"
4899#endif
4900 )
4901 {
4902 const char VBoxHeadless_exe[] = "VBoxHeadless" HOSTSUFF_EXE;
4903 Assert (sz >= sizeof (VBoxHeadless_exe));
4904 strcpy (cmd, VBoxHeadless_exe);
4905
4906 Utf8Str idStr = mData->mUuid.toString();
4907 /* Leave space for 2 args, as "headless" needs --vrdp off on non-OSE. */
4908# ifdef RT_OS_WINDOWS
4909 const char * args[] = {szPath, "--startvm", idStr.c_str(), 0, 0, 0 };
4910# else
4911 Utf8Str strName = mUserData->mName;
4912 const char * args[] ={szPath, "--comment", strName.c_str(), "--startvm", idStr.c_str(), 0, 0, 0 };
4913# endif
4914#ifdef VBOX_WITH_VRDP
4915 if (strType == "headless")
4916 {
4917 unsigned pos = RT_ELEMENTS(args) - 3;
4918 args[pos++] = "--vrdp";
4919 args[pos] = "off";
4920 }
4921#endif
4922 if (strType == "capture")
4923 {
4924 unsigned pos = RT_ELEMENTS(args) - 3;
4925 args[pos] = "--capture";
4926 }
4927 vrc = RTProcCreate(szPath, args, env, 0, &pid);
4928 }
4929#else /* !VBOX_WITH_HEADLESS */
4930 if (0)
4931 ;
4932#endif /* !VBOX_WITH_HEADLESS */
4933 else
4934 {
4935 RTEnvDestroy (env);
4936 return setError(E_INVALIDARG,
4937 tr("Invalid session type: '%s'"),
4938 strType.c_str());
4939 }
4940
4941 RTEnvDestroy (env);
4942
4943 if (RT_FAILURE(vrc))
4944 return setError(VBOX_E_IPRT_ERROR,
4945 tr("Could not launch a process for the machine '%ls' (%Rrc)"),
4946 mUserData->mName.raw(), vrc);
4947
4948 LogFlowThisFunc(("launched.pid=%d(0x%x)\n", pid, pid));
4949
4950 /*
4951 * Note that we don't leave the lock here before calling the client,
4952 * because it doesn't need to call us back if called with a NULL argument.
4953 * Leaving the lock herer is dangerous because we didn't prepare the
4954 * launch data yet, but the client we've just started may happen to be
4955 * too fast and call openSession() that will fail (because of PID, etc.),
4956 * so that the Machine will never get out of the Spawning session state.
4957 */
4958
4959 /* inform the session that it will be a remote one */
4960 LogFlowThisFunc(("Calling AssignMachine (NULL)...\n"));
4961 HRESULT rc = aControl->AssignMachine (NULL);
4962 LogFlowThisFunc(("AssignMachine (NULL) returned %08X\n", rc));
4963
4964 if (FAILED(rc))
4965 {
4966 /* restore the session state */
4967 mData->mSession.mState = SessionState_Closed;
4968 /* The failure may occur w/o any error info (from RPC), so provide one */
4969 return setError(VBOX_E_VM_ERROR,
4970 tr("Failed to assign the machine to the session (%Rrc)"), rc);
4971 }
4972
4973 /* attach launch data to the machine */
4974 Assert (mData->mSession.mPid == NIL_RTPROCESS);
4975 mData->mSession.mRemoteControls.push_back (aControl);
4976 mData->mSession.mProgress = aProgress;
4977 mData->mSession.mPid = pid;
4978 mData->mSession.mState = SessionState_Spawning;
4979 mData->mSession.mType = strType;
4980
4981 LogFlowThisFuncLeave();
4982 return S_OK;
4983}
4984
4985/**
4986 * @note Locks this object for writing, calls the client process
4987 * (outside the lock).
4988 */
4989HRESULT Machine::openExistingSession (IInternalSessionControl *aControl)
4990{
4991 LogFlowThisFuncEnter();
4992
4993 AssertReturn(aControl, E_FAIL);
4994
4995 AutoCaller autoCaller(this);
4996 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4997
4998 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4999
5000 if (!mData->mRegistered)
5001 return setError (E_UNEXPECTED,
5002 tr ("The machine '%ls' is not registered"), mUserData->mName.raw());
5003
5004 LogFlowThisFunc(("mSession.state=%s\n", Global::stringifySessionState(mData->mSession.mState)));
5005
5006 if (mData->mSession.mState != SessionState_Open)
5007 return setError (VBOX_E_INVALID_SESSION_STATE,
5008 tr ("The machine '%ls' does not have an open session"),
5009 mUserData->mName.raw());
5010
5011 ComAssertRet (!mData->mSession.mDirectControl.isNull(), E_FAIL);
5012
5013 /*
5014 * Get the console from the direct session (note that we don't leave the
5015 * lock here because GetRemoteConsole must not call us back).
5016 */
5017 ComPtr<IConsole> console;
5018 HRESULT rc = mData->mSession.mDirectControl->
5019 GetRemoteConsole (console.asOutParam());
5020 if (FAILED (rc))
5021 {
5022 /* The failure may occur w/o any error info (from RPC), so provide one */
5023 return setError (VBOX_E_VM_ERROR,
5024 tr ("Failed to get a console object from the direct session (%Rrc)"), rc);
5025 }
5026
5027 ComAssertRet (!console.isNull(), E_FAIL);
5028
5029 ComObjPtr<SessionMachine> sessionMachine = mData->mSession.mMachine;
5030 AssertReturn(!sessionMachine.isNull(), E_FAIL);
5031
5032 /*
5033 * Leave the lock before calling the client process. It's safe here
5034 * since the only thing to do after we get the lock again is to add
5035 * the remote control to the list (which doesn't directly influence
5036 * anything).
5037 */
5038 alock.leave();
5039
5040 /* attach the remote session to the machine */
5041 LogFlowThisFunc(("Calling AssignRemoteMachine()...\n"));
5042 rc = aControl->AssignRemoteMachine (sessionMachine, console);
5043 LogFlowThisFunc(("AssignRemoteMachine() returned %08X\n", rc));
5044
5045 /* The failure may occur w/o any error info (from RPC), so provide one */
5046 if (FAILED(rc))
5047 return setError(VBOX_E_VM_ERROR,
5048 tr("Failed to assign the machine to the session (%Rrc)"),
5049 rc);
5050
5051 alock.enter();
5052
5053 /* need to revalidate the state after entering the lock again */
5054 if (mData->mSession.mState != SessionState_Open)
5055 {
5056 aControl->Uninitialize();
5057
5058 return setError(VBOX_E_INVALID_SESSION_STATE,
5059 tr("The machine '%ls' does not have an open session"),
5060 mUserData->mName.raw());
5061 }
5062
5063 /* store the control in the list */
5064 mData->mSession.mRemoteControls.push_back (aControl);
5065
5066 LogFlowThisFuncLeave();
5067 return S_OK;
5068}
5069
5070/**
5071 * Returns @c true if the given machine has an open direct session and returns
5072 * the session machine instance and additional session data (on some platforms)
5073 * if so.
5074 *
5075 * Note that when the method returns @c false, the arguments remain unchanged.
5076 *
5077 * @param aMachine Session machine object.
5078 * @param aControl Direct session control object (optional).
5079 * @param aIPCSem Mutex IPC semaphore handle for this machine (optional).
5080 *
5081 * @note locks this object for reading.
5082 */
5083#if defined (RT_OS_WINDOWS)
5084bool Machine::isSessionOpen (ComObjPtr<SessionMachine> &aMachine,
5085 ComPtr<IInternalSessionControl> *aControl /*= NULL*/,
5086 HANDLE *aIPCSem /*= NULL*/,
5087 bool aAllowClosing /*= false*/)
5088#elif defined (RT_OS_OS2)
5089bool Machine::isSessionOpen (ComObjPtr<SessionMachine> &aMachine,
5090 ComPtr<IInternalSessionControl> *aControl /*= NULL*/,
5091 HMTX *aIPCSem /*= NULL*/,
5092 bool aAllowClosing /*= false*/)
5093#else
5094bool Machine::isSessionOpen (ComObjPtr<SessionMachine> &aMachine,
5095 ComPtr<IInternalSessionControl> *aControl /*= NULL*/,
5096 bool aAllowClosing /*= false*/)
5097#endif
5098{
5099 AutoLimitedCaller autoCaller(this);
5100 AssertComRCReturn (autoCaller.rc(), false);
5101
5102 /* just return false for inaccessible machines */
5103 if (autoCaller.state() != Ready)
5104 return false;
5105
5106 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5107
5108 if (mData->mSession.mState == SessionState_Open ||
5109 (aAllowClosing && mData->mSession.mState == SessionState_Closing))
5110 {
5111 AssertReturn(!mData->mSession.mMachine.isNull(), false);
5112
5113 aMachine = mData->mSession.mMachine;
5114
5115 if (aControl != NULL)
5116 *aControl = mData->mSession.mDirectControl;
5117
5118#if defined (RT_OS_WINDOWS) || defined (RT_OS_OS2)
5119 /* Additional session data */
5120 if (aIPCSem != NULL)
5121 *aIPCSem = aMachine->mIPCSem;
5122#endif
5123 return true;
5124 }
5125
5126 return false;
5127}
5128
5129/**
5130 * Returns @c true if the given machine has an spawning direct session and
5131 * returns and additional session data (on some platforms) if so.
5132 *
5133 * Note that when the method returns @c false, the arguments remain unchanged.
5134 *
5135 * @param aPID PID of the spawned direct session process.
5136 *
5137 * @note locks this object for reading.
5138 */
5139#if defined (RT_OS_WINDOWS) || defined (RT_OS_OS2)
5140bool Machine::isSessionSpawning (RTPROCESS *aPID /*= NULL*/)
5141#else
5142bool Machine::isSessionSpawning()
5143#endif
5144{
5145 AutoLimitedCaller autoCaller(this);
5146 AssertComRCReturn (autoCaller.rc(), false);
5147
5148 /* just return false for inaccessible machines */
5149 if (autoCaller.state() != Ready)
5150 return false;
5151
5152 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5153
5154 if (mData->mSession.mState == SessionState_Spawning)
5155 {
5156#if defined (RT_OS_WINDOWS) || defined (RT_OS_OS2)
5157 /* Additional session data */
5158 if (aPID != NULL)
5159 {
5160 AssertReturn(mData->mSession.mPid != NIL_RTPROCESS, false);
5161 *aPID = mData->mSession.mPid;
5162 }
5163#endif
5164 return true;
5165 }
5166
5167 return false;
5168}
5169
5170/**
5171 * Called from the client watcher thread to check for unexpected client process
5172 * death during Session_Spawning state (e.g. before it successfully opened a
5173 * direct session).
5174 *
5175 * On Win32 and on OS/2, this method is called only when we've got the
5176 * direct client's process termination notification, so it always returns @c
5177 * true.
5178 *
5179 * On other platforms, this method returns @c true if the client process is
5180 * terminated and @c false if it's still alive.
5181 *
5182 * @note Locks this object for writing.
5183 */
5184bool Machine::checkForSpawnFailure()
5185{
5186 AutoCaller autoCaller(this);
5187 if (!autoCaller.isOk())
5188 {
5189 /* nothing to do */
5190 LogFlowThisFunc(("Already uninitialized!\n"));
5191 return true;
5192 }
5193
5194 /* VirtualBox::addProcessToReap() needs a write lock */
5195 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
5196
5197 if (mData->mSession.mState != SessionState_Spawning)
5198 {
5199 /* nothing to do */
5200 LogFlowThisFunc(("Not spawning any more!\n"));
5201 return true;
5202 }
5203
5204 HRESULT rc = S_OK;
5205
5206#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
5207
5208 /* the process was already unexpectedly terminated, we just need to set an
5209 * error and finalize session spawning */
5210 rc = setError(E_FAIL,
5211 tr("Virtual machine '%ls' has terminated unexpectedly during startup"),
5212 getName().raw());
5213#else
5214
5215 /* PID not yet initialized, skip check. */
5216 if (mData->mSession.mPid == NIL_RTPROCESS)
5217 return false;
5218
5219 RTPROCSTATUS status;
5220 int vrc = ::RTProcWait(mData->mSession.mPid, RTPROCWAIT_FLAGS_NOBLOCK,
5221 &status);
5222
5223 if (vrc != VERR_PROCESS_RUNNING)
5224 rc = setError(E_FAIL,
5225 tr("Virtual machine '%ls' has terminated unexpectedly during startup"),
5226 getName().raw());
5227#endif
5228
5229 if (FAILED(rc))
5230 {
5231 /* Close the remote session, remove the remote control from the list
5232 * and reset session state to Closed (@note keep the code in sync with
5233 * the relevant part in checkForSpawnFailure()). */
5234
5235 Assert(mData->mSession.mRemoteControls.size() == 1);
5236 if (mData->mSession.mRemoteControls.size() == 1)
5237 {
5238 ErrorInfoKeeper eik;
5239 mData->mSession.mRemoteControls.front()->Uninitialize();
5240 }
5241
5242 mData->mSession.mRemoteControls.clear();
5243 mData->mSession.mState = SessionState_Closed;
5244
5245 /* finalize the progress after setting the state, for consistency */
5246 if (!mData->mSession.mProgress.isNull())
5247 {
5248 mData->mSession.mProgress->notifyComplete(rc);
5249 mData->mSession.mProgress.setNull();
5250 }
5251
5252 mParent->addProcessToReap(mData->mSession.mPid);
5253 mData->mSession.mPid = NIL_RTPROCESS;
5254
5255 mParent->onSessionStateChange(mData->mUuid, SessionState_Closed);
5256 return true;
5257 }
5258
5259 return false;
5260}
5261
5262/**
5263 * Checks that the registered flag of the machine can be set according to
5264 * the argument and sets it. On success, commits and saves all settings.
5265 *
5266 * @note When this machine is inaccessible, the only valid value for \a
5267 * aRegistered is FALSE (i.e. unregister the machine) because unregistered
5268 * inaccessible machines are not currently supported. Note that unregistering
5269 * an inaccessible machine will \b uninitialize this machine object. Therefore,
5270 * the caller must make sure there are no active Machine::addCaller() calls
5271 * on the current thread because this will block Machine::uninit().
5272 *
5273 * @note Must be called from mParent's write lock. Locks this object and
5274 * children for writing.
5275 */
5276HRESULT Machine::trySetRegistered(BOOL argNewRegistered)
5277{
5278 AssertReturn(mParent->isWriteLockOnCurrentThread(), E_FAIL);
5279
5280 AutoLimitedCaller autoCaller(this);
5281 AssertComRCReturnRC(autoCaller.rc());
5282
5283 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5284
5285 /* wait for state dependants to drop to zero */
5286 ensureNoStateDependencies();
5287
5288 ComAssertRet(mData->mRegistered != argNewRegistered, E_FAIL);
5289
5290 if (!mData->mAccessible)
5291 {
5292 /* A special case: the machine is not accessible. */
5293
5294 /* inaccessible machines can only be unregistered */
5295 AssertReturn(!argNewRegistered, E_FAIL);
5296
5297 /* Uninitialize ourselves here because currently there may be no
5298 * unregistered that are inaccessible (this state combination is not
5299 * supported). Note releasing the caller and leaving the lock before
5300 * calling uninit() */
5301
5302 alock.leave();
5303 autoCaller.release();
5304
5305 uninit();
5306
5307 return S_OK;
5308 }
5309
5310 AssertReturn(autoCaller.state() == Ready, E_FAIL);
5311
5312 if (argNewRegistered)
5313 {
5314 if (mData->mRegistered)
5315 return setError(VBOX_E_INVALID_OBJECT_STATE,
5316 tr("The machine '%ls' with UUID {%s} is already registered"),
5317 mUserData->mName.raw(),
5318 mData->mUuid.toString().raw());
5319 }
5320 else
5321 {
5322 if (mData->mMachineState == MachineState_Saved)
5323 return setError(VBOX_E_INVALID_VM_STATE,
5324 tr("Cannot unregister the machine '%ls' because it is in the Saved state"),
5325 mUserData->mName.raw());
5326
5327 size_t snapshotCount = 0;
5328 if (mData->mFirstSnapshot)
5329 snapshotCount = mData->mFirstSnapshot->getAllChildrenCount() + 1;
5330 if (snapshotCount)
5331 return setError(VBOX_E_INVALID_OBJECT_STATE,
5332 tr("Cannot unregister the machine '%ls' because it has %d snapshots"),
5333 mUserData->mName.raw(), snapshotCount);
5334
5335 if (mData->mSession.mState != SessionState_Closed)
5336 return setError(VBOX_E_INVALID_OBJECT_STATE,
5337 tr("Cannot unregister the machine '%ls' because it has an open session"),
5338 mUserData->mName.raw());
5339
5340 if (mMediaData->mAttachments.size() != 0)
5341 return setError(VBOX_E_INVALID_OBJECT_STATE,
5342 tr("Cannot unregister the machine '%ls' because it has %d medium attachments"),
5343 mUserData->mName.raw(),
5344 mMediaData->mAttachments.size());
5345
5346 /* Note that we do not prevent unregistration of a DVD or Floppy image
5347 * is attached: as opposed to hard disks detaching such an image
5348 * implicitly in this method (which we will do below) won't have any
5349 * side effects (like detached orphan base and diff hard disks etc).*/
5350 }
5351
5352 HRESULT rc = S_OK;
5353
5354 // Ensure the settings are saved. If we are going to be registered and
5355 // no config file exists yet, create it by calling saveSettings() too.
5356 if ( (m_flModifications)
5357 || (argNewRegistered && !mData->m_pMachineConfigFile->fileExists())
5358 )
5359 {
5360 rc = saveSettings();
5361 if (FAILED(rc)) return rc;
5362 }
5363
5364 /* more config checking goes here */
5365
5366 if (SUCCEEDED(rc))
5367 {
5368 /* we may have had implicit modifications we want to fix on success */
5369 commit();
5370
5371 mData->mRegistered = argNewRegistered;
5372 }
5373 else
5374 {
5375 /* we may have had implicit modifications we want to cancel on failure*/
5376 rollback (false /* aNotify */);
5377 }
5378
5379 return rc;
5380}
5381
5382/**
5383 * Increases the number of objects dependent on the machine state or on the
5384 * registered state. Guarantees that these two states will not change at least
5385 * until #releaseStateDependency() is called.
5386 *
5387 * Depending on the @a aDepType value, additional state checks may be made.
5388 * These checks will set extended error info on failure. See
5389 * #checkStateDependency() for more info.
5390 *
5391 * If this method returns a failure, the dependency is not added and the caller
5392 * is not allowed to rely on any particular machine state or registration state
5393 * value and may return the failed result code to the upper level.
5394 *
5395 * @param aDepType Dependency type to add.
5396 * @param aState Current machine state (NULL if not interested).
5397 * @param aRegistered Current registered state (NULL if not interested).
5398 *
5399 * @note Locks this object for writing.
5400 */
5401HRESULT Machine::addStateDependency (StateDependency aDepType /* = AnyStateDep */,
5402 MachineState_T *aState /* = NULL */,
5403 BOOL *aRegistered /* = NULL */)
5404{
5405 AutoCaller autoCaller(this);
5406 AssertComRCReturnRC(autoCaller.rc());
5407
5408 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5409
5410 HRESULT rc = checkStateDependency(aDepType);
5411 if (FAILED(rc)) return rc;
5412
5413 {
5414 if (mData->mMachineStateChangePending != 0)
5415 {
5416 /* ensureNoStateDependencies() is waiting for state dependencies to
5417 * drop to zero so don't add more. It may make sense to wait a bit
5418 * and retry before reporting an error (since the pending state
5419 * transition should be really quick) but let's just assert for
5420 * now to see if it ever happens on practice. */
5421
5422 AssertFailed();
5423
5424 return setError(E_ACCESSDENIED,
5425 tr("Machine state change is in progress. Please retry the operation later."));
5426 }
5427
5428 ++mData->mMachineStateDeps;
5429 Assert (mData->mMachineStateDeps != 0 /* overflow */);
5430 }
5431
5432 if (aState)
5433 *aState = mData->mMachineState;
5434 if (aRegistered)
5435 *aRegistered = mData->mRegistered;
5436
5437 return S_OK;
5438}
5439
5440/**
5441 * Decreases the number of objects dependent on the machine state.
5442 * Must always complete the #addStateDependency() call after the state
5443 * dependency is no more necessary.
5444 */
5445void Machine::releaseStateDependency()
5446{
5447 AutoCaller autoCaller(this);
5448 AssertComRCReturnVoid (autoCaller.rc());
5449
5450 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5451
5452 AssertReturnVoid (mData->mMachineStateDeps != 0
5453 /* releaseStateDependency() w/o addStateDependency()? */);
5454 -- mData->mMachineStateDeps;
5455
5456 if (mData->mMachineStateDeps == 0)
5457 {
5458 /* inform ensureNoStateDependencies() that there are no more deps */
5459 if (mData->mMachineStateChangePending != 0)
5460 {
5461 Assert (mData->mMachineStateDepsSem != NIL_RTSEMEVENTMULTI);
5462 RTSemEventMultiSignal (mData->mMachineStateDepsSem);
5463 }
5464 }
5465}
5466
5467// protected methods
5468/////////////////////////////////////////////////////////////////////////////
5469
5470/**
5471 * Performs machine state checks based on the @a aDepType value. If a check
5472 * fails, this method will set extended error info, otherwise it will return
5473 * S_OK. It is supposed, that on failure, the caller will immedieately return
5474 * the return value of this method to the upper level.
5475 *
5476 * When @a aDepType is AnyStateDep, this method always returns S_OK.
5477 *
5478 * When @a aDepType is MutableStateDep, this method returns S_OK only if the
5479 * current state of this machine object allows to change settings of the
5480 * machine (i.e. the machine is not registered, or registered but not running
5481 * and not saved). It is useful to call this method from Machine setters
5482 * before performing any change.
5483 *
5484 * When @a aDepType is MutableOrSavedStateDep, this method behaves the same
5485 * as for MutableStateDep except that if the machine is saved, S_OK is also
5486 * returned. This is useful in setters which allow changing machine
5487 * properties when it is in the saved state.
5488 *
5489 * @param aDepType Dependency type to check.
5490 *
5491 * @note Non Machine based classes should use #addStateDependency() and
5492 * #releaseStateDependency() methods or the smart AutoStateDependency
5493 * template.
5494 *
5495 * @note This method must be called from under this object's read or write
5496 * lock.
5497 */
5498HRESULT Machine::checkStateDependency(StateDependency aDepType)
5499{
5500 switch (aDepType)
5501 {
5502 case AnyStateDep:
5503 {
5504 break;
5505 }
5506 case MutableStateDep:
5507 {
5508 if ( mData->mRegistered
5509 && ( getClassID() != clsidSessionMachine /** @todo This was just convered raw; Check if Running and Paused should actually be included here... (Live Migration) */
5510 || ( mData->mMachineState != MachineState_Paused
5511 && mData->mMachineState != MachineState_Running
5512 && mData->mMachineState != MachineState_Aborted
5513 && mData->mMachineState != MachineState_Teleported
5514 && mData->mMachineState != MachineState_PoweredOff
5515 )
5516 )
5517 )
5518 return setError(VBOX_E_INVALID_VM_STATE,
5519 tr("The machine is not mutable (state is %s)"),
5520 Global::stringifyMachineState(mData->mMachineState));
5521 break;
5522 }
5523 case MutableOrSavedStateDep:
5524 {
5525 if ( mData->mRegistered
5526 && ( getClassID() != clsidSessionMachine /** @todo This was just convered raw; Check if Running and Paused should actually be included here... (Live Migration) */
5527 || ( mData->mMachineState != MachineState_Paused
5528 && mData->mMachineState != MachineState_Running
5529 && mData->mMachineState != MachineState_Aborted
5530 && mData->mMachineState != MachineState_Teleported
5531 && mData->mMachineState != MachineState_Saved
5532 && mData->mMachineState != MachineState_PoweredOff
5533 )
5534 )
5535 )
5536 return setError(VBOX_E_INVALID_VM_STATE,
5537 tr("The machine is not mutable (state is %s)"),
5538 Global::stringifyMachineState(mData->mMachineState));
5539 break;
5540 }
5541 }
5542
5543 return S_OK;
5544}
5545
5546/**
5547 * Helper to initialize all associated child objects and allocate data
5548 * structures.
5549 *
5550 * This method must be called as a part of the object's initialization procedure
5551 * (usually done in the #init() method).
5552 *
5553 * @note Must be called only from #init() or from #registeredInit().
5554 */
5555HRESULT Machine::initDataAndChildObjects()
5556{
5557 AutoCaller autoCaller(this);
5558 AssertComRCReturnRC(autoCaller.rc());
5559 AssertComRCReturn (autoCaller.state() == InInit ||
5560 autoCaller.state() == Limited, E_FAIL);
5561
5562 AssertReturn(!mData->mAccessible, E_FAIL);
5563
5564 /* allocate data structures */
5565 mSSData.allocate();
5566 mUserData.allocate();
5567 mHWData.allocate();
5568 mMediaData.allocate();
5569 mStorageControllers.allocate();
5570
5571 /* initialize mOSTypeId */
5572 mUserData->mOSTypeId = mParent->getUnknownOSType()->id();
5573
5574 /* create associated BIOS settings object */
5575 unconst(mBIOSSettings).createObject();
5576 mBIOSSettings->init (this);
5577
5578#ifdef VBOX_WITH_VRDP
5579 /* create an associated VRDPServer object (default is disabled) */
5580 unconst(mVRDPServer).createObject();
5581 mVRDPServer->init (this);
5582#endif
5583
5584 /* create associated serial port objects */
5585 for (ULONG slot = 0; slot < RT_ELEMENTS (mSerialPorts); slot ++)
5586 {
5587 unconst(mSerialPorts [slot]).createObject();
5588 mSerialPorts [slot]->init (this, slot);
5589 }
5590
5591 /* create associated parallel port objects */
5592 for (ULONG slot = 0; slot < RT_ELEMENTS (mParallelPorts); slot ++)
5593 {
5594 unconst(mParallelPorts [slot]).createObject();
5595 mParallelPorts [slot]->init (this, slot);
5596 }
5597
5598 /* create the audio adapter object (always present, default is disabled) */
5599 unconst(mAudioAdapter).createObject();
5600 mAudioAdapter->init (this);
5601
5602 /* create the USB controller object (always present, default is disabled) */
5603 unconst(mUSBController).createObject();
5604 mUSBController->init (this);
5605
5606 /* create associated network adapter objects */
5607 for (ULONG slot = 0; slot < RT_ELEMENTS (mNetworkAdapters); slot ++)
5608 {
5609 unconst(mNetworkAdapters [slot]).createObject();
5610 mNetworkAdapters [slot]->init (this, slot);
5611 }
5612
5613 return S_OK;
5614}
5615
5616/**
5617 * Helper to uninitialize all associated child objects and to free all data
5618 * structures.
5619 *
5620 * This method must be called as a part of the object's uninitialization
5621 * procedure (usually done in the #uninit() method).
5622 *
5623 * @note Must be called only from #uninit() or from #registeredInit().
5624 */
5625void Machine::uninitDataAndChildObjects()
5626{
5627 AutoCaller autoCaller(this);
5628 AssertComRCReturnVoid(autoCaller.rc());
5629 AssertComRCReturnVoid( autoCaller.state() == InUninit
5630 || autoCaller.state() == Limited);
5631
5632 /* uninit all children using addDependentChild()/removeDependentChild()
5633 * in their init()/uninit() methods */
5634 uninitDependentChildren();
5635
5636 /* tell all our other child objects we've been uninitialized */
5637
5638 for (ULONG slot = 0; slot < RT_ELEMENTS(mNetworkAdapters); slot++)
5639 {
5640 if (mNetworkAdapters[slot])
5641 {
5642 mNetworkAdapters[slot]->uninit();
5643 unconst(mNetworkAdapters[slot]).setNull();
5644 }
5645 }
5646
5647 if (mUSBController)
5648 {
5649 mUSBController->uninit();
5650 unconst(mUSBController).setNull();
5651 }
5652
5653 if (mAudioAdapter)
5654 {
5655 mAudioAdapter->uninit();
5656 unconst(mAudioAdapter).setNull();
5657 }
5658
5659 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
5660 {
5661 if (mParallelPorts[slot])
5662 {
5663 mParallelPorts[slot]->uninit();
5664 unconst(mParallelPorts[slot]).setNull();
5665 }
5666 }
5667
5668 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
5669 {
5670 if (mSerialPorts[slot])
5671 {
5672 mSerialPorts[slot]->uninit();
5673 unconst(mSerialPorts[slot]).setNull();
5674 }
5675 }
5676
5677#ifdef VBOX_WITH_VRDP
5678 if (mVRDPServer)
5679 {
5680 mVRDPServer->uninit();
5681 unconst(mVRDPServer).setNull();
5682 }
5683#endif
5684
5685 if (mBIOSSettings)
5686 {
5687 mBIOSSettings->uninit();
5688 unconst(mBIOSSettings).setNull();
5689 }
5690
5691 /* Deassociate hard disks (only when a real Machine or a SnapshotMachine
5692 * instance is uninitialized; SessionMachine instances refer to real
5693 * Machine hard disks). This is necessary for a clean re-initialization of
5694 * the VM after successfully re-checking the accessibility state. Note
5695 * that in case of normal Machine or SnapshotMachine uninitialization (as
5696 * a result of unregistering or discarding the snapshot), outdated hard
5697 * disk attachments will already be uninitialized and deleted, so this
5698 * code will not affect them. */
5699 VBoxClsID clsid = getClassID();
5700 if ( !!mMediaData
5701 && (clsid == clsidMachine || clsid == clsidSnapshotMachine)
5702 )
5703 {
5704 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
5705 it != mMediaData->mAttachments.end();
5706 ++it)
5707 {
5708 ComObjPtr<Medium> hd = (*it)->getMedium();
5709 if (hd.isNull())
5710 continue;
5711 HRESULT rc = hd->detachFrom(mData->mUuid, getSnapshotId());
5712 AssertComRC (rc);
5713 }
5714 }
5715
5716 if (getClassID() == clsidMachine)
5717 {
5718 // clean up the snapshots list (Snapshot::uninit() will handle the snapshot's children recursively)
5719 if (mData->mFirstSnapshot)
5720 {
5721 // snapshots tree is protected by media write lock; strictly
5722 // this isn't necessary here since we're deleting the entire
5723 // machine, but otherwise we assert in Snapshot::uninit()
5724 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5725 mData->mFirstSnapshot->uninit();
5726 mData->mFirstSnapshot.setNull();
5727 }
5728
5729 mData->mCurrentSnapshot.setNull();
5730 }
5731
5732 /* free data structures (the essential mData structure is not freed here
5733 * since it may be still in use) */
5734 mMediaData.free();
5735 mStorageControllers.free();
5736 mHWData.free();
5737 mUserData.free();
5738 mSSData.free();
5739}
5740
5741/**
5742 * Makes sure that there are no machine state dependants. If necessary, waits
5743 * for the number of dependants to drop to zero.
5744 *
5745 * Make sure this method is called from under this object's write lock to
5746 * guarantee that no new dependants may be added when this method returns
5747 * control to the caller.
5748 *
5749 * @note Locks this object for writing. The lock will be released while waiting
5750 * (if necessary).
5751 *
5752 * @warning To be used only in methods that change the machine state!
5753 */
5754void Machine::ensureNoStateDependencies()
5755{
5756 AssertReturnVoid (isWriteLockOnCurrentThread());
5757
5758 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5759
5760 /* Wait for all state dependants if necessary */
5761 if (mData->mMachineStateDeps != 0)
5762 {
5763 /* lazy semaphore creation */
5764 if (mData->mMachineStateDepsSem == NIL_RTSEMEVENTMULTI)
5765 RTSemEventMultiCreate (&mData->mMachineStateDepsSem);
5766
5767 LogFlowThisFunc(("Waiting for state deps (%d) to drop to zero...\n",
5768 mData->mMachineStateDeps));
5769
5770 ++mData->mMachineStateChangePending;
5771
5772 /* reset the semaphore before waiting, the last dependant will signal
5773 * it */
5774 RTSemEventMultiReset (mData->mMachineStateDepsSem);
5775
5776 alock.leave();
5777
5778 RTSemEventMultiWait (mData->mMachineStateDepsSem, RT_INDEFINITE_WAIT);
5779
5780 alock.enter();
5781
5782 -- mData->mMachineStateChangePending;
5783 }
5784}
5785
5786/**
5787 * Changes the machine state and informs callbacks.
5788 *
5789 * This method is not intended to fail so it either returns S_OK or asserts (and
5790 * returns a failure).
5791 *
5792 * @note Locks this object for writing.
5793 */
5794HRESULT Machine::setMachineState (MachineState_T aMachineState)
5795{
5796 LogFlowThisFuncEnter();
5797 LogFlowThisFunc(("aMachineState=%s\n", Global::stringifyMachineState(aMachineState) ));
5798
5799 AutoCaller autoCaller(this);
5800 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
5801
5802 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5803
5804 /* wait for state dependants to drop to zero */
5805 ensureNoStateDependencies();
5806
5807 if (mData->mMachineState != aMachineState)
5808 {
5809 mData->mMachineState = aMachineState;
5810
5811 RTTimeNow (&mData->mLastStateChange);
5812
5813 mParent->onMachineStateChange(mData->mUuid, aMachineState);
5814 }
5815
5816 LogFlowThisFuncLeave();
5817 return S_OK;
5818}
5819
5820/**
5821 * Searches for a shared folder with the given logical name
5822 * in the collection of shared folders.
5823 *
5824 * @param aName logical name of the shared folder
5825 * @param aSharedFolder where to return the found object
5826 * @param aSetError whether to set the error info if the folder is
5827 * not found
5828 * @return
5829 * S_OK when found or VBOX_E_OBJECT_NOT_FOUND when not found
5830 *
5831 * @note
5832 * must be called from under the object's lock!
5833 */
5834HRESULT Machine::findSharedFolder (CBSTR aName,
5835 ComObjPtr<SharedFolder> &aSharedFolder,
5836 bool aSetError /* = false */)
5837{
5838 bool found = false;
5839 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
5840 !found && it != mHWData->mSharedFolders.end();
5841 ++it)
5842 {
5843 AutoWriteLock alock(*it COMMA_LOCKVAL_SRC_POS);
5844 found = (*it)->getName() == aName;
5845 if (found)
5846 aSharedFolder = *it;
5847 }
5848
5849 HRESULT rc = found ? S_OK : VBOX_E_OBJECT_NOT_FOUND;
5850
5851 if (aSetError && !found)
5852 setError(rc, tr("Could not find a shared folder named '%ls'"), aName);
5853
5854 return rc;
5855}
5856
5857/**
5858 * Loads all the VM settings by walking down the <Machine> node.
5859 *
5860 * @param aRegistered true when the machine is being loaded on VirtualBox
5861 * startup
5862 *
5863 * @note This method is intended to be called only from init(), so it assumes
5864 * all machine data fields have appropriate default values when it is called.
5865 *
5866 * @note Doesn't lock any objects.
5867 */
5868HRESULT Machine::loadSettings(bool aRegistered)
5869{
5870 LogFlowThisFuncEnter();
5871 AssertReturn(getClassID() == clsidMachine, E_FAIL);
5872
5873 AutoCaller autoCaller(this);
5874 AssertReturn(autoCaller.state() == InInit, E_FAIL);
5875
5876 HRESULT rc = S_OK;
5877
5878 try
5879 {
5880 Assert(mData->m_pMachineConfigFile == NULL);
5881
5882 // load and parse machine XML; this will throw on XML or logic errors
5883 mData->m_pMachineConfigFile = new settings::MachineConfigFile(&mData->m_strConfigFileFull);
5884
5885 /* If the stored UUID is not empty, it means the registered machine
5886 * is being loaded. Compare the loaded UUID with the stored one taken
5887 * from the global registry. */
5888 if (!mData->mUuid.isEmpty())
5889 {
5890 if (mData->mUuid != mData->m_pMachineConfigFile->uuid)
5891 {
5892 throw setError(E_FAIL,
5893 tr("Machine UUID {%RTuuid} in '%s' doesn't match its UUID {%s} in the registry file '%s'"),
5894 mData->m_pMachineConfigFile->uuid.raw(),
5895 mData->m_strConfigFileFull.raw(),
5896 mData->mUuid.toString().raw(),
5897 mParent->settingsFilePath().raw());
5898 }
5899 }
5900 else
5901 unconst (mData->mUuid) = mData->m_pMachineConfigFile->uuid;
5902
5903 /* name (required) */
5904 mUserData->mName = mData->m_pMachineConfigFile->strName;
5905
5906 /* nameSync (optional, default is true) */
5907 mUserData->mNameSync = mData->m_pMachineConfigFile->fNameSync;
5908
5909 mUserData->mDescription = mData->m_pMachineConfigFile->strDescription;
5910
5911 // guest OS type
5912 mUserData->mOSTypeId = mData->m_pMachineConfigFile->strOsType;
5913 /* look up the object by Id to check it is valid */
5914 ComPtr<IGuestOSType> guestOSType;
5915 rc = mParent->GetGuestOSType(mUserData->mOSTypeId,
5916 guestOSType.asOutParam());
5917 if (FAILED(rc)) throw rc;
5918
5919 // stateFile (optional)
5920 if (mData->m_pMachineConfigFile->strStateFile.isEmpty())
5921 mSSData->mStateFilePath.setNull();
5922 else
5923 {
5924 Utf8Str stateFilePathFull(mData->m_pMachineConfigFile->strStateFile);
5925 int vrc = calculateFullPath(stateFilePathFull, stateFilePathFull);
5926 if (RT_FAILURE(vrc))
5927 throw setError(E_FAIL,
5928 tr("Invalid saved state file path '%s' (%Rrc)"),
5929 mData->m_pMachineConfigFile->strStateFile.raw(),
5930 vrc);
5931 mSSData->mStateFilePath = stateFilePathFull;
5932 }
5933
5934 /* snapshotFolder (optional) */
5935 rc = COMSETTER(SnapshotFolder)(Bstr(mData->m_pMachineConfigFile->strSnapshotFolder));
5936 if (FAILED(rc)) throw rc;
5937
5938 /* currentStateModified (optional, default is true) */
5939 mData->mCurrentStateModified = mData->m_pMachineConfigFile->fCurrentStateModified;
5940
5941 mData->mLastStateChange = mData->m_pMachineConfigFile->timeLastStateChange;
5942
5943 /* teleportation */
5944 mUserData->mTeleporterEnabled = mData->m_pMachineConfigFile->fTeleporterEnabled;
5945 mUserData->mTeleporterPort = mData->m_pMachineConfigFile->uTeleporterPort;
5946 mUserData->mTeleporterAddress = mData->m_pMachineConfigFile->strTeleporterAddress;
5947 mUserData->mTeleporterPassword = mData->m_pMachineConfigFile->strTeleporterPassword;
5948
5949 /* RTC */
5950 mUserData->mRTCUseUTC = mData->m_pMachineConfigFile->fRTCUseUTC;
5951
5952 /*
5953 * note: all mUserData members must be assigned prior this point because
5954 * we need to commit changes in order to let mUserData be shared by all
5955 * snapshot machine instances.
5956 */
5957 mUserData.commitCopy();
5958
5959 /* Snapshot node (optional) */
5960 size_t cRootSnapshots;
5961 if ((cRootSnapshots = mData->m_pMachineConfigFile->llFirstSnapshot.size()))
5962 {
5963 // there must be only one root snapshot
5964 Assert(cRootSnapshots == 1);
5965
5966 settings::Snapshot &snap = mData->m_pMachineConfigFile->llFirstSnapshot.front();
5967
5968 rc = loadSnapshot(snap,
5969 mData->m_pMachineConfigFile->uuidCurrentSnapshot,
5970 NULL); // no parent == first snapshot
5971 if (FAILED(rc)) throw rc;
5972 }
5973
5974 /* Hardware node (required) */
5975 rc = loadHardware(mData->m_pMachineConfigFile->hardwareMachine);
5976 if (FAILED(rc)) throw rc;
5977
5978 /* Load storage controllers */
5979 rc = loadStorageControllers(mData->m_pMachineConfigFile->storageMachine, aRegistered);
5980 if (FAILED(rc)) throw rc;
5981
5982 /*
5983 * NOTE: the assignment below must be the last thing to do,
5984 * otherwise it will be not possible to change the settings
5985 * somewehere in the code above because all setters will be
5986 * blocked by checkStateDependency(MutableStateDep).
5987 */
5988
5989 /* set the machine state to Aborted or Saved when appropriate */
5990 if (mData->m_pMachineConfigFile->fAborted)
5991 {
5992 Assert(!mSSData->mStateFilePath.isEmpty());
5993 mSSData->mStateFilePath.setNull();
5994
5995 /* no need to use setMachineState() during init() */
5996 mData->mMachineState = MachineState_Aborted;
5997 }
5998 else if (!mSSData->mStateFilePath.isEmpty())
5999 {
6000 /* no need to use setMachineState() during init() */
6001 mData->mMachineState = MachineState_Saved;
6002 }
6003
6004 // after loading settings, we are no longer different from the XML on disk
6005 m_flModifications = 0;
6006 }
6007 catch (HRESULT err)
6008 {
6009 /* we assume that error info is set by the thrower */
6010 rc = err;
6011 }
6012 catch (...)
6013 {
6014 rc = VirtualBox::handleUnexpectedExceptions(RT_SRC_POS);
6015 }
6016
6017 LogFlowThisFuncLeave();
6018 return rc;
6019}
6020
6021/**
6022 * Recursively loads all snapshots starting from the given.
6023 *
6024 * @param aNode <Snapshot> node.
6025 * @param aCurSnapshotId Current snapshot ID from the settings file.
6026 * @param aParentSnapshot Parent snapshot.
6027 */
6028HRESULT Machine::loadSnapshot(const settings::Snapshot &data,
6029 const Guid &aCurSnapshotId,
6030 Snapshot *aParentSnapshot)
6031{
6032 AssertReturn(getClassID() == clsidMachine, E_FAIL);
6033
6034 HRESULT rc = S_OK;
6035
6036 Utf8Str strStateFile;
6037 if (!data.strStateFile.isEmpty())
6038 {
6039 /* optional */
6040 strStateFile = data.strStateFile;
6041 int vrc = calculateFullPath(strStateFile, strStateFile);
6042 if (RT_FAILURE(vrc))
6043 return setError(E_FAIL,
6044 tr("Invalid saved state file path '%s' (%Rrc)"),
6045 strStateFile.raw(),
6046 vrc);
6047 }
6048
6049 /* create a snapshot machine object */
6050 ComObjPtr<SnapshotMachine> pSnapshotMachine;
6051 pSnapshotMachine.createObject();
6052 rc = pSnapshotMachine->init(this,
6053 data.hardware,
6054 data.storage,
6055 data.uuid,
6056 strStateFile);
6057 if (FAILED(rc)) return rc;
6058
6059 /* create a snapshot object */
6060 ComObjPtr<Snapshot> pSnapshot;
6061 pSnapshot.createObject();
6062 /* initialize the snapshot */
6063 rc = pSnapshot->init(mParent, // VirtualBox object
6064 data.uuid,
6065 data.strName,
6066 data.strDescription,
6067 data.timestamp,
6068 pSnapshotMachine,
6069 aParentSnapshot);
6070 if (FAILED(rc)) return rc;
6071
6072 /* memorize the first snapshot if necessary */
6073 if (!mData->mFirstSnapshot)
6074 mData->mFirstSnapshot = pSnapshot;
6075
6076 /* memorize the current snapshot when appropriate */
6077 if ( !mData->mCurrentSnapshot
6078 && pSnapshot->getId() == aCurSnapshotId
6079 )
6080 mData->mCurrentSnapshot = pSnapshot;
6081
6082 // now create the children
6083 for (settings::SnapshotsList::const_iterator it = data.llChildSnapshots.begin();
6084 it != data.llChildSnapshots.end();
6085 ++it)
6086 {
6087 const settings::Snapshot &childData = *it;
6088 // recurse
6089 rc = loadSnapshot(childData,
6090 aCurSnapshotId,
6091 pSnapshot); // parent = the one we created above
6092 if (FAILED(rc)) return rc;
6093 }
6094
6095 return rc;
6096}
6097
6098/**
6099 * @param aNode <Hardware> node.
6100 */
6101HRESULT Machine::loadHardware(const settings::Hardware &data)
6102{
6103 AssertReturn(getClassID() == clsidMachine || getClassID() == clsidSnapshotMachine, E_FAIL);
6104
6105 HRESULT rc = S_OK;
6106
6107 try
6108 {
6109 /* The hardware version attribute (optional). */
6110 mHWData->mHWVersion = data.strVersion;
6111 mHWData->mHardwareUUID = data.uuid;
6112
6113 mHWData->mHWVirtExEnabled = data.fHardwareVirt;
6114 mHWData->mHWVirtExExclusive = data.fHardwareVirtExclusive;
6115 mHWData->mHWVirtExNestedPagingEnabled = data.fNestedPaging;
6116 mHWData->mHWVirtExVPIDEnabled = data.fVPID;
6117 mHWData->mPAEEnabled = data.fPAE;
6118 mHWData->mSyntheticCpu = data.fSyntheticCpu;
6119
6120 mHWData->mCPUCount = data.cCPUs;
6121 mHWData->mCPUHotPlugEnabled = data.fCpuHotPlug;
6122
6123 // cpu
6124 if (mHWData->mCPUHotPlugEnabled)
6125 {
6126 for (settings::CpuList::const_iterator it = data.llCpus.begin();
6127 it != data.llCpus.end();
6128 ++it)
6129 {
6130 const settings::Cpu &cpu = *it;
6131
6132 mHWData->mCPUAttached[cpu.ulId] = true;
6133 }
6134 }
6135
6136 // cpuid leafs
6137 for (settings::CpuIdLeafsList::const_iterator it = data.llCpuIdLeafs.begin();
6138 it != data.llCpuIdLeafs.end();
6139 ++it)
6140 {
6141 const settings::CpuIdLeaf &leaf = *it;
6142
6143 switch (leaf.ulId)
6144 {
6145 case 0x0:
6146 case 0x1:
6147 case 0x2:
6148 case 0x3:
6149 case 0x4:
6150 case 0x5:
6151 case 0x6:
6152 case 0x7:
6153 case 0x8:
6154 case 0x9:
6155 case 0xA:
6156 mHWData->mCpuIdStdLeafs[leaf.ulId] = leaf;
6157 break;
6158
6159 case 0x80000000:
6160 case 0x80000001:
6161 case 0x80000002:
6162 case 0x80000003:
6163 case 0x80000004:
6164 case 0x80000005:
6165 case 0x80000006:
6166 case 0x80000007:
6167 case 0x80000008:
6168 case 0x80000009:
6169 case 0x8000000A:
6170 mHWData->mCpuIdExtLeafs[leaf.ulId - 0x80000000] = leaf;
6171 break;
6172
6173 default:
6174 /* just ignore */
6175 break;
6176 }
6177 }
6178
6179 mHWData->mMemorySize = data.ulMemorySizeMB;
6180
6181 // boot order
6182 for (size_t i = 0;
6183 i < RT_ELEMENTS(mHWData->mBootOrder);
6184 i++)
6185 {
6186 settings::BootOrderMap::const_iterator it = data.mapBootOrder.find(i);
6187 if (it == data.mapBootOrder.end())
6188 mHWData->mBootOrder[i] = DeviceType_Null;
6189 else
6190 mHWData->mBootOrder[i] = it->second;
6191 }
6192
6193 mHWData->mVRAMSize = data.ulVRAMSizeMB;
6194 mHWData->mMonitorCount = data.cMonitors;
6195 mHWData->mAccelerate3DEnabled = data.fAccelerate3D;
6196 mHWData->mAccelerate2DVideoEnabled = data.fAccelerate2DVideo;
6197 mHWData->mFirmwareType = data.firmwareType;
6198
6199#ifdef VBOX_WITH_VRDP
6200 /* RemoteDisplay */
6201 rc = mVRDPServer->loadSettings(data.vrdpSettings);
6202 if (FAILED(rc)) return rc;
6203#endif
6204
6205 /* BIOS */
6206 rc = mBIOSSettings->loadSettings(data.biosSettings);
6207 if (FAILED(rc)) return rc;
6208
6209 /* USB Controller */
6210 rc = mUSBController->loadSettings(data.usbController);
6211 if (FAILED(rc)) return rc;
6212
6213 // network adapters
6214 for (settings::NetworkAdaptersList::const_iterator it = data.llNetworkAdapters.begin();
6215 it != data.llNetworkAdapters.end();
6216 ++it)
6217 {
6218 const settings::NetworkAdapter &nic = *it;
6219
6220 /* slot unicity is guaranteed by XML Schema */
6221 AssertBreak(nic.ulSlot < RT_ELEMENTS(mNetworkAdapters));
6222 rc = mNetworkAdapters[nic.ulSlot]->loadSettings(nic);
6223 if (FAILED(rc)) return rc;
6224 }
6225
6226 // serial ports
6227 for (settings::SerialPortsList::const_iterator it = data.llSerialPorts.begin();
6228 it != data.llSerialPorts.end();
6229 ++it)
6230 {
6231 const settings::SerialPort &s = *it;
6232
6233 AssertBreak(s.ulSlot < RT_ELEMENTS(mSerialPorts));
6234 rc = mSerialPorts[s.ulSlot]->loadSettings(s);
6235 if (FAILED(rc)) return rc;
6236 }
6237
6238 // parallel ports (optional)
6239 for (settings::ParallelPortsList::const_iterator it = data.llParallelPorts.begin();
6240 it != data.llParallelPorts.end();
6241 ++it)
6242 {
6243 const settings::ParallelPort &p = *it;
6244
6245 AssertBreak(p.ulSlot < RT_ELEMENTS(mParallelPorts));
6246 rc = mParallelPorts[p.ulSlot]->loadSettings(p);
6247 if (FAILED(rc)) return rc;
6248 }
6249
6250 /* AudioAdapter */
6251 rc = mAudioAdapter->loadSettings(data.audioAdapter);
6252 if (FAILED(rc)) return rc;
6253
6254 for (settings::SharedFoldersList::const_iterator it = data.llSharedFolders.begin();
6255 it != data.llSharedFolders.end();
6256 ++it)
6257 {
6258 const settings::SharedFolder &sf = *it;
6259 rc = CreateSharedFolder(Bstr(sf.strName), Bstr(sf.strHostPath), sf.fWritable);
6260 if (FAILED(rc)) return rc;
6261 }
6262
6263 // Clipboard
6264 mHWData->mClipboardMode = data.clipboardMode;
6265
6266 // guest settings
6267 mHWData->mMemoryBalloonSize = data.ulMemoryBalloonSize;
6268 mHWData->mStatisticsUpdateInterval = data.ulStatisticsUpdateInterval;
6269
6270#ifdef VBOX_WITH_GUEST_PROPS
6271 /* Guest properties (optional) */
6272 for (settings::GuestPropertiesList::const_iterator it = data.llGuestProperties.begin();
6273 it != data.llGuestProperties.end();
6274 ++it)
6275 {
6276 const settings::GuestProperty &prop = *it;
6277 uint32_t fFlags = guestProp::NILFLAG;
6278 guestProp::validateFlags(prop.strFlags.c_str(), &fFlags);
6279 HWData::GuestProperty property = { prop.strName, prop.strValue, prop.timestamp, fFlags };
6280 mHWData->mGuestProperties.push_back(property);
6281 }
6282
6283 mHWData->mPropertyServiceActive = false;
6284 mHWData->mGuestPropertyNotificationPatterns = data.strNotificationPatterns;
6285#endif /* VBOX_WITH_GUEST_PROPS defined */
6286 }
6287 catch(std::bad_alloc &)
6288 {
6289 return E_OUTOFMEMORY;
6290 }
6291
6292 AssertComRC(rc);
6293 return rc;
6294}
6295
6296 /**
6297 * @param aNode <StorageControllers> node.
6298 */
6299HRESULT Machine::loadStorageControllers(const settings::Storage &data,
6300 bool aRegistered,
6301 const Guid *aSnapshotId /* = NULL */)
6302{
6303 AssertReturn(getClassID() == clsidMachine || getClassID() == clsidSnapshotMachine, E_FAIL);
6304
6305 HRESULT rc = S_OK;
6306
6307 for (settings::StorageControllersList::const_iterator it = data.llStorageControllers.begin();
6308 it != data.llStorageControllers.end();
6309 ++it)
6310 {
6311 const settings::StorageController &ctlData = *it;
6312
6313 ComObjPtr<StorageController> pCtl;
6314 /* Try to find one with the name first. */
6315 rc = getStorageControllerByName(ctlData.strName, pCtl, false /* aSetError */);
6316 if (SUCCEEDED(rc))
6317 return setError(VBOX_E_OBJECT_IN_USE,
6318 tr("Storage controller named '%s' already exists"),
6319 ctlData.strName.raw());
6320
6321 pCtl.createObject();
6322 rc = pCtl->init(this,
6323 ctlData.strName,
6324 ctlData.storageBus,
6325 ctlData.ulInstance);
6326 if (FAILED(rc)) return rc;
6327
6328 mStorageControllers->push_back(pCtl);
6329
6330 rc = pCtl->COMSETTER(ControllerType)(ctlData.controllerType);
6331 if (FAILED(rc)) return rc;
6332
6333 rc = pCtl->COMSETTER(PortCount)(ctlData.ulPortCount);
6334 if (FAILED(rc)) return rc;
6335
6336 /* Set IDE emulation settings (only for AHCI controller). */
6337 if (ctlData.controllerType == StorageControllerType_IntelAhci)
6338 {
6339 if ( (FAILED(rc = pCtl->SetIDEEmulationPort(0, ctlData.lIDE0MasterEmulationPort)))
6340 || (FAILED(rc = pCtl->SetIDEEmulationPort(1, ctlData.lIDE0SlaveEmulationPort)))
6341 || (FAILED(rc = pCtl->SetIDEEmulationPort(2, ctlData.lIDE1MasterEmulationPort)))
6342 || (FAILED(rc = pCtl->SetIDEEmulationPort(3, ctlData.lIDE1SlaveEmulationPort)))
6343 )
6344 return rc;
6345 }
6346
6347 /* Load the attached devices now. */
6348 rc = loadStorageDevices(pCtl,
6349 ctlData,
6350 aRegistered,
6351 aSnapshotId);
6352 if (FAILED(rc)) return rc;
6353 }
6354
6355 return S_OK;
6356}
6357
6358/**
6359 * @param aNode <HardDiskAttachments> node.
6360 * @param aRegistered true when the machine is being loaded on VirtualBox
6361 * startup, or when a snapshot is being loaded (wchich
6362 * currently can happen on startup only)
6363 * @param aSnapshotId pointer to the snapshot ID if this is a snapshot machine
6364 *
6365 * @note Lock mParent for reading and hard disks for writing before calling.
6366 */
6367HRESULT Machine::loadStorageDevices(StorageController *aStorageController,
6368 const settings::StorageController &data,
6369 bool aRegistered,
6370 const Guid *aSnapshotId /*= NULL*/)
6371{
6372 AssertReturn( (getClassID() == clsidMachine && aSnapshotId == NULL)
6373 || (getClassID() == clsidSnapshotMachine && aSnapshotId != NULL),
6374 E_FAIL);
6375
6376 HRESULT rc = S_OK;
6377
6378 if (!aRegistered && data.llAttachedDevices.size() > 0)
6379 /* when the machine is being loaded (opened) from a file, it cannot
6380 * have hard disks attached (this should not happen normally,
6381 * because we don't allow to attach hard disks to an unregistered
6382 * VM at all */
6383 return setError(E_FAIL,
6384 tr("Unregistered machine '%ls' cannot have storage devices attached (found %d attachments)"),
6385 mUserData->mName.raw(),
6386 data.llAttachedDevices.size());
6387
6388 /* paranoia: detect duplicate attachments */
6389 for (settings::AttachedDevicesList::const_iterator it = data.llAttachedDevices.begin();
6390 it != data.llAttachedDevices.end();
6391 ++it)
6392 {
6393 for (settings::AttachedDevicesList::const_iterator it2 = it;
6394 it2 != data.llAttachedDevices.end();
6395 ++it2)
6396 {
6397 if (it == it2)
6398 continue;
6399
6400 if ( (*it).lPort == (*it2).lPort
6401 && (*it).lDevice == (*it2).lDevice)
6402 {
6403 return setError(E_FAIL,
6404 tr("Duplicate attachments for storage controller '%s', port %d, device %d of the virtual machine '%ls'"),
6405 aStorageController->getName().raw(), (*it).lPort, (*it).lDevice, mUserData->mName.raw());
6406 }
6407 }
6408 }
6409
6410 for (settings::AttachedDevicesList::const_iterator it = data.llAttachedDevices.begin();
6411 it != data.llAttachedDevices.end();
6412 ++it)
6413 {
6414 const settings::AttachedDevice &dev = *it;
6415 ComObjPtr<Medium> medium;
6416
6417 switch (dev.deviceType)
6418 {
6419 case DeviceType_Floppy:
6420 /* find a floppy by UUID */
6421 if (!dev.uuid.isEmpty())
6422 rc = mParent->findFloppyImage(&dev.uuid, NULL, true /* aDoSetError */, &medium);
6423 /* find a floppy by host device name */
6424 else if (!dev.strHostDriveSrc.isEmpty())
6425 {
6426 SafeIfaceArray<IMedium> drivevec;
6427 rc = mParent->host()->COMGETTER(FloppyDrives)(ComSafeArrayAsOutParam(drivevec));
6428 if (SUCCEEDED(rc))
6429 {
6430 for (size_t i = 0; i < drivevec.size(); ++i)
6431 {
6432 /// @todo eliminate this conversion
6433 ComObjPtr<Medium> med = (Medium *)drivevec[i];
6434 if ( dev.strHostDriveSrc == med->getName()
6435 || dev.strHostDriveSrc == med->getLocation())
6436 {
6437 medium = med;
6438 break;
6439 }
6440 }
6441 }
6442 }
6443 break;
6444
6445 case DeviceType_DVD:
6446 /* find a DVD by UUID */
6447 if (!dev.uuid.isEmpty())
6448 rc = mParent->findDVDImage(&dev.uuid, NULL, true /* aDoSetError */, &medium);
6449 /* find a DVD by host device name */
6450 else if (!dev.strHostDriveSrc.isEmpty())
6451 {
6452 SafeIfaceArray<IMedium> drivevec;
6453 rc = mParent->host()->COMGETTER(DVDDrives)(ComSafeArrayAsOutParam(drivevec));
6454 if (SUCCEEDED(rc))
6455 {
6456 for (size_t i = 0; i < drivevec.size(); ++i)
6457 {
6458 Bstr hostDriveSrc(dev.strHostDriveSrc);
6459 /// @todo eliminate this conversion
6460 ComObjPtr<Medium> med = (Medium *)drivevec[i];
6461 if ( hostDriveSrc == med->getName()
6462 || hostDriveSrc == med->getLocation())
6463 {
6464 medium = med;
6465 break;
6466 }
6467 }
6468 }
6469 }
6470 break;
6471
6472 case DeviceType_HardDisk:
6473 {
6474 /* find a hard disk by UUID */
6475 rc = mParent->findHardDisk(&dev.uuid, NULL, true /* aDoSetError */, &medium);
6476 if (FAILED(rc))
6477 {
6478 VBoxClsID clsid = getClassID();
6479 if (clsid == clsidSnapshotMachine)
6480 {
6481 // wrap another error message around the "cannot find hard disk" set by findHardDisk
6482 // so the user knows that the bad disk is in a snapshot somewhere
6483 com::ErrorInfo info;
6484 return setError(E_FAIL,
6485 tr("A differencing image of snapshot {%RTuuid} could not be found. %ls"),
6486 aSnapshotId->raw(),
6487 info.getText().raw());
6488 }
6489 else
6490 return rc;
6491 }
6492
6493 AutoWriteLock hdLock(medium COMMA_LOCKVAL_SRC_POS);
6494
6495 if (medium->getType() == MediumType_Immutable)
6496 {
6497 if (getClassID() == clsidSnapshotMachine)
6498 return setError(E_FAIL,
6499 tr("Immutable hard disk '%s' with UUID {%RTuuid} cannot be directly attached to snapshot with UUID {%RTuuid} "
6500 "of the virtual machine '%ls' ('%s')"),
6501 medium->getLocationFull().raw(),
6502 dev.uuid.raw(),
6503 aSnapshotId->raw(),
6504 mUserData->mName.raw(),
6505 mData->m_strConfigFileFull.raw());
6506
6507 return setError(E_FAIL,
6508 tr("Immutable hard disk '%s' with UUID {%RTuuid} cannot be directly attached to the virtual machine '%ls' ('%s')"),
6509 medium->getLocationFull().raw(),
6510 dev.uuid.raw(),
6511 mUserData->mName.raw(),
6512 mData->m_strConfigFileFull.raw());
6513 }
6514
6515 if ( getClassID() != clsidSnapshotMachine
6516 && medium->getChildren().size() != 0
6517 )
6518 return setError(E_FAIL,
6519 tr("Hard disk '%s' with UUID {%RTuuid} cannot be directly attached to the virtual machine '%ls' ('%s') "
6520 "because it has %d differencing child hard disks"),
6521 medium->getLocationFull().raw(),
6522 dev.uuid.raw(),
6523 mUserData->mName.raw(),
6524 mData->m_strConfigFileFull.raw(),
6525 medium->getChildren().size());
6526
6527 if (findAttachment(mMediaData->mAttachments,
6528 medium))
6529 return setError(E_FAIL,
6530 tr("Hard disk '%s' with UUID {%RTuuid} is already attached to the virtual machine '%ls' ('%s')"),
6531 medium->getLocationFull().raw(),
6532 dev.uuid.raw(),
6533 mUserData->mName.raw(),
6534 mData->m_strConfigFileFull.raw());
6535
6536 break;
6537 }
6538
6539 default:
6540 return setError(E_FAIL,
6541 tr("Device with unknown type is attached to the virtual machine '%s' ('%s')"),
6542 medium->getLocationFull().raw(),
6543 mUserData->mName.raw(),
6544 mData->m_strConfigFileFull.raw());
6545 }
6546
6547 if (FAILED(rc))
6548 break;
6549
6550 const Bstr controllerName = aStorageController->getName();
6551 ComObjPtr<MediumAttachment> pAttachment;
6552 pAttachment.createObject();
6553 rc = pAttachment->init(this,
6554 medium,
6555 controllerName,
6556 dev.lPort,
6557 dev.lDevice,
6558 dev.deviceType,
6559 dev.fPassThrough);
6560 if (FAILED(rc)) break;
6561
6562 /* associate the medium with this machine and snapshot */
6563 if (!medium.isNull())
6564 {
6565 if (getClassID() == clsidSnapshotMachine)
6566 rc = medium->attachTo(mData->mUuid, *aSnapshotId);
6567 else
6568 rc = medium->attachTo(mData->mUuid);
6569 }
6570
6571 if (FAILED(rc))
6572 break;
6573
6574 /* back up mMediaData to let registeredInit() properly rollback on failure
6575 * (= limited accessibility) */
6576 setModified(IsModified_Storage);
6577 mMediaData.backup();
6578 mMediaData->mAttachments.push_back(pAttachment);
6579 }
6580
6581 return rc;
6582}
6583
6584/**
6585 * Returns the snapshot with the given UUID or fails of no such snapshot exists.
6586 *
6587 * @param aId snapshot UUID to find (empty UUID refers the first snapshot)
6588 * @param aSnapshot where to return the found snapshot
6589 * @param aSetError true to set extended error info on failure
6590 */
6591HRESULT Machine::findSnapshot(const Guid &aId,
6592 ComObjPtr<Snapshot> &aSnapshot,
6593 bool aSetError /* = false */)
6594{
6595 AutoReadLock chlock(this COMMA_LOCKVAL_SRC_POS);
6596
6597 if (!mData->mFirstSnapshot)
6598 {
6599 if (aSetError)
6600 return setError(E_FAIL,
6601 tr("This machine does not have any snapshots"));
6602 return E_FAIL;
6603 }
6604
6605 if (aId.isEmpty())
6606 aSnapshot = mData->mFirstSnapshot;
6607 else
6608 aSnapshot = mData->mFirstSnapshot->findChildOrSelf(aId);
6609
6610 if (!aSnapshot)
6611 {
6612 if (aSetError)
6613 return setError(E_FAIL,
6614 tr("Could not find a snapshot with UUID {%s}"),
6615 aId.toString().raw());
6616 return E_FAIL;
6617 }
6618
6619 return S_OK;
6620}
6621
6622/**
6623 * Returns the snapshot with the given name or fails of no such snapshot.
6624 *
6625 * @param aName snapshot name to find
6626 * @param aSnapshot where to return the found snapshot
6627 * @param aSetError true to set extended error info on failure
6628 */
6629HRESULT Machine::findSnapshot(IN_BSTR aName,
6630 ComObjPtr<Snapshot> &aSnapshot,
6631 bool aSetError /* = false */)
6632{
6633 AssertReturn(aName, E_INVALIDARG);
6634
6635 AutoReadLock chlock(this COMMA_LOCKVAL_SRC_POS);
6636
6637 if (!mData->mFirstSnapshot)
6638 {
6639 if (aSetError)
6640 return setError(VBOX_E_OBJECT_NOT_FOUND,
6641 tr("This machine does not have any snapshots"));
6642 return VBOX_E_OBJECT_NOT_FOUND;
6643 }
6644
6645 aSnapshot = mData->mFirstSnapshot->findChildOrSelf (aName);
6646
6647 if (!aSnapshot)
6648 {
6649 if (aSetError)
6650 return setError(VBOX_E_OBJECT_NOT_FOUND,
6651 tr("Could not find a snapshot named '%ls'"), aName);
6652 return VBOX_E_OBJECT_NOT_FOUND;
6653 }
6654
6655 return S_OK;
6656}
6657
6658/**
6659 * Returns a storage controller object with the given name.
6660 *
6661 * @param aName storage controller name to find
6662 * @param aStorageController where to return the found storage controller
6663 * @param aSetError true to set extended error info on failure
6664 */
6665HRESULT Machine::getStorageControllerByName(const Utf8Str &aName,
6666 ComObjPtr<StorageController> &aStorageController,
6667 bool aSetError /* = false */)
6668{
6669 AssertReturn (!aName.isEmpty(), E_INVALIDARG);
6670
6671 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
6672 it != mStorageControllers->end();
6673 ++it)
6674 {
6675 if ((*it)->getName() == aName)
6676 {
6677 aStorageController = (*it);
6678 return S_OK;
6679 }
6680 }
6681
6682 if (aSetError)
6683 return setError(VBOX_E_OBJECT_NOT_FOUND,
6684 tr("Could not find a storage controller named '%s'"),
6685 aName.raw());
6686 return VBOX_E_OBJECT_NOT_FOUND;
6687}
6688
6689HRESULT Machine::getMediumAttachmentsOfController(CBSTR aName,
6690 MediaData::AttachmentList &atts)
6691{
6692 AutoCaller autoCaller(this);
6693 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6694
6695 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6696
6697 for (MediaData::AttachmentList::iterator it = mMediaData->mAttachments.begin();
6698 it != mMediaData->mAttachments.end();
6699 ++it)
6700 {
6701 if ((*it)->getControllerName() == aName)
6702 atts.push_back(*it);
6703 }
6704
6705 return S_OK;
6706}
6707
6708/**
6709 * Helper for #saveSettings. Cares about renaming the settings directory and
6710 * file if the machine name was changed and about creating a new settings file
6711 * if this is a new machine.
6712 *
6713 * @note Must be never called directly but only from #saveSettings().
6714 *
6715 * @param aRenamed receives |true| if the name was changed and the settings
6716 * file was renamed as a result, or |false| otherwise. The
6717 * value makes sense only on success.
6718 * @param aNew receives |true| if a virgin settings file was created.
6719 */
6720HRESULT Machine::prepareSaveSettings(bool &aRenamed,
6721 bool &aNew)
6722{
6723 /* Note: tecnhically, mParent needs to be locked only when the machine is
6724 * registered (see prepareSaveSettings() for details) but we don't
6725 * currently differentiate it in callers of saveSettings() so we don't
6726 * make difference here too. */
6727 AssertReturn(mParent->isWriteLockOnCurrentThread(), E_FAIL);
6728 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6729
6730 HRESULT rc = S_OK;
6731
6732 aRenamed = false;
6733
6734 /* if we're ready and isConfigLocked() is FALSE then it means
6735 * that no config file exists yet (we will create a virgin one) */
6736 aNew = !mData->m_pMachineConfigFile->fileExists();
6737
6738 /* attempt to rename the settings file if machine name is changed */
6739 if ( mUserData->mNameSync
6740 && mUserData.isBackedUp()
6741 && mUserData.backedUpData()->mName != mUserData->mName
6742 )
6743 {
6744 aRenamed = true;
6745
6746 bool dirRenamed = false;
6747 bool fileRenamed = false;
6748
6749 Utf8Str configFile, newConfigFile;
6750 Utf8Str configDir, newConfigDir;
6751
6752 do
6753 {
6754 int vrc = VINF_SUCCESS;
6755
6756 Utf8Str name = mUserData.backedUpData()->mName;
6757 Utf8Str newName = mUserData->mName;
6758
6759 configFile = mData->m_strConfigFileFull;
6760
6761 /* first, rename the directory if it matches the machine name */
6762 configDir = configFile;
6763 configDir.stripFilename();
6764 newConfigDir = configDir;
6765 if (!strcmp(RTPathFilename(configDir.c_str()), name.c_str()))
6766 {
6767 newConfigDir.stripFilename();
6768 newConfigDir = Utf8StrFmt ("%s%c%s",
6769 newConfigDir.raw(), RTPATH_DELIMITER, newName.raw());
6770 /* new dir and old dir cannot be equal here because of 'if'
6771 * above and because name != newName */
6772 Assert (configDir != newConfigDir);
6773 if (!aNew)
6774 {
6775 /* perform real rename only if the machine is not new */
6776 vrc = RTPathRename (configDir.raw(), newConfigDir.raw(), 0);
6777 if (RT_FAILURE(vrc))
6778 {
6779 rc = setError(E_FAIL,
6780 tr("Could not rename the directory '%s' to '%s' to save the settings file (%Rrc)"),
6781 configDir.raw(),
6782 newConfigDir.raw(),
6783 vrc);
6784 break;
6785 }
6786 dirRenamed = true;
6787 }
6788 }
6789
6790 newConfigFile = Utf8StrFmt ("%s%c%s.xml",
6791 newConfigDir.raw(), RTPATH_DELIMITER, newName.raw());
6792
6793 /* then try to rename the settings file itself */
6794 if (newConfigFile != configFile)
6795 {
6796 /* get the path to old settings file in renamed directory */
6797 configFile = Utf8StrFmt("%s%c%s",
6798 newConfigDir.raw(),
6799 RTPATH_DELIMITER,
6800 RTPathFilename(configFile.c_str()));
6801 if (!aNew)
6802 {
6803 /* perform real rename only if the machine is not new */
6804 vrc = RTFileRename (configFile.raw(), newConfigFile.raw(), 0);
6805 if (RT_FAILURE(vrc))
6806 {
6807 rc = setError(E_FAIL,
6808 tr("Could not rename the settings file '%s' to '%s' (%Rrc)"),
6809 configFile.raw(),
6810 newConfigFile.raw(),
6811 vrc);
6812 break;
6813 }
6814 fileRenamed = true;
6815 }
6816 }
6817
6818 /* update m_strConfigFileFull amd mConfigFile */
6819 Utf8Str oldConfigFileFull = mData->m_strConfigFileFull;
6820 Utf8Str oldConfigFile = mData->m_strConfigFile;
6821 mData->m_strConfigFileFull = newConfigFile;
6822 /* try to get the relative path for mConfigFile */
6823 Utf8Str path = newConfigFile;
6824 mParent->calculateRelativePath (path, path);
6825 mData->m_strConfigFile = path;
6826
6827 /* last, try to update the global settings with the new path */
6828 if (mData->mRegistered)
6829 {
6830 rc = mParent->updateSettings(configDir.c_str(), newConfigDir.c_str());
6831 if (FAILED(rc))
6832 {
6833 /* revert to old values */
6834 mData->m_strConfigFileFull = oldConfigFileFull;
6835 mData->m_strConfigFile = oldConfigFile;
6836 break;
6837 }
6838 }
6839
6840 /* update the snapshot folder */
6841 path = mUserData->mSnapshotFolderFull;
6842 if (RTPathStartsWith(path.c_str(), configDir.c_str()))
6843 {
6844 path = Utf8StrFmt("%s%s", newConfigDir.raw(),
6845 path.raw() + configDir.length());
6846 mUserData->mSnapshotFolderFull = path;
6847 calculateRelativePath (path, path);
6848 mUserData->mSnapshotFolder = path;
6849 }
6850
6851 /* update the saved state file path */
6852 path = mSSData->mStateFilePath;
6853 if (RTPathStartsWith(path.c_str(), configDir.c_str()))
6854 {
6855 path = Utf8StrFmt("%s%s", newConfigDir.raw(),
6856 path.raw() + configDir.length());
6857 mSSData->mStateFilePath = path;
6858 }
6859
6860 /* Update saved state file paths of all online snapshots.
6861 * Note that saveSettings() will recognize name change
6862 * and will save all snapshots in this case. */
6863 if (mData->mFirstSnapshot)
6864 mData->mFirstSnapshot->updateSavedStatePaths(configDir.c_str(),
6865 newConfigDir.c_str());
6866 }
6867 while (0);
6868
6869 if (FAILED(rc))
6870 {
6871 /* silently try to rename everything back */
6872 if (fileRenamed)
6873 RTFileRename(newConfigFile.raw(), configFile.raw(), 0);
6874 if (dirRenamed)
6875 RTPathRename(newConfigDir.raw(), configDir.raw(), 0);
6876 }
6877
6878 if (FAILED(rc)) return rc;
6879 }
6880
6881 if (aNew)
6882 {
6883 /* create a virgin config file */
6884 int vrc = VINF_SUCCESS;
6885
6886 /* ensure the settings directory exists */
6887 Utf8Str path(mData->m_strConfigFileFull);
6888 path.stripFilename();
6889 if (!RTDirExists(path.c_str()))
6890 {
6891 vrc = RTDirCreateFullPath(path.c_str(), 0777);
6892 if (RT_FAILURE(vrc))
6893 {
6894 return setError(E_FAIL,
6895 tr("Could not create a directory '%s' to save the settings file (%Rrc)"),
6896 path.raw(),
6897 vrc);
6898 }
6899 }
6900
6901 /* Note: open flags must correlate with RTFileOpen() in lockConfig() */
6902 path = Utf8Str(mData->m_strConfigFileFull);
6903 vrc = RTFileOpen(&mData->mHandleCfgFile, path.c_str(),
6904 RTFILE_O_READWRITE | RTFILE_O_CREATE | RTFILE_O_DENY_WRITE);
6905 if (RT_FAILURE(vrc))
6906 {
6907 mData->mHandleCfgFile = NIL_RTFILE;
6908 return setError(E_FAIL,
6909 tr("Could not create the settings file '%s' (%Rrc)"),
6910 path.raw(),
6911 vrc);
6912 }
6913 RTFileClose(mData->mHandleCfgFile);
6914 }
6915
6916 return rc;
6917}
6918
6919/**
6920 * Saves and commits machine data, user data and hardware data.
6921 *
6922 * Note that on failure, the data remains uncommitted.
6923 *
6924 * @a aFlags may combine the following flags:
6925 *
6926 * - SaveS_ResetCurStateModified: Resets mData->mCurrentStateModified to FALSE.
6927 * Used when saving settings after an operation that makes them 100%
6928 * correspond to the settings from the current snapshot.
6929 * - SaveS_InformCallbacksAnyway: Callbacks will be informed even if
6930 * #isReallyModified() returns false. This is necessary for cases when we
6931 * change machine data directly, not through the backup()/commit() mechanism.
6932 *
6933 * @note Must be called from under mParent write lock (sometimes needed by
6934 * #prepareSaveSettings()) and this object's write lock. Locks children for
6935 * writing. There is one exception when mParent is unused and therefore may be
6936 * left unlocked: if this machine is an unregistered one.
6937 */
6938HRESULT Machine::saveSettings(int aFlags /*= 0*/)
6939{
6940 LogFlowThisFuncEnter();
6941
6942 /* Note: tecnhically, mParent needs to be locked only when the machine is
6943 * registered (see prepareSaveSettings() for details) but we don't
6944 * currently differentiate it in callers of saveSettings() so we don't
6945 * make difference here too. */
6946 AssertReturn(mParent->isWriteLockOnCurrentThread(), E_FAIL);
6947 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6948
6949 /* make sure child objects are unable to modify the settings while we are
6950 * saving them */
6951 ensureNoStateDependencies();
6952
6953 AssertReturn( getClassID() == clsidMachine
6954 || getClassID() == clsidSessionMachine,
6955 E_FAIL);
6956
6957 HRESULT rc = S_OK;
6958 bool fNeedsWrite = false;
6959
6960 /* First, prepare to save settings. It will care about renaming the
6961 * settings directory and file if the machine name was changed and about
6962 * creating a new settings file if this is a new machine. */
6963 bool fIsRenamed = false;
6964 bool fIsNew = false;
6965 rc = prepareSaveSettings(fIsRenamed, fIsNew);
6966 if (FAILED(rc)) return rc;
6967
6968 // keep a pointer to the current settings structures
6969 settings::MachineConfigFile *pOldConfig = mData->m_pMachineConfigFile;
6970
6971 try
6972 {
6973 // make a fresh one to have everyone write stuff into
6974 mData->m_pMachineConfigFile = new settings::MachineConfigFile(NULL);
6975 mData->m_pMachineConfigFile->copyBaseFrom(*pOldConfig);
6976
6977 // deep copy extradata
6978 mData->m_pMachineConfigFile->mapExtraDataItems = pOldConfig->mapExtraDataItems;
6979
6980 mData->m_pMachineConfigFile->uuid = mData->mUuid;
6981 mData->m_pMachineConfigFile->strName = mUserData->mName;
6982 mData->m_pMachineConfigFile->fNameSync = !!mUserData->mNameSync;
6983 mData->m_pMachineConfigFile->strDescription = mUserData->mDescription;
6984 mData->m_pMachineConfigFile->strOsType = mUserData->mOSTypeId;
6985
6986 if ( mData->mMachineState == MachineState_Saved
6987 || mData->mMachineState == MachineState_Restoring
6988 // when deleting a snapshot we may or may not have a saved state in the current state,
6989 // so let's not assert here please
6990 || ( (mData->mMachineState == MachineState_DeletingSnapshot)
6991 && (!mSSData->mStateFilePath.isEmpty())
6992 )
6993 )
6994 {
6995 Assert(!mSSData->mStateFilePath.isEmpty());
6996 /* try to make the file name relative to the settings file dir */
6997 calculateRelativePath(mSSData->mStateFilePath, mData->m_pMachineConfigFile->strStateFile);
6998 }
6999 else
7000 {
7001 Assert(mSSData->mStateFilePath.isEmpty());
7002 mData->m_pMachineConfigFile->strStateFile.setNull();
7003 }
7004
7005 if (mData->mCurrentSnapshot)
7006 mData->m_pMachineConfigFile->uuidCurrentSnapshot = mData->mCurrentSnapshot->getId();
7007 else
7008 mData->m_pMachineConfigFile->uuidCurrentSnapshot.clear();
7009
7010 mData->m_pMachineConfigFile->strSnapshotFolder = mUserData->mSnapshotFolder;
7011 // mData->m_pMachineConfigFile->fCurrentStateModified is special, see below
7012 mData->m_pMachineConfigFile->timeLastStateChange = mData->mLastStateChange;
7013 mData->m_pMachineConfigFile->fAborted = (mData->mMachineState == MachineState_Aborted);
7014/// @todo Live Migration: mData->m_pMachineConfigFile->fTeleported = (mData->mMachineState == MachineState_Teleported);
7015
7016 mData->m_pMachineConfigFile->fTeleporterEnabled = !!mUserData->mTeleporterEnabled;
7017 mData->m_pMachineConfigFile->uTeleporterPort = mUserData->mTeleporterPort;
7018 mData->m_pMachineConfigFile->strTeleporterAddress = mUserData->mTeleporterAddress;
7019 mData->m_pMachineConfigFile->strTeleporterPassword = mUserData->mTeleporterPassword;
7020
7021 mData->m_pMachineConfigFile->fRTCUseUTC = !!mUserData->mRTCUseUTC;
7022
7023 rc = saveHardware(mData->m_pMachineConfigFile->hardwareMachine);
7024 if (FAILED(rc)) throw rc;
7025
7026 rc = saveStorageControllers(mData->m_pMachineConfigFile->storageMachine);
7027 if (FAILED(rc)) throw rc;
7028
7029 // save snapshots
7030 rc = saveAllSnapshots();
7031 if (FAILED(rc)) throw rc;
7032
7033 if (aFlags & SaveS_ResetCurStateModified)
7034 {
7035 // this gets set by restoreSnapshot()
7036 mData->mCurrentStateModified = FALSE;
7037 fNeedsWrite = true; // always, no need to compare
7038 }
7039 else
7040 {
7041 if (!mData->mCurrentStateModified)
7042 {
7043 // do a deep compare of the settings that we just saved with the settings
7044 // previously stored in the config file; this invokes MachineConfigFile::operator==
7045 // which does a deep compare of all the settings, which is expensive but less expensive
7046 // than writing out XML in vain
7047 bool fAnySettingsChanged = (*mData->m_pMachineConfigFile == *pOldConfig);
7048
7049 // could still be modified if any settings changed
7050 mData->mCurrentStateModified = fAnySettingsChanged;
7051
7052 fNeedsWrite = fAnySettingsChanged;
7053 }
7054 else
7055 fNeedsWrite = true;
7056 }
7057
7058 mData->m_pMachineConfigFile->fCurrentStateModified = !!mData->mCurrentStateModified;
7059
7060 if (fNeedsWrite)
7061 // now spit it all out!
7062 mData->m_pMachineConfigFile->write(mData->m_strConfigFileFull);
7063
7064 // after saving settings, we are no longer different from the XML on disk
7065 m_flModifications = 0;
7066 }
7067 catch (HRESULT err)
7068 {
7069 // we assume that error info is set by the thrower
7070 rc = err;
7071
7072 // restore old config
7073 delete mData->m_pMachineConfigFile;
7074 mData->m_pMachineConfigFile = pOldConfig;
7075 }
7076 catch (...)
7077 {
7078 rc = VirtualBox::handleUnexpectedExceptions (RT_SRC_POS);
7079 }
7080
7081 if (SUCCEEDED(rc))
7082 {
7083 commit();
7084 delete pOldConfig;
7085 }
7086
7087 if (fNeedsWrite || (aFlags & SaveS_InformCallbacksAnyway))
7088 {
7089 /* Fire the data change event, even on failure (since we've already
7090 * committed all data). This is done only for SessionMachines because
7091 * mutable Machine instances are always not registered (i.e. private
7092 * to the client process that creates them) and thus don't need to
7093 * inform callbacks. */
7094 if (getClassID() == clsidSessionMachine)
7095 mParent->onMachineDataChange(mData->mUuid);
7096 }
7097
7098 LogFlowThisFunc(("rc=%08X\n", rc));
7099 LogFlowThisFuncLeave();
7100 return rc;
7101}
7102
7103HRESULT Machine::saveAllSnapshots()
7104{
7105 AssertReturn (isWriteLockOnCurrentThread(), E_FAIL);
7106
7107 HRESULT rc = S_OK;
7108
7109 try
7110 {
7111 mData->m_pMachineConfigFile->llFirstSnapshot.clear();
7112
7113 if (mData->mFirstSnapshot)
7114 {
7115 settings::Snapshot snapNew;
7116 mData->m_pMachineConfigFile->llFirstSnapshot.push_back(snapNew);
7117
7118 // get reference to the fresh copy of the snapshot on the list and
7119 // work on that copy directly to avoid excessive copying later
7120 settings::Snapshot &snap = mData->m_pMachineConfigFile->llFirstSnapshot.front();
7121
7122 rc = mData->mFirstSnapshot->saveSnapshot(snap, false /*aAttrsOnly*/);
7123 if (FAILED(rc)) throw rc;
7124 }
7125
7126// if (mType == IsSessionMachine)
7127// mParent->onMachineDataChange(mData->mUuid); @todo is this necessary?
7128
7129 }
7130 catch (HRESULT err)
7131 {
7132 /* we assume that error info is set by the thrower */
7133 rc = err;
7134 }
7135 catch (...)
7136 {
7137 rc = VirtualBox::handleUnexpectedExceptions (RT_SRC_POS);
7138 }
7139
7140 return rc;
7141}
7142
7143/**
7144 * Saves the VM hardware configuration. It is assumed that the
7145 * given node is empty.
7146 *
7147 * @param aNode <Hardware> node to save the VM hardware confguration to.
7148 */
7149HRESULT Machine::saveHardware(settings::Hardware &data)
7150{
7151 HRESULT rc = S_OK;
7152
7153 try
7154 {
7155 /* The hardware version attribute (optional).
7156 Automatically upgrade from 1 to 2 when there is no saved state. (ugly!) */
7157 if ( mHWData->mHWVersion == "1"
7158 && mSSData->mStateFilePath.isEmpty()
7159 )
7160 mHWData->mHWVersion = "2"; /** @todo Is this safe, to update mHWVersion here? If not some other point needs to be found where this can be done. */
7161
7162 data.strVersion = mHWData->mHWVersion;
7163 data.uuid = mHWData->mHardwareUUID;
7164
7165 // CPU
7166 data.fHardwareVirt = !!mHWData->mHWVirtExEnabled;
7167 data.fHardwareVirtExclusive = !!mHWData->mHWVirtExExclusive;
7168 data.fNestedPaging = !!mHWData->mHWVirtExNestedPagingEnabled;
7169 data.fVPID = !!mHWData->mHWVirtExVPIDEnabled;
7170 data.fPAE = !!mHWData->mPAEEnabled;
7171 data.fSyntheticCpu = !!mHWData->mSyntheticCpu;
7172
7173 /* Standard and Extended CPUID leafs. */
7174 data.llCpuIdLeafs.clear();
7175 for (unsigned idx = 0; idx < RT_ELEMENTS(mHWData->mCpuIdStdLeafs); idx++)
7176 {
7177 if (mHWData->mCpuIdStdLeafs[idx].ulId != UINT32_MAX)
7178 data.llCpuIdLeafs.push_back(mHWData->mCpuIdStdLeafs[idx]);
7179 }
7180 for (unsigned idx = 0; idx < RT_ELEMENTS(mHWData->mCpuIdExtLeafs); idx++)
7181 {
7182 if (mHWData->mCpuIdExtLeafs[idx].ulId != UINT32_MAX)
7183 data.llCpuIdLeafs.push_back(mHWData->mCpuIdExtLeafs[idx]);
7184 }
7185
7186 data.cCPUs = mHWData->mCPUCount;
7187 data.fCpuHotPlug = mHWData->mCPUHotPlugEnabled;
7188
7189 data.llCpus.clear();
7190 if (data.fCpuHotPlug)
7191 {
7192 for (unsigned idx = 0; idx < data.cCPUs; idx++)
7193 {
7194 if (mHWData->mCPUAttached[idx])
7195 {
7196 settings::Cpu cpu;
7197 cpu.ulId = idx;
7198 data.llCpus.push_back(cpu);
7199 }
7200 }
7201 }
7202
7203 // memory
7204 data.ulMemorySizeMB = mHWData->mMemorySize;
7205
7206 // firmware
7207 data.firmwareType = mHWData->mFirmwareType;
7208
7209 // boot order
7210 data.mapBootOrder.clear();
7211 for (size_t i = 0;
7212 i < RT_ELEMENTS(mHWData->mBootOrder);
7213 ++i)
7214 data.mapBootOrder[i] = mHWData->mBootOrder[i];
7215
7216 // display
7217 data.ulVRAMSizeMB = mHWData->mVRAMSize;
7218 data.cMonitors = mHWData->mMonitorCount;
7219 data.fAccelerate3D = !!mHWData->mAccelerate3DEnabled;
7220 data.fAccelerate2DVideo = !!mHWData->mAccelerate2DVideoEnabled;
7221
7222#ifdef VBOX_WITH_VRDP
7223 /* VRDP settings (optional) */
7224 rc = mVRDPServer->saveSettings(data.vrdpSettings);
7225 if (FAILED(rc)) throw rc;
7226#endif
7227
7228 /* BIOS (required) */
7229 rc = mBIOSSettings->saveSettings(data.biosSettings);
7230 if (FAILED(rc)) throw rc;
7231
7232 /* USB Controller (required) */
7233 rc = mUSBController->saveSettings(data.usbController);
7234 if (FAILED(rc)) throw rc;
7235
7236 /* Network adapters (required) */
7237 data.llNetworkAdapters.clear();
7238 for (ULONG slot = 0;
7239 slot < RT_ELEMENTS(mNetworkAdapters);
7240 ++slot)
7241 {
7242 settings::NetworkAdapter nic;
7243 nic.ulSlot = slot;
7244 rc = mNetworkAdapters[slot]->saveSettings(nic);
7245 if (FAILED(rc)) throw rc;
7246
7247 data.llNetworkAdapters.push_back(nic);
7248 }
7249
7250 /* Serial ports */
7251 data.llSerialPorts.clear();
7252 for (ULONG slot = 0;
7253 slot < RT_ELEMENTS(mSerialPorts);
7254 ++slot)
7255 {
7256 settings::SerialPort s;
7257 s.ulSlot = slot;
7258 rc = mSerialPorts[slot]->saveSettings(s);
7259 if (FAILED(rc)) return rc;
7260
7261 data.llSerialPorts.push_back(s);
7262 }
7263
7264 /* Parallel ports */
7265 data.llParallelPorts.clear();
7266 for (ULONG slot = 0;
7267 slot < RT_ELEMENTS(mParallelPorts);
7268 ++slot)
7269 {
7270 settings::ParallelPort p;
7271 p.ulSlot = slot;
7272 rc = mParallelPorts[slot]->saveSettings(p);
7273 if (FAILED(rc)) return rc;
7274
7275 data.llParallelPorts.push_back(p);
7276 }
7277
7278 /* Audio adapter */
7279 rc = mAudioAdapter->saveSettings(data.audioAdapter);
7280 if (FAILED(rc)) return rc;
7281
7282 /* Shared folders */
7283 data.llSharedFolders.clear();
7284 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
7285 it != mHWData->mSharedFolders.end();
7286 ++it)
7287 {
7288 ComObjPtr<SharedFolder> pFolder = *it;
7289 settings::SharedFolder sf;
7290 sf.strName = pFolder->getName();
7291 sf.strHostPath = pFolder->getHostPath();
7292 sf.fWritable = !!pFolder->isWritable();
7293
7294 data.llSharedFolders.push_back(sf);
7295 }
7296
7297 // clipboard
7298 data.clipboardMode = mHWData->mClipboardMode;
7299
7300 /* Guest */
7301 data.ulMemoryBalloonSize = mHWData->mMemoryBalloonSize;
7302 data.ulStatisticsUpdateInterval = mHWData->mStatisticsUpdateInterval;
7303
7304 // guest properties
7305 data.llGuestProperties.clear();
7306#ifdef VBOX_WITH_GUEST_PROPS
7307 for (HWData::GuestPropertyList::const_iterator it = mHWData->mGuestProperties.begin();
7308 it != mHWData->mGuestProperties.end();
7309 ++it)
7310 {
7311 HWData::GuestProperty property = *it;
7312
7313 settings::GuestProperty prop;
7314 prop.strName = property.strName;
7315 prop.strValue = property.strValue;
7316 prop.timestamp = property.mTimestamp;
7317 char szFlags[guestProp::MAX_FLAGS_LEN + 1];
7318 guestProp::writeFlags(property.mFlags, szFlags);
7319 prop.strFlags = szFlags;
7320
7321 data.llGuestProperties.push_back(prop);
7322 }
7323
7324 data.strNotificationPatterns = mHWData->mGuestPropertyNotificationPatterns;
7325#endif /* VBOX_WITH_GUEST_PROPS defined */
7326 }
7327 catch(std::bad_alloc &)
7328 {
7329 return E_OUTOFMEMORY;
7330 }
7331
7332 AssertComRC(rc);
7333 return rc;
7334}
7335
7336/**
7337 * Saves the storage controller configuration.
7338 *
7339 * @param aNode <StorageControllers> node to save the VM hardware confguration to.
7340 */
7341HRESULT Machine::saveStorageControllers(settings::Storage &data)
7342{
7343 data.llStorageControllers.clear();
7344
7345 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
7346 it != mStorageControllers->end();
7347 ++it)
7348 {
7349 HRESULT rc;
7350 ComObjPtr<StorageController> pCtl = *it;
7351
7352 settings::StorageController ctl;
7353 ctl.strName = pCtl->getName();
7354 ctl.controllerType = pCtl->getControllerType();
7355 ctl.storageBus = pCtl->getStorageBus();
7356 ctl.ulInstance = pCtl->getInstance();
7357
7358 /* Save the port count. */
7359 ULONG portCount;
7360 rc = pCtl->COMGETTER(PortCount)(&portCount);
7361 ComAssertComRCRet(rc, rc);
7362 ctl.ulPortCount = portCount;
7363
7364 /* Save IDE emulation settings. */
7365 if (ctl.controllerType == StorageControllerType_IntelAhci)
7366 {
7367 if ( (FAILED(rc = pCtl->GetIDEEmulationPort(0, (LONG*)&ctl.lIDE0MasterEmulationPort)))
7368 || (FAILED(rc = pCtl->GetIDEEmulationPort(1, (LONG*)&ctl.lIDE0SlaveEmulationPort)))
7369 || (FAILED(rc = pCtl->GetIDEEmulationPort(2, (LONG*)&ctl.lIDE1MasterEmulationPort)))
7370 || (FAILED(rc = pCtl->GetIDEEmulationPort(3, (LONG*)&ctl.lIDE1SlaveEmulationPort)))
7371 )
7372 ComAssertComRCRet(rc, rc);
7373 }
7374
7375 /* save the devices now. */
7376 rc = saveStorageDevices(pCtl, ctl);
7377 ComAssertComRCRet(rc, rc);
7378
7379 data.llStorageControllers.push_back(ctl);
7380 }
7381
7382 return S_OK;
7383}
7384
7385/**
7386 * Saves the hard disk confguration.
7387 */
7388HRESULT Machine::saveStorageDevices(ComObjPtr<StorageController> aStorageController,
7389 settings::StorageController &data)
7390{
7391 MediaData::AttachmentList atts;
7392
7393 HRESULT rc = getMediumAttachmentsOfController(Bstr(aStorageController->getName()), atts);
7394 if (FAILED(rc)) return rc;
7395
7396 data.llAttachedDevices.clear();
7397 for (MediaData::AttachmentList::const_iterator it = atts.begin();
7398 it != atts.end();
7399 ++it)
7400 {
7401 settings::AttachedDevice dev;
7402
7403 MediumAttachment *pAttach = *it;
7404 Medium *pMedium = pAttach->getMedium();
7405
7406 dev.deviceType = pAttach->getType();
7407 dev.lPort = pAttach->getPort();
7408 dev.lDevice = pAttach->getDevice();
7409 if (pMedium)
7410 {
7411 BOOL fHostDrive = FALSE;
7412 rc = pMedium->COMGETTER(HostDrive)(&fHostDrive);
7413 if (FAILED(rc))
7414 return rc;
7415 if (fHostDrive)
7416 dev.strHostDriveSrc = pMedium->getLocation();
7417 else
7418 dev.uuid = pMedium->getId();
7419 dev.fPassThrough = pAttach->getPassthrough();
7420 }
7421
7422 data.llAttachedDevices.push_back(dev);
7423 }
7424
7425 return S_OK;
7426}
7427
7428/**
7429 * Saves machine state settings as defined by aFlags
7430 * (SaveSTS_* values).
7431 *
7432 * @param aFlags Combination of SaveSTS_* flags.
7433 *
7434 * @note Locks objects for writing.
7435 */
7436HRESULT Machine::saveStateSettings(int aFlags)
7437{
7438 if (aFlags == 0)
7439 return S_OK;
7440
7441 AutoCaller autoCaller(this);
7442 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
7443
7444 /* This object's write lock is also necessary to serialize file access
7445 * (prevent concurrent reads and writes) */
7446 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7447
7448 HRESULT rc = S_OK;
7449
7450 Assert(mData->m_pMachineConfigFile);
7451
7452 try
7453 {
7454 if (aFlags & SaveSTS_CurStateModified)
7455 mData->m_pMachineConfigFile->fCurrentStateModified = true;
7456
7457 if (aFlags & SaveSTS_StateFilePath)
7458 {
7459 if (!mSSData->mStateFilePath.isEmpty())
7460 /* try to make the file name relative to the settings file dir */
7461 calculateRelativePath(mSSData->mStateFilePath, mData->m_pMachineConfigFile->strStateFile);
7462 else
7463 mData->m_pMachineConfigFile->strStateFile.setNull();
7464 }
7465
7466 if (aFlags & SaveSTS_StateTimeStamp)
7467 {
7468 Assert( mData->mMachineState != MachineState_Aborted
7469 || mSSData->mStateFilePath.isEmpty());
7470
7471 mData->m_pMachineConfigFile->timeLastStateChange = mData->mLastStateChange;
7472
7473 mData->m_pMachineConfigFile->fAborted = (mData->mMachineState == MachineState_Aborted);
7474//@todo live migration mData->m_pMachineConfigFile->fTeleported = (mData->mMachineState == MachineState_Teleported);
7475 }
7476
7477 mData->m_pMachineConfigFile->write(mData->m_strConfigFileFull);
7478 }
7479 catch (...)
7480 {
7481 rc = VirtualBox::handleUnexpectedExceptions (RT_SRC_POS);
7482 }
7483
7484 return rc;
7485}
7486
7487/**
7488 * Creates differencing hard disks for all normal hard disks attached to this
7489 * machine and a new set of attachments to refer to created disks.
7490 *
7491 * Used when taking a snapshot or when discarding the current state.
7492 *
7493 * This method assumes that mMediaData contains the original hard disk attachments
7494 * it needs to create diffs for. On success, these attachments will be replaced
7495 * with the created diffs. On failure, #deleteImplicitDiffs() is implicitly
7496 * called to delete created diffs which will also rollback mMediaData and restore
7497 * whatever was backed up before calling this method.
7498 *
7499 * Attachments with non-normal hard disks are left as is.
7500 *
7501 * If @a aOnline is @c false then the original hard disks that require implicit
7502 * diffs will be locked for reading. Otherwise it is assumed that they are
7503 * already locked for writing (when the VM was started). Note that in the latter
7504 * case it is responsibility of the caller to lock the newly created diffs for
7505 * writing if this method succeeds.
7506 *
7507 * @param aFolder Folder where to create diff hard disks.
7508 * @param aProgress Progress object to run (must contain at least as
7509 * many operations left as the number of hard disks
7510 * attached).
7511 * @param aOnline Whether the VM was online prior to this operation.
7512 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
7513 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
7514 *
7515 * @note The progress object is not marked as completed, neither on success nor
7516 * on failure. This is a responsibility of the caller.
7517 *
7518 * @note Locks this object for writing.
7519 */
7520HRESULT Machine::createImplicitDiffs(const Bstr &aFolder,
7521 IProgress *aProgress,
7522 ULONG aWeight,
7523 bool aOnline,
7524 bool *pfNeedsSaveSettings)
7525{
7526 AssertReturn(!aFolder.isEmpty(), E_FAIL);
7527
7528 LogFlowThisFunc(("aFolder='%ls', aOnline=%d\n", aFolder.raw(), aOnline));
7529
7530 AutoCaller autoCaller(this);
7531 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7532
7533 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7534
7535 /* must be in a protective state because we leave the lock below */
7536 AssertReturn( mData->mMachineState == MachineState_Saving
7537 || mData->mMachineState == MachineState_LiveSnapshotting
7538 || mData->mMachineState == MachineState_RestoringSnapshot
7539 || mData->mMachineState == MachineState_DeletingSnapshot
7540 , E_FAIL);
7541
7542 HRESULT rc = S_OK;
7543
7544 MediaList lockedMedia;
7545
7546 try
7547 {
7548 if (!aOnline)
7549 {
7550 /* lock all attached hard disks early to detect "in use"
7551 * situations before creating actual diffs */
7552 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
7553 it != mMediaData->mAttachments.end();
7554 ++it)
7555 {
7556 MediumAttachment* pAtt = *it;
7557 if (pAtt->getType() == DeviceType_HardDisk)
7558 {
7559 Medium* pHD = pAtt->getMedium();
7560 Assert(pHD);
7561 rc = pHD->LockRead (NULL);
7562 if (FAILED(rc)) throw rc;
7563 lockedMedia.push_back(pHD);
7564 }
7565 }
7566 }
7567
7568 /* remember the current list (note that we don't use backup() since
7569 * mMediaData may be already backed up) */
7570 MediaData::AttachmentList atts = mMediaData->mAttachments;
7571
7572 /* start from scratch */
7573 mMediaData->mAttachments.clear();
7574
7575 /* go through remembered attachments and create diffs for normal hard
7576 * disks and attach them */
7577 for (MediaData::AttachmentList::const_iterator it = atts.begin();
7578 it != atts.end();
7579 ++it)
7580 {
7581 MediumAttachment* pAtt = *it;
7582
7583 DeviceType_T devType = pAtt->getType();
7584 Medium* medium = pAtt->getMedium();
7585
7586 if ( devType != DeviceType_HardDisk
7587 || medium == NULL
7588 || medium->getType() != MediumType_Normal)
7589 {
7590 /* copy the attachment as is */
7591
7592 /** @todo the progress object created in Console::TakeSnaphot
7593 * only expects operations for hard disks. Later other
7594 * device types need to show up in the progress as well. */
7595 if (devType == DeviceType_HardDisk)
7596 {
7597 if (medium == NULL)
7598 aProgress->SetNextOperation(Bstr(tr("Skipping attachment without medium")),
7599 aWeight); // weight
7600 else
7601 aProgress->SetNextOperation(BstrFmt(tr("Skipping medium '%s'"),
7602 medium->getBase()->getName().raw()),
7603 aWeight); // weight
7604 }
7605
7606 mMediaData->mAttachments.push_back(pAtt);
7607 continue;
7608 }
7609
7610 /* need a diff */
7611 aProgress->SetNextOperation(BstrFmt(tr("Creating differencing hard disk for '%s'"),
7612 medium->getBase()->getName().raw()),
7613 aWeight); // weight
7614
7615 ComObjPtr<Medium> diff;
7616 diff.createObject();
7617 rc = diff->init(mParent,
7618 medium->preferredDiffFormat().raw(),
7619 BstrFmt("%ls"RTPATH_SLASH_STR,
7620 mUserData->mSnapshotFolderFull.raw()).raw(),
7621 pfNeedsSaveSettings);
7622 if (FAILED(rc)) throw rc;
7623
7624 /* leave the lock before the potentially lengthy operation */
7625 alock.leave();
7626
7627 rc = medium->createDiffStorageAndWait(diff,
7628 MediumVariant_Standard,
7629 pfNeedsSaveSettings);
7630
7631 /** @todo r=bird: How is the locking and diff image cleaned up if we fail before
7632 * the push_back? Looks like we're going to leave medium with the
7633 * wrong kind of lock (general issue with if we fail anywhere at all)
7634 * and an orphaned VDI in the snapshots folder. */
7635 // at this point, the old image is still locked for writing, but instead
7636 // we need the new diff image locked for writing and lock the previously
7637 // current one for reading only
7638 if (aOnline)
7639 {
7640 diff->LockWrite(NULL);
7641 mData->mSession.mLockedMedia.push_back(Data::Session::LockedMedia::value_type(ComPtr<IMedium>(diff), true));
7642 medium->UnlockWrite(NULL);
7643 medium->LockRead(NULL);
7644 mData->mSession.mLockedMedia.push_back(Data::Session::LockedMedia::value_type(ComPtr<IMedium>(medium), false));
7645 }
7646
7647 if (FAILED(rc)) throw rc;
7648
7649 alock.enter();
7650
7651 rc = diff->attachTo(mData->mUuid);
7652 AssertComRCThrowRC(rc);
7653
7654 /* add a new attachment */
7655 ComObjPtr<MediumAttachment> attachment;
7656 attachment.createObject();
7657 rc = attachment->init(this,
7658 diff,
7659 pAtt->getControllerName(),
7660 pAtt->getPort(),
7661 pAtt->getDevice(),
7662 DeviceType_HardDisk,
7663 true /* aImplicit */);
7664 if (FAILED(rc)) throw rc;
7665
7666 mMediaData->mAttachments.push_back(attachment);
7667 }
7668 }
7669 catch (HRESULT aRC) { rc = aRC; }
7670
7671 /* unlock all hard disks we locked */
7672 if (!aOnline)
7673 {
7674 ErrorInfoKeeper eik;
7675
7676 for (MediaList::const_iterator it = lockedMedia.begin();
7677 it != lockedMedia.end();
7678 ++it)
7679 {
7680 HRESULT rc2 = (*it)->UnlockRead(NULL);
7681 AssertComRC(rc2);
7682 }
7683 }
7684
7685 if (FAILED(rc))
7686 {
7687 MultiResultRef mrc (rc);
7688
7689 mrc = deleteImplicitDiffs(pfNeedsSaveSettings);
7690 }
7691
7692 return rc;
7693}
7694
7695/**
7696 * Deletes implicit differencing hard disks created either by
7697 * #createImplicitDiffs() or by #AttachMedium() and rolls back mMediaData.
7698 *
7699 * Note that to delete hard disks created by #AttachMedium() this method is
7700 * called from #fixupMedia() when the changes are rolled back.
7701 *
7702 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
7703 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
7704 *
7705 * @note Locks this object for writing.
7706 */
7707HRESULT Machine::deleteImplicitDiffs(bool *pfNeedsSaveSettings)
7708{
7709 AutoCaller autoCaller(this);
7710 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
7711
7712 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7713 LogFlowThisFuncEnter();
7714
7715 AssertReturn(mMediaData.isBackedUp(), E_FAIL);
7716
7717 HRESULT rc = S_OK;
7718
7719 MediaData::AttachmentList implicitAtts;
7720
7721 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
7722
7723 /* enumerate new attachments */
7724 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
7725 it != mMediaData->mAttachments.end();
7726 ++it)
7727 {
7728 ComObjPtr<Medium> hd = (*it)->getMedium();
7729 if (hd.isNull())
7730 continue;
7731
7732 if ((*it)->isImplicit())
7733 {
7734 /* deassociate and mark for deletion */
7735 LogFlowThisFunc(("Detaching '%s', pending deletion\n", (*it)->getLogName()));
7736 rc = hd->detachFrom(mData->mUuid);
7737 AssertComRC(rc);
7738 implicitAtts.push_back (*it);
7739 continue;
7740 }
7741
7742 /* was this hard disk attached before? */
7743 if (!findAttachment(oldAtts, hd))
7744 {
7745 /* no: de-associate */
7746 LogFlowThisFunc(("Detaching '%s', no deletion\n", (*it)->getLogName()));
7747 rc = hd->detachFrom(mData->mUuid);
7748 AssertComRC(rc);
7749 continue;
7750 }
7751 LogFlowThisFunc(("Not detaching '%s'\n", (*it)->getLogName()));
7752 }
7753
7754 /* rollback hard disk changes */
7755 mMediaData.rollback();
7756
7757 MultiResult mrc (S_OK);
7758
7759 /* delete unused implicit diffs */
7760 if (implicitAtts.size() != 0)
7761 {
7762 /* will leave the lock before the potentially lengthy
7763 * operation, so protect with the special state (unless already
7764 * protected) */
7765 MachineState_T oldState = mData->mMachineState;
7766 if ( oldState != MachineState_Saving
7767 && oldState != MachineState_LiveSnapshotting
7768 && oldState != MachineState_RestoringSnapshot
7769 && oldState != MachineState_DeletingSnapshot
7770 )
7771 setMachineState (MachineState_SettingUp);
7772
7773 alock.leave();
7774
7775 for (MediaData::AttachmentList::const_iterator it = implicitAtts.begin();
7776 it != implicitAtts.end();
7777 ++it)
7778 {
7779 LogFlowThisFunc(("Deleting '%s'\n", (*it)->getLogName()));
7780 ComObjPtr<Medium> hd = (*it)->getMedium();
7781
7782 rc = hd->deleteStorageAndWait(NULL /*aProgress*/, pfNeedsSaveSettings);
7783#if 1 /* HACK ALERT: Just make it kind of work */ /** @todo Fix this hack properly. The LockWrite / UnlockWrite / LockRead changes aren't undone! */
7784 if (rc == VBOX_E_INVALID_OBJECT_STATE)
7785 {
7786 LogFlowFunc(("Applying unlock hack on '%s'! FIXME!\n", (*it)->getLogName()));
7787 hd->UnlockWrite(NULL);
7788 rc = hd->deleteStorageAndWait(NULL /*aProgress*/, pfNeedsSaveSettings);
7789 }
7790#endif
7791 AssertMsg(SUCCEEDED(rc), ("rc=%Rhrc it=%s hd=%s\n", rc, (*it)->getLogName(), hd->getLocationFull().c_str() ));
7792 mrc = rc;
7793 }
7794
7795 alock.enter();
7796
7797 if (mData->mMachineState == MachineState_SettingUp)
7798 {
7799 setMachineState (oldState);
7800 }
7801 }
7802
7803 return mrc;
7804}
7805
7806/**
7807 * Looks through the given list of media attachments for one with the given parameters
7808 * and returns it, or NULL if not found. The list is a parameter so that backup lists
7809 * can be searched as well if needed.
7810 *
7811 * @param list
7812 * @param aControllerName
7813 * @param aControllerPort
7814 * @param aDevice
7815 * @return
7816 */
7817MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
7818 IN_BSTR aControllerName,
7819 LONG aControllerPort,
7820 LONG aDevice)
7821{
7822 for (MediaData::AttachmentList::const_iterator it = ll.begin();
7823 it != ll.end();
7824 ++it)
7825 {
7826 MediumAttachment *pAttach = *it;
7827 if (pAttach->matches(aControllerName, aControllerPort, aDevice))
7828 return pAttach;
7829 }
7830
7831 return NULL;
7832}
7833
7834/**
7835 * Looks through the given list of media attachments for one with the given parameters
7836 * and returns it, or NULL if not found. The list is a parameter so that backup lists
7837 * can be searched as well if needed.
7838 *
7839 * @param list
7840 * @param aControllerName
7841 * @param aControllerPort
7842 * @param aDevice
7843 * @return
7844 */
7845MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
7846 ComObjPtr<Medium> pMedium)
7847{
7848 for (MediaData::AttachmentList::const_iterator it = ll.begin();
7849 it != ll.end();
7850 ++it)
7851 {
7852 MediumAttachment *pAttach = *it;
7853 ComObjPtr<Medium> pMediumThis = pAttach->getMedium();
7854 if (pMediumThis.equalsTo(pMedium))
7855 return pAttach;
7856 }
7857
7858 return NULL;
7859}
7860
7861/**
7862 * Looks through the given list of media attachments for one with the given parameters
7863 * and returns it, or NULL if not found. The list is a parameter so that backup lists
7864 * can be searched as well if needed.
7865 *
7866 * @param list
7867 * @param aControllerName
7868 * @param aControllerPort
7869 * @param aDevice
7870 * @return
7871 */
7872MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
7873 Guid &id)
7874{
7875 for (MediaData::AttachmentList::const_iterator it = ll.begin();
7876 it != ll.end();
7877 ++it)
7878 {
7879 MediumAttachment *pAttach = *it;
7880 ComObjPtr<Medium> pMediumThis = pAttach->getMedium();
7881 if (pMediumThis->getId() == id)
7882 return pAttach;
7883 }
7884
7885 return NULL;
7886}
7887
7888/**
7889 * Perform deferred hard disk detachments.
7890 *
7891 * Does nothing if the hard disk attachment data (mMediaData) is not changed (not
7892 * backed up).
7893 *
7894 * If @a aOnline is @c true then this method will also unlock the old hard disks
7895 * for which the new implicit diffs were created and will lock these new diffs for
7896 * writing.
7897 *
7898 * @param aOnline Whether the VM was online prior to this operation.
7899 *
7900 * @note Locks this object for writing!
7901 */
7902void Machine::commitMedia(bool aOnline /*= false*/)
7903{
7904 AutoCaller autoCaller(this);
7905 AssertComRCReturnVoid (autoCaller.rc());
7906
7907 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7908
7909 LogFlowThisFunc(("Entering, aOnline=%d\n", aOnline));
7910
7911 HRESULT rc = S_OK;
7912
7913 /* no attach/detach operations -- nothing to do */
7914 if (!mMediaData.isBackedUp())
7915 return;
7916
7917 MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
7918
7919 /* enumerate new attachments */
7920 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
7921 it != mMediaData->mAttachments.end();
7922 ++it)
7923 {
7924 MediumAttachment *pAttach = *it;
7925
7926 pAttach->commit();
7927
7928 Medium* pMedium = pAttach->getMedium();
7929 bool fImplicit = pAttach->isImplicit();
7930
7931 LogFlowThisFunc(("Examining current medium '%s' (implicit: %d)\n",
7932 (pMedium) ? pMedium->getName().raw() : "NULL",
7933 fImplicit));
7934
7935 /** @todo convert all this Machine-based voodoo to MediumAttachment
7936 * based commit logic. */
7937 if (fImplicit)
7938 {
7939 /* convert implicit attachment to normal */
7940 pAttach->setImplicit(false);
7941
7942 if ( aOnline
7943 && pMedium
7944 && pAttach->getType() == DeviceType_HardDisk
7945 )
7946 {
7947 rc = pMedium->LockWrite(NULL);
7948 AssertComRC(rc);
7949
7950 mData->mSession.mLockedMedia.push_back(
7951 Data::Session::LockedMedia::value_type(
7952 ComPtr<IMedium>(pMedium), true));
7953
7954 /* also, relock the old hard disk which is a base for the
7955 * new diff for reading if the VM is online */
7956
7957 ComObjPtr<Medium> parent = pMedium->getParent();
7958 /* make the relock atomic */
7959 AutoWriteLock parentLock(parent COMMA_LOCKVAL_SRC_POS);
7960 rc = parent->UnlockWrite(NULL);
7961 AssertComRC(rc);
7962 rc = parent->LockRead(NULL);
7963 AssertComRC(rc);
7964
7965 /* XXX actually we should replace the old entry in that
7966 * vector (write lock => read lock) but this would take
7967 * some effort. So lets just ignore the error code in
7968 * SessionMachine::unlockMedia(). */
7969 mData->mSession.mLockedMedia.push_back(
7970 Data::Session::LockedMedia::value_type (
7971 ComPtr<IMedium>(parent), false));
7972 }
7973
7974 continue;
7975 }
7976
7977 if (pMedium)
7978 {
7979 /* was this medium attached before? */
7980 for (MediaData::AttachmentList::iterator oldIt = oldAtts.begin();
7981 oldIt != oldAtts.end();
7982 ++oldIt)
7983 {
7984 MediumAttachment *pOldAttach = *oldIt;
7985 if (pOldAttach->getMedium().equalsTo(pMedium))
7986 {
7987 LogFlowThisFunc(("--> medium '%s' was attached before, will not remove\n", pMedium->getName().raw()));
7988
7989 /* yes: remove from old to avoid de-association */
7990 oldAtts.erase(oldIt);
7991 break;
7992 }
7993 }
7994 }
7995 }
7996
7997 /* enumerate remaining old attachments and de-associate from the
7998 * current machine state */
7999 for (MediaData::AttachmentList::const_iterator it = oldAtts.begin();
8000 it != oldAtts.end();
8001 ++it)
8002 {
8003 MediumAttachment *pAttach = *it;
8004 Medium* pMedium = pAttach->getMedium();
8005
8006 /* Detach only hard disks, since DVD/floppy media is detached
8007 * instantly in MountMedium. */
8008 if (pAttach->getType() == DeviceType_HardDisk && pMedium)
8009 {
8010 LogFlowThisFunc(("detaching medium '%s' from machine\n", pMedium->getName().raw()));
8011
8012 /* now de-associate from the current machine state */
8013 rc = pMedium->detachFrom(mData->mUuid);
8014 AssertComRC(rc);
8015
8016 if ( aOnline
8017 && pAttach->getType() == DeviceType_HardDisk)
8018 {
8019 /* unlock since not used anymore */
8020 MediumState_T state;
8021 rc = pMedium->UnlockWrite(&state);
8022 /* the disk may be alredy relocked for reading above */
8023 Assert (SUCCEEDED(rc) || state == MediumState_LockedRead);
8024 }
8025 }
8026 }
8027
8028 /* commit the hard disk changes */
8029 mMediaData.commit();
8030
8031 if (getClassID() == clsidSessionMachine)
8032 {
8033 /* attach new data to the primary machine and reshare it */
8034 mPeer->mMediaData.attach(mMediaData);
8035 }
8036
8037 return;
8038}
8039
8040/**
8041 * Perform deferred deletion of implicitly created diffs.
8042 *
8043 * Does nothing if the hard disk attachment data (mMediaData) is not changed (not
8044 * backed up).
8045 *
8046 * @param pfNeedsSaveSettings Optional pointer to a bool that must have been initialized to false and that will be set to true
8047 * by this function if the caller should invoke VirtualBox::saveSettings() because the global settings have changed.
8048 *
8049 * @note Locks this object for writing!
8050 */
8051void Machine::rollbackMedia()
8052{
8053 AutoCaller autoCaller(this);
8054 AssertComRCReturnVoid (autoCaller.rc());
8055
8056 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8057
8058 LogFlowThisFunc(("Entering\n"));
8059
8060 HRESULT rc = S_OK;
8061
8062 /* no attach/detach operations -- nothing to do */
8063 if (!mMediaData.isBackedUp())
8064 return;
8065
8066 /* enumerate new attachments */
8067 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
8068 it != mMediaData->mAttachments.end();
8069 ++it)
8070 {
8071 MediumAttachment *pAttach = *it;
8072 /* Fix up the backrefs for DVD/floppy media. */
8073 if (pAttach->getType() != DeviceType_HardDisk)
8074 {
8075 Medium* pMedium = pAttach->getMedium();
8076 if (pMedium)
8077 {
8078 rc = pMedium->detachFrom(mData->mUuid);
8079 AssertComRC(rc);
8080 }
8081 }
8082
8083 (*it)->rollback();
8084
8085 pAttach = *it;
8086 /* Fix up the backrefs for DVD/floppy media. */
8087 if (pAttach->getType() != DeviceType_HardDisk)
8088 {
8089 Medium* pMedium = pAttach->getMedium();
8090 if (pMedium)
8091 {
8092 rc = pMedium->attachTo(mData->mUuid);
8093 AssertComRC(rc);
8094 }
8095 }
8096 }
8097
8098 /** @todo convert all this Machine-based voodoo to MediumAttachment
8099 * based rollback logic. */
8100 // @todo r=dj the below totally fails if this gets called from Machine::rollback(),
8101 // which gets called if Machine::registeredInit() fails...
8102 deleteImplicitDiffs(NULL /*pfNeedsSaveSettings*/);
8103
8104 return;
8105}
8106
8107/**
8108 * Returns true if the settings file is located in the directory named exactly
8109 * as the machine. This will be true if the machine settings structure was
8110 * created by default in #openConfigLoader().
8111 *
8112 * @param aSettingsDir if not NULL, the full machine settings file directory
8113 * name will be assigned there.
8114 *
8115 * @note Doesn't lock anything.
8116 * @note Not thread safe (must be called from this object's lock).
8117 */
8118bool Machine::isInOwnDir(Utf8Str *aSettingsDir /* = NULL */)
8119{
8120 Utf8Str settingsDir = mData->m_strConfigFileFull;
8121 settingsDir.stripFilename();
8122 char *dirName = RTPathFilename(settingsDir.c_str());
8123
8124 AssertReturn(dirName, false);
8125
8126 /* if we don't rename anything on name change, return false shorlty */
8127 if (!mUserData->mNameSync)
8128 return false;
8129
8130 if (aSettingsDir)
8131 *aSettingsDir = settingsDir;
8132
8133 return Bstr (dirName) == mUserData->mName;
8134}
8135
8136/**
8137 * Discards all changes to machine settings.
8138 *
8139 * @param aNotify Whether to notify the direct session about changes or not.
8140 *
8141 * @note Locks objects for writing!
8142 */
8143void Machine::rollback(bool aNotify)
8144{
8145 AutoCaller autoCaller(this);
8146 AssertComRCReturn (autoCaller.rc(), (void) 0);
8147
8148 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8149
8150 if (!mStorageControllers.isNull())
8151 {
8152 if (mStorageControllers.isBackedUp())
8153 {
8154 /* unitialize all new devices (absent in the backed up list). */
8155 StorageControllerList::const_iterator it = mStorageControllers->begin();
8156 StorageControllerList *backedList = mStorageControllers.backedUpData();
8157 while (it != mStorageControllers->end())
8158 {
8159 if ( std::find(backedList->begin(), backedList->end(), *it)
8160 == backedList->end()
8161 )
8162 {
8163 (*it)->uninit();
8164 }
8165 ++it;
8166 }
8167
8168 /* restore the list */
8169 mStorageControllers.rollback();
8170 }
8171
8172 /* rollback any changes to devices after restoring the list */
8173 if (m_flModifications & IsModified_Storage)
8174 {
8175 StorageControllerList::const_iterator it = mStorageControllers->begin();
8176 while (it != mStorageControllers->end())
8177 {
8178 (*it)->rollback();
8179 ++it;
8180 }
8181 }
8182 }
8183
8184 mUserData.rollback();
8185
8186 mHWData.rollback();
8187
8188 if (m_flModifications & IsModified_Storage)
8189 rollbackMedia();
8190
8191 if (mBIOSSettings)
8192 mBIOSSettings->rollback();
8193
8194#ifdef VBOX_WITH_VRDP
8195 if (mVRDPServer && (m_flModifications & IsModified_VRDPServer))
8196 mVRDPServer->rollback();
8197#endif
8198
8199 if (mAudioAdapter)
8200 mAudioAdapter->rollback();
8201
8202 if (mUSBController && (m_flModifications & IsModified_USB))
8203 mUSBController->rollback();
8204
8205 ComPtr<INetworkAdapter> networkAdapters[RT_ELEMENTS(mNetworkAdapters)];
8206 ComPtr<ISerialPort> serialPorts[RT_ELEMENTS(mSerialPorts)];
8207 ComPtr<IParallelPort> parallelPorts[RT_ELEMENTS(mParallelPorts)];
8208
8209 if (m_flModifications & IsModified_NetworkAdapters)
8210 for (ULONG slot = 0; slot < RT_ELEMENTS(mNetworkAdapters); slot++)
8211 if ( mNetworkAdapters[slot]
8212 && mNetworkAdapters[slot]->isModified())
8213 {
8214 mNetworkAdapters[slot]->rollback();
8215 networkAdapters[slot] = mNetworkAdapters[slot];
8216 }
8217
8218 if (m_flModifications & IsModified_SerialPorts)
8219 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
8220 if ( mSerialPorts[slot]
8221 && mSerialPorts[slot]->isModified())
8222 {
8223 mSerialPorts[slot]->rollback();
8224 serialPorts[slot] = mSerialPorts[slot];
8225 }
8226
8227 if (m_flModifications & IsModified_ParallelPorts)
8228 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
8229 if ( mParallelPorts[slot]
8230 && mParallelPorts[slot]->isModified())
8231 {
8232 mParallelPorts[slot]->rollback();
8233 parallelPorts[slot] = mParallelPorts[slot];
8234 }
8235
8236 if (aNotify)
8237 {
8238 /* inform the direct session about changes */
8239
8240 ComObjPtr<Machine> that = this;
8241 uint32_t flModifications = m_flModifications;
8242 alock.leave();
8243
8244 if (flModifications & IsModified_SharedFolders)
8245 that->onSharedFolderChange();
8246
8247 if (flModifications & IsModified_VRDPServer)
8248 that->onVRDPServerChange();
8249 if (flModifications & IsModified_USB)
8250 that->onUSBControllerChange();
8251
8252 for (ULONG slot = 0; slot < RT_ELEMENTS(networkAdapters); slot ++)
8253 if (networkAdapters[slot])
8254 that->onNetworkAdapterChange(networkAdapters[slot], FALSE);
8255 for (ULONG slot = 0; slot < RT_ELEMENTS(serialPorts); slot ++)
8256 if (serialPorts[slot])
8257 that->onSerialPortChange(serialPorts[slot]);
8258 for (ULONG slot = 0; slot < RT_ELEMENTS(parallelPorts); slot ++)
8259 if (parallelPorts[slot])
8260 that->onParallelPortChange(parallelPorts[slot]);
8261
8262 if (flModifications & IsModified_Storage)
8263 that->onStorageControllerChange();
8264 }
8265}
8266
8267/**
8268 * Commits all the changes to machine settings.
8269 *
8270 * Note that this operation is supposed to never fail.
8271 *
8272 * @note Locks this object and children for writing.
8273 */
8274void Machine::commit()
8275{
8276 AutoCaller autoCaller(this);
8277 AssertComRCReturnVoid (autoCaller.rc());
8278
8279 AutoCaller peerCaller (mPeer);
8280 AssertComRCReturnVoid (peerCaller.rc());
8281
8282 AutoMultiWriteLock2 alock(mPeer, this COMMA_LOCKVAL_SRC_POS);
8283
8284 /*
8285 * use safe commit to ensure Snapshot machines (that share mUserData)
8286 * will still refer to a valid memory location
8287 */
8288 mUserData.commitCopy();
8289
8290 mHWData.commit();
8291
8292 if (mMediaData.isBackedUp())
8293 commitMedia();
8294
8295 mBIOSSettings->commit();
8296#ifdef VBOX_WITH_VRDP
8297 mVRDPServer->commit();
8298#endif
8299 mAudioAdapter->commit();
8300 mUSBController->commit();
8301
8302 for (ULONG slot = 0; slot < RT_ELEMENTS (mNetworkAdapters); slot ++)
8303 mNetworkAdapters [slot]->commit();
8304 for (ULONG slot = 0; slot < RT_ELEMENTS (mSerialPorts); slot ++)
8305 mSerialPorts [slot]->commit();
8306 for (ULONG slot = 0; slot < RT_ELEMENTS (mParallelPorts); slot ++)
8307 mParallelPorts [slot]->commit();
8308
8309 bool commitStorageControllers = false;
8310
8311 if (mStorageControllers.isBackedUp())
8312 {
8313 mStorageControllers.commit();
8314
8315 if (mPeer)
8316 {
8317 AutoWriteLock peerlock(mPeer COMMA_LOCKVAL_SRC_POS);
8318
8319 /* Commit all changes to new controllers (this will reshare data with
8320 * peers for thos who have peers) */
8321 StorageControllerList *newList = new StorageControllerList();
8322 StorageControllerList::const_iterator it = mStorageControllers->begin();
8323 while (it != mStorageControllers->end())
8324 {
8325 (*it)->commit();
8326
8327 /* look if this controller has a peer device */
8328 ComObjPtr<StorageController> peer = (*it)->getPeer();
8329 if (!peer)
8330 {
8331 /* no peer means the device is a newly created one;
8332 * create a peer owning data this device share it with */
8333 peer.createObject();
8334 peer->init (mPeer, *it, true /* aReshare */);
8335 }
8336 else
8337 {
8338 /* remove peer from the old list */
8339 mPeer->mStorageControllers->remove (peer);
8340 }
8341 /* and add it to the new list */
8342 newList->push_back(peer);
8343
8344 ++it;
8345 }
8346
8347 /* uninit old peer's controllers that are left */
8348 it = mPeer->mStorageControllers->begin();
8349 while (it != mPeer->mStorageControllers->end())
8350 {
8351 (*it)->uninit();
8352 ++it;
8353 }
8354
8355 /* attach new list of controllers to our peer */
8356 mPeer->mStorageControllers.attach (newList);
8357 }
8358 else
8359 {
8360 /* we have no peer (our parent is the newly created machine);
8361 * just commit changes to devices */
8362 commitStorageControllers = true;
8363 }
8364 }
8365 else
8366 {
8367 /* the list of controllers itself is not changed,
8368 * just commit changes to controllers themselves */
8369 commitStorageControllers = true;
8370 }
8371
8372 if (commitStorageControllers)
8373 {
8374 StorageControllerList::const_iterator it = mStorageControllers->begin();
8375 while (it != mStorageControllers->end())
8376 {
8377 (*it)->commit();
8378 ++it;
8379 }
8380 }
8381
8382 if (getClassID() == clsidSessionMachine)
8383 {
8384 /* attach new data to the primary machine and reshare it */
8385 mPeer->mUserData.attach (mUserData);
8386 mPeer->mHWData.attach (mHWData);
8387 /* mMediaData is reshared by fixupMedia */
8388 // mPeer->mMediaData.attach(mMediaData);
8389 Assert(mPeer->mMediaData.data() == mMediaData.data());
8390 }
8391}
8392
8393/**
8394 * Copies all the hardware data from the given machine.
8395 *
8396 * Currently, only called when the VM is being restored from a snapshot. In
8397 * particular, this implies that the VM is not running during this method's
8398 * call.
8399 *
8400 * @note This method must be called from under this object's lock.
8401 *
8402 * @note This method doesn't call #commit(), so all data remains backed up and
8403 * unsaved.
8404 */
8405void Machine::copyFrom(Machine *aThat)
8406{
8407 AssertReturnVoid(getClassID() == clsidMachine || getClassID() == clsidSessionMachine);
8408 AssertReturnVoid(aThat->getClassID() == clsidSnapshotMachine);
8409
8410 AssertReturnVoid(!Global::IsOnline (mData->mMachineState));
8411
8412 mHWData.assignCopy(aThat->mHWData);
8413
8414 // create copies of all shared folders (mHWData after attiching a copy
8415 // contains just references to original objects)
8416 for (HWData::SharedFolderList::iterator it = mHWData->mSharedFolders.begin();
8417 it != mHWData->mSharedFolders.end();
8418 ++it)
8419 {
8420 ComObjPtr<SharedFolder> folder;
8421 folder.createObject();
8422 HRESULT rc = folder->initCopy(getMachine(), *it);
8423 AssertComRC (rc);
8424 *it = folder;
8425 }
8426
8427 mBIOSSettings->copyFrom(aThat->mBIOSSettings);
8428#ifdef VBOX_WITH_VRDP
8429 mVRDPServer->copyFrom(aThat->mVRDPServer);
8430#endif
8431 mAudioAdapter->copyFrom(aThat->mAudioAdapter);
8432 mUSBController->copyFrom(aThat->mUSBController);
8433
8434 /* create private copies of all controllers */
8435 mStorageControllers.backup();
8436 mStorageControllers->clear();
8437 for (StorageControllerList::iterator it = aThat->mStorageControllers->begin();
8438 it != aThat->mStorageControllers->end();
8439 ++it)
8440 {
8441 ComObjPtr<StorageController> ctrl;
8442 ctrl.createObject();
8443 ctrl->initCopy (this, *it);
8444 mStorageControllers->push_back(ctrl);
8445 }
8446
8447 for (ULONG slot = 0; slot < RT_ELEMENTS (mNetworkAdapters); slot ++)
8448 mNetworkAdapters[slot]->copyFrom (aThat->mNetworkAdapters [slot]);
8449 for (ULONG slot = 0; slot < RT_ELEMENTS (mSerialPorts); slot ++)
8450 mSerialPorts[slot]->copyFrom (aThat->mSerialPorts [slot]);
8451 for (ULONG slot = 0; slot < RT_ELEMENTS (mParallelPorts); slot ++)
8452 mParallelPorts[slot]->copyFrom (aThat->mParallelPorts [slot]);
8453}
8454
8455#ifdef VBOX_WITH_RESOURCE_USAGE_API
8456void Machine::registerMetrics (PerformanceCollector *aCollector, Machine *aMachine, RTPROCESS pid)
8457{
8458 pm::CollectorHAL *hal = aCollector->getHAL();
8459 /* Create sub metrics */
8460 pm::SubMetric *cpuLoadUser = new pm::SubMetric ("CPU/Load/User",
8461 "Percentage of processor time spent in user mode by VM process.");
8462 pm::SubMetric *cpuLoadKernel = new pm::SubMetric ("CPU/Load/Kernel",
8463 "Percentage of processor time spent in kernel mode by VM process.");
8464 pm::SubMetric *ramUsageUsed = new pm::SubMetric ("RAM/Usage/Used",
8465 "Size of resident portion of VM process in memory.");
8466 /* Create and register base metrics */
8467 pm::BaseMetric *cpuLoad = new pm::MachineCpuLoadRaw (hal, aMachine, pid,
8468 cpuLoadUser, cpuLoadKernel);
8469 aCollector->registerBaseMetric (cpuLoad);
8470 pm::BaseMetric *ramUsage = new pm::MachineRamUsage (hal, aMachine, pid,
8471 ramUsageUsed);
8472 aCollector->registerBaseMetric (ramUsage);
8473
8474 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadUser, 0));
8475 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadUser,
8476 new pm::AggregateAvg()));
8477 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadUser,
8478 new pm::AggregateMin()));
8479 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadUser,
8480 new pm::AggregateMax()));
8481 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadKernel, 0));
8482 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadKernel,
8483 new pm::AggregateAvg()));
8484 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadKernel,
8485 new pm::AggregateMin()));
8486 aCollector->registerMetric (new pm::Metric (cpuLoad, cpuLoadKernel,
8487 new pm::AggregateMax()));
8488
8489 aCollector->registerMetric (new pm::Metric (ramUsage, ramUsageUsed, 0));
8490 aCollector->registerMetric (new pm::Metric (ramUsage, ramUsageUsed,
8491 new pm::AggregateAvg()));
8492 aCollector->registerMetric (new pm::Metric (ramUsage, ramUsageUsed,
8493 new pm::AggregateMin()));
8494 aCollector->registerMetric (new pm::Metric (ramUsage, ramUsageUsed,
8495 new pm::AggregateMax()));
8496};
8497
8498void Machine::unregisterMetrics (PerformanceCollector *aCollector, Machine *aMachine)
8499{
8500 aCollector->unregisterMetricsFor (aMachine);
8501 aCollector->unregisterBaseMetricsFor (aMachine);
8502};
8503#endif /* VBOX_WITH_RESOURCE_USAGE_API */
8504
8505
8506////////////////////////////////////////////////////////////////////////////////
8507
8508DEFINE_EMPTY_CTOR_DTOR(SessionMachine)
8509
8510HRESULT SessionMachine::FinalConstruct()
8511{
8512 LogFlowThisFunc(("\n"));
8513
8514#if defined(RT_OS_WINDOWS)
8515 mIPCSem = NULL;
8516#elif defined(RT_OS_OS2)
8517 mIPCSem = NULLHANDLE;
8518#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
8519 mIPCSem = -1;
8520#else
8521# error "Port me!"
8522#endif
8523
8524 return S_OK;
8525}
8526
8527void SessionMachine::FinalRelease()
8528{
8529 LogFlowThisFunc(("\n"));
8530
8531 uninit (Uninit::Unexpected);
8532}
8533
8534/**
8535 * @note Must be called only by Machine::openSession() from its own write lock.
8536 */
8537HRESULT SessionMachine::init (Machine *aMachine)
8538{
8539 LogFlowThisFuncEnter();
8540 LogFlowThisFunc(("mName={%ls}\n", aMachine->mUserData->mName.raw()));
8541
8542 AssertReturn(aMachine, E_INVALIDARG);
8543
8544 AssertReturn(aMachine->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
8545
8546 /* Enclose the state transition NotReady->InInit->Ready */
8547 AutoInitSpan autoInitSpan(this);
8548 AssertReturn(autoInitSpan.isOk(), E_FAIL);
8549
8550 /* create the interprocess semaphore */
8551#if defined(RT_OS_WINDOWS)
8552 mIPCSemName = aMachine->mData->m_strConfigFileFull;
8553 for (size_t i = 0; i < mIPCSemName.length(); i++)
8554 if (mIPCSemName[i] == '\\')
8555 mIPCSemName[i] = '/';
8556 mIPCSem = ::CreateMutex (NULL, FALSE, mIPCSemName);
8557 ComAssertMsgRet (mIPCSem,
8558 ("Cannot create IPC mutex '%ls', err=%d",
8559 mIPCSemName.raw(), ::GetLastError()),
8560 E_FAIL);
8561#elif defined(RT_OS_OS2)
8562 Utf8Str ipcSem = Utf8StrFmt ("\\SEM32\\VBOX\\VM\\{%RTuuid}",
8563 aMachine->mData->mUuid.raw());
8564 mIPCSemName = ipcSem;
8565 APIRET arc = ::DosCreateMutexSem ((PSZ) ipcSem.raw(), &mIPCSem, 0, FALSE);
8566 ComAssertMsgRet (arc == NO_ERROR,
8567 ("Cannot create IPC mutex '%s', arc=%ld",
8568 ipcSem.raw(), arc),
8569 E_FAIL);
8570#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
8571# ifdef VBOX_WITH_NEW_SYS_V_KEYGEN
8572# if defined(RT_OS_FREEBSD) && (HC_ARCH_BITS == 64)
8573 /** @todo Check that this still works correctly. */
8574 AssertCompileSize(key_t, 8);
8575# else
8576 AssertCompileSize(key_t, 4);
8577# endif
8578 key_t key;
8579 mIPCSem = -1;
8580 mIPCKey = "0";
8581 for (uint32_t i = 0; i < 1 << 24; i++)
8582 {
8583 key = ((uint32_t)'V' << 24) | i;
8584 int sem = ::semget (key, 1, S_IRUSR | S_IWUSR | IPC_CREAT | IPC_EXCL);
8585 if (sem >= 0 || (errno != EEXIST && errno != EACCES))
8586 {
8587 mIPCSem = sem;
8588 if (sem >= 0)
8589 mIPCKey = BstrFmt ("%u", key);
8590 break;
8591 }
8592 }
8593# else /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
8594 Utf8Str semName = aMachine->mData->m_strConfigFileFull;
8595 char *pszSemName = NULL;
8596 RTStrUtf8ToCurrentCP (&pszSemName, semName);
8597 key_t key = ::ftok (pszSemName, 'V');
8598 RTStrFree (pszSemName);
8599
8600 mIPCSem = ::semget (key, 1, S_IRWXU | S_IRWXG | S_IRWXO | IPC_CREAT);
8601# endif /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
8602
8603 int errnoSave = errno;
8604 if (mIPCSem < 0 && errnoSave == ENOSYS)
8605 {
8606 setError(E_FAIL,
8607 tr("Cannot create IPC semaphore. Most likely your host kernel lacks "
8608 "support for SysV IPC. Check the host kernel configuration for "
8609 "CONFIG_SYSVIPC=y"));
8610 return E_FAIL;
8611 }
8612 /* ENOSPC can also be the result of VBoxSVC crashes without properly freeing
8613 * the IPC semaphores */
8614 if (mIPCSem < 0 && errnoSave == ENOSPC)
8615 {
8616#ifdef RT_OS_LINUX
8617 setError(E_FAIL,
8618 tr("Cannot create IPC semaphore because the system limit for the "
8619 "maximum number of semaphore sets (SEMMNI), or the system wide "
8620 "maximum number of sempahores (SEMMNS) would be exceeded. The "
8621 "current set of SysV IPC semaphores can be determined from "
8622 "the file /proc/sysvipc/sem"));
8623#else
8624 setError(E_FAIL,
8625 tr("Cannot create IPC semaphore because the system-imposed limit "
8626 "on the maximum number of allowed semaphores or semaphore "
8627 "identifiers system-wide would be exceeded"));
8628#endif
8629 return E_FAIL;
8630 }
8631 ComAssertMsgRet (mIPCSem >= 0, ("Cannot create IPC semaphore, errno=%d", errnoSave),
8632 E_FAIL);
8633 /* set the initial value to 1 */
8634 int rv = ::semctl (mIPCSem, 0, SETVAL, 1);
8635 ComAssertMsgRet (rv == 0, ("Cannot init IPC semaphore, errno=%d", errno),
8636 E_FAIL);
8637#else
8638# error "Port me!"
8639#endif
8640
8641 /* memorize the peer Machine */
8642 unconst(mPeer) = aMachine;
8643 /* share the parent pointer */
8644 unconst(mParent) = aMachine->mParent;
8645
8646 /* take the pointers to data to share */
8647 mData.share (aMachine->mData);
8648 mSSData.share (aMachine->mSSData);
8649
8650 mUserData.share (aMachine->mUserData);
8651 mHWData.share (aMachine->mHWData);
8652 mMediaData.share(aMachine->mMediaData);
8653
8654 mStorageControllers.allocate();
8655 for (StorageControllerList::const_iterator it = aMachine->mStorageControllers->begin();
8656 it != aMachine->mStorageControllers->end();
8657 ++it)
8658 {
8659 ComObjPtr<StorageController> ctl;
8660 ctl.createObject();
8661 ctl->init(this, *it);
8662 mStorageControllers->push_back (ctl);
8663 }
8664
8665 unconst(mBIOSSettings).createObject();
8666 mBIOSSettings->init (this, aMachine->mBIOSSettings);
8667#ifdef VBOX_WITH_VRDP
8668 /* create another VRDPServer object that will be mutable */
8669 unconst(mVRDPServer).createObject();
8670 mVRDPServer->init (this, aMachine->mVRDPServer);
8671#endif
8672 /* create another audio adapter object that will be mutable */
8673 unconst(mAudioAdapter).createObject();
8674 mAudioAdapter->init (this, aMachine->mAudioAdapter);
8675 /* create a list of serial ports that will be mutable */
8676 for (ULONG slot = 0; slot < RT_ELEMENTS (mSerialPorts); slot ++)
8677 {
8678 unconst(mSerialPorts [slot]).createObject();
8679 mSerialPorts [slot]->init (this, aMachine->mSerialPorts [slot]);
8680 }
8681 /* create a list of parallel ports that will be mutable */
8682 for (ULONG slot = 0; slot < RT_ELEMENTS (mParallelPorts); slot ++)
8683 {
8684 unconst(mParallelPorts [slot]).createObject();
8685 mParallelPorts [slot]->init (this, aMachine->mParallelPorts [slot]);
8686 }
8687 /* create another USB controller object that will be mutable */
8688 unconst(mUSBController).createObject();
8689 mUSBController->init(this, aMachine->mUSBController);
8690
8691 /* create a list of network adapters that will be mutable */
8692 for (ULONG slot = 0; slot < RT_ELEMENTS(mNetworkAdapters); slot++)
8693 {
8694 unconst(mNetworkAdapters [slot]).createObject();
8695 mNetworkAdapters[slot]->init (this, aMachine->mNetworkAdapters [slot]);
8696 }
8697
8698 /* default is to delete saved state on Saved -> PoweredOff transition */
8699 mRemoveSavedState = true;
8700
8701 /* Confirm a successful initialization when it's the case */
8702 autoInitSpan.setSucceeded();
8703
8704 LogFlowThisFuncLeave();
8705 return S_OK;
8706}
8707
8708/**
8709 * Uninitializes this session object. If the reason is other than
8710 * Uninit::Unexpected, then this method MUST be called from #checkForDeath().
8711 *
8712 * @param aReason uninitialization reason
8713 *
8714 * @note Locks mParent + this object for writing.
8715 */
8716void SessionMachine::uninit (Uninit::Reason aReason)
8717{
8718 LogFlowThisFuncEnter();
8719 LogFlowThisFunc(("reason=%d\n", aReason));
8720
8721 /*
8722 * Strongly reference ourselves to prevent this object deletion after
8723 * mData->mSession.mMachine.setNull() below (which can release the last
8724 * reference and call the destructor). Important: this must be done before
8725 * accessing any members (and before AutoUninitSpan that does it as well).
8726 * This self reference will be released as the very last step on return.
8727 */
8728 ComObjPtr<SessionMachine> selfRef = this;
8729
8730 /* Enclose the state transition Ready->InUninit->NotReady */
8731 AutoUninitSpan autoUninitSpan(this);
8732 if (autoUninitSpan.uninitDone())
8733 {
8734 LogFlowThisFunc(("Already uninitialized\n"));
8735 LogFlowThisFuncLeave();
8736 return;
8737 }
8738
8739 if (autoUninitSpan.initFailed())
8740 {
8741 /* We've been called by init() because it's failed. It's not really
8742 * necessary (nor it's safe) to perform the regular uninit sequense
8743 * below, the following is enough.
8744 */
8745 LogFlowThisFunc(("Initialization failed.\n"));
8746#if defined(RT_OS_WINDOWS)
8747 if (mIPCSem)
8748 ::CloseHandle (mIPCSem);
8749 mIPCSem = NULL;
8750#elif defined(RT_OS_OS2)
8751 if (mIPCSem != NULLHANDLE)
8752 ::DosCloseMutexSem (mIPCSem);
8753 mIPCSem = NULLHANDLE;
8754#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
8755 if (mIPCSem >= 0)
8756 ::semctl (mIPCSem, 0, IPC_RMID);
8757 mIPCSem = -1;
8758# ifdef VBOX_WITH_NEW_SYS_V_KEYGEN
8759 mIPCKey = "0";
8760# endif /* VBOX_WITH_NEW_SYS_V_KEYGEN */
8761#else
8762# error "Port me!"
8763#endif
8764 uninitDataAndChildObjects();
8765 mData.free();
8766 unconst(mParent).setNull();
8767 unconst(mPeer).setNull();
8768 LogFlowThisFuncLeave();
8769 return;
8770 }
8771
8772 /* We need to lock this object in uninit() because the lock is shared
8773 * with mPeer (as well as data we modify below). mParent->addProcessToReap()
8774 * and others need mParent lock, and USB needs host lock. */
8775 AutoMultiWriteLock3 alock(mParent, mParent->host(), this COMMA_LOCKVAL_SRC_POS);
8776
8777#ifdef VBOX_WITH_RESOURCE_USAGE_API
8778 unregisterMetrics(mParent->performanceCollector(), mPeer);
8779#endif /* VBOX_WITH_RESOURCE_USAGE_API */
8780
8781 MachineState_T lastState = mData->mMachineState;
8782 NOREF(lastState);
8783
8784 if (aReason == Uninit::Abnormal)
8785 {
8786 LogWarningThisFunc(("ABNORMAL client termination! (wasBusy=%d)\n",
8787 Global::IsOnlineOrTransient (lastState)));
8788
8789 /* reset the state to Aborted */
8790 if (mData->mMachineState != MachineState_Aborted)
8791 setMachineState(MachineState_Aborted);
8792 }
8793
8794 // any machine settings modified?
8795 if (m_flModifications)
8796 {
8797 LogWarningThisFunc(("Discarding unsaved settings changes!\n"));
8798 rollback (false /* aNotify */);
8799 }
8800
8801 Assert(mSnapshotData.mStateFilePath.isEmpty() || !mSnapshotData.mSnapshot);
8802 if (!mSnapshotData.mStateFilePath.isEmpty())
8803 {
8804 LogWarningThisFunc(("canceling failed save state request!\n"));
8805 endSavingState(FALSE /* aSuccess */);
8806 }
8807 else if (!mSnapshotData.mSnapshot.isNull())
8808 {
8809 LogWarningThisFunc(("canceling untaken snapshot!\n"));
8810
8811 /* delete all differencing hard disks created (this will also attach
8812 * their parents back by rolling back mMediaData) */
8813 rollbackMedia();
8814 /* delete the saved state file (it might have been already created) */
8815 if (mSnapshotData.mSnapshot->stateFilePath().length())
8816 RTFileDelete(mSnapshotData.mSnapshot->stateFilePath().c_str());
8817
8818 mSnapshotData.mSnapshot->uninit();
8819 }
8820
8821#ifdef VBOX_WITH_USB
8822 /* release all captured USB devices */
8823 if (aReason == Uninit::Abnormal && Global::IsOnline (lastState))
8824 {
8825 /* Console::captureUSBDevices() is called in the VM process only after
8826 * setting the machine state to Starting or Restoring.
8827 * Console::detachAllUSBDevices() will be called upon successful
8828 * termination. So, we need to release USB devices only if there was
8829 * an abnormal termination of a running VM.
8830 *
8831 * This is identical to SessionMachine::DetachAllUSBDevices except
8832 * for the aAbnormal argument. */
8833 HRESULT rc = mUSBController->notifyProxy (false /* aInsertFilters */);
8834 AssertComRC(rc);
8835 NOREF (rc);
8836
8837 USBProxyService *service = mParent->host()->usbProxyService();
8838 if (service)
8839 service->detachAllDevicesFromVM (this, true /* aDone */, true /* aAbnormal */);
8840 }
8841#endif /* VBOX_WITH_USB */
8842
8843 if (!mData->mSession.mType.isNull())
8844 {
8845 /* mType is not null when this machine's process has been started by
8846 * VirtualBox::OpenRemoteSession(), therefore it is our child. We
8847 * need to queue the PID to reap the process (and avoid zombies on
8848 * Linux). */
8849 Assert (mData->mSession.mPid != NIL_RTPROCESS);
8850 mParent->addProcessToReap (mData->mSession.mPid);
8851 }
8852
8853 mData->mSession.mPid = NIL_RTPROCESS;
8854
8855 if (aReason == Uninit::Unexpected)
8856 {
8857 /* Uninitialization didn't come from #checkForDeath(), so tell the
8858 * client watcher thread to update the set of machines that have open
8859 * sessions. */
8860 mParent->updateClientWatcher();
8861 }
8862
8863 /* uninitialize all remote controls */
8864 if (mData->mSession.mRemoteControls.size())
8865 {
8866 LogFlowThisFunc(("Closing remote sessions (%d):\n",
8867 mData->mSession.mRemoteControls.size()));
8868
8869 Data::Session::RemoteControlList::iterator it =
8870 mData->mSession.mRemoteControls.begin();
8871 while (it != mData->mSession.mRemoteControls.end())
8872 {
8873 LogFlowThisFunc((" Calling remoteControl->Uninitialize()...\n"));
8874 HRESULT rc = (*it)->Uninitialize();
8875 LogFlowThisFunc((" remoteControl->Uninitialize() returned %08X\n", rc));
8876 if (FAILED (rc))
8877 LogWarningThisFunc(("Forgot to close the remote session?\n"));
8878 ++it;
8879 }
8880 mData->mSession.mRemoteControls.clear();
8881 }
8882
8883 /*
8884 * An expected uninitialization can come only from #checkForDeath().
8885 * Otherwise it means that something's got really wrong (for examlple,
8886 * the Session implementation has released the VirtualBox reference
8887 * before it triggered #OnSessionEnd(), or before releasing IPC semaphore,
8888 * etc). However, it's also possible, that the client releases the IPC
8889 * semaphore correctly (i.e. before it releases the VirtualBox reference),
8890 * but the VirtualBox release event comes first to the server process.
8891 * This case is practically possible, so we should not assert on an
8892 * unexpected uninit, just log a warning.
8893 */
8894
8895 if ((aReason == Uninit::Unexpected))
8896 LogWarningThisFunc(("Unexpected SessionMachine uninitialization!\n"));
8897
8898 if (aReason != Uninit::Normal)
8899 {
8900 mData->mSession.mDirectControl.setNull();
8901 }
8902 else
8903 {
8904 /* this must be null here (see #OnSessionEnd()) */
8905 Assert (mData->mSession.mDirectControl.isNull());
8906 Assert (mData->mSession.mState == SessionState_Closing);
8907 Assert (!mData->mSession.mProgress.isNull());
8908
8909 mData->mSession.mProgress->notifyComplete (S_OK);
8910 mData->mSession.mProgress.setNull();
8911 }
8912
8913 /* remove the association between the peer machine and this session machine */
8914 Assert (mData->mSession.mMachine == this ||
8915 aReason == Uninit::Unexpected);
8916
8917 /* reset the rest of session data */
8918 mData->mSession.mMachine.setNull();
8919 mData->mSession.mState = SessionState_Closed;
8920 mData->mSession.mType.setNull();
8921
8922 /* close the interprocess semaphore before leaving the exclusive lock */
8923#if defined(RT_OS_WINDOWS)
8924 if (mIPCSem)
8925 ::CloseHandle (mIPCSem);
8926 mIPCSem = NULL;
8927#elif defined(RT_OS_OS2)
8928 if (mIPCSem != NULLHANDLE)
8929 ::DosCloseMutexSem (mIPCSem);
8930 mIPCSem = NULLHANDLE;
8931#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
8932 if (mIPCSem >= 0)
8933 ::semctl (mIPCSem, 0, IPC_RMID);
8934 mIPCSem = -1;
8935# ifdef VBOX_WITH_NEW_SYS_V_KEYGEN
8936 mIPCKey = "0";
8937# endif /* VBOX_WITH_NEW_SYS_V_KEYGEN */
8938#else
8939# error "Port me!"
8940#endif
8941
8942 /* fire an event */
8943 mParent->onSessionStateChange (mData->mUuid, SessionState_Closed);
8944
8945 uninitDataAndChildObjects();
8946
8947 /* free the essential data structure last */
8948 mData.free();
8949
8950 /* leave the exclusive lock before setting the below two to NULL */
8951 alock.leave();
8952
8953 unconst(mParent).setNull();
8954 unconst(mPeer).setNull();
8955
8956 LogFlowThisFuncLeave();
8957}
8958
8959// util::Lockable interface
8960////////////////////////////////////////////////////////////////////////////////
8961
8962/**
8963 * Overrides VirtualBoxBase::lockHandle() in order to share the lock handle
8964 * with the primary Machine instance (mPeer).
8965 */
8966RWLockHandle *SessionMachine::lockHandle() const
8967{
8968 AssertReturn(!mPeer.isNull(), NULL);
8969 return mPeer->lockHandle();
8970}
8971
8972// IInternalMachineControl methods
8973////////////////////////////////////////////////////////////////////////////////
8974
8975/**
8976 * @note Locks this object for writing.
8977 */
8978STDMETHODIMP SessionMachine::SetRemoveSavedState(BOOL aRemove)
8979{
8980 AutoCaller autoCaller(this);
8981 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
8982
8983 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8984
8985 mRemoveSavedState = aRemove;
8986
8987 return S_OK;
8988}
8989
8990/**
8991 * @note Locks the same as #setMachineState() does.
8992 */
8993STDMETHODIMP SessionMachine::UpdateState (MachineState_T aMachineState)
8994{
8995 return setMachineState (aMachineState);
8996}
8997
8998/**
8999 * @note Locks this object for reading.
9000 */
9001STDMETHODIMP SessionMachine::GetIPCId (BSTR *aId)
9002{
9003 AutoCaller autoCaller(this);
9004 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9005
9006 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9007
9008#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
9009 mIPCSemName.cloneTo(aId);
9010 return S_OK;
9011#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
9012# ifdef VBOX_WITH_NEW_SYS_V_KEYGEN
9013 mIPCKey.cloneTo(aId);
9014# else /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
9015 mData->m_strConfigFileFull.cloneTo(aId);
9016# endif /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
9017 return S_OK;
9018#else
9019# error "Port me!"
9020#endif
9021}
9022
9023/**
9024 * Goes through the USB filters of the given machine to see if the given
9025 * device matches any filter or not.
9026 *
9027 * @note Locks the same as USBController::hasMatchingFilter() does.
9028 */
9029STDMETHODIMP SessionMachine::RunUSBDeviceFilters (IUSBDevice *aUSBDevice,
9030 BOOL *aMatched,
9031 ULONG *aMaskedIfs)
9032{
9033 LogFlowThisFunc(("\n"));
9034
9035 CheckComArgNotNull (aUSBDevice);
9036 CheckComArgOutPointerValid(aMatched);
9037
9038 AutoCaller autoCaller(this);
9039 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9040
9041#ifdef VBOX_WITH_USB
9042 *aMatched = mUSBController->hasMatchingFilter (aUSBDevice, aMaskedIfs);
9043#else
9044 NOREF(aUSBDevice);
9045 NOREF(aMaskedIfs);
9046 *aMatched = FALSE;
9047#endif
9048
9049 return S_OK;
9050}
9051
9052/**
9053 * @note Locks the same as Host::captureUSBDevice() does.
9054 */
9055STDMETHODIMP SessionMachine::CaptureUSBDevice (IN_BSTR aId)
9056{
9057 LogFlowThisFunc(("\n"));
9058
9059 AutoCaller autoCaller(this);
9060 AssertComRCReturnRC(autoCaller.rc());
9061
9062#ifdef VBOX_WITH_USB
9063 /* if captureDeviceForVM() fails, it must have set extended error info */
9064 MultiResult rc = mParent->host()->checkUSBProxyService();
9065 if (FAILED(rc)) return rc;
9066
9067 USBProxyService *service = mParent->host()->usbProxyService();
9068 AssertReturn(service, E_FAIL);
9069 return service->captureDeviceForVM (this, Guid(aId));
9070#else
9071 NOREF(aId);
9072 return E_NOTIMPL;
9073#endif
9074}
9075
9076/**
9077 * @note Locks the same as Host::detachUSBDevice() does.
9078 */
9079STDMETHODIMP SessionMachine::DetachUSBDevice (IN_BSTR aId, BOOL aDone)
9080{
9081 LogFlowThisFunc(("\n"));
9082
9083 AutoCaller autoCaller(this);
9084 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9085
9086#ifdef VBOX_WITH_USB
9087 USBProxyService *service = mParent->host()->usbProxyService();
9088 AssertReturn(service, E_FAIL);
9089 return service->detachDeviceFromVM (this, Guid(aId), !!aDone);
9090#else
9091 NOREF(aId);
9092 NOREF(aDone);
9093 return E_NOTIMPL;
9094#endif
9095}
9096
9097/**
9098 * Inserts all machine filters to the USB proxy service and then calls
9099 * Host::autoCaptureUSBDevices().
9100 *
9101 * Called by Console from the VM process upon VM startup.
9102 *
9103 * @note Locks what called methods lock.
9104 */
9105STDMETHODIMP SessionMachine::AutoCaptureUSBDevices()
9106{
9107 LogFlowThisFunc(("\n"));
9108
9109 AutoCaller autoCaller(this);
9110 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9111
9112#ifdef VBOX_WITH_USB
9113 HRESULT rc = mUSBController->notifyProxy (true /* aInsertFilters */);
9114 AssertComRC(rc);
9115 NOREF (rc);
9116
9117 USBProxyService *service = mParent->host()->usbProxyService();
9118 AssertReturn(service, E_FAIL);
9119 return service->autoCaptureDevicesForVM (this);
9120#else
9121 return S_OK;
9122#endif
9123}
9124
9125/**
9126 * Removes all machine filters from the USB proxy service and then calls
9127 * Host::detachAllUSBDevices().
9128 *
9129 * Called by Console from the VM process upon normal VM termination or by
9130 * SessionMachine::uninit() upon abnormal VM termination (from under the
9131 * Machine/SessionMachine lock).
9132 *
9133 * @note Locks what called methods lock.
9134 */
9135STDMETHODIMP SessionMachine::DetachAllUSBDevices (BOOL aDone)
9136{
9137 LogFlowThisFunc(("\n"));
9138
9139 AutoCaller autoCaller(this);
9140 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9141
9142#ifdef VBOX_WITH_USB
9143 HRESULT rc = mUSBController->notifyProxy (false /* aInsertFilters */);
9144 AssertComRC(rc);
9145 NOREF (rc);
9146
9147 USBProxyService *service = mParent->host()->usbProxyService();
9148 AssertReturn(service, E_FAIL);
9149 return service->detachAllDevicesFromVM (this, !!aDone, false /* aAbnormal */);
9150#else
9151 NOREF(aDone);
9152 return S_OK;
9153#endif
9154}
9155
9156/**
9157 * @note Locks this object for writing.
9158 */
9159STDMETHODIMP SessionMachine::OnSessionEnd (ISession *aSession,
9160 IProgress **aProgress)
9161{
9162 LogFlowThisFuncEnter();
9163
9164 AssertReturn(aSession, E_INVALIDARG);
9165 AssertReturn(aProgress, E_INVALIDARG);
9166
9167 AutoCaller autoCaller(this);
9168
9169 LogFlowThisFunc(("callerstate=%d\n", autoCaller.state()));
9170 /*
9171 * We don't assert below because it might happen that a non-direct session
9172 * informs us it is closed right after we've been uninitialized -- it's ok.
9173 */
9174 if (FAILED(autoCaller.rc())) return autoCaller.rc();
9175
9176 /* get IInternalSessionControl interface */
9177 ComPtr<IInternalSessionControl> control (aSession);
9178
9179 ComAssertRet (!control.isNull(), E_INVALIDARG);
9180
9181 /* Creating a Progress object requires the VirtualBox lock, and
9182 * thus locking it here is required by the lock order rules. */
9183 AutoMultiWriteLock2 alock(mParent->lockHandle(), this->lockHandle() COMMA_LOCKVAL_SRC_POS);
9184
9185 if (control.equalsTo(mData->mSession.mDirectControl))
9186 {
9187 ComAssertRet (aProgress, E_POINTER);
9188
9189 /* The direct session is being normally closed by the client process
9190 * ----------------------------------------------------------------- */
9191
9192 /* go to the closing state (essential for all open*Session() calls and
9193 * for #checkForDeath()) */
9194 Assert (mData->mSession.mState == SessionState_Open);
9195 mData->mSession.mState = SessionState_Closing;
9196
9197 /* set direct control to NULL to release the remote instance */
9198 mData->mSession.mDirectControl.setNull();
9199 LogFlowThisFunc(("Direct control is set to NULL\n"));
9200
9201 /* Create the progress object the client will use to wait until
9202 * #checkForDeath() is called to uninitialize this session object after
9203 * it releases the IPC semaphore. */
9204 ComObjPtr<Progress> progress;
9205 progress.createObject();
9206 progress->init (mParent, static_cast <IMachine *> (mPeer),
9207 Bstr (tr ("Closing session")), FALSE /* aCancelable */);
9208 progress.queryInterfaceTo(aProgress);
9209 mData->mSession.mProgress = progress;
9210 }
9211 else
9212 {
9213 /* the remote session is being normally closed */
9214 Data::Session::RemoteControlList::iterator it =
9215 mData->mSession.mRemoteControls.begin();
9216 while (it != mData->mSession.mRemoteControls.end())
9217 {
9218 if (control.equalsTo (*it))
9219 break;
9220 ++it;
9221 }
9222 BOOL found = it != mData->mSession.mRemoteControls.end();
9223 ComAssertMsgRet (found, ("The session is not found in the session list!"),
9224 E_INVALIDARG);
9225 mData->mSession.mRemoteControls.remove (*it);
9226 }
9227
9228 LogFlowThisFuncLeave();
9229 return S_OK;
9230}
9231
9232/**
9233 * @note Locks this object for writing.
9234 */
9235STDMETHODIMP SessionMachine::BeginSavingState (IProgress *aProgress, BSTR *aStateFilePath)
9236{
9237 LogFlowThisFuncEnter();
9238
9239 AssertReturn(aProgress, E_INVALIDARG);
9240 AssertReturn(aStateFilePath, E_POINTER);
9241
9242 AutoCaller autoCaller(this);
9243 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9244
9245 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9246
9247 AssertReturn( mData->mMachineState == MachineState_Paused
9248 && mSnapshotData.mLastState == MachineState_Null
9249 && mSnapshotData.mProgressId.isEmpty()
9250 && mSnapshotData.mStateFilePath.isEmpty(),
9251 E_FAIL);
9252
9253 /* memorize the progress ID and add it to the global collection */
9254 Bstr progressId;
9255 HRESULT rc = aProgress->COMGETTER(Id) (progressId.asOutParam());
9256 AssertComRCReturn (rc, rc);
9257 rc = mParent->addProgress (aProgress);
9258 AssertComRCReturn (rc, rc);
9259
9260 Bstr stateFilePath;
9261 /* stateFilePath is null when the machine is not running */
9262 if (mData->mMachineState == MachineState_Paused)
9263 {
9264 stateFilePath = Utf8StrFmt ("%ls%c{%RTuuid}.sav",
9265 mUserData->mSnapshotFolderFull.raw(),
9266 RTPATH_DELIMITER, mData->mUuid.raw());
9267 }
9268
9269 /* fill in the snapshot data */
9270 mSnapshotData.mLastState = mData->mMachineState;
9271 mSnapshotData.mProgressId = Guid(progressId);
9272 mSnapshotData.mStateFilePath = stateFilePath;
9273
9274 /* set the state to Saving (this is expected by Console::SaveState()) */
9275 setMachineState (MachineState_Saving);
9276
9277 stateFilePath.cloneTo(aStateFilePath);
9278
9279 return S_OK;
9280}
9281
9282/**
9283 * @note Locks mParent + this object for writing.
9284 */
9285STDMETHODIMP SessionMachine::EndSavingState (BOOL aSuccess)
9286{
9287 LogFlowThisFunc(("\n"));
9288
9289 AutoCaller autoCaller(this);
9290 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9291
9292 /* endSavingState() need mParent lock */
9293 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
9294
9295 AssertReturn( mData->mMachineState == MachineState_Saving
9296 && mSnapshotData.mLastState != MachineState_Null
9297 && !mSnapshotData.mProgressId.isEmpty()
9298 && !mSnapshotData.mStateFilePath.isEmpty(),
9299 E_FAIL);
9300
9301 /*
9302 * on success, set the state to Saved;
9303 * on failure, set the state to the state we had when BeginSavingState() was
9304 * called (this is expected by Console::SaveState() and
9305 * Console::saveStateThread())
9306 */
9307 if (aSuccess)
9308 setMachineState (MachineState_Saved);
9309 else
9310 setMachineState (mSnapshotData.mLastState);
9311
9312 return endSavingState (aSuccess);
9313}
9314
9315/**
9316 * @note Locks this object for writing.
9317 */
9318STDMETHODIMP SessionMachine::AdoptSavedState (IN_BSTR aSavedStateFile)
9319{
9320 LogFlowThisFunc(("\n"));
9321
9322 CheckComArgStrNotEmptyOrNull(aSavedStateFile);
9323
9324 AutoCaller autoCaller(this);
9325 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9326
9327 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9328
9329 AssertReturn( mData->mMachineState == MachineState_PoweredOff
9330 || mData->mMachineState == MachineState_Teleported
9331 || mData->mMachineState == MachineState_Aborted
9332 , E_FAIL); /** @todo setError. */
9333
9334 Utf8Str stateFilePathFull = aSavedStateFile;
9335 int vrc = calculateFullPath(stateFilePathFull, stateFilePathFull);
9336 if (RT_FAILURE(vrc))
9337 return setError(VBOX_E_FILE_ERROR,
9338 tr("Invalid saved state file path '%ls' (%Rrc)"),
9339 aSavedStateFile,
9340 vrc);
9341
9342 mSSData->mStateFilePath = stateFilePathFull;
9343
9344 /* The below setMachineState() will detect the state transition and will
9345 * update the settings file */
9346
9347 return setMachineState (MachineState_Saved);
9348}
9349
9350STDMETHODIMP SessionMachine::PullGuestProperties(ComSafeArrayOut(BSTR, aNames),
9351 ComSafeArrayOut(BSTR, aValues),
9352 ComSafeArrayOut(ULONG64, aTimestamps),
9353 ComSafeArrayOut(BSTR, aFlags))
9354{
9355 LogFlowThisFunc(("\n"));
9356
9357#ifdef VBOX_WITH_GUEST_PROPS
9358 using namespace guestProp;
9359
9360 AutoCaller autoCaller(this);
9361 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9362
9363 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9364
9365 AssertReturn(!ComSafeArrayOutIsNull(aNames), E_POINTER);
9366 AssertReturn(!ComSafeArrayOutIsNull(aValues), E_POINTER);
9367 AssertReturn(!ComSafeArrayOutIsNull(aTimestamps), E_POINTER);
9368 AssertReturn(!ComSafeArrayOutIsNull(aFlags), E_POINTER);
9369
9370 size_t cEntries = mHWData->mGuestProperties.size();
9371 com::SafeArray<BSTR> names (cEntries);
9372 com::SafeArray<BSTR> values (cEntries);
9373 com::SafeArray<ULONG64> timestamps (cEntries);
9374 com::SafeArray<BSTR> flags (cEntries);
9375 unsigned i = 0;
9376 for (HWData::GuestPropertyList::iterator it = mHWData->mGuestProperties.begin();
9377 it != mHWData->mGuestProperties.end();
9378 ++it)
9379 {
9380 char szFlags[MAX_FLAGS_LEN + 1];
9381 it->strName.cloneTo(&names[i]);
9382 it->strValue.cloneTo(&values[i]);
9383 timestamps[i] = it->mTimestamp;
9384 /* If it is NULL, keep it NULL. */
9385 if (it->mFlags)
9386 {
9387 writeFlags(it->mFlags, szFlags);
9388 Bstr(szFlags).cloneTo(&flags[i]);
9389 }
9390 else
9391 flags[i] = NULL;
9392 ++i;
9393 }
9394 names.detachTo(ComSafeArrayOutArg(aNames));
9395 values.detachTo(ComSafeArrayOutArg(aValues));
9396 timestamps.detachTo(ComSafeArrayOutArg(aTimestamps));
9397 flags.detachTo(ComSafeArrayOutArg(aFlags));
9398 mHWData->mPropertyServiceActive = true;
9399 return S_OK;
9400#else
9401 ReturnComNotImplemented();
9402#endif
9403}
9404
9405STDMETHODIMP SessionMachine::PushGuestProperties(ComSafeArrayIn(IN_BSTR, aNames),
9406 ComSafeArrayIn(IN_BSTR, aValues),
9407 ComSafeArrayIn(ULONG64, aTimestamps),
9408 ComSafeArrayIn(IN_BSTR, aFlags))
9409{
9410 LogFlowThisFunc(("\n"));
9411
9412#ifdef VBOX_WITH_GUEST_PROPS
9413 using namespace guestProp;
9414
9415 AssertReturn(!ComSafeArrayInIsNull(aNames), E_POINTER);
9416 AssertReturn(!ComSafeArrayInIsNull(aValues), E_POINTER);
9417 AssertReturn(!ComSafeArrayInIsNull(aTimestamps), E_POINTER);
9418 AssertReturn(!ComSafeArrayInIsNull(aFlags), E_POINTER);
9419
9420 AutoCaller autoCaller(this);
9421 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
9422
9423 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9424
9425 /*
9426 * Temporarily reset the registered flag, so that our machine state
9427 * changes (i.e. mHWData.backup()) succeed. (isMutable() used in all
9428 * setters will return FALSE for a Machine instance if mRegistered is TRUE).
9429 *
9430 * This is copied from registeredInit(), and may or may not be the right
9431 * way to handle this.
9432 *
9433 * @todo r=dj review this, this gets called during machine power-down when
9434 * we have already saved the machine settings, there's no need to do this
9435 * twice.
9436 */
9437 Assert(mData->mRegistered);
9438 mData->mRegistered = FALSE;
9439
9440 HRESULT rc = checkStateDependency(MutableStateDep);
9441 AssertLogRelMsgReturn(SUCCEEDED(rc), ("%Rhrc\n", rc), rc);
9442
9443 com::SafeArray<IN_BSTR> names( ComSafeArrayInArg(aNames));
9444 com::SafeArray<IN_BSTR> values( ComSafeArrayInArg(aValues));
9445 com::SafeArray<ULONG64> timestamps(ComSafeArrayInArg(aTimestamps));
9446 com::SafeArray<IN_BSTR> flags( ComSafeArrayInArg(aFlags));
9447
9448 DiscardSettings();
9449 setModified(IsModified_MachineData);
9450 mHWData.backup();
9451
9452 mHWData->mGuestProperties.erase(mHWData->mGuestProperties.begin(),
9453 mHWData->mGuestProperties.end());
9454 for (unsigned i = 0; i < names.size(); ++i)
9455 {
9456 uint32_t fFlags = NILFLAG;
9457 validateFlags(Utf8Str(flags[i]).raw(), &fFlags);
9458 HWData::GuestProperty property = { names[i], values[i], timestamps[i], fFlags };
9459 mHWData->mGuestProperties.push_back(property);
9460 }
9461
9462 mHWData->mPropertyServiceActive = false;
9463
9464 alock.release();
9465 SaveSettings();
9466
9467 /* Restore the mRegistered flag. */
9468 alock.acquire();
9469 mData->mRegistered = TRUE;
9470
9471 return S_OK;
9472#else
9473 ReturnComNotImplemented();
9474#endif
9475}
9476
9477STDMETHODIMP SessionMachine::PushGuestProperty(IN_BSTR aName,
9478 IN_BSTR aValue,
9479 ULONG64 aTimestamp,
9480 IN_BSTR aFlags)
9481{
9482 LogFlowThisFunc(("\n"));
9483
9484#ifdef VBOX_WITH_GUEST_PROPS
9485 using namespace guestProp;
9486
9487 CheckComArgNotNull(aName);
9488 if (aValue != NULL && (!VALID_PTR(aValue) || !VALID_PTR(aFlags)))
9489 return E_POINTER; /* aValue can be NULL to indicate deletion */
9490
9491 try
9492 {
9493 /*
9494 * Convert input up front.
9495 */
9496 Utf8Str utf8Name(aName);
9497 uint32_t fFlags = NILFLAG;
9498 if (aFlags)
9499 {
9500 Utf8Str utf8Flags(aFlags);
9501 int vrc = validateFlags(utf8Flags.raw(), &fFlags);
9502 AssertRCReturn(vrc, E_INVALIDARG);
9503 }
9504
9505 /*
9506 * Now grab the object lock, validate the state and do the update.
9507 */
9508 AutoCaller autoCaller(this);
9509 if (FAILED(autoCaller.rc())) return autoCaller.rc();
9510
9511 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9512
9513 AssertReturn(mHWData->mPropertyServiceActive, VBOX_E_INVALID_OBJECT_STATE);
9514 switch (mData->mMachineState)
9515 {
9516 case MachineState_Paused:
9517 case MachineState_Running:
9518 case MachineState_Teleporting:
9519 case MachineState_TeleportingPausedVM:
9520 case MachineState_LiveSnapshotting:
9521 case MachineState_Saving:
9522 break;
9523
9524 default:
9525 AssertMsgFailedReturn(("%s\n", Global::stringifyMachineState(mData->mMachineState)),
9526 VBOX_E_INVALID_VM_STATE);
9527 }
9528
9529 setModified(IsModified_MachineData);
9530 mHWData.backup();
9531
9532 /** @todo r=bird: The careful memory handling doesn't work out here because
9533 * the catch block won't undo any damange we've done. So, if push_back throws
9534 * bad_alloc then you've lost the value.
9535 *
9536 * Another thing. Doing a linear search here isn't extremely efficient, esp.
9537 * since values that changes actually bubbles to the end of the list. Using
9538 * something that has an efficient lookup and can tollerate a bit of updates
9539 * would be nice. RTStrSpace is one suggestion (it's not perfect). Some
9540 * combination of RTStrCache (for sharing names and getting uniqueness into
9541 * the bargain) and hash/tree is another. */
9542 for (HWData::GuestPropertyList::iterator iter = mHWData->mGuestProperties.begin();
9543 iter != mHWData->mGuestProperties.end();
9544 ++iter)
9545 if (utf8Name == iter->strName)
9546 {
9547 mHWData->mGuestProperties.erase(iter);
9548 break;
9549 }
9550 if (aValue != NULL)
9551 {
9552 HWData::GuestProperty property = { aName, aValue, aTimestamp, fFlags };
9553 mHWData->mGuestProperties.push_back(property);
9554 }
9555
9556 /*
9557 * Send a callback notification if appropriate
9558 */
9559 if ( mHWData->mGuestPropertyNotificationPatterns.isEmpty()
9560 || RTStrSimplePatternMultiMatch(mHWData->mGuestPropertyNotificationPatterns.raw(),
9561 RTSTR_MAX,
9562 utf8Name.raw(),
9563 RTSTR_MAX, NULL)
9564 )
9565 {
9566 alock.leave();
9567
9568 mParent->onGuestPropertyChange(mData->mUuid,
9569 aName,
9570 aValue,
9571 aFlags);
9572 }
9573 }
9574 catch (...)
9575 {
9576 return VirtualBox::handleUnexpectedExceptions(RT_SRC_POS);
9577 }
9578 return S_OK;
9579#else
9580 ReturnComNotImplemented();
9581#endif
9582}
9583
9584// public methods only for internal purposes
9585/////////////////////////////////////////////////////////////////////////////
9586
9587/**
9588 * Called from the client watcher thread to check for expected or unexpected
9589 * death of the client process that has a direct session to this machine.
9590 *
9591 * On Win32 and on OS/2, this method is called only when we've got the
9592 * mutex (i.e. the client has either died or terminated normally) so it always
9593 * returns @c true (the client is terminated, the session machine is
9594 * uninitialized).
9595 *
9596 * On other platforms, the method returns @c true if the client process has
9597 * terminated normally or abnormally and the session machine was uninitialized,
9598 * and @c false if the client process is still alive.
9599 *
9600 * @note Locks this object for writing.
9601 */
9602bool SessionMachine::checkForDeath()
9603{
9604 Uninit::Reason reason;
9605 bool terminated = false;
9606
9607 /* Enclose autoCaller with a block because calling uninit() from under it
9608 * will deadlock. */
9609 {
9610 AutoCaller autoCaller(this);
9611 if (!autoCaller.isOk())
9612 {
9613 /* return true if not ready, to cause the client watcher to exclude
9614 * the corresponding session from watching */
9615 LogFlowThisFunc(("Already uninitialized!\n"));
9616 return true;
9617 }
9618
9619 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
9620
9621 /* Determine the reason of death: if the session state is Closing here,
9622 * everything is fine. Otherwise it means that the client did not call
9623 * OnSessionEnd() before it released the IPC semaphore. This may happen
9624 * either because the client process has abnormally terminated, or
9625 * because it simply forgot to call ISession::Close() before exiting. We
9626 * threat the latter also as an abnormal termination (see
9627 * Session::uninit() for details). */
9628 reason = mData->mSession.mState == SessionState_Closing ?
9629 Uninit::Normal :
9630 Uninit::Abnormal;
9631
9632#if defined(RT_OS_WINDOWS)
9633
9634 AssertMsg (mIPCSem, ("semaphore must be created"));
9635
9636 /* release the IPC mutex */
9637 ::ReleaseMutex (mIPCSem);
9638
9639 terminated = true;
9640
9641#elif defined(RT_OS_OS2)
9642
9643 AssertMsg (mIPCSem, ("semaphore must be created"));
9644
9645 /* release the IPC mutex */
9646 ::DosReleaseMutexSem (mIPCSem);
9647
9648 terminated = true;
9649
9650#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
9651
9652 AssertMsg (mIPCSem >= 0, ("semaphore must be created"));
9653
9654 int val = ::semctl (mIPCSem, 0, GETVAL);
9655 if (val > 0)
9656 {
9657 /* the semaphore is signaled, meaning the session is terminated */
9658 terminated = true;
9659 }
9660
9661#else
9662# error "Port me!"
9663#endif
9664
9665 } /* AutoCaller block */
9666
9667 if (terminated)
9668 uninit (reason);
9669
9670 return terminated;
9671}
9672
9673/**
9674 * @note Locks this object for reading.
9675 */
9676HRESULT SessionMachine::onNetworkAdapterChange (INetworkAdapter *networkAdapter, BOOL changeAdapter)
9677{
9678 LogFlowThisFunc(("\n"));
9679
9680 AutoCaller autoCaller(this);
9681 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9682
9683 ComPtr<IInternalSessionControl> directControl;
9684 {
9685 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9686 directControl = mData->mSession.mDirectControl;
9687 }
9688
9689 /* ignore notifications sent after #OnSessionEnd() is called */
9690 if (!directControl)
9691 return S_OK;
9692
9693 return directControl->OnNetworkAdapterChange (networkAdapter, changeAdapter);
9694}
9695
9696/**
9697 * @note Locks this object for reading.
9698 */
9699HRESULT SessionMachine::onSerialPortChange (ISerialPort *serialPort)
9700{
9701 LogFlowThisFunc(("\n"));
9702
9703 AutoCaller autoCaller(this);
9704 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9705
9706 ComPtr<IInternalSessionControl> directControl;
9707 {
9708 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9709 directControl = mData->mSession.mDirectControl;
9710 }
9711
9712 /* ignore notifications sent after #OnSessionEnd() is called */
9713 if (!directControl)
9714 return S_OK;
9715
9716 return directControl->OnSerialPortChange (serialPort);
9717}
9718
9719/**
9720 * @note Locks this object for reading.
9721 */
9722HRESULT SessionMachine::onParallelPortChange (IParallelPort *parallelPort)
9723{
9724 LogFlowThisFunc(("\n"));
9725
9726 AutoCaller autoCaller(this);
9727 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
9728
9729 ComPtr<IInternalSessionControl> directControl;
9730 {
9731 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9732 directControl = mData->mSession.mDirectControl;
9733 }
9734
9735 /* ignore notifications sent after #OnSessionEnd() is called */
9736 if (!directControl)
9737 return S_OK;
9738
9739 return directControl->OnParallelPortChange(parallelPort);
9740}
9741
9742/**
9743 * @note Locks this object for reading.
9744 */
9745HRESULT SessionMachine::onStorageControllerChange ()
9746{
9747 LogFlowThisFunc(("\n"));
9748
9749 AutoCaller autoCaller(this);
9750 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9751
9752 ComPtr<IInternalSessionControl> directControl;
9753 {
9754 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9755 directControl = mData->mSession.mDirectControl;
9756 }
9757
9758 /* ignore notifications sent after #OnSessionEnd() is called */
9759 if (!directControl)
9760 return S_OK;
9761
9762 return directControl->OnStorageControllerChange ();
9763}
9764
9765/**
9766 * @note Locks this object for reading.
9767 */
9768HRESULT SessionMachine::onMediumChange(IMediumAttachment *aAttachment, BOOL aForce)
9769{
9770 LogFlowThisFunc(("\n"));
9771
9772 AutoCaller autoCaller(this);
9773 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9774
9775 ComPtr<IInternalSessionControl> directControl;
9776 {
9777 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9778 directControl = mData->mSession.mDirectControl;
9779 }
9780
9781 /* ignore notifications sent after #OnSessionEnd() is called */
9782 if (!directControl)
9783 return S_OK;
9784
9785 return directControl->OnMediumChange(aAttachment, aForce);
9786}
9787
9788/**
9789 * @note Locks this object for reading.
9790 */
9791HRESULT SessionMachine::onCPUChange(ULONG aCPU, BOOL aRemove)
9792{
9793 LogFlowThisFunc(("\n"));
9794
9795 AutoCaller autoCaller(this);
9796 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9797
9798 ComPtr<IInternalSessionControl> directControl;
9799 {
9800 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9801 directControl = mData->mSession.mDirectControl;
9802 }
9803
9804 /* ignore notifications sent after #OnSessionEnd() is called */
9805 if (!directControl)
9806 return S_OK;
9807
9808 return directControl->OnCPUChange(aCPU, aRemove);
9809}
9810
9811/**
9812 * @note Locks this object for reading.
9813 */
9814HRESULT SessionMachine::onVRDPServerChange()
9815{
9816 LogFlowThisFunc(("\n"));
9817
9818 AutoCaller autoCaller(this);
9819 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9820
9821 ComPtr<IInternalSessionControl> directControl;
9822 {
9823 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9824 directControl = mData->mSession.mDirectControl;
9825 }
9826
9827 /* ignore notifications sent after #OnSessionEnd() is called */
9828 if (!directControl)
9829 return S_OK;
9830
9831 return directControl->OnVRDPServerChange();
9832}
9833
9834/**
9835 * @note Locks this object for reading.
9836 */
9837HRESULT SessionMachine::onUSBControllerChange()
9838{
9839 LogFlowThisFunc(("\n"));
9840
9841 AutoCaller autoCaller(this);
9842 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9843
9844 ComPtr<IInternalSessionControl> directControl;
9845 {
9846 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9847 directControl = mData->mSession.mDirectControl;
9848 }
9849
9850 /* ignore notifications sent after #OnSessionEnd() is called */
9851 if (!directControl)
9852 return S_OK;
9853
9854 return directControl->OnUSBControllerChange();
9855}
9856
9857/**
9858 * @note Locks this object for reading.
9859 */
9860HRESULT SessionMachine::onSharedFolderChange()
9861{
9862 LogFlowThisFunc(("\n"));
9863
9864 AutoCaller autoCaller(this);
9865 AssertComRCReturnRC(autoCaller.rc());
9866
9867 ComPtr<IInternalSessionControl> directControl;
9868 {
9869 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9870 directControl = mData->mSession.mDirectControl;
9871 }
9872
9873 /* ignore notifications sent after #OnSessionEnd() is called */
9874 if (!directControl)
9875 return S_OK;
9876
9877 return directControl->OnSharedFolderChange (FALSE /* aGlobal */);
9878}
9879
9880/**
9881 * Returns @c true if this machine's USB controller reports it has a matching
9882 * filter for the given USB device and @c false otherwise.
9883 *
9884 * @note Caller must have requested machine read lock.
9885 */
9886bool SessionMachine::hasMatchingUSBFilter (const ComObjPtr<HostUSBDevice> &aDevice, ULONG *aMaskedIfs)
9887{
9888 AutoCaller autoCaller(this);
9889 /* silently return if not ready -- this method may be called after the
9890 * direct machine session has been called */
9891 if (!autoCaller.isOk())
9892 return false;
9893
9894 AssertReturn(isWriteLockOnCurrentThread(), false);
9895
9896#ifdef VBOX_WITH_USB
9897 switch (mData->mMachineState)
9898 {
9899 case MachineState_Starting:
9900 case MachineState_Restoring:
9901 case MachineState_TeleportingIn:
9902 case MachineState_Paused:
9903 case MachineState_Running:
9904 /** @todo Live Migration: snapshoting & teleporting. Need to fend things of
9905 * elsewhere... */
9906 return mUSBController->hasMatchingFilter (aDevice, aMaskedIfs);
9907 default: break;
9908 }
9909#else
9910 NOREF(aDevice);
9911 NOREF(aMaskedIfs);
9912#endif
9913 return false;
9914}
9915
9916/**
9917 * @note The calls shall hold no locks. Will temporarily lock this object for reading.
9918 */
9919HRESULT SessionMachine::onUSBDeviceAttach (IUSBDevice *aDevice,
9920 IVirtualBoxErrorInfo *aError,
9921 ULONG aMaskedIfs)
9922{
9923 LogFlowThisFunc(("\n"));
9924
9925 AutoCaller autoCaller(this);
9926
9927 /* This notification may happen after the machine object has been
9928 * uninitialized (the session was closed), so don't assert. */
9929 if (FAILED(autoCaller.rc())) return autoCaller.rc();
9930
9931 ComPtr<IInternalSessionControl> directControl;
9932 {
9933 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9934 directControl = mData->mSession.mDirectControl;
9935 }
9936
9937 /* fail on notifications sent after #OnSessionEnd() is called, it is
9938 * expected by the caller */
9939 if (!directControl)
9940 return E_FAIL;
9941
9942 /* No locks should be held at this point. */
9943 AssertMsg (RTLockValidatorWriteLockGetCount (RTThreadSelf()) == 0, ("%d\n", RTLockValidatorWriteLockGetCount (RTThreadSelf())));
9944 AssertMsg (RTLockValidatorReadLockGetCount (RTThreadSelf()) == 0, ("%d\n", RTLockValidatorReadLockGetCount (RTThreadSelf())));
9945
9946 return directControl->OnUSBDeviceAttach (aDevice, aError, aMaskedIfs);
9947}
9948
9949/**
9950 * @note The calls shall hold no locks. Will temporarily lock this object for reading.
9951 */
9952HRESULT SessionMachine::onUSBDeviceDetach (IN_BSTR aId,
9953 IVirtualBoxErrorInfo *aError)
9954{
9955 LogFlowThisFunc(("\n"));
9956
9957 AutoCaller autoCaller(this);
9958
9959 /* This notification may happen after the machine object has been
9960 * uninitialized (the session was closed), so don't assert. */
9961 if (FAILED(autoCaller.rc())) return autoCaller.rc();
9962
9963 ComPtr<IInternalSessionControl> directControl;
9964 {
9965 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9966 directControl = mData->mSession.mDirectControl;
9967 }
9968
9969 /* fail on notifications sent after #OnSessionEnd() is called, it is
9970 * expected by the caller */
9971 if (!directControl)
9972 return E_FAIL;
9973
9974 /* No locks should be held at this point. */
9975 AssertMsg (RTLockValidatorWriteLockGetCount (RTThreadSelf()) == 0, ("%d\n", RTLockValidatorWriteLockGetCount (RTThreadSelf())));
9976 AssertMsg (RTLockValidatorReadLockGetCount (RTThreadSelf()) == 0, ("%d\n", RTLockValidatorReadLockGetCount (RTThreadSelf())));
9977
9978 return directControl->OnUSBDeviceDetach (aId, aError);
9979}
9980
9981// protected methods
9982/////////////////////////////////////////////////////////////////////////////
9983
9984/**
9985 * Helper method to finalize saving the state.
9986 *
9987 * @note Must be called from under this object's lock.
9988 *
9989 * @param aSuccess TRUE if the snapshot has been taken successfully
9990 *
9991 * @note Locks mParent + this objects for writing.
9992 */
9993HRESULT SessionMachine::endSavingState (BOOL aSuccess)
9994{
9995 LogFlowThisFuncEnter();
9996
9997 AutoCaller autoCaller(this);
9998 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
9999
10000 /* saveSettings() needs mParent lock */
10001 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
10002
10003 HRESULT rc = S_OK;
10004
10005 if (aSuccess)
10006 {
10007 mSSData->mStateFilePath = mSnapshotData.mStateFilePath;
10008
10009 /* save all VM settings */
10010 rc = saveSettings();
10011 }
10012 else
10013 {
10014 /* delete the saved state file (it might have been already created) */
10015 RTFileDelete(mSnapshotData.mStateFilePath.c_str());
10016 }
10017
10018 /* remove the completed progress object */
10019 mParent->removeProgress(mSnapshotData.mProgressId);
10020
10021 /* clear out the temporary saved state data */
10022 mSnapshotData.mLastState = MachineState_Null;
10023 mSnapshotData.mProgressId.clear();
10024 mSnapshotData.mStateFilePath.setNull();
10025
10026 LogFlowThisFuncLeave();
10027 return rc;
10028}
10029
10030/**
10031 * Locks the attached media.
10032 *
10033 * All attached hard disks are locked for writing and DVD/floppy are locked for
10034 * reading. Parents of attached hard disks (if any) are locked for reading.
10035 *
10036 * This method also performs accessibility check of all media it locks: if some
10037 * media is inaccessible, the method will return a failure and a bunch of
10038 * extended error info objects per each inaccessible medium.
10039 *
10040 * Note that this method is atomic: if it returns a success, all media are
10041 * locked as described above; on failure no media is locked at all (all
10042 * succeeded individual locks will be undone).
10043 *
10044 * This method is intended to be called when the machine is in Starting or
10045 * Restoring state and asserts otherwise.
10046 *
10047 * The locks made by this method must be undone by calling #unlockMedia() when
10048 * no more needed.
10049 */
10050HRESULT SessionMachine::lockMedia()
10051{
10052 AutoCaller autoCaller(this);
10053 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
10054
10055 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
10056
10057 AssertReturn( mData->mMachineState == MachineState_Starting
10058 || mData->mMachineState == MachineState_Restoring
10059 || mData->mMachineState == MachineState_TeleportingIn, E_FAIL);
10060
10061 try
10062 {
10063 HRESULT rc = S_OK;
10064
10065 ErrorInfoKeeper eik(true /* aIsNull */);
10066 MultiResult mrc(S_OK);
10067
10068 /* Lock all medium objects attached to the VM.
10069 * Get status for inaccessible media as well. */
10070 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
10071 it != mMediaData->mAttachments.end();
10072 ++it)
10073 {
10074 DeviceType_T devType = (*it)->getType();
10075 ComObjPtr<Medium> medium = (*it)->getMedium();
10076
10077 bool first = true;
10078
10079 /** @todo split out the media locking, and put it into
10080 * MediumImpl.cpp, as it needs this functionality too. */
10081 while (!medium.isNull())
10082 {
10083 MediumState_T mediumState = medium->getState();
10084
10085 /* accessibility check must be first, otherwise locking
10086 * interferes with getting the medium state. */
10087 if (mediumState == MediumState_Inaccessible)
10088 {
10089 rc = medium->RefreshState(&mediumState);
10090 if (FAILED(rc)) throw rc;
10091
10092 if (mediumState == MediumState_Inaccessible)
10093 {
10094 Bstr error;
10095 rc = medium->COMGETTER(LastAccessError)(error.asOutParam());
10096 if (FAILED(rc)) throw rc;
10097
10098 Bstr loc;
10099 rc = medium->COMGETTER(Location)(loc.asOutParam());
10100 if (FAILED(rc)) throw rc;
10101
10102 /* collect multiple errors */
10103 eik.restore();
10104
10105 /* be in sync with MediumBase::setStateError() */
10106 Assert(!error.isEmpty());
10107 mrc = setError(E_FAIL,
10108 tr("Medium '%ls' is not accessible. %ls"),
10109 loc.raw(),
10110 error.raw());
10111
10112 eik.fetch();
10113 }
10114 }
10115
10116 if (first)
10117 {
10118 if (devType != DeviceType_DVD)
10119 {
10120 /* HardDisk and Floppy medium must be locked for writing */
10121 rc = medium->LockWrite(NULL);
10122 if (FAILED(rc)) throw rc;
10123 }
10124 else
10125 {
10126 /* DVD medium must be locked for reading */
10127 rc = medium->LockRead(NULL);
10128 if (FAILED(rc)) throw rc;
10129 }
10130
10131 mData->mSession.mLockedMedia.push_back(
10132 Data::Session::LockedMedia::value_type(
10133 ComPtr<IMedium>(medium), true));
10134
10135 first = false;
10136 }
10137 else
10138 {
10139 rc = medium->LockRead(NULL);
10140 if (FAILED(rc)) throw rc;
10141
10142 mData->mSession.mLockedMedia.push_back(
10143 Data::Session::LockedMedia::value_type(
10144 ComPtr<IMedium>(medium), false));
10145 }
10146
10147
10148 /* no locks or callers here since there should be no way to
10149 * change the hard disk parent at this point (as it is still
10150 * attached to the machine) */
10151 medium = medium->getParent();
10152 }
10153 }
10154
10155 /* @todo r=dj is this correct? first restoring the eik and then throwing? */
10156 eik.restore();
10157 HRESULT rc2 = (HRESULT)mrc;
10158 if (FAILED(rc2)) throw rc2;
10159 }
10160 catch (HRESULT aRC)
10161 {
10162 /* Unlock all locked media on failure */
10163 unlockMedia();
10164 return aRC;
10165 }
10166
10167 return S_OK;
10168}
10169
10170/**
10171 * Undoes the locks made by by #lockMedia().
10172 */
10173void SessionMachine::unlockMedia()
10174{
10175 AutoCaller autoCaller(this);
10176 AssertComRCReturnVoid (autoCaller.rc());
10177
10178 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
10179
10180 /* we may be holding important error info on the current thread;
10181 * preserve it */
10182 ErrorInfoKeeper eik;
10183
10184 HRESULT rc = S_OK;
10185
10186 for (Data::Session::LockedMedia::const_iterator
10187 it = mData->mSession.mLockedMedia.begin();
10188 it != mData->mSession.mLockedMedia.end(); ++it)
10189 {
10190 MediumState_T state;
10191 if (it->second)
10192 rc = it->first->UnlockWrite (&state);
10193 else
10194 rc = it->first->UnlockRead (&state);
10195
10196 /* The second can happen if an object was re-locked in
10197 * Machine::fixupMedia(). The last can happen when e.g a DVD/Floppy
10198 * image was unmounted at runtime. */
10199 Assert (SUCCEEDED(rc) || state == MediumState_LockedRead || state == MediumState_Created);
10200 }
10201
10202 mData->mSession.mLockedMedia.clear();
10203}
10204
10205/**
10206 * Helper to change the machine state (reimplementation).
10207 *
10208 * @note Locks this object for writing.
10209 */
10210HRESULT SessionMachine::setMachineState (MachineState_T aMachineState)
10211{
10212 LogFlowThisFuncEnter();
10213 LogFlowThisFunc(("aMachineState=%s\n", Global::stringifyMachineState(aMachineState) ));
10214
10215 AutoCaller autoCaller(this);
10216 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
10217
10218 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
10219
10220 MachineState_T oldMachineState = mData->mMachineState;
10221
10222 AssertMsgReturn(oldMachineState != aMachineState,
10223 ("oldMachineState=%s, aMachineState=%s\n",
10224 Global::stringifyMachineState(oldMachineState), Global::stringifyMachineState(aMachineState)),
10225 E_FAIL);
10226
10227 HRESULT rc = S_OK;
10228
10229 int stsFlags = 0;
10230 bool deleteSavedState = false;
10231
10232 /* detect some state transitions */
10233
10234 if ( ( oldMachineState == MachineState_Saved
10235 && aMachineState == MachineState_Restoring)
10236 || ( ( oldMachineState == MachineState_PoweredOff
10237 || oldMachineState == MachineState_Teleported
10238 || oldMachineState == MachineState_Aborted
10239 )
10240 && ( aMachineState == MachineState_TeleportingIn
10241 || aMachineState == MachineState_Starting
10242 )
10243 )
10244 )
10245 {
10246 /* The EMT thread is about to start */
10247
10248 /* Nothing to do here for now... */
10249
10250 /// @todo NEWMEDIA don't let mDVDDrive and other children
10251 /// change anything when in the Starting/Restoring state
10252 }
10253 else if ( ( oldMachineState == MachineState_Running
10254 || oldMachineState == MachineState_Paused
10255 || oldMachineState == MachineState_Teleporting
10256 || oldMachineState == MachineState_LiveSnapshotting
10257 || oldMachineState == MachineState_Stuck
10258 || oldMachineState == MachineState_Starting
10259 || oldMachineState == MachineState_Stopping
10260 || oldMachineState == MachineState_Saving
10261 || oldMachineState == MachineState_Restoring
10262 || oldMachineState == MachineState_TeleportingPausedVM
10263 || oldMachineState == MachineState_TeleportingIn
10264 )
10265 && ( aMachineState == MachineState_PoweredOff
10266 || aMachineState == MachineState_Saved
10267 || aMachineState == MachineState_Teleported
10268 || aMachineState == MachineState_Aborted
10269 )
10270 /* ignore PoweredOff->Saving->PoweredOff transition when taking a
10271 * snapshot */
10272 && ( mSnapshotData.mSnapshot.isNull()
10273 || mSnapshotData.mLastState >= MachineState_Running /** @todo Live Migration: clean up (lazy bird) */
10274 )
10275 )
10276 {
10277 /* The EMT thread has just stopped, unlock attached media. Note that as
10278 * opposed to locking that is done from Console, we do unlocking here
10279 * because the VM process may have aborted before having a chance to
10280 * properly unlock all media it locked. */
10281
10282 unlockMedia();
10283 }
10284
10285 if (oldMachineState == MachineState_Restoring)
10286 {
10287 if (aMachineState != MachineState_Saved)
10288 {
10289 /*
10290 * delete the saved state file once the machine has finished
10291 * restoring from it (note that Console sets the state from
10292 * Restoring to Saved if the VM couldn't restore successfully,
10293 * to give the user an ability to fix an error and retry --
10294 * we keep the saved state file in this case)
10295 */
10296 deleteSavedState = true;
10297 }
10298 }
10299 else if ( oldMachineState == MachineState_Saved
10300 && ( aMachineState == MachineState_PoweredOff
10301 || aMachineState == MachineState_Aborted
10302 || aMachineState == MachineState_Teleported
10303 )
10304 )
10305 {
10306 /*
10307 * delete the saved state after Console::DiscardSavedState() is called
10308 * or if the VM process (owning a direct VM session) crashed while the
10309 * VM was Saved
10310 */
10311
10312 /// @todo (dmik)
10313 // Not sure that deleting the saved state file just because of the
10314 // client death before it attempted to restore the VM is a good
10315 // thing. But when it crashes we need to go to the Aborted state
10316 // which cannot have the saved state file associated... The only
10317 // way to fix this is to make the Aborted condition not a VM state
10318 // but a bool flag: i.e., when a crash occurs, set it to true and
10319 // change the state to PoweredOff or Saved depending on the
10320 // saved state presence.
10321
10322 deleteSavedState = true;
10323 mData->mCurrentStateModified = TRUE;
10324 stsFlags |= SaveSTS_CurStateModified;
10325 }
10326
10327 if ( aMachineState == MachineState_Starting
10328 || aMachineState == MachineState_Restoring
10329 || aMachineState == MachineState_TeleportingIn
10330 )
10331 {
10332 /* set the current state modified flag to indicate that the current
10333 * state is no more identical to the state in the
10334 * current snapshot */
10335 if (!mData->mCurrentSnapshot.isNull())
10336 {
10337 mData->mCurrentStateModified = TRUE;
10338 stsFlags |= SaveSTS_CurStateModified;
10339 }
10340 }
10341
10342 if (deleteSavedState)
10343 {
10344 if (mRemoveSavedState)
10345 {
10346 Assert(!mSSData->mStateFilePath.isEmpty());
10347 RTFileDelete(mSSData->mStateFilePath.c_str());
10348 }
10349 mSSData->mStateFilePath.setNull();
10350 stsFlags |= SaveSTS_StateFilePath;
10351 }
10352
10353 /* redirect to the underlying peer machine */
10354 mPeer->setMachineState (aMachineState);
10355
10356 if ( aMachineState == MachineState_PoweredOff
10357 || aMachineState == MachineState_Teleported
10358 || aMachineState == MachineState_Aborted
10359 || aMachineState == MachineState_Saved)
10360 {
10361 /* the machine has stopped execution
10362 * (or the saved state file was adopted) */
10363 stsFlags |= SaveSTS_StateTimeStamp;
10364 }
10365
10366 if ( ( oldMachineState == MachineState_PoweredOff
10367 || oldMachineState == MachineState_Aborted
10368 || oldMachineState == MachineState_Teleported
10369 )
10370 && aMachineState == MachineState_Saved)
10371 {
10372 /* the saved state file was adopted */
10373 Assert(!mSSData->mStateFilePath.isEmpty());
10374 stsFlags |= SaveSTS_StateFilePath;
10375 }
10376
10377 rc = saveStateSettings (stsFlags);
10378
10379 if ( ( oldMachineState != MachineState_PoweredOff
10380 && oldMachineState != MachineState_Aborted
10381 && oldMachineState != MachineState_Teleported
10382 )
10383 && ( aMachineState == MachineState_PoweredOff
10384 || aMachineState == MachineState_Aborted
10385 || aMachineState == MachineState_Teleported
10386 )
10387 )
10388 {
10389 /* we've been shut down for any reason */
10390 /* no special action so far */
10391 }
10392
10393 LogFlowThisFunc(("rc=%Rhrc [%s]\n", rc, Global::stringifyMachineState(mData->mMachineState) ));
10394 LogFlowThisFuncLeave();
10395 return rc;
10396}
10397
10398/**
10399 * Sends the current machine state value to the VM process.
10400 *
10401 * @note Locks this object for reading, then calls a client process.
10402 */
10403HRESULT SessionMachine::updateMachineStateOnClient()
10404{
10405 AutoCaller autoCaller(this);
10406 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
10407
10408 ComPtr<IInternalSessionControl> directControl;
10409 {
10410 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
10411 AssertReturn(!!mData, E_FAIL);
10412 directControl = mData->mSession.mDirectControl;
10413
10414 /* directControl may be already set to NULL here in #OnSessionEnd()
10415 * called too early by the direct session process while there is still
10416 * some operation (like discarding the snapshot) in progress. The client
10417 * process in this case is waiting inside Session::close() for the
10418 * "end session" process object to complete, while #uninit() called by
10419 * #checkForDeath() on the Watcher thread is waiting for the pending
10420 * operation to complete. For now, we accept this inconsitent behavior
10421 * and simply do nothing here. */
10422
10423 if (mData->mSession.mState == SessionState_Closing)
10424 return S_OK;
10425
10426 AssertReturn(!directControl.isNull(), E_FAIL);
10427 }
10428
10429 return directControl->UpdateMachineState (mData->mMachineState);
10430}
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