VirtualBox

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

Last change on this file since 27124 was 26951, checked in by vboxsync, 15 years ago

introduced VBOX_WITH_VUSB

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