VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/MachineImpl.cpp@ 48654

Last change on this file since 48654 was 48538, checked in by vboxsync, 11 years ago

Main/NetworkAdapter+Machine+Appliance+SystemProperties+Medium+Console+Settings+IDL: make NAT networking a separate network attachment type which improves the user experience, store the necessary settings, plus changing the design of the methods which will move images and entire VMs, they lacked a progress object
Frontends/VirtualBox: adapted fully, can configure NAT networks with proper drop down list support
Frontends/VBoxManage: also supports NAT networks completely, and adds the long missing code to list intnets

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 501.2 KB
Line 
1/* $Id: MachineImpl.cpp 48538 2013-09-19 15:17:43Z vboxsync $ */
2/** @file
3 * Implementation of IMachine in VBoxSVC.
4 */
5
6/*
7 * Copyright (C) 2004-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/* Make sure all the stdint.h macros are included - must come first! */
19#ifndef __STDC_LIMIT_MACROS
20# define __STDC_LIMIT_MACROS
21#endif
22#ifndef __STDC_CONSTANT_MACROS
23# define __STDC_CONSTANT_MACROS
24#endif
25
26#include "Logging.h"
27#include "VirtualBoxImpl.h"
28#include "MachineImpl.h"
29#include "ClientToken.h"
30#include "ProgressImpl.h"
31#include "ProgressProxyImpl.h"
32#include "MediumAttachmentImpl.h"
33#include "MediumImpl.h"
34#include "MediumLock.h"
35#include "USBControllerImpl.h"
36#include "USBDeviceFiltersImpl.h"
37#include "HostImpl.h"
38#include "SharedFolderImpl.h"
39#include "GuestOSTypeImpl.h"
40#include "VirtualBoxErrorInfoImpl.h"
41#include "GuestImpl.h"
42#include "StorageControllerImpl.h"
43#include "DisplayImpl.h"
44#include "DisplayUtils.h"
45#include "MachineImplCloneVM.h"
46#include "AutostartDb.h"
47#include "SystemPropertiesImpl.h"
48
49// generated header
50#include "VBoxEvents.h"
51
52#ifdef VBOX_WITH_USB
53# include "USBProxyService.h"
54#endif
55
56#include "AutoCaller.h"
57#include "HashedPw.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/cpp/xml.h> /* xml::XmlFileWriter::s_psz*Suff. */
68#include <iprt/sha.h>
69#include <iprt/string.h>
70#include <iprt/base64.h>
71
72#include <VBox/com/array.h>
73#include <VBox/com/list.h>
74
75#include <VBox/err.h>
76#include <VBox/param.h>
77#include <VBox/settings.h>
78#include <VBox/vmm/ssm.h>
79
80#ifdef VBOX_WITH_GUEST_PROPS
81# include <VBox/HostServices/GuestPropertySvc.h>
82# include <VBox/com/array.h>
83#endif
84
85#include "VBox/com/MultiResult.h"
86
87#include <algorithm>
88
89#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
90# define HOSTSUFF_EXE ".exe"
91#else /* !RT_OS_WINDOWS */
92# define HOSTSUFF_EXE ""
93#endif /* !RT_OS_WINDOWS */
94
95// defines / prototypes
96/////////////////////////////////////////////////////////////////////////////
97
98/////////////////////////////////////////////////////////////////////////////
99// Machine::Data structure
100/////////////////////////////////////////////////////////////////////////////
101
102Machine::Data::Data()
103{
104 mRegistered = FALSE;
105 pMachineConfigFile = NULL;
106 /* Contains hints on what has changed when the user is using the VM (config
107 * changes, running the VM, ...). This is used to decide if a config needs
108 * to be written to disk. */
109 flModifications = 0;
110 /* VM modification usually also trigger setting the current state to
111 * "Modified". Although this is not always the case. An e.g. is the VM
112 * initialization phase or when snapshot related data is changed. The
113 * actually behavior is controlled by the following flag. */
114 m_fAllowStateModification = false;
115 mAccessible = FALSE;
116 /* mUuid is initialized in Machine::init() */
117
118 mMachineState = MachineState_PoweredOff;
119 RTTimeNow(&mLastStateChange);
120
121 mMachineStateDeps = 0;
122 mMachineStateDepsSem = NIL_RTSEMEVENTMULTI;
123 mMachineStateChangePending = 0;
124
125 mCurrentStateModified = TRUE;
126 mGuestPropertiesModified = FALSE;
127
128 mSession.mPID = NIL_RTPROCESS;
129 mSession.mState = SessionState_Unlocked;
130}
131
132Machine::Data::~Data()
133{
134 if (mMachineStateDepsSem != NIL_RTSEMEVENTMULTI)
135 {
136 RTSemEventMultiDestroy(mMachineStateDepsSem);
137 mMachineStateDepsSem = NIL_RTSEMEVENTMULTI;
138 }
139 if (pMachineConfigFile)
140 {
141 delete pMachineConfigFile;
142 pMachineConfigFile = NULL;
143 }
144}
145
146/////////////////////////////////////////////////////////////////////////////
147// Machine::HWData structure
148/////////////////////////////////////////////////////////////////////////////
149
150Machine::HWData::HWData()
151{
152 /* default values for a newly created machine */
153 mHWVersion = "2"; /** @todo get the default from the schema if that is possible. */
154 mMemorySize = 128;
155 mCPUCount = 1;
156 mCPUHotPlugEnabled = false;
157 mMemoryBalloonSize = 0;
158 mPageFusionEnabled = false;
159 mGraphicsControllerType = GraphicsControllerType_VBoxVGA;
160 mVRAMSize = 8;
161 mAccelerate3DEnabled = false;
162 mAccelerate2DVideoEnabled = false;
163 mMonitorCount = 1;
164 mVideoCaptureWidth = 1024;
165 mVideoCaptureHeight = 768;
166 mVideoCaptureRate = 512;
167 mVideoCaptureFPS = 25;
168 mVideoCaptureEnabled = false;
169 for (unsigned i = 0; i < RT_ELEMENTS(maVideoCaptureScreens); i++)
170 maVideoCaptureScreens[i] = true;
171
172 mHWVirtExEnabled = true;
173 mHWVirtExNestedPagingEnabled = true;
174#if HC_ARCH_BITS == 64 && !defined(RT_OS_LINUX)
175 mHWVirtExLargePagesEnabled = true;
176#else
177 /* Not supported on 32 bits hosts. */
178 mHWVirtExLargePagesEnabled = false;
179#endif
180 mHWVirtExVPIDEnabled = true;
181 mHWVirtExUXEnabled = true;
182 mHWVirtExForceEnabled = false;
183#if HC_ARCH_BITS == 64 || defined(RT_OS_WINDOWS) || defined(RT_OS_DARWIN)
184 mPAEEnabled = true;
185#else
186 mPAEEnabled = false;
187#endif
188 mLongMode = HC_ARCH_BITS == 64 ? settings::Hardware::LongMode_Enabled : settings::Hardware::LongMode_Disabled;
189 mSyntheticCpu = false;
190 mHPETEnabled = false;
191
192 /* default boot order: floppy - DVD - HDD */
193 mBootOrder[0] = DeviceType_Floppy;
194 mBootOrder[1] = DeviceType_DVD;
195 mBootOrder[2] = DeviceType_HardDisk;
196 for (size_t i = 3; i < RT_ELEMENTS(mBootOrder); ++i)
197 mBootOrder[i] = DeviceType_Null;
198
199 mClipboardMode = ClipboardMode_Disabled;
200 mDragAndDropMode = DragAndDropMode_Disabled;
201 mGuestPropertyNotificationPatterns = "";
202
203 mFirmwareType = FirmwareType_BIOS;
204 mKeyboardHIDType = KeyboardHIDType_PS2Keyboard;
205 mPointingHIDType = PointingHIDType_PS2Mouse;
206 mChipsetType = ChipsetType_PIIX3;
207 mEmulatedUSBCardReaderEnabled = FALSE;
208
209 for (size_t i = 0; i < RT_ELEMENTS(mCPUAttached); i++)
210 mCPUAttached[i] = false;
211
212 mIOCacheEnabled = true;
213 mIOCacheSize = 5; /* 5MB */
214
215 /* Maximum CPU execution cap by default. */
216 mCpuExecutionCap = 100;
217}
218
219Machine::HWData::~HWData()
220{
221}
222
223/////////////////////////////////////////////////////////////////////////////
224// Machine::HDData structure
225/////////////////////////////////////////////////////////////////////////////
226
227Machine::MediaData::MediaData()
228{
229}
230
231Machine::MediaData::~MediaData()
232{
233}
234
235/////////////////////////////////////////////////////////////////////////////
236// Machine class
237/////////////////////////////////////////////////////////////////////////////
238
239// constructor / destructor
240/////////////////////////////////////////////////////////////////////////////
241
242Machine::Machine() :
243#ifdef VBOX_WITH_RESOURCE_USAGE_API
244 mCollectorGuest(NULL),
245#endif
246 mPeer(NULL),
247 mParent(NULL),
248 mSerialPorts(),
249 mParallelPorts(),
250 uRegistryNeedsSaving(0)
251{}
252
253Machine::~Machine()
254{}
255
256HRESULT Machine::FinalConstruct()
257{
258 LogFlowThisFunc(("\n"));
259 return BaseFinalConstruct();
260}
261
262void Machine::FinalRelease()
263{
264 LogFlowThisFunc(("\n"));
265 uninit();
266 BaseFinalRelease();
267}
268
269/**
270 * Initializes a new machine instance; this init() variant creates a new, empty machine.
271 * This gets called from VirtualBox::CreateMachine().
272 *
273 * @param aParent Associated parent object
274 * @param strConfigFile Local file system path to the VM settings file (can
275 * be relative to the VirtualBox config directory).
276 * @param strName name for the machine
277 * @param llGroups list of groups for the machine
278 * @param aOsType OS Type of this machine or NULL.
279 * @param aId UUID for the new machine.
280 * @param fForceOverwrite Whether to overwrite an existing machine settings file.
281 *
282 * @return Success indicator. if not S_OK, the machine object is invalid
283 */
284HRESULT Machine::init(VirtualBox *aParent,
285 const Utf8Str &strConfigFile,
286 const Utf8Str &strName,
287 const StringsList &llGroups,
288 GuestOSType *aOsType,
289 const Guid &aId,
290 bool fForceOverwrite,
291 bool fDirectoryIncludesUUID)
292{
293 LogFlowThisFuncEnter();
294 LogFlowThisFunc(("(Init_New) aConfigFile='%s'\n", strConfigFile.c_str()));
295
296 /* Enclose the state transition NotReady->InInit->Ready */
297 AutoInitSpan autoInitSpan(this);
298 AssertReturn(autoInitSpan.isOk(), E_FAIL);
299
300 HRESULT rc = initImpl(aParent, strConfigFile);
301 if (FAILED(rc)) return rc;
302
303 rc = tryCreateMachineConfigFile(fForceOverwrite);
304 if (FAILED(rc)) return rc;
305
306 if (SUCCEEDED(rc))
307 {
308 // create an empty machine config
309 mData->pMachineConfigFile = new settings::MachineConfigFile(NULL);
310
311 rc = initDataAndChildObjects();
312 }
313
314 if (SUCCEEDED(rc))
315 {
316 // set to true now to cause uninit() to call uninitDataAndChildObjects() on failure
317 mData->mAccessible = TRUE;
318
319 unconst(mData->mUuid) = aId;
320
321 mUserData->s.strName = strName;
322
323 mUserData->s.llGroups = llGroups;
324
325 mUserData->s.fDirectoryIncludesUUID = fDirectoryIncludesUUID;
326 // the "name sync" flag determines whether the machine directory gets renamed along
327 // with the machine file; say so if the settings file name is the same as the
328 // settings file parent directory (machine directory)
329 mUserData->s.fNameSync = isInOwnDir();
330
331 // initialize the default snapshots folder
332 rc = COMSETTER(SnapshotFolder)(NULL);
333 AssertComRC(rc);
334
335 if (aOsType)
336 {
337 /* Store OS type */
338 mUserData->s.strOsType = aOsType->id();
339
340 /* Apply BIOS defaults */
341 mBIOSSettings->applyDefaults(aOsType);
342
343 /* Apply network adapters defaults */
344 for (ULONG slot = 0; slot < mNetworkAdapters.size(); ++slot)
345 mNetworkAdapters[slot]->applyDefaults(aOsType);
346
347 /* Apply serial port defaults */
348 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); ++slot)
349 mSerialPorts[slot]->applyDefaults(aOsType);
350
351 /* Let the OS type select 64-bit ness. */
352 mHWData->mLongMode = aOsType->is64Bit()
353 ? settings::Hardware::LongMode_Enabled : settings::Hardware::LongMode_Disabled;
354 }
355
356 /* At this point the changing of the current state modification
357 * flag is allowed. */
358 allowStateModification();
359
360 /* commit all changes made during the initialization */
361 commit();
362 }
363
364 /* Confirm a successful initialization when it's the case */
365 if (SUCCEEDED(rc))
366 {
367 if (mData->mAccessible)
368 autoInitSpan.setSucceeded();
369 else
370 autoInitSpan.setLimited();
371 }
372
373 LogFlowThisFunc(("mName='%s', mRegistered=%RTbool, mAccessible=%RTbool, rc=%08X\n",
374 !!mUserData ? mUserData->s.strName.c_str() : "NULL",
375 mData->mRegistered,
376 mData->mAccessible,
377 rc));
378
379 LogFlowThisFuncLeave();
380
381 return rc;
382}
383
384/**
385 * Initializes a new instance with data from machine XML (formerly Init_Registered).
386 * Gets called in two modes:
387 *
388 * -- from VirtualBox::initMachines() during VirtualBox startup; in that case, the
389 * UUID is specified and we mark the machine as "registered";
390 *
391 * -- from the public VirtualBox::OpenMachine() API, in which case the UUID is NULL
392 * and the machine remains unregistered until RegisterMachine() is called.
393 *
394 * @param aParent Associated parent object
395 * @param aConfigFile Local file system path to the VM settings file (can
396 * be relative to the VirtualBox config directory).
397 * @param aId UUID of the machine or NULL (see above).
398 *
399 * @return Success indicator. if not S_OK, the machine object is invalid
400 */
401HRESULT Machine::initFromSettings(VirtualBox *aParent,
402 const Utf8Str &strConfigFile,
403 const Guid *aId)
404{
405 LogFlowThisFuncEnter();
406 LogFlowThisFunc(("(Init_Registered) aConfigFile='%s\n", strConfigFile.c_str()));
407
408 /* Enclose the state transition NotReady->InInit->Ready */
409 AutoInitSpan autoInitSpan(this);
410 AssertReturn(autoInitSpan.isOk(), E_FAIL);
411
412 HRESULT rc = initImpl(aParent, strConfigFile);
413 if (FAILED(rc)) return rc;
414
415 if (aId)
416 {
417 // loading a registered VM:
418 unconst(mData->mUuid) = *aId;
419 mData->mRegistered = TRUE;
420 // now load the settings from XML:
421 rc = registeredInit();
422 // this calls initDataAndChildObjects() and loadSettings()
423 }
424 else
425 {
426 // opening an unregistered VM (VirtualBox::OpenMachine()):
427 rc = initDataAndChildObjects();
428
429 if (SUCCEEDED(rc))
430 {
431 // set to true now to cause uninit() to call uninitDataAndChildObjects() on failure
432 mData->mAccessible = TRUE;
433
434 try
435 {
436 // load and parse machine XML; this will throw on XML or logic errors
437 mData->pMachineConfigFile = new settings::MachineConfigFile(&mData->m_strConfigFileFull);
438
439 // reject VM UUID duplicates, they can happen if someone
440 // tries to register an already known VM config again
441 if (aParent->findMachine(mData->pMachineConfigFile->uuid,
442 true /* fPermitInaccessible */,
443 false /* aDoSetError */,
444 NULL) != VBOX_E_OBJECT_NOT_FOUND)
445 {
446 throw setError(E_FAIL,
447 tr("Trying to open a VM config '%s' which has the same UUID as an existing virtual machine"),
448 mData->m_strConfigFile.c_str());
449 }
450
451 // use UUID from machine config
452 unconst(mData->mUuid) = mData->pMachineConfigFile->uuid;
453
454 rc = loadMachineDataFromSettings(*mData->pMachineConfigFile,
455 NULL /* puuidRegistry */);
456 if (FAILED(rc)) throw rc;
457
458 /* At this point the changing of the current state modification
459 * flag is allowed. */
460 allowStateModification();
461
462 commit();
463 }
464 catch (HRESULT err)
465 {
466 /* we assume that error info is set by the thrower */
467 rc = err;
468 }
469 catch (...)
470 {
471 rc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
472 }
473 }
474 }
475
476 /* Confirm a successful initialization when it's the case */
477 if (SUCCEEDED(rc))
478 {
479 if (mData->mAccessible)
480 autoInitSpan.setSucceeded();
481 else
482 {
483 autoInitSpan.setLimited();
484
485 // uninit media from this machine's media registry, or else
486 // reloading the settings will fail
487 mParent->unregisterMachineMedia(getId());
488 }
489 }
490
491 LogFlowThisFunc(("mName='%s', mRegistered=%RTbool, mAccessible=%RTbool "
492 "rc=%08X\n",
493 !!mUserData ? mUserData->s.strName.c_str() : "NULL",
494 mData->mRegistered, mData->mAccessible, rc));
495
496 LogFlowThisFuncLeave();
497
498 return rc;
499}
500
501/**
502 * Initializes a new instance from a machine config that is already in memory
503 * (import OVF case). Since we are importing, the UUID in the machine
504 * config is ignored and we always generate a fresh one.
505 *
506 * @param strName Name for the new machine; this overrides what is specified in config and is used
507 * for the settings file as well.
508 * @param config Machine configuration loaded and parsed from XML.
509 *
510 * @return Success indicator. if not S_OK, the machine object is invalid
511 */
512HRESULT Machine::init(VirtualBox *aParent,
513 const Utf8Str &strName,
514 const settings::MachineConfigFile &config)
515{
516 LogFlowThisFuncEnter();
517
518 /* Enclose the state transition NotReady->InInit->Ready */
519 AutoInitSpan autoInitSpan(this);
520 AssertReturn(autoInitSpan.isOk(), E_FAIL);
521
522 Utf8Str strConfigFile;
523 aParent->getDefaultMachineFolder(strConfigFile);
524 strConfigFile.append(RTPATH_DELIMITER);
525 strConfigFile.append(strName);
526 strConfigFile.append(RTPATH_DELIMITER);
527 strConfigFile.append(strName);
528 strConfigFile.append(".vbox");
529
530 HRESULT rc = initImpl(aParent, strConfigFile);
531 if (FAILED(rc)) return rc;
532
533 rc = tryCreateMachineConfigFile(false /* fForceOverwrite */);
534 if (FAILED(rc)) return rc;
535
536 rc = initDataAndChildObjects();
537
538 if (SUCCEEDED(rc))
539 {
540 // set to true now to cause uninit() to call uninitDataAndChildObjects() on failure
541 mData->mAccessible = TRUE;
542
543 // create empty machine config for instance data
544 mData->pMachineConfigFile = new settings::MachineConfigFile(NULL);
545
546 // generate fresh UUID, ignore machine config
547 unconst(mData->mUuid).create();
548
549 rc = loadMachineDataFromSettings(config,
550 &mData->mUuid); // puuidRegistry: initialize media with this registry ID
551
552 // override VM name as well, it may be different
553 mUserData->s.strName = strName;
554
555 if (SUCCEEDED(rc))
556 {
557 /* At this point the changing of the current state modification
558 * flag is allowed. */
559 allowStateModification();
560
561 /* commit all changes made during the initialization */
562 commit();
563 }
564 }
565
566 /* Confirm a successful initialization when it's the case */
567 if (SUCCEEDED(rc))
568 {
569 if (mData->mAccessible)
570 autoInitSpan.setSucceeded();
571 else
572 {
573 autoInitSpan.setLimited();
574
575 // uninit media from this machine's media registry, or else
576 // reloading the settings will fail
577 mParent->unregisterMachineMedia(getId());
578 }
579 }
580
581 LogFlowThisFunc(("mName='%s', mRegistered=%RTbool, mAccessible=%RTbool "
582 "rc=%08X\n",
583 !!mUserData ? mUserData->s.strName.c_str() : "NULL",
584 mData->mRegistered, mData->mAccessible, rc));
585
586 LogFlowThisFuncLeave();
587
588 return rc;
589}
590
591/**
592 * Shared code between the various init() implementations.
593 * @param aParent
594 * @return
595 */
596HRESULT Machine::initImpl(VirtualBox *aParent,
597 const Utf8Str &strConfigFile)
598{
599 LogFlowThisFuncEnter();
600
601 AssertReturn(aParent, E_INVALIDARG);
602 AssertReturn(!strConfigFile.isEmpty(), E_INVALIDARG);
603
604 HRESULT rc = S_OK;
605
606 /* share the parent weakly */
607 unconst(mParent) = aParent;
608
609 /* allocate the essential machine data structure (the rest will be
610 * allocated later by initDataAndChildObjects() */
611 mData.allocate();
612
613 /* memorize the config file name (as provided) */
614 mData->m_strConfigFile = strConfigFile;
615
616 /* get the full file name */
617 int vrc1 = mParent->calculateFullPath(strConfigFile, mData->m_strConfigFileFull);
618 if (RT_FAILURE(vrc1))
619 return setError(VBOX_E_FILE_ERROR,
620 tr("Invalid machine settings file name '%s' (%Rrc)"),
621 strConfigFile.c_str(),
622 vrc1);
623
624 LogFlowThisFuncLeave();
625
626 return rc;
627}
628
629/**
630 * Tries to create a machine settings file in the path stored in the machine
631 * instance data. Used when a new machine is created to fail gracefully if
632 * the settings file could not be written (e.g. because machine dir is read-only).
633 * @return
634 */
635HRESULT Machine::tryCreateMachineConfigFile(bool fForceOverwrite)
636{
637 HRESULT rc = S_OK;
638
639 // when we create a new machine, we must be able to create the settings file
640 RTFILE f = NIL_RTFILE;
641 int vrc = RTFileOpen(&f, mData->m_strConfigFileFull.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
642 if ( RT_SUCCESS(vrc)
643 || vrc == VERR_SHARING_VIOLATION
644 )
645 {
646 if (RT_SUCCESS(vrc))
647 RTFileClose(f);
648 if (!fForceOverwrite)
649 rc = setError(VBOX_E_FILE_ERROR,
650 tr("Machine settings file '%s' already exists"),
651 mData->m_strConfigFileFull.c_str());
652 else
653 {
654 /* try to delete the config file, as otherwise the creation
655 * of a new settings file will fail. */
656 int vrc2 = RTFileDelete(mData->m_strConfigFileFull.c_str());
657 if (RT_FAILURE(vrc2))
658 rc = setError(VBOX_E_FILE_ERROR,
659 tr("Could not delete the existing settings file '%s' (%Rrc)"),
660 mData->m_strConfigFileFull.c_str(), vrc2);
661 }
662 }
663 else if ( vrc != VERR_FILE_NOT_FOUND
664 && vrc != VERR_PATH_NOT_FOUND
665 )
666 rc = setError(VBOX_E_FILE_ERROR,
667 tr("Invalid machine settings file name '%s' (%Rrc)"),
668 mData->m_strConfigFileFull.c_str(),
669 vrc);
670 return rc;
671}
672
673/**
674 * Initializes the registered machine by loading the settings file.
675 * This method is separated from #init() in order to make it possible to
676 * retry the operation after VirtualBox startup instead of refusing to
677 * startup the whole VirtualBox server in case if the settings file of some
678 * registered VM is invalid or inaccessible.
679 *
680 * @note Must be always called from this object's write lock
681 * (unless called from #init() that doesn't need any locking).
682 * @note Locks the mUSBController method for writing.
683 * @note Subclasses must not call this method.
684 */
685HRESULT Machine::registeredInit()
686{
687 AssertReturn(!isSessionMachine(), E_FAIL);
688 AssertReturn(!isSnapshotMachine(), E_FAIL);
689 AssertReturn(mData->mUuid.isValid(), E_FAIL);
690 AssertReturn(!mData->mAccessible, E_FAIL);
691
692 HRESULT rc = initDataAndChildObjects();
693
694 if (SUCCEEDED(rc))
695 {
696 /* Temporarily reset the registered flag in order to let setters
697 * potentially called from loadSettings() succeed (isMutable() used in
698 * all setters will return FALSE for a Machine instance if mRegistered
699 * is TRUE). */
700 mData->mRegistered = FALSE;
701
702 try
703 {
704 // load and parse machine XML; this will throw on XML or logic errors
705 mData->pMachineConfigFile = new settings::MachineConfigFile(&mData->m_strConfigFileFull);
706
707 if (mData->mUuid != mData->pMachineConfigFile->uuid)
708 throw setError(E_FAIL,
709 tr("Machine UUID {%RTuuid} in '%s' doesn't match its UUID {%s} in the registry file '%s'"),
710 mData->pMachineConfigFile->uuid.raw(),
711 mData->m_strConfigFileFull.c_str(),
712 mData->mUuid.toString().c_str(),
713 mParent->settingsFilePath().c_str());
714
715 rc = loadMachineDataFromSettings(*mData->pMachineConfigFile,
716 NULL /* const Guid *puuidRegistry */);
717 if (FAILED(rc)) throw rc;
718 }
719 catch (HRESULT err)
720 {
721 /* we assume that error info is set by the thrower */
722 rc = err;
723 }
724 catch (...)
725 {
726 rc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
727 }
728
729 /* Restore the registered flag (even on failure) */
730 mData->mRegistered = TRUE;
731 }
732
733 if (SUCCEEDED(rc))
734 {
735 /* Set mAccessible to TRUE only if we successfully locked and loaded
736 * the settings file */
737 mData->mAccessible = TRUE;
738
739 /* commit all changes made during loading the settings file */
740 commit(); // @todo r=dj why do we need a commit during init?!? this is very expensive
741 /// @todo r=klaus for some reason the settings loading logic backs up
742 // the settings, and therefore a commit is needed. Should probably be changed.
743 }
744 else
745 {
746 /* If the machine is registered, then, instead of returning a
747 * failure, we mark it as inaccessible and set the result to
748 * success to give it a try later */
749
750 /* fetch the current error info */
751 mData->mAccessError = com::ErrorInfo();
752 LogWarning(("Machine {%RTuuid} is inaccessible! [%ls]\n",
753 mData->mUuid.raw(),
754 mData->mAccessError.getText().raw()));
755
756 /* rollback all changes */
757 rollback(false /* aNotify */);
758
759 // uninit media from this machine's media registry, or else
760 // reloading the settings will fail
761 mParent->unregisterMachineMedia(getId());
762
763 /* uninitialize the common part to make sure all data is reset to
764 * default (null) values */
765 uninitDataAndChildObjects();
766
767 rc = S_OK;
768 }
769
770 return rc;
771}
772
773/**
774 * Uninitializes the instance.
775 * Called either from FinalRelease() or by the parent when it gets destroyed.
776 *
777 * @note The caller of this method must make sure that this object
778 * a) doesn't have active callers on the current thread and b) is not locked
779 * by the current thread; otherwise uninit() will hang either a) due to
780 * AutoUninitSpan waiting for a number of calls to drop to zero or b) due to
781 * a dead-lock caused by this thread waiting for all callers on the other
782 * threads are done but preventing them from doing so by holding a lock.
783 */
784void Machine::uninit()
785{
786 LogFlowThisFuncEnter();
787
788 Assert(!isWriteLockOnCurrentThread());
789
790 Assert(!uRegistryNeedsSaving);
791 if (uRegistryNeedsSaving)
792 {
793 AutoCaller autoCaller(this);
794 if (SUCCEEDED(autoCaller.rc()))
795 {
796 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
797 saveSettings(NULL, Machine::SaveS_Force);
798 }
799 }
800
801 /* Enclose the state transition Ready->InUninit->NotReady */
802 AutoUninitSpan autoUninitSpan(this);
803 if (autoUninitSpan.uninitDone())
804 return;
805
806 Assert(!isSnapshotMachine());
807 Assert(!isSessionMachine());
808 Assert(!!mData);
809
810 LogFlowThisFunc(("initFailed()=%d\n", autoUninitSpan.initFailed()));
811 LogFlowThisFunc(("mRegistered=%d\n", mData->mRegistered));
812
813 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
814
815 if (!mData->mSession.mMachine.isNull())
816 {
817 /* Theoretically, this can only happen if the VirtualBox server has been
818 * terminated while there were clients running that owned open direct
819 * sessions. Since in this case we are definitely called by
820 * VirtualBox::uninit(), we may be sure that SessionMachine::uninit()
821 * won't happen on the client watcher thread (because it does
822 * VirtualBox::addCaller() for the duration of the
823 * SessionMachine::checkForDeath() call, so that VirtualBox::uninit()
824 * cannot happen until the VirtualBox caller is released). This is
825 * important, because SessionMachine::uninit() cannot correctly operate
826 * after we return from this method (it expects the Machine instance is
827 * still valid). We'll call it ourselves below.
828 */
829 LogWarningThisFunc(("Session machine is not NULL (%p), the direct session is still open!\n",
830 (SessionMachine*)mData->mSession.mMachine));
831
832 if (Global::IsOnlineOrTransient(mData->mMachineState))
833 {
834 LogWarningThisFunc(("Setting state to Aborted!\n"));
835 /* set machine state using SessionMachine reimplementation */
836 static_cast<Machine*>(mData->mSession.mMachine)->setMachineState(MachineState_Aborted);
837 }
838
839 /*
840 * Uninitialize SessionMachine using public uninit() to indicate
841 * an unexpected uninitialization.
842 */
843 mData->mSession.mMachine->uninit();
844 /* SessionMachine::uninit() must set mSession.mMachine to null */
845 Assert(mData->mSession.mMachine.isNull());
846 }
847
848 // uninit media from this machine's media registry, if they're still there
849 Guid uuidMachine(getId());
850
851 /* the lock is no more necessary (SessionMachine is uninitialized) */
852 alock.release();
853
854 /* XXX This will fail with
855 * "cannot be closed because it is still attached to 1 virtual machines"
856 * because at this point we did not call uninitDataAndChildObjects() yet
857 * and therefore also removeBackReference() for all these mediums was not called! */
858
859 if (uuidMachine.isValid() && !uuidMachine.isZero()) // can be empty if we're called from a failure of Machine::init
860 mParent->unregisterMachineMedia(uuidMachine);
861
862 // has machine been modified?
863 if (mData->flModifications)
864 {
865 LogWarningThisFunc(("Discarding unsaved settings changes!\n"));
866 rollback(false /* aNotify */);
867 }
868
869 if (mData->mAccessible)
870 uninitDataAndChildObjects();
871
872 /* free the essential data structure last */
873 mData.free();
874
875 LogFlowThisFuncLeave();
876}
877
878// IMachine properties
879/////////////////////////////////////////////////////////////////////////////
880
881STDMETHODIMP Machine::COMGETTER(Parent)(IVirtualBox **aParent)
882{
883 CheckComArgOutPointerValid(aParent);
884
885 AutoLimitedCaller autoCaller(this);
886 if (FAILED(autoCaller.rc())) return autoCaller.rc();
887
888 /* mParent is constant during life time, no need to lock */
889 ComObjPtr<VirtualBox> pVirtualBox(mParent);
890 pVirtualBox.queryInterfaceTo(aParent);
891
892 return S_OK;
893}
894
895STDMETHODIMP Machine::COMGETTER(Accessible)(BOOL *aAccessible)
896{
897 CheckComArgOutPointerValid(aAccessible);
898
899 AutoLimitedCaller autoCaller(this);
900 if (FAILED(autoCaller.rc())) return autoCaller.rc();
901
902 LogFlowThisFunc(("ENTER\n"));
903
904 /* In some cases (medium registry related), it is necessary to be able to
905 * go through the list of all machines. Happens when an inaccessible VM
906 * has a sensible medium registry. */
907 AutoReadLock mllock(mParent->getMachinesListLockHandle() COMMA_LOCKVAL_SRC_POS);
908 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
909
910 HRESULT rc = S_OK;
911
912 if (!mData->mAccessible)
913 {
914 /* try to initialize the VM once more if not accessible */
915
916 AutoReinitSpan autoReinitSpan(this);
917 AssertReturn(autoReinitSpan.isOk(), E_FAIL);
918
919#ifdef DEBUG
920 LogFlowThisFunc(("Dumping media backreferences\n"));
921 mParent->dumpAllBackRefs();
922#endif
923
924 if (mData->pMachineConfigFile)
925 {
926 // reset the XML file to force loadSettings() (called from registeredInit())
927 // to parse it again; the file might have changed
928 delete mData->pMachineConfigFile;
929 mData->pMachineConfigFile = NULL;
930 }
931
932 rc = registeredInit();
933
934 if (SUCCEEDED(rc) && mData->mAccessible)
935 {
936 autoReinitSpan.setSucceeded();
937
938 /* make sure interesting parties will notice the accessibility
939 * state change */
940 mParent->onMachineStateChange(mData->mUuid, mData->mMachineState);
941 mParent->onMachineDataChange(mData->mUuid);
942 }
943 }
944
945 if (SUCCEEDED(rc))
946 *aAccessible = mData->mAccessible;
947
948 LogFlowThisFuncLeave();
949
950 return rc;
951}
952
953STDMETHODIMP Machine::COMGETTER(AccessError)(IVirtualBoxErrorInfo **aAccessError)
954{
955 CheckComArgOutPointerValid(aAccessError);
956
957 AutoLimitedCaller autoCaller(this);
958 if (FAILED(autoCaller.rc())) return autoCaller.rc();
959
960 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
961
962 if (mData->mAccessible || !mData->mAccessError.isBasicAvailable())
963 {
964 /* return shortly */
965 aAccessError = NULL;
966 return S_OK;
967 }
968
969 HRESULT rc = S_OK;
970
971 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
972 rc = errorInfo.createObject();
973 if (SUCCEEDED(rc))
974 {
975 errorInfo->init(mData->mAccessError.getResultCode(),
976 mData->mAccessError.getInterfaceID().ref(),
977 Utf8Str(mData->mAccessError.getComponent()).c_str(),
978 Utf8Str(mData->mAccessError.getText()));
979 rc = errorInfo.queryInterfaceTo(aAccessError);
980 }
981
982 return rc;
983}
984
985STDMETHODIMP Machine::COMGETTER(Name)(BSTR *aName)
986{
987 CheckComArgOutPointerValid(aName);
988
989 AutoCaller autoCaller(this);
990 if (FAILED(autoCaller.rc())) return autoCaller.rc();
991
992 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
993
994 mUserData->s.strName.cloneTo(aName);
995
996 return S_OK;
997}
998
999STDMETHODIMP Machine::COMSETTER(Name)(IN_BSTR aName)
1000{
1001 CheckComArgStrNotEmptyOrNull(aName);
1002
1003 AutoCaller autoCaller(this);
1004 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1005
1006 // prohibit setting a UUID only as the machine name, or else it can
1007 // never be found by findMachine()
1008 Guid test(aName);
1009
1010 if (test.isValid())
1011 return setError(E_INVALIDARG, tr("A machine cannot have a UUID as its name"));
1012
1013 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1014
1015 HRESULT rc = checkStateDependency(MutableStateDep);
1016 if (FAILED(rc)) return rc;
1017
1018 setModified(IsModified_MachineData);
1019 mUserData.backup();
1020 mUserData->s.strName = aName;
1021
1022 return S_OK;
1023}
1024
1025STDMETHODIMP Machine::COMGETTER(Description)(BSTR *aDescription)
1026{
1027 CheckComArgOutPointerValid(aDescription);
1028
1029 AutoCaller autoCaller(this);
1030 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1031
1032 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1033
1034 mUserData->s.strDescription.cloneTo(aDescription);
1035
1036 return S_OK;
1037}
1038
1039STDMETHODIMP Machine::COMSETTER(Description)(IN_BSTR aDescription)
1040{
1041 AutoCaller autoCaller(this);
1042 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1043
1044 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1045
1046 // this can be done in principle in any state as it doesn't affect the VM
1047 // significantly, but play safe by not messing around while complex
1048 // activities are going on
1049 HRESULT rc = checkStateDependency(MutableOrSavedStateDep);
1050 if (FAILED(rc)) return rc;
1051
1052 setModified(IsModified_MachineData);
1053 mUserData.backup();
1054 mUserData->s.strDescription = aDescription;
1055
1056 return S_OK;
1057}
1058
1059STDMETHODIMP Machine::COMGETTER(Id)(BSTR *aId)
1060{
1061 CheckComArgOutPointerValid(aId);
1062
1063 AutoLimitedCaller autoCaller(this);
1064 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1065
1066 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1067
1068 mData->mUuid.toUtf16().cloneTo(aId);
1069
1070 return S_OK;
1071}
1072
1073STDMETHODIMP Machine::COMGETTER(Groups)(ComSafeArrayOut(BSTR, aGroups))
1074{
1075 CheckComArgOutSafeArrayPointerValid(aGroups);
1076
1077 AutoCaller autoCaller(this);
1078 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1079
1080 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1081 SafeArray<BSTR> groups(mUserData->s.llGroups.size());
1082 size_t i = 0;
1083 for (StringsList::const_iterator it = mUserData->s.llGroups.begin();
1084 it != mUserData->s.llGroups.end();
1085 ++it, i++)
1086 {
1087 Bstr tmp = *it;
1088 tmp.cloneTo(&groups[i]);
1089 }
1090 groups.detachTo(ComSafeArrayOutArg(aGroups));
1091
1092 return S_OK;
1093}
1094
1095STDMETHODIMP Machine::COMSETTER(Groups)(ComSafeArrayIn(IN_BSTR, aGroups))
1096{
1097 AutoCaller autoCaller(this);
1098 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1099
1100 StringsList llGroups;
1101 HRESULT rc = mParent->convertMachineGroups(ComSafeArrayInArg(aGroups), &llGroups);
1102 if (FAILED(rc))
1103 return rc;
1104
1105 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1106
1107 // changing machine groups is possible while the VM is offline
1108 rc = checkStateDependency(OfflineStateDep);
1109 if (FAILED(rc)) return rc;
1110
1111 setModified(IsModified_MachineData);
1112 mUserData.backup();
1113 mUserData->s.llGroups = llGroups;
1114
1115 return S_OK;
1116}
1117
1118STDMETHODIMP Machine::COMGETTER(OSTypeId)(BSTR *aOSTypeId)
1119{
1120 CheckComArgOutPointerValid(aOSTypeId);
1121
1122 AutoCaller autoCaller(this);
1123 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1124
1125 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1126
1127 mUserData->s.strOsType.cloneTo(aOSTypeId);
1128
1129 return S_OK;
1130}
1131
1132STDMETHODIMP Machine::COMSETTER(OSTypeId)(IN_BSTR aOSTypeId)
1133{
1134 CheckComArgStrNotEmptyOrNull(aOSTypeId);
1135
1136 AutoCaller autoCaller(this);
1137 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1138
1139 /* look up the object by Id to check it is valid */
1140 ComPtr<IGuestOSType> guestOSType;
1141 HRESULT rc = mParent->GetGuestOSType(aOSTypeId, guestOSType.asOutParam());
1142 if (FAILED(rc)) return rc;
1143
1144 /* when setting, always use the "etalon" value for consistency -- lookup
1145 * by ID is case-insensitive and the input value may have different case */
1146 Bstr osTypeId;
1147 rc = guestOSType->COMGETTER(Id)(osTypeId.asOutParam());
1148 if (FAILED(rc)) return rc;
1149
1150 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1151
1152 rc = checkStateDependency(MutableStateDep);
1153 if (FAILED(rc)) return rc;
1154
1155 setModified(IsModified_MachineData);
1156 mUserData.backup();
1157 mUserData->s.strOsType = osTypeId;
1158
1159 return S_OK;
1160}
1161
1162
1163STDMETHODIMP Machine::COMGETTER(FirmwareType)(FirmwareType_T *aFirmwareType)
1164{
1165 CheckComArgOutPointerValid(aFirmwareType);
1166
1167 AutoCaller autoCaller(this);
1168 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1169
1170 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1171
1172 *aFirmwareType = mHWData->mFirmwareType;
1173
1174 return S_OK;
1175}
1176
1177STDMETHODIMP Machine::COMSETTER(FirmwareType)(FirmwareType_T aFirmwareType)
1178{
1179 AutoCaller autoCaller(this);
1180 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1181 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1182
1183 HRESULT rc = checkStateDependency(MutableStateDep);
1184 if (FAILED(rc)) return rc;
1185
1186 setModified(IsModified_MachineData);
1187 mHWData.backup();
1188 mHWData->mFirmwareType = aFirmwareType;
1189
1190 return S_OK;
1191}
1192
1193STDMETHODIMP Machine::COMGETTER(KeyboardHIDType)(KeyboardHIDType_T *aKeyboardHIDType)
1194{
1195 CheckComArgOutPointerValid(aKeyboardHIDType);
1196
1197 AutoCaller autoCaller(this);
1198 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1199
1200 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1201
1202 *aKeyboardHIDType = mHWData->mKeyboardHIDType;
1203
1204 return S_OK;
1205}
1206
1207STDMETHODIMP Machine::COMSETTER(KeyboardHIDType)(KeyboardHIDType_T aKeyboardHIDType)
1208{
1209 AutoCaller autoCaller(this);
1210 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1211 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1212
1213 HRESULT rc = checkStateDependency(MutableStateDep);
1214 if (FAILED(rc)) return rc;
1215
1216 setModified(IsModified_MachineData);
1217 mHWData.backup();
1218 mHWData->mKeyboardHIDType = aKeyboardHIDType;
1219
1220 return S_OK;
1221}
1222
1223STDMETHODIMP Machine::COMGETTER(PointingHIDType)(PointingHIDType_T *aPointingHIDType)
1224{
1225 CheckComArgOutPointerValid(aPointingHIDType);
1226
1227 AutoCaller autoCaller(this);
1228 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1229
1230 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1231
1232 *aPointingHIDType = mHWData->mPointingHIDType;
1233
1234 return S_OK;
1235}
1236
1237STDMETHODIMP Machine::COMSETTER(PointingHIDType)(PointingHIDType_T aPointingHIDType)
1238{
1239 AutoCaller autoCaller(this);
1240 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1241 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1242
1243 HRESULT rc = checkStateDependency(MutableStateDep);
1244 if (FAILED(rc)) return rc;
1245
1246 setModified(IsModified_MachineData);
1247 mHWData.backup();
1248 mHWData->mPointingHIDType = aPointingHIDType;
1249
1250 return S_OK;
1251}
1252
1253STDMETHODIMP Machine::COMGETTER(ChipsetType)(ChipsetType_T *aChipsetType)
1254{
1255 CheckComArgOutPointerValid(aChipsetType);
1256
1257 AutoCaller autoCaller(this);
1258 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1259
1260 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1261
1262 *aChipsetType = mHWData->mChipsetType;
1263
1264 return S_OK;
1265}
1266
1267STDMETHODIMP Machine::COMSETTER(ChipsetType)(ChipsetType_T aChipsetType)
1268{
1269 AutoCaller autoCaller(this);
1270 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1271 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1272
1273 HRESULT rc = checkStateDependency(MutableStateDep);
1274 if (FAILED(rc)) return rc;
1275
1276 if (aChipsetType != mHWData->mChipsetType)
1277 {
1278 setModified(IsModified_MachineData);
1279 mHWData.backup();
1280 mHWData->mChipsetType = aChipsetType;
1281
1282 // Resize network adapter array, to be finalized on commit/rollback.
1283 // We must not throw away entries yet, otherwise settings are lost
1284 // without a way to roll back.
1285 size_t newCount = Global::getMaxNetworkAdapters(aChipsetType);
1286 size_t oldCount = mNetworkAdapters.size();
1287 if (newCount > oldCount)
1288 {
1289 mNetworkAdapters.resize(newCount);
1290 for (ULONG slot = oldCount; slot < mNetworkAdapters.size(); slot++)
1291 {
1292 unconst(mNetworkAdapters[slot]).createObject();
1293 mNetworkAdapters[slot]->init(this, slot);
1294 }
1295 }
1296 }
1297
1298 return S_OK;
1299}
1300
1301STDMETHODIMP Machine::COMGETTER(HardwareVersion)(BSTR *aHWVersion)
1302{
1303 CheckComArgOutPointerValid(aHWVersion);
1304
1305 AutoCaller autoCaller(this);
1306 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1307
1308 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1309
1310 mHWData->mHWVersion.cloneTo(aHWVersion);
1311
1312 return S_OK;
1313}
1314
1315STDMETHODIMP Machine::COMSETTER(HardwareVersion)(IN_BSTR aHWVersion)
1316{
1317 /* check known version */
1318 Utf8Str hwVersion = aHWVersion;
1319 if ( hwVersion.compare("1") != 0
1320 && hwVersion.compare("2") != 0)
1321 return setError(E_INVALIDARG,
1322 tr("Invalid hardware version: %ls\n"), aHWVersion);
1323
1324 AutoCaller autoCaller(this);
1325 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1326
1327 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1328
1329 HRESULT rc = checkStateDependency(MutableStateDep);
1330 if (FAILED(rc)) return rc;
1331
1332 setModified(IsModified_MachineData);
1333 mHWData.backup();
1334 mHWData->mHWVersion = hwVersion;
1335
1336 return S_OK;
1337}
1338
1339STDMETHODIMP Machine::COMGETTER(HardwareUUID)(BSTR *aUUID)
1340{
1341 CheckComArgOutPointerValid(aUUID);
1342
1343 AutoCaller autoCaller(this);
1344 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1345
1346 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1347
1348 if (mHWData->mHardwareUUID.isValid())
1349 mHWData->mHardwareUUID.toUtf16().cloneTo(aUUID);
1350 else
1351 mData->mUuid.toUtf16().cloneTo(aUUID);
1352
1353 return S_OK;
1354}
1355
1356STDMETHODIMP Machine::COMSETTER(HardwareUUID)(IN_BSTR aUUID)
1357{
1358 Guid hardwareUUID(aUUID);
1359 if (!hardwareUUID.isValid())
1360 return E_INVALIDARG;
1361
1362 AutoCaller autoCaller(this);
1363 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1364
1365 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1366
1367 HRESULT rc = checkStateDependency(MutableStateDep);
1368 if (FAILED(rc)) return rc;
1369
1370 setModified(IsModified_MachineData);
1371 mHWData.backup();
1372 if (hardwareUUID == mData->mUuid)
1373 mHWData->mHardwareUUID.clear();
1374 else
1375 mHWData->mHardwareUUID = hardwareUUID;
1376
1377 return S_OK;
1378}
1379
1380STDMETHODIMP Machine::COMGETTER(MemorySize)(ULONG *memorySize)
1381{
1382 CheckComArgOutPointerValid(memorySize);
1383
1384 AutoCaller autoCaller(this);
1385 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1386
1387 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1388
1389 *memorySize = mHWData->mMemorySize;
1390
1391 return S_OK;
1392}
1393
1394STDMETHODIMP Machine::COMSETTER(MemorySize)(ULONG memorySize)
1395{
1396 /* check RAM limits */
1397 if ( memorySize < MM_RAM_MIN_IN_MB
1398 || memorySize > MM_RAM_MAX_IN_MB
1399 )
1400 return setError(E_INVALIDARG,
1401 tr("Invalid RAM size: %lu MB (must be in range [%lu, %lu] MB)"),
1402 memorySize, MM_RAM_MIN_IN_MB, MM_RAM_MAX_IN_MB);
1403
1404 AutoCaller autoCaller(this);
1405 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1406
1407 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1408
1409 HRESULT rc = checkStateDependency(MutableStateDep);
1410 if (FAILED(rc)) return rc;
1411
1412 setModified(IsModified_MachineData);
1413 mHWData.backup();
1414 mHWData->mMemorySize = memorySize;
1415
1416 return S_OK;
1417}
1418
1419STDMETHODIMP Machine::COMGETTER(CPUCount)(ULONG *CPUCount)
1420{
1421 CheckComArgOutPointerValid(CPUCount);
1422
1423 AutoCaller autoCaller(this);
1424 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1425
1426 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1427
1428 *CPUCount = mHWData->mCPUCount;
1429
1430 return S_OK;
1431}
1432
1433STDMETHODIMP Machine::COMSETTER(CPUCount)(ULONG CPUCount)
1434{
1435 /* check CPU limits */
1436 if ( CPUCount < SchemaDefs::MinCPUCount
1437 || CPUCount > SchemaDefs::MaxCPUCount
1438 )
1439 return setError(E_INVALIDARG,
1440 tr("Invalid virtual CPU count: %lu (must be in range [%lu, %lu])"),
1441 CPUCount, SchemaDefs::MinCPUCount, SchemaDefs::MaxCPUCount);
1442
1443 AutoCaller autoCaller(this);
1444 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1445
1446 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1447
1448 /* We cant go below the current number of CPUs attached if hotplug is enabled*/
1449 if (mHWData->mCPUHotPlugEnabled)
1450 {
1451 for (unsigned idx = CPUCount; idx < SchemaDefs::MaxCPUCount; idx++)
1452 {
1453 if (mHWData->mCPUAttached[idx])
1454 return setError(E_INVALIDARG,
1455 tr("There is still a CPU attached to socket %lu."
1456 "Detach the CPU before removing the socket"),
1457 CPUCount, idx+1);
1458 }
1459 }
1460
1461 HRESULT rc = checkStateDependency(MutableStateDep);
1462 if (FAILED(rc)) return rc;
1463
1464 setModified(IsModified_MachineData);
1465 mHWData.backup();
1466 mHWData->mCPUCount = CPUCount;
1467
1468 return S_OK;
1469}
1470
1471STDMETHODIMP Machine::COMGETTER(CPUExecutionCap)(ULONG *aExecutionCap)
1472{
1473 CheckComArgOutPointerValid(aExecutionCap);
1474
1475 AutoCaller autoCaller(this);
1476 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1477
1478 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1479
1480 *aExecutionCap = mHWData->mCpuExecutionCap;
1481
1482 return S_OK;
1483}
1484
1485STDMETHODIMP Machine::COMSETTER(CPUExecutionCap)(ULONG aExecutionCap)
1486{
1487 HRESULT rc = S_OK;
1488
1489 /* check throttle limits */
1490 if ( aExecutionCap < 1
1491 || aExecutionCap > 100
1492 )
1493 return setError(E_INVALIDARG,
1494 tr("Invalid CPU execution cap value: %lu (must be in range [%lu, %lu])"),
1495 aExecutionCap, 1, 100);
1496
1497 AutoCaller autoCaller(this);
1498 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1499
1500 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1501
1502 alock.release();
1503 rc = onCPUExecutionCapChange(aExecutionCap);
1504 alock.acquire();
1505 if (FAILED(rc)) return rc;
1506
1507 setModified(IsModified_MachineData);
1508 mHWData.backup();
1509 mHWData->mCpuExecutionCap = aExecutionCap;
1510
1511 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
1512 if (Global::IsOnline(mData->mMachineState))
1513 saveSettings(NULL);
1514
1515 return S_OK;
1516}
1517
1518
1519STDMETHODIMP Machine::COMGETTER(CPUHotPlugEnabled)(BOOL *aEnabled)
1520{
1521 CheckComArgOutPointerValid(aEnabled);
1522
1523 AutoCaller autoCaller(this);
1524 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1525
1526 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1527
1528 *aEnabled = mHWData->mCPUHotPlugEnabled;
1529
1530 return S_OK;
1531}
1532
1533STDMETHODIMP Machine::COMSETTER(CPUHotPlugEnabled)(BOOL aEnabled)
1534{
1535 HRESULT rc = S_OK;
1536
1537 AutoCaller autoCaller(this);
1538 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1539
1540 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1541
1542 rc = checkStateDependency(MutableStateDep);
1543 if (FAILED(rc)) return rc;
1544
1545 if (mHWData->mCPUHotPlugEnabled != aEnabled)
1546 {
1547 if (aEnabled)
1548 {
1549 setModified(IsModified_MachineData);
1550 mHWData.backup();
1551
1552 /* Add the amount of CPUs currently attached */
1553 for (unsigned i = 0; i < mHWData->mCPUCount; i++)
1554 {
1555 mHWData->mCPUAttached[i] = true;
1556 }
1557 }
1558 else
1559 {
1560 /*
1561 * We can disable hotplug only if the amount of maximum CPUs is equal
1562 * to the amount of attached CPUs
1563 */
1564 unsigned cCpusAttached = 0;
1565 unsigned iHighestId = 0;
1566
1567 for (unsigned i = 0; i < SchemaDefs::MaxCPUCount; i++)
1568 {
1569 if (mHWData->mCPUAttached[i])
1570 {
1571 cCpusAttached++;
1572 iHighestId = i;
1573 }
1574 }
1575
1576 if ( (cCpusAttached != mHWData->mCPUCount)
1577 || (iHighestId >= mHWData->mCPUCount))
1578 return setError(E_INVALIDARG,
1579 tr("CPU hotplugging can't be disabled because the maximum number of CPUs is not equal to the amount of CPUs attached"));
1580
1581 setModified(IsModified_MachineData);
1582 mHWData.backup();
1583 }
1584 }
1585
1586 mHWData->mCPUHotPlugEnabled = aEnabled;
1587
1588 return rc;
1589}
1590
1591STDMETHODIMP Machine::COMGETTER(EmulatedUSBCardReaderEnabled)(BOOL *aEnabled)
1592{
1593#ifdef VBOX_WITH_USB_CARDREADER
1594 CheckComArgOutPointerValid(aEnabled);
1595
1596 AutoCaller autoCaller(this);
1597 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1598
1599 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1600
1601 *aEnabled = mHWData->mEmulatedUSBCardReaderEnabled;
1602
1603 return S_OK;
1604#else
1605 NOREF(aEnabled);
1606 return E_NOTIMPL;
1607#endif
1608}
1609
1610STDMETHODIMP Machine::COMSETTER(EmulatedUSBCardReaderEnabled)(BOOL aEnabled)
1611{
1612#ifdef VBOX_WITH_USB_CARDREADER
1613 AutoCaller autoCaller(this);
1614 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1615 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1616
1617 HRESULT rc = checkStateDependency(MutableStateDep);
1618 if (FAILED(rc)) return rc;
1619
1620 setModified(IsModified_MachineData);
1621 mHWData.backup();
1622 mHWData->mEmulatedUSBCardReaderEnabled = aEnabled;
1623
1624 return S_OK;
1625#else
1626 NOREF(aEnabled);
1627 return E_NOTIMPL;
1628#endif
1629}
1630
1631STDMETHODIMP Machine::COMGETTER(HPETEnabled)(BOOL *aEnabled)
1632{
1633 CheckComArgOutPointerValid(aEnabled);
1634
1635 AutoCaller autoCaller(this);
1636 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1637 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1638
1639 *aEnabled = mHWData->mHPETEnabled;
1640
1641 return S_OK;
1642}
1643
1644STDMETHODIMP Machine::COMSETTER(HPETEnabled)(BOOL aEnabled)
1645{
1646 HRESULT rc = S_OK;
1647
1648 AutoCaller autoCaller(this);
1649 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1650 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1651
1652 rc = checkStateDependency(MutableStateDep);
1653 if (FAILED(rc)) return rc;
1654
1655 setModified(IsModified_MachineData);
1656 mHWData.backup();
1657
1658 mHWData->mHPETEnabled = aEnabled;
1659
1660 return rc;
1661}
1662
1663STDMETHODIMP Machine::COMGETTER(VideoCaptureEnabled)(BOOL *fEnabled)
1664{
1665 AutoCaller autoCaller(this);
1666 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1667
1668 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1669
1670 *fEnabled = mHWData->mVideoCaptureEnabled;
1671 return S_OK;
1672}
1673
1674STDMETHODIMP Machine::COMSETTER(VideoCaptureEnabled)(BOOL fEnabled)
1675{
1676 HRESULT rc = S_OK;
1677
1678 AutoCaller autoCaller(this);
1679 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1680 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1681
1682 setModified(IsModified_MachineData);
1683 mHWData.backup();
1684 mHWData->mVideoCaptureEnabled = fEnabled;
1685
1686 alock.release();
1687 rc = onVideoCaptureChange();
1688 alock.acquire();
1689 if (FAILED(rc))
1690 {
1691 /*
1692 * Normally we would do the actual change _after_ onVideoCaptureChange() succeeded.
1693 * We cannot do this because that function uses Machine::GetVideoCaptureEnabled to
1694 * determine if it should start or stop capturing. Therefore we need to manually
1695 * undo change.
1696 */
1697 mHWData->mVideoCaptureEnabled = mHWData.backedUpData()->mVideoCaptureEnabled;
1698 return rc;
1699 }
1700
1701 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
1702 if (Global::IsOnline(mData->mMachineState))
1703 saveSettings(NULL);
1704
1705 return rc;
1706}
1707
1708STDMETHODIMP Machine::COMGETTER(VideoCaptureScreens)(ComSafeArrayOut(BOOL, aScreens))
1709{
1710 CheckComArgOutSafeArrayPointerValid(aScreens);
1711
1712 AutoCaller autoCaller(this);
1713 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1714
1715 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1716
1717 SafeArray<BOOL> screens(mHWData->mMonitorCount);
1718 for (unsigned i = 0; i < screens.size(); i++)
1719 screens[i] = mHWData->maVideoCaptureScreens[i];
1720 screens.detachTo(ComSafeArrayOutArg(aScreens));
1721 return S_OK;
1722}
1723
1724STDMETHODIMP Machine::COMSETTER(VideoCaptureScreens)(ComSafeArrayIn(BOOL, aScreens))
1725{
1726 SafeArray<BOOL> screens(ComSafeArrayInArg(aScreens));
1727 AssertReturn(screens.size() <= RT_ELEMENTS(mHWData->maVideoCaptureScreens), E_INVALIDARG);
1728 bool fChanged = false;
1729
1730 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1731
1732 for (unsigned i = 0; i < screens.size(); i++)
1733 {
1734 if (mHWData->maVideoCaptureScreens[i] != RT_BOOL(screens[i]))
1735 {
1736 mHWData->maVideoCaptureScreens[i] = RT_BOOL(screens[i]);
1737 fChanged = true;
1738 }
1739 }
1740 if (fChanged)
1741 {
1742 alock.release();
1743 HRESULT rc = onVideoCaptureChange();
1744 alock.acquire();
1745 if (FAILED(rc)) return rc;
1746 setModified(IsModified_MachineData);
1747
1748 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
1749 if (Global::IsOnline(mData->mMachineState))
1750 saveSettings(NULL);
1751 }
1752
1753 return S_OK;
1754}
1755
1756STDMETHODIMP Machine::COMGETTER(VideoCaptureFile)(BSTR *apFile)
1757{
1758 AutoCaller autoCaller(this);
1759 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1760
1761 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1762 if (mHWData->mVideoCaptureFile.isEmpty())
1763 {
1764 Utf8Str defaultFile;
1765 getDefaultVideoCaptureFile(defaultFile);
1766 defaultFile.cloneTo(apFile);
1767 }
1768 else
1769 mHWData->mVideoCaptureFile.cloneTo(apFile);
1770 return S_OK;
1771}
1772
1773STDMETHODIMP Machine::COMSETTER(VideoCaptureFile)(IN_BSTR aFile)
1774{
1775 Utf8Str strFile(aFile);
1776 AutoCaller autoCaller(this);
1777 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1778
1779 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1780
1781 if ( Global::IsOnline(mData->mMachineState)
1782 && mHWData->mVideoCaptureEnabled)
1783 return setError(E_INVALIDARG, tr("Cannot change parameters while capturing is enabled"));
1784
1785 if (!RTPathStartsWithRoot(strFile.c_str()))
1786 return setError(E_INVALIDARG, tr("Video capture file name '%s' is not absolute"), strFile.c_str());
1787
1788 if (!strFile.isEmpty())
1789 {
1790 Utf8Str defaultFile;
1791 getDefaultVideoCaptureFile(defaultFile);
1792 if (!RTPathCompare(strFile.c_str(), defaultFile.c_str()))
1793 strFile.setNull();
1794 }
1795
1796 setModified(IsModified_MachineData);
1797 mHWData.backup();
1798 mHWData->mVideoCaptureFile = strFile;
1799
1800 return S_OK;
1801}
1802
1803STDMETHODIMP Machine::COMGETTER(VideoCaptureWidth)(ULONG *aHorzRes)
1804{
1805 AutoCaller autoCaller(this);
1806 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1807
1808 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1809 *aHorzRes = mHWData->mVideoCaptureWidth;
1810 return S_OK;
1811}
1812
1813STDMETHODIMP Machine::COMSETTER(VideoCaptureWidth)(ULONG aHorzRes)
1814{
1815 AutoCaller autoCaller(this);
1816 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1817
1818 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1819
1820 if ( Global::IsOnline(mData->mMachineState)
1821 && mHWData->mVideoCaptureEnabled)
1822 return setError(E_INVALIDARG, tr("Cannot change parameters while capturing is enabled"));
1823
1824 setModified(IsModified_MachineData);
1825 mHWData.backup();
1826 mHWData->mVideoCaptureWidth = aHorzRes;
1827
1828 return S_OK;
1829}
1830
1831STDMETHODIMP Machine::COMGETTER(VideoCaptureHeight)(ULONG *aVertRes)
1832{
1833 AutoCaller autoCaller(this);
1834 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1835
1836 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1837 *aVertRes = mHWData->mVideoCaptureHeight;
1838 return S_OK;
1839}
1840
1841STDMETHODIMP Machine::COMSETTER(VideoCaptureHeight)(ULONG aVertRes)
1842{
1843 AutoCaller autoCaller(this);
1844 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1845
1846 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1847
1848 if ( Global::IsOnline(mData->mMachineState)
1849 && mHWData->mVideoCaptureEnabled)
1850 return setError(E_INVALIDARG, tr("Cannot change parameters while capturing is enabled"));
1851
1852 setModified(IsModified_MachineData);
1853 mHWData.backup();
1854 mHWData->mVideoCaptureHeight = aVertRes;
1855
1856 return S_OK;
1857}
1858
1859STDMETHODIMP Machine::COMGETTER(VideoCaptureRate)(ULONG *aRate)
1860{
1861 AutoCaller autoCaller(this);
1862 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1863
1864 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1865 *aRate = mHWData->mVideoCaptureRate;
1866 return S_OK;
1867}
1868
1869STDMETHODIMP Machine::COMSETTER(VideoCaptureRate)(ULONG aRate)
1870{
1871 AutoCaller autoCaller(this);
1872 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1873
1874 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1875
1876 if ( Global::IsOnline(mData->mMachineState)
1877 && mHWData->mVideoCaptureEnabled)
1878 return setError(E_INVALIDARG, tr("Cannot change parameters while capturing is enabled"));
1879
1880 setModified(IsModified_MachineData);
1881 mHWData.backup();
1882 mHWData->mVideoCaptureRate = aRate;
1883
1884 return S_OK;
1885}
1886
1887STDMETHODIMP Machine::COMGETTER(VideoCaptureFPS)(ULONG *aFPS)
1888{
1889 AutoCaller autoCaller(this);
1890 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1891
1892 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1893 *aFPS = mHWData->mVideoCaptureFPS;
1894 return S_OK;
1895}
1896
1897STDMETHODIMP Machine::COMSETTER(VideoCaptureFPS)(ULONG aFPS)
1898{
1899 AutoCaller autoCaller(this);
1900 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1901
1902 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1903
1904 if ( Global::IsOnline(mData->mMachineState)
1905 && mHWData->mVideoCaptureEnabled)
1906 return setError(E_INVALIDARG, tr("Cannot change parameters while capturing is enabled"));
1907
1908 setModified(IsModified_MachineData);
1909 mHWData.backup();
1910 mHWData->mVideoCaptureFPS = aFPS;
1911
1912 return S_OK;
1913}
1914
1915STDMETHODIMP Machine::COMGETTER(GraphicsControllerType)(GraphicsControllerType_T *aGraphicsControllerType)
1916{
1917 CheckComArgOutPointerValid(aGraphicsControllerType);
1918
1919 AutoCaller autoCaller(this);
1920 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1921
1922 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1923
1924 *aGraphicsControllerType = mHWData->mGraphicsControllerType;
1925
1926 return S_OK;
1927}
1928
1929STDMETHODIMP Machine::COMSETTER(GraphicsControllerType)(GraphicsControllerType_T aGraphicsControllerType)
1930{
1931 switch (aGraphicsControllerType)
1932 {
1933 case GraphicsControllerType_Null:
1934 case GraphicsControllerType_VBoxVGA:
1935 break;
1936 default:
1937 return setError(E_INVALIDARG, tr("The graphics controller type (%d) is invalid"), aGraphicsControllerType);
1938 }
1939
1940 AutoCaller autoCaller(this);
1941 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1942
1943 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1944
1945 HRESULT rc = checkStateDependency(MutableStateDep);
1946 if (FAILED(rc)) return rc;
1947
1948 setModified(IsModified_MachineData);
1949 mHWData.backup();
1950 mHWData->mGraphicsControllerType = aGraphicsControllerType;
1951
1952 return S_OK;
1953}
1954
1955STDMETHODIMP Machine::COMGETTER(VRAMSize)(ULONG *memorySize)
1956{
1957 CheckComArgOutPointerValid(memorySize);
1958
1959 AutoCaller autoCaller(this);
1960 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1961
1962 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1963
1964 *memorySize = mHWData->mVRAMSize;
1965
1966 return S_OK;
1967}
1968
1969STDMETHODIMP Machine::COMSETTER(VRAMSize)(ULONG memorySize)
1970{
1971 /* check VRAM limits */
1972 if (memorySize < SchemaDefs::MinGuestVRAM ||
1973 memorySize > SchemaDefs::MaxGuestVRAM)
1974 return setError(E_INVALIDARG,
1975 tr("Invalid VRAM size: %lu MB (must be in range [%lu, %lu] MB)"),
1976 memorySize, SchemaDefs::MinGuestVRAM, SchemaDefs::MaxGuestVRAM);
1977
1978 AutoCaller autoCaller(this);
1979 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1980
1981 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1982
1983 HRESULT rc = checkStateDependency(MutableStateDep);
1984 if (FAILED(rc)) return rc;
1985
1986 setModified(IsModified_MachineData);
1987 mHWData.backup();
1988 mHWData->mVRAMSize = memorySize;
1989
1990 return S_OK;
1991}
1992
1993/** @todo this method should not be public */
1994STDMETHODIMP Machine::COMGETTER(MemoryBalloonSize)(ULONG *memoryBalloonSize)
1995{
1996 CheckComArgOutPointerValid(memoryBalloonSize);
1997
1998 AutoCaller autoCaller(this);
1999 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2000
2001 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2002
2003 *memoryBalloonSize = mHWData->mMemoryBalloonSize;
2004
2005 return S_OK;
2006}
2007
2008/**
2009 * Set the memory balloon size.
2010 *
2011 * This method is also called from IGuest::COMSETTER(MemoryBalloonSize) so
2012 * we have to make sure that we never call IGuest from here.
2013 */
2014STDMETHODIMP Machine::COMSETTER(MemoryBalloonSize)(ULONG memoryBalloonSize)
2015{
2016 /* This must match GMMR0Init; currently we only support memory ballooning on all 64-bit hosts except Mac OS X */
2017#if HC_ARCH_BITS == 64 && (defined(RT_OS_WINDOWS) || defined(RT_OS_SOLARIS) || defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD))
2018 /* check limits */
2019 if (memoryBalloonSize >= VMMDEV_MAX_MEMORY_BALLOON(mHWData->mMemorySize))
2020 return setError(E_INVALIDARG,
2021 tr("Invalid memory balloon size: %lu MB (must be in range [%lu, %lu] MB)"),
2022 memoryBalloonSize, 0, VMMDEV_MAX_MEMORY_BALLOON(mHWData->mMemorySize));
2023
2024 AutoCaller autoCaller(this);
2025 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2026
2027 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2028
2029 setModified(IsModified_MachineData);
2030 mHWData.backup();
2031 mHWData->mMemoryBalloonSize = memoryBalloonSize;
2032
2033 return S_OK;
2034#else
2035 NOREF(memoryBalloonSize);
2036 return setError(E_NOTIMPL, tr("Memory ballooning is only supported on 64-bit hosts"));
2037#endif
2038}
2039
2040STDMETHODIMP Machine::COMGETTER(PageFusionEnabled) (BOOL *aEnabled)
2041{
2042 CheckComArgOutPointerValid(aEnabled);
2043
2044 AutoCaller autoCaller(this);
2045 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2046
2047 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2048
2049 *aEnabled = mHWData->mPageFusionEnabled;
2050 return S_OK;
2051}
2052
2053STDMETHODIMP Machine::COMSETTER(PageFusionEnabled) (BOOL aEnabled)
2054{
2055#ifdef VBOX_WITH_PAGE_SHARING
2056 AutoCaller autoCaller(this);
2057 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2058
2059 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2060
2061 /** @todo must support changes for running vms and keep this in sync with IGuest. */
2062 setModified(IsModified_MachineData);
2063 mHWData.backup();
2064 mHWData->mPageFusionEnabled = aEnabled;
2065 return S_OK;
2066#else
2067 NOREF(aEnabled);
2068 return setError(E_NOTIMPL, tr("Page fusion is only supported on 64-bit hosts"));
2069#endif
2070}
2071
2072STDMETHODIMP Machine::COMGETTER(Accelerate3DEnabled)(BOOL *aEnabled)
2073{
2074 CheckComArgOutPointerValid(aEnabled);
2075
2076 AutoCaller autoCaller(this);
2077 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2078
2079 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2080
2081 *aEnabled = mHWData->mAccelerate3DEnabled;
2082
2083 return S_OK;
2084}
2085
2086STDMETHODIMP Machine::COMSETTER(Accelerate3DEnabled)(BOOL enable)
2087{
2088 AutoCaller autoCaller(this);
2089 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2090
2091 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2092
2093 HRESULT rc = checkStateDependency(MutableStateDep);
2094 if (FAILED(rc)) return rc;
2095
2096 /** @todo check validity! */
2097
2098 setModified(IsModified_MachineData);
2099 mHWData.backup();
2100 mHWData->mAccelerate3DEnabled = enable;
2101
2102 return S_OK;
2103}
2104
2105
2106STDMETHODIMP Machine::COMGETTER(Accelerate2DVideoEnabled)(BOOL *aEnabled)
2107{
2108 CheckComArgOutPointerValid(aEnabled);
2109
2110 AutoCaller autoCaller(this);
2111 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2112
2113 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2114
2115 *aEnabled = mHWData->mAccelerate2DVideoEnabled;
2116
2117 return S_OK;
2118}
2119
2120STDMETHODIMP Machine::COMSETTER(Accelerate2DVideoEnabled)(BOOL enable)
2121{
2122 AutoCaller autoCaller(this);
2123 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2124
2125 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2126
2127 HRESULT rc = checkStateDependency(MutableStateDep);
2128 if (FAILED(rc)) return rc;
2129
2130 /** @todo check validity! */
2131
2132 setModified(IsModified_MachineData);
2133 mHWData.backup();
2134 mHWData->mAccelerate2DVideoEnabled = enable;
2135
2136 return S_OK;
2137}
2138
2139STDMETHODIMP Machine::COMGETTER(MonitorCount)(ULONG *monitorCount)
2140{
2141 CheckComArgOutPointerValid(monitorCount);
2142
2143 AutoCaller autoCaller(this);
2144 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2145
2146 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2147
2148 *monitorCount = mHWData->mMonitorCount;
2149
2150 return S_OK;
2151}
2152
2153STDMETHODIMP Machine::COMSETTER(MonitorCount)(ULONG monitorCount)
2154{
2155 /* make sure monitor count is a sensible number */
2156 if (monitorCount < 1 || monitorCount > SchemaDefs::MaxGuestMonitors)
2157 return setError(E_INVALIDARG,
2158 tr("Invalid monitor count: %lu (must be in range [%lu, %lu])"),
2159 monitorCount, 1, SchemaDefs::MaxGuestMonitors);
2160
2161 AutoCaller autoCaller(this);
2162 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2163
2164 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2165
2166 HRESULT rc = checkStateDependency(MutableStateDep);
2167 if (FAILED(rc)) return rc;
2168
2169 setModified(IsModified_MachineData);
2170 mHWData.backup();
2171 mHWData->mMonitorCount = monitorCount;
2172
2173 return S_OK;
2174}
2175
2176STDMETHODIMP Machine::COMGETTER(BIOSSettings)(IBIOSSettings **biosSettings)
2177{
2178 CheckComArgOutPointerValid(biosSettings);
2179
2180 AutoCaller autoCaller(this);
2181 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2182
2183 /* mBIOSSettings is constant during life time, no need to lock */
2184 mBIOSSettings.queryInterfaceTo(biosSettings);
2185
2186 return S_OK;
2187}
2188
2189STDMETHODIMP Machine::GetCPUProperty(CPUPropertyType_T property, BOOL *aVal)
2190{
2191 CheckComArgOutPointerValid(aVal);
2192
2193 AutoCaller autoCaller(this);
2194 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2195
2196 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2197
2198 switch (property)
2199 {
2200 case CPUPropertyType_PAE:
2201 *aVal = mHWData->mPAEEnabled;
2202 break;
2203
2204 case CPUPropertyType_Synthetic:
2205 *aVal = mHWData->mSyntheticCpu;
2206 break;
2207
2208 case CPUPropertyType_LongMode:
2209 if (mHWData->mLongMode == settings::Hardware::LongMode_Enabled)
2210 *aVal = TRUE;
2211 else if (mHWData->mLongMode == settings::Hardware::LongMode_Disabled)
2212 *aVal = FALSE;
2213#if HC_ARCH_BITS == 64
2214 else
2215 *aVal = TRUE;
2216#else
2217 else
2218 {
2219 *aVal = FALSE;
2220
2221 ComPtr<IGuestOSType> ptrGuestOSType;
2222 HRESULT hrc2 = mParent->GetGuestOSType(Bstr(mUserData->s.strOsType).raw(), ptrGuestOSType.asOutParam());
2223 if (SUCCEEDED(hrc2))
2224 {
2225 BOOL fIs64Bit = FALSE;
2226 hrc2 = ptrGuestOSType->COMGETTER(Is64Bit)(&fIs64Bit); AssertComRC(hrc2);
2227 if (SUCCEEDED(hrc2) && fIs64Bit)
2228 {
2229 ComObjPtr<Host> ptrHost = mParent->host();
2230 alock.release();
2231
2232 hrc2 = ptrHost->GetProcessorFeature(ProcessorFeature_LongMode, aVal); AssertComRC(hrc2);
2233 if (FAILED(hrc2))
2234 *aVal = FALSE;
2235 }
2236 }
2237 }
2238#endif
2239 break;
2240
2241 default:
2242 return E_INVALIDARG;
2243 }
2244 return S_OK;
2245}
2246
2247STDMETHODIMP Machine::SetCPUProperty(CPUPropertyType_T property, BOOL aVal)
2248{
2249 AutoCaller autoCaller(this);
2250 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2251
2252 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2253
2254 HRESULT rc = checkStateDependency(MutableStateDep);
2255 if (FAILED(rc)) return rc;
2256
2257 switch (property)
2258 {
2259 case CPUPropertyType_PAE:
2260 setModified(IsModified_MachineData);
2261 mHWData.backup();
2262 mHWData->mPAEEnabled = !!aVal;
2263 break;
2264
2265 case CPUPropertyType_Synthetic:
2266 setModified(IsModified_MachineData);
2267 mHWData.backup();
2268 mHWData->mSyntheticCpu = !!aVal;
2269 break;
2270
2271 case CPUPropertyType_LongMode:
2272 setModified(IsModified_MachineData);
2273 mHWData.backup();
2274 mHWData->mLongMode = !aVal ? settings::Hardware::LongMode_Disabled : settings::Hardware::LongMode_Enabled;
2275 break;
2276
2277 default:
2278 return E_INVALIDARG;
2279 }
2280 return S_OK;
2281}
2282
2283STDMETHODIMP Machine::GetCPUIDLeaf(ULONG aId, ULONG *aValEax, ULONG *aValEbx, ULONG *aValEcx, ULONG *aValEdx)
2284{
2285 CheckComArgOutPointerValid(aValEax);
2286 CheckComArgOutPointerValid(aValEbx);
2287 CheckComArgOutPointerValid(aValEcx);
2288 CheckComArgOutPointerValid(aValEdx);
2289
2290 AutoCaller autoCaller(this);
2291 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2292
2293 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2294
2295 switch(aId)
2296 {
2297 case 0x0:
2298 case 0x1:
2299 case 0x2:
2300 case 0x3:
2301 case 0x4:
2302 case 0x5:
2303 case 0x6:
2304 case 0x7:
2305 case 0x8:
2306 case 0x9:
2307 case 0xA:
2308 if (mHWData->mCpuIdStdLeafs[aId].ulId != aId)
2309 return E_INVALIDARG;
2310
2311 *aValEax = mHWData->mCpuIdStdLeafs[aId].ulEax;
2312 *aValEbx = mHWData->mCpuIdStdLeafs[aId].ulEbx;
2313 *aValEcx = mHWData->mCpuIdStdLeafs[aId].ulEcx;
2314 *aValEdx = mHWData->mCpuIdStdLeafs[aId].ulEdx;
2315 break;
2316
2317 case 0x80000000:
2318 case 0x80000001:
2319 case 0x80000002:
2320 case 0x80000003:
2321 case 0x80000004:
2322 case 0x80000005:
2323 case 0x80000006:
2324 case 0x80000007:
2325 case 0x80000008:
2326 case 0x80000009:
2327 case 0x8000000A:
2328 if (mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId != aId)
2329 return E_INVALIDARG;
2330
2331 *aValEax = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEax;
2332 *aValEbx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEbx;
2333 *aValEcx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEcx;
2334 *aValEdx = mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEdx;
2335 break;
2336
2337 default:
2338 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
2339 }
2340 return S_OK;
2341}
2342
2343STDMETHODIMP Machine::SetCPUIDLeaf(ULONG aId, ULONG aValEax, ULONG aValEbx, ULONG aValEcx, ULONG aValEdx)
2344{
2345 AutoCaller autoCaller(this);
2346 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2347
2348 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2349
2350 HRESULT rc = checkStateDependency(MutableStateDep);
2351 if (FAILED(rc)) return rc;
2352
2353 switch(aId)
2354 {
2355 case 0x0:
2356 case 0x1:
2357 case 0x2:
2358 case 0x3:
2359 case 0x4:
2360 case 0x5:
2361 case 0x6:
2362 case 0x7:
2363 case 0x8:
2364 case 0x9:
2365 case 0xA:
2366 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdStdLeafs) == 0xB);
2367 AssertRelease(aId < RT_ELEMENTS(mHWData->mCpuIdStdLeafs));
2368 setModified(IsModified_MachineData);
2369 mHWData.backup();
2370 mHWData->mCpuIdStdLeafs[aId].ulId = aId;
2371 mHWData->mCpuIdStdLeafs[aId].ulEax = aValEax;
2372 mHWData->mCpuIdStdLeafs[aId].ulEbx = aValEbx;
2373 mHWData->mCpuIdStdLeafs[aId].ulEcx = aValEcx;
2374 mHWData->mCpuIdStdLeafs[aId].ulEdx = aValEdx;
2375 break;
2376
2377 case 0x80000000:
2378 case 0x80000001:
2379 case 0x80000002:
2380 case 0x80000003:
2381 case 0x80000004:
2382 case 0x80000005:
2383 case 0x80000006:
2384 case 0x80000007:
2385 case 0x80000008:
2386 case 0x80000009:
2387 case 0x8000000A:
2388 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdExtLeafs) == 0xB);
2389 AssertRelease(aId - 0x80000000 < RT_ELEMENTS(mHWData->mCpuIdExtLeafs));
2390 setModified(IsModified_MachineData);
2391 mHWData.backup();
2392 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId = aId;
2393 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEax = aValEax;
2394 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEbx = aValEbx;
2395 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEcx = aValEcx;
2396 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulEdx = aValEdx;
2397 break;
2398
2399 default:
2400 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
2401 }
2402 return S_OK;
2403}
2404
2405STDMETHODIMP Machine::RemoveCPUIDLeaf(ULONG aId)
2406{
2407 AutoCaller autoCaller(this);
2408 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2409
2410 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2411
2412 HRESULT rc = checkStateDependency(MutableStateDep);
2413 if (FAILED(rc)) return rc;
2414
2415 switch(aId)
2416 {
2417 case 0x0:
2418 case 0x1:
2419 case 0x2:
2420 case 0x3:
2421 case 0x4:
2422 case 0x5:
2423 case 0x6:
2424 case 0x7:
2425 case 0x8:
2426 case 0x9:
2427 case 0xA:
2428 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdStdLeafs) == 0xB);
2429 AssertRelease(aId < RT_ELEMENTS(mHWData->mCpuIdStdLeafs));
2430 setModified(IsModified_MachineData);
2431 mHWData.backup();
2432 /* Invalidate leaf. */
2433 mHWData->mCpuIdStdLeafs[aId].ulId = UINT32_MAX;
2434 break;
2435
2436 case 0x80000000:
2437 case 0x80000001:
2438 case 0x80000002:
2439 case 0x80000003:
2440 case 0x80000004:
2441 case 0x80000005:
2442 case 0x80000006:
2443 case 0x80000007:
2444 case 0x80000008:
2445 case 0x80000009:
2446 case 0x8000000A:
2447 AssertCompile(RT_ELEMENTS(mHWData->mCpuIdExtLeafs) == 0xB);
2448 AssertRelease(aId - 0x80000000 < RT_ELEMENTS(mHWData->mCpuIdExtLeafs));
2449 setModified(IsModified_MachineData);
2450 mHWData.backup();
2451 /* Invalidate leaf. */
2452 mHWData->mCpuIdExtLeafs[aId - 0x80000000].ulId = UINT32_MAX;
2453 break;
2454
2455 default:
2456 return setError(E_INVALIDARG, tr("CpuId override leaf %#x is out of range"), aId);
2457 }
2458 return S_OK;
2459}
2460
2461STDMETHODIMP Machine::RemoveAllCPUIDLeaves()
2462{
2463 AutoCaller autoCaller(this);
2464 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2465
2466 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2467
2468 HRESULT rc = checkStateDependency(MutableStateDep);
2469 if (FAILED(rc)) return rc;
2470
2471 setModified(IsModified_MachineData);
2472 mHWData.backup();
2473
2474 /* Invalidate all standard leafs. */
2475 for (unsigned i = 0; i < RT_ELEMENTS(mHWData->mCpuIdStdLeafs); i++)
2476 mHWData->mCpuIdStdLeafs[i].ulId = UINT32_MAX;
2477
2478 /* Invalidate all extended leafs. */
2479 for (unsigned i = 0; i < RT_ELEMENTS(mHWData->mCpuIdExtLeafs); i++)
2480 mHWData->mCpuIdExtLeafs[i].ulId = UINT32_MAX;
2481
2482 return S_OK;
2483}
2484
2485STDMETHODIMP Machine::GetHWVirtExProperty(HWVirtExPropertyType_T property, BOOL *aVal)
2486{
2487 CheckComArgOutPointerValid(aVal);
2488
2489 AutoCaller autoCaller(this);
2490 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2491
2492 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2493
2494 switch(property)
2495 {
2496 case HWVirtExPropertyType_Enabled:
2497 *aVal = mHWData->mHWVirtExEnabled;
2498 break;
2499
2500 case HWVirtExPropertyType_VPID:
2501 *aVal = mHWData->mHWVirtExVPIDEnabled;
2502 break;
2503
2504 case HWVirtExPropertyType_NestedPaging:
2505 *aVal = mHWData->mHWVirtExNestedPagingEnabled;
2506 break;
2507
2508 case HWVirtExPropertyType_UnrestrictedExecution:
2509 *aVal = mHWData->mHWVirtExUXEnabled;
2510 break;
2511
2512 case HWVirtExPropertyType_LargePages:
2513 *aVal = mHWData->mHWVirtExLargePagesEnabled;
2514#if defined(DEBUG_bird) && defined(RT_OS_LINUX) /* This feature is deadly here */
2515 *aVal = FALSE;
2516#endif
2517 break;
2518
2519 case HWVirtExPropertyType_Force:
2520 *aVal = mHWData->mHWVirtExForceEnabled;
2521 break;
2522
2523 default:
2524 return E_INVALIDARG;
2525 }
2526 return S_OK;
2527}
2528
2529STDMETHODIMP Machine::SetHWVirtExProperty(HWVirtExPropertyType_T property, BOOL aVal)
2530{
2531 AutoCaller autoCaller(this);
2532 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2533
2534 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2535
2536 HRESULT rc = checkStateDependency(MutableStateDep);
2537 if (FAILED(rc)) return rc;
2538
2539 switch(property)
2540 {
2541 case HWVirtExPropertyType_Enabled:
2542 setModified(IsModified_MachineData);
2543 mHWData.backup();
2544 mHWData->mHWVirtExEnabled = !!aVal;
2545 break;
2546
2547 case HWVirtExPropertyType_VPID:
2548 setModified(IsModified_MachineData);
2549 mHWData.backup();
2550 mHWData->mHWVirtExVPIDEnabled = !!aVal;
2551 break;
2552
2553 case HWVirtExPropertyType_NestedPaging:
2554 setModified(IsModified_MachineData);
2555 mHWData.backup();
2556 mHWData->mHWVirtExNestedPagingEnabled = !!aVal;
2557 break;
2558
2559 case HWVirtExPropertyType_UnrestrictedExecution:
2560 setModified(IsModified_MachineData);
2561 mHWData.backup();
2562 mHWData->mHWVirtExUXEnabled = !!aVal;
2563 break;
2564
2565 case HWVirtExPropertyType_LargePages:
2566 setModified(IsModified_MachineData);
2567 mHWData.backup();
2568 mHWData->mHWVirtExLargePagesEnabled = !!aVal;
2569 break;
2570
2571 case HWVirtExPropertyType_Force:
2572 setModified(IsModified_MachineData);
2573 mHWData.backup();
2574 mHWData->mHWVirtExForceEnabled = !!aVal;
2575 break;
2576
2577 default:
2578 return E_INVALIDARG;
2579 }
2580
2581 return S_OK;
2582}
2583
2584STDMETHODIMP Machine::COMGETTER(SnapshotFolder)(BSTR *aSnapshotFolder)
2585{
2586 CheckComArgOutPointerValid(aSnapshotFolder);
2587
2588 AutoCaller autoCaller(this);
2589 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2590
2591 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2592
2593 Utf8Str strFullSnapshotFolder;
2594 calculateFullPath(mUserData->s.strSnapshotFolder, strFullSnapshotFolder);
2595 strFullSnapshotFolder.cloneTo(aSnapshotFolder);
2596
2597 return S_OK;
2598}
2599
2600STDMETHODIMP Machine::COMSETTER(SnapshotFolder)(IN_BSTR aSnapshotFolder)
2601{
2602 /* @todo (r=dmik):
2603 * 1. Allow to change the name of the snapshot folder containing snapshots
2604 * 2. Rename the folder on disk instead of just changing the property
2605 * value (to be smart and not to leave garbage). Note that it cannot be
2606 * done here because the change may be rolled back. Thus, the right
2607 * place is #saveSettings().
2608 */
2609
2610 AutoCaller autoCaller(this);
2611 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2612
2613 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2614
2615 HRESULT rc = checkStateDependency(MutableStateDep);
2616 if (FAILED(rc)) return rc;
2617
2618 if (!mData->mCurrentSnapshot.isNull())
2619 return setError(E_FAIL,
2620 tr("The snapshot folder of a machine with snapshots cannot be changed (please delete all snapshots first)"));
2621
2622 Utf8Str strSnapshotFolder0(aSnapshotFolder); // keep original
2623
2624 Utf8Str strSnapshotFolder(strSnapshotFolder0);
2625 if (strSnapshotFolder.isEmpty())
2626 strSnapshotFolder = "Snapshots";
2627 int vrc = calculateFullPath(strSnapshotFolder,
2628 strSnapshotFolder);
2629 if (RT_FAILURE(vrc))
2630 return setError(E_FAIL,
2631 tr("Invalid snapshot folder '%ls' (%Rrc)"),
2632 aSnapshotFolder, vrc);
2633
2634 setModified(IsModified_MachineData);
2635 mUserData.backup();
2636
2637 copyPathRelativeToMachine(strSnapshotFolder, mUserData->s.strSnapshotFolder);
2638
2639 return S_OK;
2640}
2641
2642STDMETHODIMP Machine::COMGETTER(MediumAttachments)(ComSafeArrayOut(IMediumAttachment*, aAttachments))
2643{
2644 CheckComArgOutSafeArrayPointerValid(aAttachments);
2645
2646 AutoCaller autoCaller(this);
2647 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2648
2649 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2650
2651 SafeIfaceArray<IMediumAttachment> attachments(mMediaData->mAttachments);
2652 attachments.detachTo(ComSafeArrayOutArg(aAttachments));
2653
2654 return S_OK;
2655}
2656
2657STDMETHODIMP Machine::COMGETTER(VRDEServer)(IVRDEServer **vrdeServer)
2658{
2659 CheckComArgOutPointerValid(vrdeServer);
2660
2661 AutoCaller autoCaller(this);
2662 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2663
2664 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2665
2666 Assert(!!mVRDEServer);
2667 mVRDEServer.queryInterfaceTo(vrdeServer);
2668
2669 return S_OK;
2670}
2671
2672STDMETHODIMP Machine::COMGETTER(AudioAdapter)(IAudioAdapter **audioAdapter)
2673{
2674 CheckComArgOutPointerValid(audioAdapter);
2675
2676 AutoCaller autoCaller(this);
2677 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2678
2679 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2680
2681 mAudioAdapter.queryInterfaceTo(audioAdapter);
2682 return S_OK;
2683}
2684
2685STDMETHODIMP Machine::COMGETTER(USBControllers)(ComSafeArrayOut(IUSBController *, aUSBControllers))
2686{
2687#ifdef VBOX_WITH_VUSB
2688 CheckComArgOutPointerValid(aUSBControllers);
2689
2690 AutoCaller autoCaller(this);
2691 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2692
2693 clearError();
2694 MultiResult rc(S_OK);
2695
2696# ifdef VBOX_WITH_USB
2697 rc = mParent->host()->checkUSBProxyService();
2698 if (FAILED(rc)) return rc;
2699# endif
2700
2701 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2702
2703 SafeIfaceArray<IUSBController> ctrls(*mUSBControllers.data());
2704 ctrls.detachTo(ComSafeArrayOutArg(aUSBControllers));
2705 return S_OK;
2706#else
2707 /* Note: The GUI depends on this method returning E_NOTIMPL with no
2708 * extended error info to indicate that USB is simply not available
2709 * (w/o treating it as a failure), for example, as in OSE */
2710 NOREF(aUSBControllers);
2711 ReturnComNotImplemented();
2712#endif /* VBOX_WITH_VUSB */
2713}
2714
2715STDMETHODIMP Machine::COMGETTER(USBDeviceFilters)(IUSBDeviceFilters **aUSBDeviceFilters)
2716{
2717#ifdef VBOX_WITH_VUSB
2718 CheckComArgOutPointerValid(aUSBDeviceFilters);
2719
2720 AutoCaller autoCaller(this);
2721 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2722
2723 clearError();
2724 MultiResult rc(S_OK);
2725
2726# ifdef VBOX_WITH_USB
2727 rc = mParent->host()->checkUSBProxyService();
2728 if (FAILED(rc)) return rc;
2729# endif
2730
2731 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2732
2733 return rc = mUSBDeviceFilters.queryInterfaceTo(aUSBDeviceFilters);
2734#else
2735 /* Note: The GUI depends on this method returning E_NOTIMPL with no
2736 * extended error info to indicate that USB is simply not available
2737 * (w/o treating it as a failure), for example, as in OSE */
2738 NOREF(aUSBDeviceFilters);
2739 ReturnComNotImplemented();
2740#endif /* VBOX_WITH_VUSB */
2741}
2742
2743STDMETHODIMP Machine::COMGETTER(SettingsFilePath)(BSTR *aFilePath)
2744{
2745 CheckComArgOutPointerValid(aFilePath);
2746
2747 AutoLimitedCaller autoCaller(this);
2748 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2749
2750 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2751
2752 mData->m_strConfigFileFull.cloneTo(aFilePath);
2753 return S_OK;
2754}
2755
2756STDMETHODIMP Machine::COMGETTER(SettingsModified)(BOOL *aModified)
2757{
2758 CheckComArgOutPointerValid(aModified);
2759
2760 AutoCaller autoCaller(this);
2761 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2762
2763 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2764
2765 HRESULT rc = checkStateDependency(MutableStateDep);
2766 if (FAILED(rc)) return rc;
2767
2768 if (!mData->pMachineConfigFile->fileExists())
2769 // this is a new machine, and no config file exists yet:
2770 *aModified = TRUE;
2771 else
2772 *aModified = (mData->flModifications != 0);
2773
2774 return S_OK;
2775}
2776
2777STDMETHODIMP Machine::COMGETTER(SessionState)(SessionState_T *aSessionState)
2778{
2779 CheckComArgOutPointerValid(aSessionState);
2780
2781 AutoCaller autoCaller(this);
2782 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2783
2784 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2785
2786 *aSessionState = mData->mSession.mState;
2787
2788 return S_OK;
2789}
2790
2791STDMETHODIMP Machine::COMGETTER(SessionType)(BSTR *aSessionType)
2792{
2793 CheckComArgOutPointerValid(aSessionType);
2794
2795 AutoCaller autoCaller(this);
2796 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2797
2798 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2799
2800 mData->mSession.mType.cloneTo(aSessionType);
2801
2802 return S_OK;
2803}
2804
2805STDMETHODIMP Machine::COMGETTER(SessionPID)(ULONG *aSessionPID)
2806{
2807 CheckComArgOutPointerValid(aSessionPID);
2808
2809 AutoCaller autoCaller(this);
2810 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2811
2812 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2813
2814 *aSessionPID = mData->mSession.mPID;
2815
2816 return S_OK;
2817}
2818
2819STDMETHODIMP Machine::COMGETTER(State)(MachineState_T *machineState)
2820{
2821 CheckComArgOutPointerValid(machineState);
2822
2823 AutoCaller autoCaller(this);
2824 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2825
2826 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2827
2828 *machineState = mData->mMachineState;
2829
2830 return S_OK;
2831}
2832
2833STDMETHODIMP Machine::COMGETTER(LastStateChange)(LONG64 *aLastStateChange)
2834{
2835 CheckComArgOutPointerValid(aLastStateChange);
2836
2837 AutoCaller autoCaller(this);
2838 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2839
2840 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2841
2842 *aLastStateChange = RTTimeSpecGetMilli(&mData->mLastStateChange);
2843
2844 return S_OK;
2845}
2846
2847STDMETHODIMP Machine::COMGETTER(StateFilePath)(BSTR *aStateFilePath)
2848{
2849 CheckComArgOutPointerValid(aStateFilePath);
2850
2851 AutoCaller autoCaller(this);
2852 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2853
2854 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2855
2856 mSSData->strStateFilePath.cloneTo(aStateFilePath);
2857
2858 return S_OK;
2859}
2860
2861STDMETHODIMP Machine::COMGETTER(LogFolder)(BSTR *aLogFolder)
2862{
2863 CheckComArgOutPointerValid(aLogFolder);
2864
2865 AutoCaller autoCaller(this);
2866 AssertComRCReturnRC(autoCaller.rc());
2867
2868 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2869
2870 Utf8Str logFolder;
2871 getLogFolder(logFolder);
2872 logFolder.cloneTo(aLogFolder);
2873
2874 return S_OK;
2875}
2876
2877STDMETHODIMP Machine::COMGETTER(CurrentSnapshot) (ISnapshot **aCurrentSnapshot)
2878{
2879 CheckComArgOutPointerValid(aCurrentSnapshot);
2880
2881 AutoCaller autoCaller(this);
2882 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2883
2884 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2885
2886 mData->mCurrentSnapshot.queryInterfaceTo(aCurrentSnapshot);
2887
2888 return S_OK;
2889}
2890
2891STDMETHODIMP Machine::COMGETTER(SnapshotCount)(ULONG *aSnapshotCount)
2892{
2893 CheckComArgOutPointerValid(aSnapshotCount);
2894
2895 AutoCaller autoCaller(this);
2896 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2897
2898 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2899
2900 *aSnapshotCount = mData->mFirstSnapshot.isNull()
2901 ? 0
2902 : mData->mFirstSnapshot->getAllChildrenCount() + 1;
2903
2904 return S_OK;
2905}
2906
2907STDMETHODIMP Machine::COMGETTER(CurrentStateModified)(BOOL *aCurrentStateModified)
2908{
2909 CheckComArgOutPointerValid(aCurrentStateModified);
2910
2911 AutoCaller autoCaller(this);
2912 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2913
2914 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2915
2916 /* Note: for machines with no snapshots, we always return FALSE
2917 * (mData->mCurrentStateModified will be TRUE in this case, for historical
2918 * reasons :) */
2919
2920 *aCurrentStateModified = mData->mFirstSnapshot.isNull()
2921 ? FALSE
2922 : mData->mCurrentStateModified;
2923
2924 return S_OK;
2925}
2926
2927STDMETHODIMP Machine::COMGETTER(SharedFolders)(ComSafeArrayOut(ISharedFolder *, aSharedFolders))
2928{
2929 CheckComArgOutSafeArrayPointerValid(aSharedFolders);
2930
2931 AutoCaller autoCaller(this);
2932 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2933
2934 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2935
2936 SafeIfaceArray<ISharedFolder> folders(mHWData->mSharedFolders);
2937 folders.detachTo(ComSafeArrayOutArg(aSharedFolders));
2938
2939 return S_OK;
2940}
2941
2942STDMETHODIMP Machine::COMGETTER(ClipboardMode)(ClipboardMode_T *aClipboardMode)
2943{
2944 CheckComArgOutPointerValid(aClipboardMode);
2945
2946 AutoCaller autoCaller(this);
2947 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2948
2949 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2950
2951 *aClipboardMode = mHWData->mClipboardMode;
2952
2953 return S_OK;
2954}
2955
2956STDMETHODIMP Machine::COMSETTER(ClipboardMode)(ClipboardMode_T aClipboardMode)
2957{
2958 HRESULT rc = S_OK;
2959
2960 AutoCaller autoCaller(this);
2961 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2962
2963 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2964
2965 alock.release();
2966 rc = onClipboardModeChange(aClipboardMode);
2967 alock.acquire();
2968 if (FAILED(rc)) return rc;
2969
2970 setModified(IsModified_MachineData);
2971 mHWData.backup();
2972 mHWData->mClipboardMode = aClipboardMode;
2973
2974 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
2975 if (Global::IsOnline(mData->mMachineState))
2976 saveSettings(NULL);
2977
2978 return S_OK;
2979}
2980
2981STDMETHODIMP Machine::COMGETTER(DragAndDropMode)(DragAndDropMode_T *aDragAndDropMode)
2982{
2983 CheckComArgOutPointerValid(aDragAndDropMode);
2984
2985 AutoCaller autoCaller(this);
2986 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2987
2988 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2989
2990 *aDragAndDropMode = mHWData->mDragAndDropMode;
2991
2992 return S_OK;
2993}
2994
2995STDMETHODIMP Machine::COMSETTER(DragAndDropMode)(DragAndDropMode_T aDragAndDropMode)
2996{
2997 HRESULT rc = S_OK;
2998
2999 AutoCaller autoCaller(this);
3000 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3001
3002 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3003
3004 alock.release();
3005 rc = onDragAndDropModeChange(aDragAndDropMode);
3006 alock.acquire();
3007 if (FAILED(rc)) return rc;
3008
3009 setModified(IsModified_MachineData);
3010 mHWData.backup();
3011 mHWData->mDragAndDropMode = aDragAndDropMode;
3012
3013 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
3014 if (Global::IsOnline(mData->mMachineState))
3015 saveSettings(NULL);
3016
3017 return S_OK;
3018}
3019
3020STDMETHODIMP Machine::COMGETTER(GuestPropertyNotificationPatterns)(BSTR *aPatterns)
3021{
3022 CheckComArgOutPointerValid(aPatterns);
3023
3024 AutoCaller autoCaller(this);
3025 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3026
3027 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3028
3029 try
3030 {
3031 mHWData->mGuestPropertyNotificationPatterns.cloneTo(aPatterns);
3032 }
3033 catch (...)
3034 {
3035 return VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
3036 }
3037
3038 return S_OK;
3039}
3040
3041STDMETHODIMP Machine::COMSETTER(GuestPropertyNotificationPatterns)(IN_BSTR aPatterns)
3042{
3043 AutoCaller autoCaller(this);
3044 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3045
3046 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3047
3048 HRESULT rc = checkStateDependency(MutableStateDep);
3049 if (FAILED(rc)) return rc;
3050
3051 setModified(IsModified_MachineData);
3052 mHWData.backup();
3053 mHWData->mGuestPropertyNotificationPatterns = aPatterns;
3054 return rc;
3055}
3056
3057STDMETHODIMP Machine::COMGETTER(StorageControllers)(ComSafeArrayOut(IStorageController *, aStorageControllers))
3058{
3059 CheckComArgOutSafeArrayPointerValid(aStorageControllers);
3060
3061 AutoCaller autoCaller(this);
3062 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3063
3064 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3065
3066 SafeIfaceArray<IStorageController> ctrls(*mStorageControllers.data());
3067 ctrls.detachTo(ComSafeArrayOutArg(aStorageControllers));
3068
3069 return S_OK;
3070}
3071
3072STDMETHODIMP Machine::COMGETTER(TeleporterEnabled)(BOOL *aEnabled)
3073{
3074 CheckComArgOutPointerValid(aEnabled);
3075
3076 AutoCaller autoCaller(this);
3077 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3078
3079 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3080
3081 *aEnabled = mUserData->s.fTeleporterEnabled;
3082
3083 return S_OK;
3084}
3085
3086STDMETHODIMP Machine::COMSETTER(TeleporterEnabled)(BOOL aEnabled)
3087{
3088 AutoCaller autoCaller(this);
3089 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3090
3091 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3092
3093 /* Only allow it to be set to true when PoweredOff or Aborted.
3094 (Clearing it is always permitted.) */
3095 if ( aEnabled
3096 && mData->mRegistered
3097 && ( !isSessionMachine()
3098 || ( mData->mMachineState != MachineState_PoweredOff
3099 && mData->mMachineState != MachineState_Teleported
3100 && mData->mMachineState != MachineState_Aborted
3101 )
3102 )
3103 )
3104 return setError(VBOX_E_INVALID_VM_STATE,
3105 tr("The machine is not powered off (state is %s)"),
3106 Global::stringifyMachineState(mData->mMachineState));
3107
3108 setModified(IsModified_MachineData);
3109 mUserData.backup();
3110 mUserData->s.fTeleporterEnabled = !!aEnabled;
3111
3112 return S_OK;
3113}
3114
3115STDMETHODIMP Machine::COMGETTER(TeleporterPort)(ULONG *aPort)
3116{
3117 CheckComArgOutPointerValid(aPort);
3118
3119 AutoCaller autoCaller(this);
3120 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3121
3122 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3123
3124 *aPort = (ULONG)mUserData->s.uTeleporterPort;
3125
3126 return S_OK;
3127}
3128
3129STDMETHODIMP Machine::COMSETTER(TeleporterPort)(ULONG aPort)
3130{
3131 if (aPort >= _64K)
3132 return setError(E_INVALIDARG, tr("Invalid port number %d"), aPort);
3133
3134 AutoCaller autoCaller(this);
3135 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3136
3137 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3138
3139 HRESULT rc = checkStateDependency(MutableStateDep);
3140 if (FAILED(rc)) return rc;
3141
3142 setModified(IsModified_MachineData);
3143 mUserData.backup();
3144 mUserData->s.uTeleporterPort = (uint32_t)aPort;
3145
3146 return S_OK;
3147}
3148
3149STDMETHODIMP Machine::COMGETTER(TeleporterAddress)(BSTR *aAddress)
3150{
3151 CheckComArgOutPointerValid(aAddress);
3152
3153 AutoCaller autoCaller(this);
3154 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3155
3156 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3157
3158 mUserData->s.strTeleporterAddress.cloneTo(aAddress);
3159
3160 return S_OK;
3161}
3162
3163STDMETHODIMP Machine::COMSETTER(TeleporterAddress)(IN_BSTR aAddress)
3164{
3165 AutoCaller autoCaller(this);
3166 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3167
3168 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3169
3170 HRESULT rc = checkStateDependency(MutableStateDep);
3171 if (FAILED(rc)) return rc;
3172
3173 setModified(IsModified_MachineData);
3174 mUserData.backup();
3175 mUserData->s.strTeleporterAddress = aAddress;
3176
3177 return S_OK;
3178}
3179
3180STDMETHODIMP Machine::COMGETTER(TeleporterPassword)(BSTR *aPassword)
3181{
3182 CheckComArgOutPointerValid(aPassword);
3183
3184 AutoCaller autoCaller(this);
3185 HRESULT hrc = autoCaller.rc();
3186 if (SUCCEEDED(hrc))
3187 {
3188 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3189 mUserData->s.strTeleporterPassword.cloneTo(aPassword);
3190 }
3191
3192 return hrc;
3193}
3194
3195STDMETHODIMP Machine::COMSETTER(TeleporterPassword)(IN_BSTR aPassword)
3196{
3197 /*
3198 * Hash the password first.
3199 */
3200 Utf8Str strPassword(aPassword);
3201 if (!strPassword.isEmpty())
3202 {
3203 if (VBoxIsPasswordHashed(&strPassword))
3204 return setError(E_INVALIDARG, tr("Cannot set an already hashed password, only plain text password please"));
3205 VBoxHashPassword(&strPassword);
3206 }
3207
3208 /*
3209 * Do the update.
3210 */
3211 AutoCaller autoCaller(this);
3212 HRESULT hrc = autoCaller.rc();
3213 if (SUCCEEDED(hrc))
3214 {
3215 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3216 hrc = checkStateDependency(MutableStateDep);
3217 if (SUCCEEDED(hrc))
3218 {
3219 setModified(IsModified_MachineData);
3220 mUserData.backup();
3221 mUserData->s.strTeleporterPassword = strPassword;
3222 }
3223 }
3224
3225 return hrc;
3226}
3227
3228STDMETHODIMP Machine::COMGETTER(FaultToleranceState)(FaultToleranceState_T *aState)
3229{
3230 CheckComArgOutPointerValid(aState);
3231
3232 AutoCaller autoCaller(this);
3233 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3234
3235 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3236
3237 *aState = mUserData->s.enmFaultToleranceState;
3238 return S_OK;
3239}
3240
3241STDMETHODIMP Machine::COMSETTER(FaultToleranceState)(FaultToleranceState_T aState)
3242{
3243 AutoCaller autoCaller(this);
3244 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3245
3246 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3247
3248 /* @todo deal with running state change. */
3249 HRESULT rc = checkStateDependency(MutableStateDep);
3250 if (FAILED(rc)) return rc;
3251
3252 setModified(IsModified_MachineData);
3253 mUserData.backup();
3254 mUserData->s.enmFaultToleranceState = aState;
3255 return S_OK;
3256}
3257
3258STDMETHODIMP Machine::COMGETTER(FaultToleranceAddress)(BSTR *aAddress)
3259{
3260 CheckComArgOutPointerValid(aAddress);
3261
3262 AutoCaller autoCaller(this);
3263 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3264
3265 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3266
3267 mUserData->s.strFaultToleranceAddress.cloneTo(aAddress);
3268 return S_OK;
3269}
3270
3271STDMETHODIMP Machine::COMSETTER(FaultToleranceAddress)(IN_BSTR aAddress)
3272{
3273 AutoCaller autoCaller(this);
3274 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3275
3276 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3277
3278 /* @todo deal with running state change. */
3279 HRESULT rc = checkStateDependency(MutableStateDep);
3280 if (FAILED(rc)) return rc;
3281
3282 setModified(IsModified_MachineData);
3283 mUserData.backup();
3284 mUserData->s.strFaultToleranceAddress = aAddress;
3285 return S_OK;
3286}
3287
3288STDMETHODIMP Machine::COMGETTER(FaultTolerancePort)(ULONG *aPort)
3289{
3290 CheckComArgOutPointerValid(aPort);
3291
3292 AutoCaller autoCaller(this);
3293 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3294
3295 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3296
3297 *aPort = mUserData->s.uFaultTolerancePort;
3298 return S_OK;
3299}
3300
3301STDMETHODIMP Machine::COMSETTER(FaultTolerancePort)(ULONG aPort)
3302{
3303 AutoCaller autoCaller(this);
3304 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3305
3306 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3307
3308 /* @todo deal with running state change. */
3309 HRESULT rc = checkStateDependency(MutableStateDep);
3310 if (FAILED(rc)) return rc;
3311
3312 setModified(IsModified_MachineData);
3313 mUserData.backup();
3314 mUserData->s.uFaultTolerancePort = aPort;
3315 return S_OK;
3316}
3317
3318STDMETHODIMP Machine::COMGETTER(FaultTolerancePassword)(BSTR *aPassword)
3319{
3320 CheckComArgOutPointerValid(aPassword);
3321
3322 AutoCaller autoCaller(this);
3323 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3324
3325 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3326
3327 mUserData->s.strFaultTolerancePassword.cloneTo(aPassword);
3328
3329 return S_OK;
3330}
3331
3332STDMETHODIMP Machine::COMSETTER(FaultTolerancePassword)(IN_BSTR aPassword)
3333{
3334 AutoCaller autoCaller(this);
3335 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3336
3337 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3338
3339 /* @todo deal with running state change. */
3340 HRESULT rc = checkStateDependency(MutableStateDep);
3341 if (FAILED(rc)) return rc;
3342
3343 setModified(IsModified_MachineData);
3344 mUserData.backup();
3345 mUserData->s.strFaultTolerancePassword = aPassword;
3346
3347 return S_OK;
3348}
3349
3350STDMETHODIMP Machine::COMGETTER(FaultToleranceSyncInterval)(ULONG *aInterval)
3351{
3352 CheckComArgOutPointerValid(aInterval);
3353
3354 AutoCaller autoCaller(this);
3355 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3356
3357 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3358
3359 *aInterval = mUserData->s.uFaultToleranceInterval;
3360 return S_OK;
3361}
3362
3363STDMETHODIMP Machine::COMSETTER(FaultToleranceSyncInterval)(ULONG aInterval)
3364{
3365 AutoCaller autoCaller(this);
3366 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3367
3368 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3369
3370 /* @todo deal with running state change. */
3371 HRESULT rc = checkStateDependency(MutableStateDep);
3372 if (FAILED(rc)) return rc;
3373
3374 setModified(IsModified_MachineData);
3375 mUserData.backup();
3376 mUserData->s.uFaultToleranceInterval = aInterval;
3377 return S_OK;
3378}
3379
3380STDMETHODIMP Machine::COMGETTER(RTCUseUTC)(BOOL *aEnabled)
3381{
3382 CheckComArgOutPointerValid(aEnabled);
3383
3384 AutoCaller autoCaller(this);
3385 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3386
3387 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3388
3389 *aEnabled = mUserData->s.fRTCUseUTC;
3390
3391 return S_OK;
3392}
3393
3394STDMETHODIMP Machine::COMSETTER(RTCUseUTC)(BOOL aEnabled)
3395{
3396 AutoCaller autoCaller(this);
3397 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3398
3399 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3400
3401 /* Only allow it to be set to true when PoweredOff or Aborted.
3402 (Clearing it is always permitted.) */
3403 if ( aEnabled
3404 && mData->mRegistered
3405 && ( !isSessionMachine()
3406 || ( mData->mMachineState != MachineState_PoweredOff
3407 && mData->mMachineState != MachineState_Teleported
3408 && mData->mMachineState != MachineState_Aborted
3409 )
3410 )
3411 )
3412 return setError(VBOX_E_INVALID_VM_STATE,
3413 tr("The machine is not powered off (state is %s)"),
3414 Global::stringifyMachineState(mData->mMachineState));
3415
3416 setModified(IsModified_MachineData);
3417 mUserData.backup();
3418 mUserData->s.fRTCUseUTC = !!aEnabled;
3419
3420 return S_OK;
3421}
3422
3423STDMETHODIMP Machine::COMGETTER(IOCacheEnabled)(BOOL *aEnabled)
3424{
3425 CheckComArgOutPointerValid(aEnabled);
3426
3427 AutoCaller autoCaller(this);
3428 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3429
3430 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3431
3432 *aEnabled = mHWData->mIOCacheEnabled;
3433
3434 return S_OK;
3435}
3436
3437STDMETHODIMP Machine::COMSETTER(IOCacheEnabled)(BOOL aEnabled)
3438{
3439 AutoCaller autoCaller(this);
3440 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3441
3442 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3443
3444 HRESULT rc = checkStateDependency(MutableStateDep);
3445 if (FAILED(rc)) return rc;
3446
3447 setModified(IsModified_MachineData);
3448 mHWData.backup();
3449 mHWData->mIOCacheEnabled = aEnabled;
3450
3451 return S_OK;
3452}
3453
3454STDMETHODIMP Machine::COMGETTER(IOCacheSize)(ULONG *aIOCacheSize)
3455{
3456 CheckComArgOutPointerValid(aIOCacheSize);
3457
3458 AutoCaller autoCaller(this);
3459 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3460
3461 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3462
3463 *aIOCacheSize = mHWData->mIOCacheSize;
3464
3465 return S_OK;
3466}
3467
3468STDMETHODIMP Machine::COMSETTER(IOCacheSize)(ULONG aIOCacheSize)
3469{
3470 AutoCaller autoCaller(this);
3471 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3472
3473 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3474
3475 HRESULT rc = checkStateDependency(MutableStateDep);
3476 if (FAILED(rc)) return rc;
3477
3478 setModified(IsModified_MachineData);
3479 mHWData.backup();
3480 mHWData->mIOCacheSize = aIOCacheSize;
3481
3482 return S_OK;
3483}
3484
3485
3486/**
3487 * @note Locks objects!
3488 */
3489STDMETHODIMP Machine::LockMachine(ISession *aSession,
3490 LockType_T lockType)
3491{
3492 CheckComArgNotNull(aSession);
3493
3494 AutoCaller autoCaller(this);
3495 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3496
3497 /* check the session state */
3498 SessionState_T state;
3499 HRESULT rc = aSession->COMGETTER(State)(&state);
3500 if (FAILED(rc)) return rc;
3501
3502 if (state != SessionState_Unlocked)
3503 return setError(VBOX_E_INVALID_OBJECT_STATE,
3504 tr("The given session is busy"));
3505
3506 // get the client's IInternalSessionControl interface
3507 ComPtr<IInternalSessionControl> pSessionControl = aSession;
3508 ComAssertMsgRet(!!pSessionControl, ("No IInternalSessionControl interface"),
3509 E_INVALIDARG);
3510
3511 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3512
3513 if (!mData->mRegistered)
3514 return setError(E_UNEXPECTED,
3515 tr("The machine '%s' is not registered"),
3516 mUserData->s.strName.c_str());
3517
3518 LogFlowThisFunc(("mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
3519
3520 SessionState_T oldState = mData->mSession.mState;
3521 /* Hack: in case the session is closing and there is a progress object
3522 * which allows waiting for the session to be closed, take the opportunity
3523 * and do a limited wait (max. 1 second). This helps a lot when the system
3524 * is busy and thus session closing can take a little while. */
3525 if ( mData->mSession.mState == SessionState_Unlocking
3526 && mData->mSession.mProgress)
3527 {
3528 alock.release();
3529 mData->mSession.mProgress->WaitForCompletion(1000);
3530 alock.acquire();
3531 LogFlowThisFunc(("after waiting: mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
3532 }
3533
3534 // try again now
3535 if ( (mData->mSession.mState == SessionState_Locked) // machine is write-locked already (i.e. session machine exists)
3536 && (lockType == LockType_Shared) // caller wants a shared link to the existing session that holds the write lock:
3537 )
3538 {
3539 // OK, share the session... we are now dealing with three processes:
3540 // 1) VBoxSVC (where this code runs);
3541 // 2) process C: the caller's client process (who wants a shared session);
3542 // 3) process W: the process which already holds the write lock on the machine (write-locking session)
3543
3544 // copy pointers to W (the write-locking session) before leaving lock (these must not be NULL)
3545 ComPtr<IInternalSessionControl> pSessionW = mData->mSession.mDirectControl;
3546 ComAssertRet(!pSessionW.isNull(), E_FAIL);
3547 ComObjPtr<SessionMachine> pSessionMachine = mData->mSession.mMachine;
3548 AssertReturn(!pSessionMachine.isNull(), E_FAIL);
3549
3550 /*
3551 * Release the lock before calling the client process. It's safe here
3552 * since the only thing to do after we get the lock again is to add
3553 * the remote control to the list (which doesn't directly influence
3554 * anything).
3555 */
3556 alock.release();
3557
3558 // get the console of the session holding the write lock (this is a remote call)
3559 ComPtr<IConsole> pConsoleW;
3560 LogFlowThisFunc(("Calling GetRemoteConsole()...\n"));
3561 rc = pSessionW->GetRemoteConsole(pConsoleW.asOutParam());
3562 LogFlowThisFunc(("GetRemoteConsole() returned %08X\n", rc));
3563 if (FAILED(rc))
3564 // the failure may occur w/o any error info (from RPC), so provide one
3565 return setError(VBOX_E_VM_ERROR,
3566 tr("Failed to get a console object from the direct session (%Rhrc)"), rc);
3567
3568 ComAssertRet(!pConsoleW.isNull(), E_FAIL);
3569
3570 // share the session machine and W's console with the caller's session
3571 LogFlowThisFunc(("Calling AssignRemoteMachine()...\n"));
3572 rc = pSessionControl->AssignRemoteMachine(pSessionMachine, pConsoleW);
3573 LogFlowThisFunc(("AssignRemoteMachine() returned %08X\n", rc));
3574
3575 if (FAILED(rc))
3576 // the failure may occur w/o any error info (from RPC), so provide one
3577 return setError(VBOX_E_VM_ERROR,
3578 tr("Failed to assign the machine to the session (%Rhrc)"), rc);
3579 alock.acquire();
3580
3581 // need to revalidate the state after acquiring the lock again
3582 if (mData->mSession.mState != SessionState_Locked)
3583 {
3584 pSessionControl->Uninitialize();
3585 return setError(VBOX_E_INVALID_SESSION_STATE,
3586 tr("The machine '%s' was unlocked unexpectedly while attempting to share its session"),
3587 mUserData->s.strName.c_str());
3588 }
3589
3590 // add the caller's session to the list
3591 mData->mSession.mRemoteControls.push_back(pSessionControl);
3592 }
3593 else if ( mData->mSession.mState == SessionState_Locked
3594 || mData->mSession.mState == SessionState_Unlocking
3595 )
3596 {
3597 // sharing not permitted, or machine still unlocking:
3598 return setError(VBOX_E_INVALID_OBJECT_STATE,
3599 tr("The machine '%s' is already locked for a session (or being unlocked)"),
3600 mUserData->s.strName.c_str());
3601 }
3602 else
3603 {
3604 // machine is not locked: then write-lock the machine (create the session machine)
3605
3606 // must not be busy
3607 AssertReturn(!Global::IsOnlineOrTransient(mData->mMachineState), E_FAIL);
3608
3609 // get the caller's session PID
3610 RTPROCESS pid = NIL_RTPROCESS;
3611 AssertCompile(sizeof(ULONG) == sizeof(RTPROCESS));
3612 pSessionControl->GetPID((ULONG*)&pid);
3613 Assert(pid != NIL_RTPROCESS);
3614
3615 bool fLaunchingVMProcess = (mData->mSession.mState == SessionState_Spawning);
3616
3617 if (fLaunchingVMProcess)
3618 {
3619 if (mData->mSession.mPID == NIL_RTPROCESS)
3620 {
3621 // two or more clients racing for a lock, the one which set the
3622 // session state to Spawning will win, the others will get an
3623 // error as we can't decide here if waiting a little would help
3624 // (only for shared locks this would avoid an error)
3625 return setError(VBOX_E_INVALID_OBJECT_STATE,
3626 tr("The machine '%s' already has a lock request pending"),
3627 mUserData->s.strName.c_str());
3628 }
3629
3630 // this machine is awaiting for a spawning session to be opened:
3631 // then the calling process must be the one that got started by
3632 // LaunchVMProcess()
3633
3634 LogFlowThisFunc(("mSession.mPID=%d(0x%x)\n", mData->mSession.mPID, mData->mSession.mPID));
3635 LogFlowThisFunc(("session.pid=%d(0x%x)\n", pid, pid));
3636
3637 if (mData->mSession.mPID != pid)
3638 return setError(E_ACCESSDENIED,
3639 tr("An unexpected process (PID=0x%08X) has tried to lock the "
3640 "machine '%s', while only the process started by LaunchVMProcess (PID=0x%08X) is allowed"),
3641 pid, mUserData->s.strName.c_str(), mData->mSession.mPID);
3642 }
3643
3644 // create the mutable SessionMachine from the current machine
3645 ComObjPtr<SessionMachine> sessionMachine;
3646 sessionMachine.createObject();
3647 rc = sessionMachine->init(this);
3648 AssertComRC(rc);
3649
3650 /* NOTE: doing return from this function after this point but
3651 * before the end is forbidden since it may call SessionMachine::uninit()
3652 * (through the ComObjPtr's destructor) which requests the VirtualBox write
3653 * lock while still holding the Machine lock in alock so that a deadlock
3654 * is possible due to the wrong lock order. */
3655
3656 if (SUCCEEDED(rc))
3657 {
3658 /*
3659 * Set the session state to Spawning to protect against subsequent
3660 * attempts to open a session and to unregister the machine after
3661 * we release the lock.
3662 */
3663 SessionState_T origState = mData->mSession.mState;
3664 mData->mSession.mState = SessionState_Spawning;
3665
3666#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
3667 /* Get the client token ID to be passed to the client process */
3668 Utf8Str strTokenId;
3669 sessionMachine->getTokenId(strTokenId);
3670 Assert(!strTokenId.isEmpty());
3671#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
3672 /* Get the client token to be passed to the client process */
3673 ComPtr<IToken> pToken(sessionMachine->getToken());
3674 /* The token is now "owned" by pToken, fix refcount */
3675 if (!pToken.isNull())
3676 pToken->Release();
3677#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
3678
3679 /*
3680 * Release the lock before calling the client process -- it will call
3681 * Machine/SessionMachine methods. Releasing the lock here is quite safe
3682 * because the state is Spawning, so that LaunchVMProcess() and
3683 * LockMachine() calls will fail. This method, called before we
3684 * acquire the lock again, will fail because of the wrong PID.
3685 *
3686 * Note that mData->mSession.mRemoteControls accessed outside
3687 * the lock may not be modified when state is Spawning, so it's safe.
3688 */
3689 alock.release();
3690
3691 LogFlowThisFunc(("Calling AssignMachine()...\n"));
3692#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
3693 rc = pSessionControl->AssignMachine(sessionMachine, lockType, Bstr(strTokenId).raw());
3694#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
3695 rc = pSessionControl->AssignMachine(sessionMachine, lockType, pToken);
3696 /* Now the token is owned by the client process. */
3697 pToken.setNull();
3698#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
3699 LogFlowThisFunc(("AssignMachine() returned %08X\n", rc));
3700
3701 /* The failure may occur w/o any error info (from RPC), so provide one */
3702 if (FAILED(rc))
3703 setError(VBOX_E_VM_ERROR,
3704 tr("Failed to assign the machine to the session (%Rhrc)"), rc);
3705
3706 if ( SUCCEEDED(rc)
3707 && fLaunchingVMProcess
3708 )
3709 {
3710 /* complete the remote session initialization */
3711
3712 /* get the console from the direct session */
3713 ComPtr<IConsole> console;
3714 rc = pSessionControl->GetRemoteConsole(console.asOutParam());
3715 ComAssertComRC(rc);
3716
3717 if (SUCCEEDED(rc) && !console)
3718 {
3719 ComAssert(!!console);
3720 rc = E_FAIL;
3721 }
3722
3723 /* assign machine & console to the remote session */
3724 if (SUCCEEDED(rc))
3725 {
3726 /*
3727 * after LaunchVMProcess(), the first and the only
3728 * entry in remoteControls is that remote session
3729 */
3730 LogFlowThisFunc(("Calling AssignRemoteMachine()...\n"));
3731 rc = mData->mSession.mRemoteControls.front()->AssignRemoteMachine(sessionMachine, console);
3732 LogFlowThisFunc(("AssignRemoteMachine() returned %08X\n", rc));
3733
3734 /* The failure may occur w/o any error info (from RPC), so provide one */
3735 if (FAILED(rc))
3736 setError(VBOX_E_VM_ERROR,
3737 tr("Failed to assign the machine to the remote session (%Rhrc)"), rc);
3738 }
3739
3740 if (FAILED(rc))
3741 pSessionControl->Uninitialize();
3742 }
3743
3744 /* acquire the lock again */
3745 alock.acquire();
3746
3747 /* Restore the session state */
3748 mData->mSession.mState = origState;
3749 }
3750
3751 // finalize spawning anyway (this is why we don't return on errors above)
3752 if (fLaunchingVMProcess)
3753 {
3754 /* Note that the progress object is finalized later */
3755 /** @todo Consider checking mData->mSession.mProgress for cancellation
3756 * around here. */
3757
3758 /* We don't reset mSession.mPID here because it is necessary for
3759 * SessionMachine::uninit() to reap the child process later. */
3760
3761 if (FAILED(rc))
3762 {
3763 /* Close the remote session, remove the remote control from the list
3764 * and reset session state to Closed (@note keep the code in sync
3765 * with the relevant part in checkForSpawnFailure()). */
3766
3767 Assert(mData->mSession.mRemoteControls.size() == 1);
3768 if (mData->mSession.mRemoteControls.size() == 1)
3769 {
3770 ErrorInfoKeeper eik;
3771 mData->mSession.mRemoteControls.front()->Uninitialize();
3772 }
3773
3774 mData->mSession.mRemoteControls.clear();
3775 mData->mSession.mState = SessionState_Unlocked;
3776 }
3777 }
3778 else
3779 {
3780 /* memorize PID of the directly opened session */
3781 if (SUCCEEDED(rc))
3782 mData->mSession.mPID = pid;
3783 }
3784
3785 if (SUCCEEDED(rc))
3786 {
3787 /* memorize the direct session control and cache IUnknown for it */
3788 mData->mSession.mDirectControl = pSessionControl;
3789 mData->mSession.mState = SessionState_Locked;
3790 /* associate the SessionMachine with this Machine */
3791 mData->mSession.mMachine = sessionMachine;
3792
3793 /* request an IUnknown pointer early from the remote party for later
3794 * identity checks (it will be internally cached within mDirectControl
3795 * at least on XPCOM) */
3796 ComPtr<IUnknown> unk = mData->mSession.mDirectControl;
3797 NOREF(unk);
3798 }
3799
3800 /* Release the lock since SessionMachine::uninit() locks VirtualBox which
3801 * would break the lock order */
3802 alock.release();
3803
3804 /* uninitialize the created session machine on failure */
3805 if (FAILED(rc))
3806 sessionMachine->uninit();
3807
3808 }
3809
3810 if (SUCCEEDED(rc))
3811 {
3812 /*
3813 * tell the client watcher thread to update the set of
3814 * machines that have open sessions
3815 */
3816 mParent->updateClientWatcher();
3817
3818 if (oldState != SessionState_Locked)
3819 /* fire an event */
3820 mParent->onSessionStateChange(getId(), SessionState_Locked);
3821 }
3822
3823 return rc;
3824}
3825
3826/**
3827 * @note Locks objects!
3828 */
3829STDMETHODIMP Machine::LaunchVMProcess(ISession *aSession,
3830 IN_BSTR aFrontend,
3831 IN_BSTR aEnvironment,
3832 IProgress **aProgress)
3833{
3834 CheckComArgStr(aFrontend);
3835 Utf8Str strFrontend(aFrontend);
3836 Utf8Str strEnvironment(aEnvironment);
3837 /* "emergencystop" doesn't need the session, so skip the checks/interface
3838 * retrieval. This code doesn't quite fit in here, but introducing a
3839 * special API method would be even more effort, and would require explicit
3840 * support by every API client. It's better to hide the feature a bit. */
3841 if (strFrontend != "emergencystop")
3842 CheckComArgNotNull(aSession);
3843 CheckComArgOutPointerValid(aProgress);
3844
3845 AutoCaller autoCaller(this);
3846 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3847
3848 HRESULT rc = S_OK;
3849 if (strFrontend.isEmpty())
3850 {
3851 Bstr bstrFrontend;
3852 rc = COMGETTER(DefaultFrontend)(bstrFrontend.asOutParam());
3853 if (FAILED(rc))
3854 return rc;
3855 strFrontend = bstrFrontend;
3856 if (strFrontend.isEmpty())
3857 {
3858 ComPtr<ISystemProperties> systemProperties;
3859 rc = mParent->COMGETTER(SystemProperties)(systemProperties.asOutParam());
3860 if (FAILED(rc))
3861 return rc;
3862 rc = systemProperties->COMGETTER(DefaultFrontend)(bstrFrontend.asOutParam());
3863 if (FAILED(rc))
3864 return rc;
3865 strFrontend = bstrFrontend;
3866 }
3867 /* paranoia - emergencystop is not a valid default */
3868 if (strFrontend == "emergencystop")
3869 strFrontend = Utf8Str::Empty;
3870 }
3871 /* default frontend: Qt GUI */
3872 if (strFrontend.isEmpty())
3873 strFrontend = "GUI/Qt";
3874
3875 if (strFrontend != "emergencystop")
3876 {
3877 /* check the session state */
3878 SessionState_T state;
3879 rc = aSession->COMGETTER(State)(&state);
3880 if (FAILED(rc))
3881 return rc;
3882
3883 if (state != SessionState_Unlocked)
3884 return setError(VBOX_E_INVALID_OBJECT_STATE,
3885 tr("The given session is busy"));
3886
3887 /* get the IInternalSessionControl interface */
3888 ComPtr<IInternalSessionControl> control(aSession);
3889 ComAssertMsgRet(!control.isNull(),
3890 ("No IInternalSessionControl interface"),
3891 E_INVALIDARG);
3892
3893 /* get the teleporter enable state for the progress object init. */
3894 BOOL fTeleporterEnabled;
3895 rc = COMGETTER(TeleporterEnabled)(&fTeleporterEnabled);
3896 if (FAILED(rc))
3897 return rc;
3898
3899 /* create a progress object */
3900 ComObjPtr<ProgressProxy> progress;
3901 progress.createObject();
3902 rc = progress->init(mParent,
3903 static_cast<IMachine*>(this),
3904 Bstr(tr("Starting VM")).raw(),
3905 TRUE /* aCancelable */,
3906 fTeleporterEnabled ? 20 : 10 /* uTotalOperationsWeight */,
3907 BstrFmt(tr("Creating process for virtual machine \"%s\" (%s)"), mUserData->s.strName.c_str(), strFrontend.c_str()).raw(),
3908 2 /* uFirstOperationWeight */,
3909 fTeleporterEnabled ? 3 : 1 /* cOtherProgressObjectOperations */);
3910
3911 if (SUCCEEDED(rc))
3912 {
3913 rc = launchVMProcess(control, strFrontend, strEnvironment, progress);
3914 if (SUCCEEDED(rc))
3915 {
3916 progress.queryInterfaceTo(aProgress);
3917
3918 /* signal the client watcher thread */
3919 mParent->updateClientWatcher();
3920
3921 /* fire an event */
3922 mParent->onSessionStateChange(getId(), SessionState_Spawning);
3923 }
3924 }
3925 }
3926 else
3927 {
3928 /* no progress object - either instant success or failure */
3929 *aProgress = NULL;
3930
3931 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3932
3933 if (mData->mSession.mState != SessionState_Locked)
3934 return setError(VBOX_E_INVALID_OBJECT_STATE,
3935 tr("The machine '%s' is not locked by a session"),
3936 mUserData->s.strName.c_str());
3937
3938 /* must have a VM process associated - do not kill normal API clients
3939 * with an open session */
3940 if (!Global::IsOnline(mData->mMachineState))
3941 return setError(VBOX_E_INVALID_OBJECT_STATE,
3942 tr("The machine '%s' does not have a VM process"),
3943 mUserData->s.strName.c_str());
3944
3945 /* forcibly terminate the VM process */
3946 if (mData->mSession.mPID != NIL_RTPROCESS)
3947 RTProcTerminate(mData->mSession.mPID);
3948
3949 /* signal the client watcher thread, as most likely the client has
3950 * been terminated */
3951 mParent->updateClientWatcher();
3952 }
3953
3954 return rc;
3955}
3956
3957STDMETHODIMP Machine::SetBootOrder(ULONG aPosition, DeviceType_T aDevice)
3958{
3959 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
3960 return setError(E_INVALIDARG,
3961 tr("Invalid boot position: %lu (must be in range [1, %lu])"),
3962 aPosition, SchemaDefs::MaxBootPosition);
3963
3964 if (aDevice == DeviceType_USB)
3965 return setError(E_NOTIMPL,
3966 tr("Booting from USB device is currently not supported"));
3967
3968 AutoCaller autoCaller(this);
3969 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3970
3971 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3972
3973 HRESULT rc = checkStateDependency(MutableStateDep);
3974 if (FAILED(rc)) return rc;
3975
3976 setModified(IsModified_MachineData);
3977 mHWData.backup();
3978 mHWData->mBootOrder[aPosition - 1] = aDevice;
3979
3980 return S_OK;
3981}
3982
3983STDMETHODIMP Machine::GetBootOrder(ULONG aPosition, DeviceType_T *aDevice)
3984{
3985 if (aPosition < 1 || aPosition > SchemaDefs::MaxBootPosition)
3986 return setError(E_INVALIDARG,
3987 tr("Invalid boot position: %lu (must be in range [1, %lu])"),
3988 aPosition, SchemaDefs::MaxBootPosition);
3989
3990 AutoCaller autoCaller(this);
3991 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3992
3993 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
3994
3995 *aDevice = mHWData->mBootOrder[aPosition - 1];
3996
3997 return S_OK;
3998}
3999
4000STDMETHODIMP Machine::AttachDevice(IN_BSTR aControllerName,
4001 LONG aControllerPort,
4002 LONG aDevice,
4003 DeviceType_T aType,
4004 IMedium *aMedium)
4005{
4006 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aType=%d aMedium=%p\n",
4007 aControllerName, aControllerPort, aDevice, aType, aMedium));
4008
4009 CheckComArgStrNotEmptyOrNull(aControllerName);
4010
4011 AutoCaller autoCaller(this);
4012 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4013
4014 // request the host lock first, since might be calling Host methods for getting host drives;
4015 // next, protect the media tree all the while we're in here, as well as our member variables
4016 AutoMultiWriteLock2 alock(mParent->host(), this COMMA_LOCKVAL_SRC_POS);
4017 AutoWriteLock treeLock(&mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4018
4019 HRESULT rc = checkStateDependency(MutableStateDep);
4020 if (FAILED(rc)) return rc;
4021
4022 /// @todo NEWMEDIA implicit machine registration
4023 if (!mData->mRegistered)
4024 return setError(VBOX_E_INVALID_OBJECT_STATE,
4025 tr("Cannot attach storage devices to an unregistered machine"));
4026
4027 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4028
4029 /* Check for an existing controller. */
4030 ComObjPtr<StorageController> ctl;
4031 rc = getStorageControllerByName(aControllerName, ctl, true /* aSetError */);
4032 if (FAILED(rc)) return rc;
4033
4034 StorageControllerType_T ctrlType;
4035 rc = ctl->COMGETTER(ControllerType)(&ctrlType);
4036 if (FAILED(rc))
4037 return setError(E_FAIL,
4038 tr("Could not get type of controller '%ls'"),
4039 aControllerName);
4040
4041 bool fSilent = false;
4042 Utf8Str strReconfig;
4043
4044 /* Check whether the flag to allow silent storage attachment reconfiguration is set. */
4045 strReconfig = getExtraData(Utf8Str("VBoxInternal2/SilentReconfigureWhilePaused"));
4046 if ( mData->mMachineState == MachineState_Paused
4047 && strReconfig == "1")
4048 fSilent = true;
4049
4050 /* Check that the controller can do hotplugging if we detach the device while the VM is running. */
4051 bool fHotplug = false;
4052 if (!fSilent && Global::IsOnlineOrTransient(mData->mMachineState))
4053 fHotplug = true;
4054
4055 if (fHotplug && !isControllerHotplugCapable(ctrlType))
4056 return setError(VBOX_E_INVALID_VM_STATE,
4057 tr("Controller '%ls' does not support hotplugging"),
4058 aControllerName);
4059
4060 // check that the port and device are not out of range
4061 rc = ctl->checkPortAndDeviceValid(aControllerPort, aDevice);
4062 if (FAILED(rc)) return rc;
4063
4064 /* check if the device slot is already busy */
4065 MediumAttachment *pAttachTemp;
4066 if ((pAttachTemp = findAttachment(mMediaData->mAttachments,
4067 aControllerName,
4068 aControllerPort,
4069 aDevice)))
4070 {
4071 Medium *pMedium = pAttachTemp->getMedium();
4072 if (pMedium)
4073 {
4074 AutoReadLock mediumLock(pMedium COMMA_LOCKVAL_SRC_POS);
4075 return setError(VBOX_E_OBJECT_IN_USE,
4076 tr("Medium '%s' is already attached to port %d, device %d of controller '%ls' of this virtual machine"),
4077 pMedium->getLocationFull().c_str(),
4078 aControllerPort,
4079 aDevice,
4080 aControllerName);
4081 }
4082 else
4083 return setError(VBOX_E_OBJECT_IN_USE,
4084 tr("Device is already attached to port %d, device %d of controller '%ls' of this virtual machine"),
4085 aControllerPort, aDevice, aControllerName);
4086 }
4087
4088 ComObjPtr<Medium> medium = static_cast<Medium*>(aMedium);
4089 if (aMedium && medium.isNull())
4090 return setError(E_INVALIDARG, "The given medium pointer is invalid");
4091
4092 AutoCaller mediumCaller(medium);
4093 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
4094
4095 AutoWriteLock mediumLock(medium COMMA_LOCKVAL_SRC_POS);
4096
4097 if ( (pAttachTemp = findAttachment(mMediaData->mAttachments, medium))
4098 && !medium.isNull()
4099 )
4100 return setError(VBOX_E_OBJECT_IN_USE,
4101 tr("Medium '%s' is already attached to this virtual machine"),
4102 medium->getLocationFull().c_str());
4103
4104 if (!medium.isNull())
4105 {
4106 MediumType_T mtype = medium->getType();
4107 // MediumType_Readonly is also new, but only applies to DVDs and floppies.
4108 // For DVDs it's not written to the config file, so needs no global config
4109 // version bump. For floppies it's a new attribute "type", which is ignored
4110 // by older VirtualBox version, so needs no global config version bump either.
4111 // For hard disks this type is not accepted.
4112 if (mtype == MediumType_MultiAttach)
4113 {
4114 // This type is new with VirtualBox 4.0 and therefore requires settings
4115 // version 1.11 in the settings backend. Unfortunately it is not enough to do
4116 // the usual routine in MachineConfigFile::bumpSettingsVersionIfNeeded() for
4117 // two reasons: The medium type is a property of the media registry tree, which
4118 // can reside in the global config file (for pre-4.0 media); we would therefore
4119 // possibly need to bump the global config version. We don't want to do that though
4120 // because that might make downgrading to pre-4.0 impossible.
4121 // As a result, we can only use these two new types if the medium is NOT in the
4122 // global registry:
4123 const Guid &uuidGlobalRegistry = mParent->getGlobalRegistryId();
4124 if ( medium->isInRegistry(uuidGlobalRegistry)
4125 || !mData->pMachineConfigFile->canHaveOwnMediaRegistry()
4126 )
4127 return setError(VBOX_E_INVALID_OBJECT_STATE,
4128 tr("Cannot attach medium '%s': the media type 'MultiAttach' can only be attached "
4129 "to machines that were created with VirtualBox 4.0 or later"),
4130 medium->getLocationFull().c_str());
4131 }
4132 }
4133
4134 bool fIndirect = false;
4135 if (!medium.isNull())
4136 fIndirect = medium->isReadOnly();
4137 bool associate = true;
4138
4139 do
4140 {
4141 if ( aType == DeviceType_HardDisk
4142 && mMediaData.isBackedUp())
4143 {
4144 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
4145
4146 /* check if the medium was attached to the VM before we started
4147 * changing attachments in which case the attachment just needs to
4148 * be restored */
4149 if ((pAttachTemp = findAttachment(oldAtts, medium)))
4150 {
4151 AssertReturn(!fIndirect, E_FAIL);
4152
4153 /* see if it's the same bus/channel/device */
4154 if (pAttachTemp->matches(aControllerName, aControllerPort, aDevice))
4155 {
4156 /* the simplest case: restore the whole attachment
4157 * and return, nothing else to do */
4158 mMediaData->mAttachments.push_back(pAttachTemp);
4159
4160 /* Reattach the medium to the VM. */
4161 if (fHotplug || fSilent)
4162 {
4163 mediumLock.release();
4164 treeLock.release();
4165 alock.release();
4166
4167 MediumLockList *pMediumLockList(new MediumLockList());
4168
4169 rc = medium->createMediumLockList(true /* fFailIfInaccessible */,
4170 true /* fMediumLockWrite */,
4171 NULL,
4172 *pMediumLockList);
4173 alock.acquire();
4174 if (FAILED(rc))
4175 delete pMediumLockList;
4176 else
4177 {
4178 mData->mSession.mLockedMedia.Unlock();
4179 alock.release();
4180 rc = mData->mSession.mLockedMedia.Insert(pAttachTemp, pMediumLockList);
4181 mData->mSession.mLockedMedia.Lock();
4182 alock.acquire();
4183 }
4184 alock.release();
4185
4186 if (SUCCEEDED(rc))
4187 {
4188 rc = onStorageDeviceChange(pAttachTemp, FALSE /* aRemove */, fSilent);
4189 /* Remove lock list in case of error. */
4190 if (FAILED(rc))
4191 {
4192 mData->mSession.mLockedMedia.Unlock();
4193 mData->mSession.mLockedMedia.Remove(pAttachTemp);
4194 mData->mSession.mLockedMedia.Lock();
4195 }
4196 }
4197 }
4198
4199 return S_OK;
4200 }
4201
4202 /* bus/channel/device differ; we need a new attachment object,
4203 * but don't try to associate it again */
4204 associate = false;
4205 break;
4206 }
4207 }
4208
4209 /* go further only if the attachment is to be indirect */
4210 if (!fIndirect)
4211 break;
4212
4213 /* perform the so called smart attachment logic for indirect
4214 * attachments. Note that smart attachment is only applicable to base
4215 * hard disks. */
4216
4217 if (medium->getParent().isNull())
4218 {
4219 /* first, investigate the backup copy of the current hard disk
4220 * attachments to make it possible to re-attach existing diffs to
4221 * another device slot w/o losing their contents */
4222 if (mMediaData.isBackedUp())
4223 {
4224 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
4225
4226 MediaData::AttachmentList::const_iterator foundIt = oldAtts.end();
4227 uint32_t foundLevel = 0;
4228
4229 for (MediaData::AttachmentList::const_iterator it = oldAtts.begin();
4230 it != oldAtts.end();
4231 ++it)
4232 {
4233 uint32_t level = 0;
4234 MediumAttachment *pAttach = *it;
4235 ComObjPtr<Medium> pMedium = pAttach->getMedium();
4236 Assert(!pMedium.isNull() || pAttach->getType() != DeviceType_HardDisk);
4237 if (pMedium.isNull())
4238 continue;
4239
4240 if (pMedium->getBase(&level) == medium)
4241 {
4242 /* skip the hard disk if its currently attached (we
4243 * cannot attach the same hard disk twice) */
4244 if (findAttachment(mMediaData->mAttachments,
4245 pMedium))
4246 continue;
4247
4248 /* matched device, channel and bus (i.e. attached to the
4249 * same place) will win and immediately stop the search;
4250 * otherwise the attachment that has the youngest
4251 * descendant of medium will be used
4252 */
4253 if (pAttach->matches(aControllerName, aControllerPort, aDevice))
4254 {
4255 /* the simplest case: restore the whole attachment
4256 * and return, nothing else to do */
4257 mMediaData->mAttachments.push_back(*it);
4258
4259 /* Reattach the medium to the VM. */
4260 if (fHotplug || fSilent)
4261 {
4262 mediumLock.release();
4263 treeLock.release();
4264 alock.release();
4265
4266 MediumLockList *pMediumLockList(new MediumLockList());
4267
4268 rc = medium->createMediumLockList(true /* fFailIfInaccessible */,
4269 true /* fMediumLockWrite */,
4270 NULL,
4271 *pMediumLockList);
4272 alock.acquire();
4273 if (FAILED(rc))
4274 delete pMediumLockList;
4275 else
4276 {
4277 mData->mSession.mLockedMedia.Unlock();
4278 alock.release();
4279 rc = mData->mSession.mLockedMedia.Insert(pAttachTemp, pMediumLockList);
4280 mData->mSession.mLockedMedia.Lock();
4281 alock.acquire();
4282 }
4283 alock.release();
4284
4285 if (SUCCEEDED(rc))
4286 {
4287 rc = onStorageDeviceChange(pAttachTemp, FALSE /* aRemove */, fSilent);
4288 /* Remove lock list in case of error. */
4289 if (FAILED(rc))
4290 {
4291 mData->mSession.mLockedMedia.Unlock();
4292 mData->mSession.mLockedMedia.Remove(pAttachTemp);
4293 mData->mSession.mLockedMedia.Lock();
4294 }
4295 }
4296 }
4297
4298 return S_OK;
4299 }
4300 else if ( foundIt == oldAtts.end()
4301 || level > foundLevel /* prefer younger */
4302 )
4303 {
4304 foundIt = it;
4305 foundLevel = level;
4306 }
4307 }
4308 }
4309
4310 if (foundIt != oldAtts.end())
4311 {
4312 /* use the previously attached hard disk */
4313 medium = (*foundIt)->getMedium();
4314 mediumCaller.attach(medium);
4315 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
4316 mediumLock.attach(medium);
4317 /* not implicit, doesn't require association with this VM */
4318 fIndirect = false;
4319 associate = false;
4320 /* go right to the MediumAttachment creation */
4321 break;
4322 }
4323 }
4324
4325 /* must give up the medium lock and medium tree lock as below we
4326 * go over snapshots, which needs a lock with higher lock order. */
4327 mediumLock.release();
4328 treeLock.release();
4329
4330 /* then, search through snapshots for the best diff in the given
4331 * hard disk's chain to base the new diff on */
4332
4333 ComObjPtr<Medium> base;
4334 ComObjPtr<Snapshot> snap = mData->mCurrentSnapshot;
4335 while (snap)
4336 {
4337 AutoReadLock snapLock(snap COMMA_LOCKVAL_SRC_POS);
4338
4339 const MediaData::AttachmentList &snapAtts = snap->getSnapshotMachine()->mMediaData->mAttachments;
4340
4341 MediumAttachment *pAttachFound = NULL;
4342 uint32_t foundLevel = 0;
4343
4344 for (MediaData::AttachmentList::const_iterator it = snapAtts.begin();
4345 it != snapAtts.end();
4346 ++it)
4347 {
4348 MediumAttachment *pAttach = *it;
4349 ComObjPtr<Medium> pMedium = pAttach->getMedium();
4350 Assert(!pMedium.isNull() || pAttach->getType() != DeviceType_HardDisk);
4351 if (pMedium.isNull())
4352 continue;
4353
4354 uint32_t level = 0;
4355 if (pMedium->getBase(&level) == medium)
4356 {
4357 /* matched device, channel and bus (i.e. attached to the
4358 * same place) will win and immediately stop the search;
4359 * otherwise the attachment that has the youngest
4360 * descendant of medium will be used
4361 */
4362 if ( pAttach->getDevice() == aDevice
4363 && pAttach->getPort() == aControllerPort
4364 && pAttach->getControllerName() == aControllerName
4365 )
4366 {
4367 pAttachFound = pAttach;
4368 break;
4369 }
4370 else if ( !pAttachFound
4371 || level > foundLevel /* prefer younger */
4372 )
4373 {
4374 pAttachFound = pAttach;
4375 foundLevel = level;
4376 }
4377 }
4378 }
4379
4380 if (pAttachFound)
4381 {
4382 base = pAttachFound->getMedium();
4383 break;
4384 }
4385
4386 snap = snap->getParent();
4387 }
4388
4389 /* re-lock medium tree and the medium, as we need it below */
4390 treeLock.acquire();
4391 mediumLock.acquire();
4392
4393 /* found a suitable diff, use it as a base */
4394 if (!base.isNull())
4395 {
4396 medium = base;
4397 mediumCaller.attach(medium);
4398 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
4399 mediumLock.attach(medium);
4400 }
4401 }
4402
4403 Utf8Str strFullSnapshotFolder;
4404 calculateFullPath(mUserData->s.strSnapshotFolder, strFullSnapshotFolder);
4405
4406 ComObjPtr<Medium> diff;
4407 diff.createObject();
4408 // store this diff in the same registry as the parent
4409 Guid uuidRegistryParent;
4410 if (!medium->getFirstRegistryMachineId(uuidRegistryParent))
4411 {
4412 // parent image has no registry: this can happen if we're attaching a new immutable
4413 // image that has not yet been attached (medium then points to the base and we're
4414 // creating the diff image for the immutable, and the parent is not yet registered);
4415 // put the parent in the machine registry then
4416 mediumLock.release();
4417 treeLock.release();
4418 alock.release();
4419 addMediumToRegistry(medium);
4420 alock.acquire();
4421 treeLock.acquire();
4422 mediumLock.acquire();
4423 medium->getFirstRegistryMachineId(uuidRegistryParent);
4424 }
4425 rc = diff->init(mParent,
4426 medium->getPreferredDiffFormat(),
4427 strFullSnapshotFolder.append(RTPATH_SLASH_STR),
4428 uuidRegistryParent);
4429 if (FAILED(rc)) return rc;
4430
4431 /* Apply the normal locking logic to the entire chain. */
4432 MediumLockList *pMediumLockList(new MediumLockList());
4433 mediumLock.release();
4434 treeLock.release();
4435 rc = diff->createMediumLockList(true /* fFailIfInaccessible */,
4436 true /* fMediumLockWrite */,
4437 medium,
4438 *pMediumLockList);
4439 treeLock.acquire();
4440 mediumLock.acquire();
4441 if (SUCCEEDED(rc))
4442 {
4443 mediumLock.release();
4444 treeLock.release();
4445 rc = pMediumLockList->Lock();
4446 treeLock.acquire();
4447 mediumLock.acquire();
4448 if (FAILED(rc))
4449 setError(rc,
4450 tr("Could not lock medium when creating diff '%s'"),
4451 diff->getLocationFull().c_str());
4452 else
4453 {
4454 /* will release the lock before the potentially lengthy
4455 * operation, so protect with the special state */
4456 MachineState_T oldState = mData->mMachineState;
4457 setMachineState(MachineState_SettingUp);
4458
4459 mediumLock.release();
4460 treeLock.release();
4461 alock.release();
4462
4463 rc = medium->createDiffStorage(diff,
4464 MediumVariant_Standard,
4465 pMediumLockList,
4466 NULL /* aProgress */,
4467 true /* aWait */);
4468
4469 alock.acquire();
4470 treeLock.acquire();
4471 mediumLock.acquire();
4472
4473 setMachineState(oldState);
4474 }
4475 }
4476
4477 /* Unlock the media and free the associated memory. */
4478 delete pMediumLockList;
4479
4480 if (FAILED(rc)) return rc;
4481
4482 /* use the created diff for the actual attachment */
4483 medium = diff;
4484 mediumCaller.attach(medium);
4485 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
4486 mediumLock.attach(medium);
4487 }
4488 while (0);
4489
4490 ComObjPtr<MediumAttachment> attachment;
4491 attachment.createObject();
4492 rc = attachment->init(this,
4493 medium,
4494 aControllerName,
4495 aControllerPort,
4496 aDevice,
4497 aType,
4498 fIndirect,
4499 false /* fPassthrough */,
4500 false /* fTempEject */,
4501 false /* fNonRotational */,
4502 false /* fDiscard */,
4503 Utf8Str::Empty);
4504 if (FAILED(rc)) return rc;
4505
4506 if (associate && !medium.isNull())
4507 {
4508 // as the last step, associate the medium to the VM
4509 rc = medium->addBackReference(mData->mUuid);
4510 // here we can fail because of Deleting, or being in process of creating a Diff
4511 if (FAILED(rc)) return rc;
4512
4513 mediumLock.release();
4514 treeLock.release();
4515 alock.release();
4516 addMediumToRegistry(medium);
4517 alock.acquire();
4518 treeLock.acquire();
4519 mediumLock.acquire();
4520 }
4521
4522 /* success: finally remember the attachment */
4523 setModified(IsModified_Storage);
4524 mMediaData.backup();
4525 mMediaData->mAttachments.push_back(attachment);
4526
4527 mediumLock.release();
4528 treeLock.release();
4529 alock.release();
4530
4531 if (fHotplug || fSilent)
4532 {
4533 if (!medium.isNull())
4534 {
4535 MediumLockList *pMediumLockList(new MediumLockList());
4536
4537 rc = medium->createMediumLockList(true /* fFailIfInaccessible */,
4538 true /* fMediumLockWrite */,
4539 NULL,
4540 *pMediumLockList);
4541 alock.acquire();
4542 if (FAILED(rc))
4543 delete pMediumLockList;
4544 else
4545 {
4546 mData->mSession.mLockedMedia.Unlock();
4547 alock.release();
4548 rc = mData->mSession.mLockedMedia.Insert(attachment, pMediumLockList);
4549 mData->mSession.mLockedMedia.Lock();
4550 alock.acquire();
4551 }
4552 alock.release();
4553 }
4554
4555 if (SUCCEEDED(rc))
4556 {
4557 rc = onStorageDeviceChange(attachment, FALSE /* aRemove */, fSilent);
4558 /* Remove lock list in case of error. */
4559 if (FAILED(rc))
4560 {
4561 mData->mSession.mLockedMedia.Unlock();
4562 mData->mSession.mLockedMedia.Remove(attachment);
4563 mData->mSession.mLockedMedia.Lock();
4564 }
4565 }
4566 }
4567
4568 mParent->saveModifiedRegistries();
4569
4570 return rc;
4571}
4572
4573STDMETHODIMP Machine::DetachDevice(IN_BSTR aControllerName, LONG aControllerPort,
4574 LONG aDevice)
4575{
4576 CheckComArgStrNotEmptyOrNull(aControllerName);
4577
4578 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
4579 aControllerName, aControllerPort, aDevice));
4580
4581 AutoCaller autoCaller(this);
4582 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4583
4584 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4585
4586 HRESULT rc = checkStateDependency(MutableStateDep);
4587 if (FAILED(rc)) return rc;
4588
4589 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4590
4591 /* Check for an existing controller. */
4592 ComObjPtr<StorageController> ctl;
4593 rc = getStorageControllerByName(aControllerName, ctl, true /* aSetError */);
4594 if (FAILED(rc)) return rc;
4595
4596 StorageControllerType_T ctrlType;
4597 rc = ctl->COMGETTER(ControllerType)(&ctrlType);
4598 if (FAILED(rc))
4599 return setError(E_FAIL,
4600 tr("Could not get type of controller '%ls'"),
4601 aControllerName);
4602
4603 bool fSilent = false;
4604 Utf8Str strReconfig;
4605
4606 /* Check whether the flag to allow silent storage attachment reconfiguration is set. */
4607 strReconfig = getExtraData(Utf8Str("VBoxInternal2/SilentReconfigureWhilePaused"));
4608 if ( mData->mMachineState == MachineState_Paused
4609 && strReconfig == "1")
4610 fSilent = true;
4611
4612 /* Check that the controller can do hotplugging if we detach the device while the VM is running. */
4613 bool fHotplug = false;
4614 if (!fSilent && Global::IsOnlineOrTransient(mData->mMachineState))
4615 fHotplug = true;
4616
4617 if (fHotplug && !isControllerHotplugCapable(ctrlType))
4618 return setError(VBOX_E_INVALID_VM_STATE,
4619 tr("Controller '%ls' does not support hotplugging"),
4620 aControllerName);
4621
4622 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4623 aControllerName,
4624 aControllerPort,
4625 aDevice);
4626 if (!pAttach)
4627 return setError(VBOX_E_OBJECT_NOT_FOUND,
4628 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4629 aDevice, aControllerPort, aControllerName);
4630
4631 /*
4632 * The VM has to detach the device before we delete any implicit diffs.
4633 * If this fails we can roll back without loosing data.
4634 */
4635 if (fHotplug || fSilent)
4636 {
4637 alock.release();
4638 rc = onStorageDeviceChange(pAttach, TRUE /* aRemove */, fSilent);
4639 alock.acquire();
4640 }
4641 if (FAILED(rc)) return rc;
4642
4643 /* If we are here everything went well and we can delete the implicit now. */
4644 rc = detachDevice(pAttach, alock, NULL /* pSnapshot */);
4645
4646 alock.release();
4647
4648 mParent->saveModifiedRegistries();
4649
4650 return rc;
4651}
4652
4653STDMETHODIMP Machine::PassthroughDevice(IN_BSTR aControllerName, LONG aControllerPort,
4654 LONG aDevice, BOOL aPassthrough)
4655{
4656 CheckComArgStrNotEmptyOrNull(aControllerName);
4657
4658 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aPassthrough=%d\n",
4659 aControllerName, aControllerPort, aDevice, aPassthrough));
4660
4661 AutoCaller autoCaller(this);
4662 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4663
4664 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4665
4666 HRESULT rc = checkStateDependency(MutableStateDep);
4667 if (FAILED(rc)) return rc;
4668
4669 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4670
4671 if (Global::IsOnlineOrTransient(mData->mMachineState))
4672 return setError(VBOX_E_INVALID_VM_STATE,
4673 tr("Invalid machine state: %s"),
4674 Global::stringifyMachineState(mData->mMachineState));
4675
4676 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4677 aControllerName,
4678 aControllerPort,
4679 aDevice);
4680 if (!pAttach)
4681 return setError(VBOX_E_OBJECT_NOT_FOUND,
4682 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4683 aDevice, aControllerPort, aControllerName);
4684
4685
4686 setModified(IsModified_Storage);
4687 mMediaData.backup();
4688
4689 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
4690
4691 if (pAttach->getType() != DeviceType_DVD)
4692 return setError(E_INVALIDARG,
4693 tr("Setting passthrough rejected as the device attached to device slot %d on port %d of controller '%ls' is not a DVD"),
4694 aDevice, aControllerPort, aControllerName);
4695 pAttach->updatePassthrough(!!aPassthrough);
4696
4697 return S_OK;
4698}
4699
4700STDMETHODIMP Machine::TemporaryEjectDevice(IN_BSTR aControllerName, LONG aControllerPort,
4701 LONG aDevice, BOOL aTemporaryEject)
4702{
4703 CheckComArgStrNotEmptyOrNull(aControllerName);
4704
4705 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aTemporaryEject=%d\n",
4706 aControllerName, aControllerPort, aDevice, aTemporaryEject));
4707
4708 AutoCaller autoCaller(this);
4709 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4710
4711 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4712
4713 HRESULT rc = checkStateDependency(MutableStateDep);
4714 if (FAILED(rc)) return rc;
4715
4716 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4717 aControllerName,
4718 aControllerPort,
4719 aDevice);
4720 if (!pAttach)
4721 return setError(VBOX_E_OBJECT_NOT_FOUND,
4722 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4723 aDevice, aControllerPort, aControllerName);
4724
4725
4726 setModified(IsModified_Storage);
4727 mMediaData.backup();
4728
4729 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
4730
4731 if (pAttach->getType() != DeviceType_DVD)
4732 return setError(E_INVALIDARG,
4733 tr("Setting temporary eject flag rejected as the device attached to device slot %d on port %d of controller '%ls' is not a DVD"),
4734 aDevice, aControllerPort, aControllerName);
4735 pAttach->updateTempEject(!!aTemporaryEject);
4736
4737 return S_OK;
4738}
4739
4740STDMETHODIMP Machine::NonRotationalDevice(IN_BSTR aControllerName, LONG aControllerPort,
4741 LONG aDevice, BOOL aNonRotational)
4742{
4743 CheckComArgStrNotEmptyOrNull(aControllerName);
4744
4745 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aNonRotational=%d\n",
4746 aControllerName, aControllerPort, aDevice, aNonRotational));
4747
4748 AutoCaller autoCaller(this);
4749 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4750
4751 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4752
4753 HRESULT rc = checkStateDependency(MutableStateDep);
4754 if (FAILED(rc)) return rc;
4755
4756 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4757
4758 if (Global::IsOnlineOrTransient(mData->mMachineState))
4759 return setError(VBOX_E_INVALID_VM_STATE,
4760 tr("Invalid machine state: %s"),
4761 Global::stringifyMachineState(mData->mMachineState));
4762
4763 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4764 aControllerName,
4765 aControllerPort,
4766 aDevice);
4767 if (!pAttach)
4768 return setError(VBOX_E_OBJECT_NOT_FOUND,
4769 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4770 aDevice, aControllerPort, aControllerName);
4771
4772
4773 setModified(IsModified_Storage);
4774 mMediaData.backup();
4775
4776 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
4777
4778 if (pAttach->getType() != DeviceType_HardDisk)
4779 return setError(E_INVALIDARG,
4780 tr("Setting the non-rotational medium flag rejected as the device attached to device slot %d on port %d of controller '%ls' is not a hard disk"),
4781 aDevice, aControllerPort, aControllerName);
4782 pAttach->updateNonRotational(!!aNonRotational);
4783
4784 return S_OK;
4785}
4786
4787STDMETHODIMP Machine::SetAutoDiscardForDevice(IN_BSTR aControllerName, LONG aControllerPort,
4788 LONG aDevice, BOOL aDiscard)
4789{
4790 CheckComArgStrNotEmptyOrNull(aControllerName);
4791
4792 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aDiscard=%d\n",
4793 aControllerName, aControllerPort, aDevice, aDiscard));
4794
4795 AutoCaller autoCaller(this);
4796 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4797
4798 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4799
4800 HRESULT rc = checkStateDependency(MutableStateDep);
4801 if (FAILED(rc)) return rc;
4802
4803 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4804
4805 if (Global::IsOnlineOrTransient(mData->mMachineState))
4806 return setError(VBOX_E_INVALID_VM_STATE,
4807 tr("Invalid machine state: %s"),
4808 Global::stringifyMachineState(mData->mMachineState));
4809
4810 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4811 aControllerName,
4812 aControllerPort,
4813 aDevice);
4814 if (!pAttach)
4815 return setError(VBOX_E_OBJECT_NOT_FOUND,
4816 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4817 aDevice, aControllerPort, aControllerName);
4818
4819
4820 setModified(IsModified_Storage);
4821 mMediaData.backup();
4822
4823 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
4824
4825 if (pAttach->getType() != DeviceType_HardDisk)
4826 return setError(E_INVALIDARG,
4827 tr("Setting the discard medium flag rejected as the device attached to device slot %d on port %d of controller '%ls' is not a hard disk"),
4828 aDevice, aControllerPort, aControllerName);
4829 pAttach->updateDiscard(!!aDiscard);
4830
4831 return S_OK;
4832}
4833
4834STDMETHODIMP Machine::SetNoBandwidthGroupForDevice(IN_BSTR aControllerName, LONG aControllerPort,
4835 LONG aDevice)
4836{
4837 int rc = S_OK;
4838 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
4839 aControllerName, aControllerPort, aDevice));
4840
4841 rc = SetBandwidthGroupForDevice(aControllerName, aControllerPort, aDevice, NULL);
4842
4843 return rc;
4844}
4845
4846STDMETHODIMP Machine::SetBandwidthGroupForDevice(IN_BSTR aControllerName, LONG aControllerPort,
4847 LONG aDevice, IBandwidthGroup *aBandwidthGroup)
4848{
4849 CheckComArgStrNotEmptyOrNull(aControllerName);
4850
4851 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
4852 aControllerName, aControllerPort, aDevice));
4853
4854 AutoCaller autoCaller(this);
4855 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4856
4857 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4858
4859 HRESULT rc = checkStateDependency(MutableStateDep);
4860 if (FAILED(rc)) return rc;
4861
4862 AssertReturn(mData->mMachineState != MachineState_Saved, E_FAIL);
4863
4864 if (Global::IsOnlineOrTransient(mData->mMachineState))
4865 return setError(VBOX_E_INVALID_VM_STATE,
4866 tr("Invalid machine state: %s"),
4867 Global::stringifyMachineState(mData->mMachineState));
4868
4869 MediumAttachment *pAttach = findAttachment(mMediaData->mAttachments,
4870 aControllerName,
4871 aControllerPort,
4872 aDevice);
4873 if (!pAttach)
4874 return setError(VBOX_E_OBJECT_NOT_FOUND,
4875 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
4876 aDevice, aControllerPort, aControllerName);
4877
4878
4879 setModified(IsModified_Storage);
4880 mMediaData.backup();
4881
4882 ComObjPtr<BandwidthGroup> group = static_cast<BandwidthGroup*>(aBandwidthGroup);
4883 if (aBandwidthGroup && group.isNull())
4884 return setError(E_INVALIDARG, "The given bandwidth group pointer is invalid");
4885
4886 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
4887
4888 const Utf8Str strBandwidthGroupOld = pAttach->getBandwidthGroup();
4889 if (strBandwidthGroupOld.isNotEmpty())
4890 {
4891 /* Get the bandwidth group object and release it - this must not fail. */
4892 ComObjPtr<BandwidthGroup> pBandwidthGroupOld;
4893 rc = getBandwidthGroup(strBandwidthGroupOld, pBandwidthGroupOld, false);
4894 Assert(SUCCEEDED(rc));
4895
4896 pBandwidthGroupOld->release();
4897 pAttach->updateBandwidthGroup(Utf8Str::Empty);
4898 }
4899
4900 if (!group.isNull())
4901 {
4902 group->reference();
4903 pAttach->updateBandwidthGroup(group->getName());
4904 }
4905
4906 return S_OK;
4907}
4908
4909STDMETHODIMP Machine::AttachDeviceWithoutMedium(IN_BSTR aControllerName,
4910 LONG aControllerPort,
4911 LONG aDevice,
4912 DeviceType_T aType)
4913{
4914 HRESULT rc = S_OK;
4915
4916 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aType=%d aMedium=%p\n",
4917 aControllerName, aControllerPort, aDevice, aType));
4918
4919 rc = AttachDevice(aControllerName, aControllerPort, aDevice, aType, NULL);
4920
4921 return rc;
4922}
4923
4924
4925
4926STDMETHODIMP Machine::UnmountMedium(IN_BSTR aControllerName,
4927 LONG aControllerPort,
4928 LONG aDevice,
4929 BOOL aForce)
4930{
4931 int rc = S_OK;
4932 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d",
4933 aControllerName, aControllerPort, aForce));
4934
4935 rc = MountMedium(aControllerName, aControllerPort, aDevice, NULL, aForce);
4936
4937 return rc;
4938}
4939
4940STDMETHODIMP Machine::MountMedium(IN_BSTR aControllerName,
4941 LONG aControllerPort,
4942 LONG aDevice,
4943 IMedium *aMedium,
4944 BOOL aForce)
4945{
4946 int rc = S_OK;
4947 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d aForce=%d\n",
4948 aControllerName, aControllerPort, aDevice, aForce));
4949
4950 CheckComArgStrNotEmptyOrNull(aControllerName);
4951
4952 AutoCaller autoCaller(this);
4953 if (FAILED(autoCaller.rc())) return autoCaller.rc();
4954
4955 // request the host lock first, since might be calling Host methods for getting host drives;
4956 // next, protect the media tree all the while we're in here, as well as our member variables
4957 AutoMultiWriteLock3 multiLock(mParent->host()->lockHandle(),
4958 this->lockHandle(),
4959 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
4960
4961 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
4962 aControllerName,
4963 aControllerPort,
4964 aDevice);
4965 if (pAttach.isNull())
4966 return setError(VBOX_E_OBJECT_NOT_FOUND,
4967 tr("No drive attached to device slot %d on port %d of controller '%ls'"),
4968 aDevice, aControllerPort, aControllerName);
4969
4970 /* Remember previously mounted medium. The medium before taking the
4971 * backup is not necessarily the same thing. */
4972 ComObjPtr<Medium> oldmedium;
4973 oldmedium = pAttach->getMedium();
4974
4975 ComObjPtr<Medium> pMedium = static_cast<Medium*>(aMedium);
4976 if (aMedium && pMedium.isNull())
4977 return setError(E_INVALIDARG, "The given medium pointer is invalid");
4978
4979 AutoCaller mediumCaller(pMedium);
4980 if (FAILED(mediumCaller.rc())) return mediumCaller.rc();
4981
4982 AutoWriteLock mediumLock(pMedium COMMA_LOCKVAL_SRC_POS);
4983 if (pMedium)
4984 {
4985 DeviceType_T mediumType = pAttach->getType();
4986 switch (mediumType)
4987 {
4988 case DeviceType_DVD:
4989 case DeviceType_Floppy:
4990 break;
4991
4992 default:
4993 return setError(VBOX_E_INVALID_OBJECT_STATE,
4994 tr("The device at port %d, device %d of controller '%ls' of this virtual machine is not removeable"),
4995 aControllerPort,
4996 aDevice,
4997 aControllerName);
4998 }
4999 }
5000
5001 setModified(IsModified_Storage);
5002 mMediaData.backup();
5003
5004 {
5005 // The backup operation makes the pAttach reference point to the
5006 // old settings. Re-get the correct reference.
5007 pAttach = findAttachment(mMediaData->mAttachments,
5008 aControllerName,
5009 aControllerPort,
5010 aDevice);
5011 if (!oldmedium.isNull())
5012 oldmedium->removeBackReference(mData->mUuid);
5013 if (!pMedium.isNull())
5014 {
5015 pMedium->addBackReference(mData->mUuid);
5016
5017 mediumLock.release();
5018 multiLock.release();
5019 addMediumToRegistry(pMedium);
5020 multiLock.acquire();
5021 mediumLock.acquire();
5022 }
5023
5024 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
5025 pAttach->updateMedium(pMedium);
5026 }
5027
5028 setModified(IsModified_Storage);
5029
5030 mediumLock.release();
5031 multiLock.release();
5032 rc = onMediumChange(pAttach, aForce);
5033 multiLock.acquire();
5034 mediumLock.acquire();
5035
5036 /* On error roll back this change only. */
5037 if (FAILED(rc))
5038 {
5039 if (!pMedium.isNull())
5040 pMedium->removeBackReference(mData->mUuid);
5041 pAttach = findAttachment(mMediaData->mAttachments,
5042 aControllerName,
5043 aControllerPort,
5044 aDevice);
5045 /* If the attachment is gone in the meantime, bail out. */
5046 if (pAttach.isNull())
5047 return rc;
5048 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
5049 if (!oldmedium.isNull())
5050 oldmedium->addBackReference(mData->mUuid);
5051 pAttach->updateMedium(oldmedium);
5052 }
5053
5054 mediumLock.release();
5055 multiLock.release();
5056
5057 mParent->saveModifiedRegistries();
5058
5059 return rc;
5060}
5061
5062STDMETHODIMP Machine::GetMedium(IN_BSTR aControllerName,
5063 LONG aControllerPort,
5064 LONG aDevice,
5065 IMedium **aMedium)
5066{
5067 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
5068 aControllerName, aControllerPort, aDevice));
5069
5070 CheckComArgStrNotEmptyOrNull(aControllerName);
5071 CheckComArgOutPointerValid(aMedium);
5072
5073 AutoCaller autoCaller(this);
5074 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5075
5076 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5077
5078 *aMedium = NULL;
5079
5080 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
5081 aControllerName,
5082 aControllerPort,
5083 aDevice);
5084 if (pAttach.isNull())
5085 return setError(VBOX_E_OBJECT_NOT_FOUND,
5086 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
5087 aDevice, aControllerPort, aControllerName);
5088
5089 pAttach->getMedium().queryInterfaceTo(aMedium);
5090
5091 return S_OK;
5092}
5093
5094STDMETHODIMP Machine::GetSerialPort(ULONG slot, ISerialPort **port)
5095{
5096 CheckComArgOutPointerValid(port);
5097 CheckComArgExpr(slot, slot < RT_ELEMENTS(mSerialPorts));
5098
5099 AutoCaller autoCaller(this);
5100 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5101
5102 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5103
5104 mSerialPorts[slot].queryInterfaceTo(port);
5105
5106 return S_OK;
5107}
5108
5109STDMETHODIMP Machine::GetParallelPort(ULONG slot, IParallelPort **port)
5110{
5111 CheckComArgOutPointerValid(port);
5112 CheckComArgExpr(slot, slot < RT_ELEMENTS(mParallelPorts));
5113
5114 AutoCaller autoCaller(this);
5115 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5116
5117 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5118
5119 mParallelPorts[slot].queryInterfaceTo(port);
5120
5121 return S_OK;
5122}
5123
5124STDMETHODIMP Machine::GetNetworkAdapter(ULONG slot, INetworkAdapter **adapter)
5125{
5126 CheckComArgOutPointerValid(adapter);
5127 /* Do not assert if slot is out of range, just return the advertised
5128 status. testdriver/vbox.py triggers this in logVmInfo. */
5129 if (slot >= mNetworkAdapters.size())
5130 return setError(E_INVALIDARG,
5131 tr("No network adapter in slot %RU32 (total %RU32 adapters)"),
5132 slot, mNetworkAdapters.size());
5133
5134 AutoCaller autoCaller(this);
5135 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5136
5137 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5138
5139 mNetworkAdapters[slot].queryInterfaceTo(adapter);
5140
5141 return S_OK;
5142}
5143
5144STDMETHODIMP Machine::GetExtraDataKeys(ComSafeArrayOut(BSTR, aKeys))
5145{
5146 CheckComArgOutSafeArrayPointerValid(aKeys);
5147
5148 AutoCaller autoCaller(this);
5149 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5150
5151 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5152
5153 com::SafeArray<BSTR> saKeys(mData->pMachineConfigFile->mapExtraDataItems.size());
5154 int i = 0;
5155 for (settings::StringsMap::const_iterator it = mData->pMachineConfigFile->mapExtraDataItems.begin();
5156 it != mData->pMachineConfigFile->mapExtraDataItems.end();
5157 ++it, ++i)
5158 {
5159 const Utf8Str &strKey = it->first;
5160 strKey.cloneTo(&saKeys[i]);
5161 }
5162 saKeys.detachTo(ComSafeArrayOutArg(aKeys));
5163
5164 return S_OK;
5165 }
5166
5167 /**
5168 * @note Locks this object for reading.
5169 */
5170STDMETHODIMP Machine::GetExtraData(IN_BSTR aKey,
5171 BSTR *aValue)
5172{
5173 CheckComArgStrNotEmptyOrNull(aKey);
5174 CheckComArgOutPointerValid(aValue);
5175
5176 AutoCaller autoCaller(this);
5177 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5178
5179 /* start with nothing found */
5180 Bstr bstrResult("");
5181
5182 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5183
5184 settings::StringsMap::const_iterator it = mData->pMachineConfigFile->mapExtraDataItems.find(Utf8Str(aKey));
5185 if (it != mData->pMachineConfigFile->mapExtraDataItems.end())
5186 // found:
5187 bstrResult = it->second; // source is a Utf8Str
5188
5189 /* return the result to caller (may be empty) */
5190 bstrResult.cloneTo(aValue);
5191
5192 return S_OK;
5193}
5194
5195 /**
5196 * @note Locks mParent for writing + this object for writing.
5197 */
5198STDMETHODIMP Machine::SetExtraData(IN_BSTR aKey, IN_BSTR aValue)
5199{
5200 CheckComArgStrNotEmptyOrNull(aKey);
5201
5202 AutoCaller autoCaller(this);
5203 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5204
5205 Utf8Str strKey(aKey);
5206 Utf8Str strValue(aValue);
5207 Utf8Str strOldValue; // empty
5208
5209 // locking note: we only hold the read lock briefly to look up the old value,
5210 // then release it and call the onExtraCanChange callbacks. There is a small
5211 // chance of a race insofar as the callback might be called twice if two callers
5212 // change the same key at the same time, but that's a much better solution
5213 // than the deadlock we had here before. The actual changing of the extradata
5214 // is then performed under the write lock and race-free.
5215
5216 // look up the old value first; if nothing has changed then we need not do anything
5217 {
5218 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); // hold read lock only while looking up
5219 settings::StringsMap::const_iterator it = mData->pMachineConfigFile->mapExtraDataItems.find(strKey);
5220 if (it != mData->pMachineConfigFile->mapExtraDataItems.end())
5221 strOldValue = it->second;
5222 }
5223
5224 bool fChanged;
5225 if ((fChanged = (strOldValue != strValue)))
5226 {
5227 // ask for permission from all listeners outside the locks;
5228 // onExtraDataCanChange() only briefly requests the VirtualBox
5229 // lock to copy the list of callbacks to invoke
5230 Bstr error;
5231 Bstr bstrValue(aValue);
5232
5233 if (!mParent->onExtraDataCanChange(mData->mUuid, aKey, bstrValue.raw(), error))
5234 {
5235 const char *sep = error.isEmpty() ? "" : ": ";
5236 CBSTR err = error.raw();
5237 LogWarningFunc(("Someone vetoed! Change refused%s%ls\n",
5238 sep, err));
5239 return setError(E_ACCESSDENIED,
5240 tr("Could not set extra data because someone refused the requested change of '%ls' to '%ls'%s%ls"),
5241 aKey,
5242 bstrValue.raw(),
5243 sep,
5244 err);
5245 }
5246
5247 // data is changing and change not vetoed: then write it out under the lock
5248 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5249
5250 if (isSnapshotMachine())
5251 {
5252 HRESULT rc = checkStateDependency(MutableStateDep);
5253 if (FAILED(rc)) return rc;
5254 }
5255
5256 if (strValue.isEmpty())
5257 mData->pMachineConfigFile->mapExtraDataItems.erase(strKey);
5258 else
5259 mData->pMachineConfigFile->mapExtraDataItems[strKey] = strValue;
5260 // creates a new key if needed
5261
5262 bool fNeedsGlobalSaveSettings = false;
5263 saveSettings(&fNeedsGlobalSaveSettings);
5264
5265 if (fNeedsGlobalSaveSettings)
5266 {
5267 // save the global settings; for that we should hold only the VirtualBox lock
5268 alock.release();
5269 AutoWriteLock vboxlock(mParent COMMA_LOCKVAL_SRC_POS);
5270 mParent->saveSettings();
5271 }
5272 }
5273
5274 // fire notification outside the lock
5275 if (fChanged)
5276 mParent->onExtraDataChange(mData->mUuid, aKey, aValue);
5277
5278 return S_OK;
5279}
5280
5281STDMETHODIMP Machine::SetSettingsFilePath(IN_BSTR aFilePath, IProgress **aProgress)
5282{
5283 CheckComArgStrNotEmptyOrNull(aFilePath);
5284 CheckComArgOutPointerValid(aProgress);
5285
5286 AutoCaller autoCaller(this);
5287 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5288
5289 *aProgress = NULL;
5290 ReturnComNotImplemented();
5291}
5292
5293STDMETHODIMP Machine::SaveSettings()
5294{
5295 AutoCaller autoCaller(this);
5296 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5297
5298 AutoWriteLock mlock(this COMMA_LOCKVAL_SRC_POS);
5299
5300 /* when there was auto-conversion, we want to save the file even if
5301 * the VM is saved */
5302 HRESULT rc = checkStateDependency(MutableOrSavedStateDep);
5303 if (FAILED(rc)) return rc;
5304
5305 /* the settings file path may never be null */
5306 ComAssertRet(!mData->m_strConfigFileFull.isEmpty(), E_FAIL);
5307
5308 /* save all VM data excluding snapshots */
5309 bool fNeedsGlobalSaveSettings = false;
5310 rc = saveSettings(&fNeedsGlobalSaveSettings);
5311 mlock.release();
5312
5313 if (SUCCEEDED(rc) && fNeedsGlobalSaveSettings)
5314 {
5315 // save the global settings; for that we should hold only the VirtualBox lock
5316 AutoWriteLock vlock(mParent COMMA_LOCKVAL_SRC_POS);
5317 rc = mParent->saveSettings();
5318 }
5319
5320 return rc;
5321}
5322
5323STDMETHODIMP Machine::DiscardSettings()
5324{
5325 AutoCaller autoCaller(this);
5326 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5327
5328 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5329
5330 HRESULT rc = checkStateDependency(MutableStateDep);
5331 if (FAILED(rc)) return rc;
5332
5333 /*
5334 * during this rollback, the session will be notified if data has
5335 * been actually changed
5336 */
5337 rollback(true /* aNotify */);
5338
5339 return S_OK;
5340}
5341
5342/** @note Locks objects! */
5343STDMETHODIMP Machine::Unregister(CleanupMode_T cleanupMode,
5344 ComSafeArrayOut(IMedium*, aMedia))
5345{
5346 // use AutoLimitedCaller because this call is valid on inaccessible machines as well
5347 AutoLimitedCaller autoCaller(this);
5348 AssertComRCReturnRC(autoCaller.rc());
5349
5350 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5351
5352 Guid id(getId());
5353
5354 if (mData->mSession.mState != SessionState_Unlocked)
5355 return setError(VBOX_E_INVALID_OBJECT_STATE,
5356 tr("Cannot unregister the machine '%s' while it is locked"),
5357 mUserData->s.strName.c_str());
5358
5359 // wait for state dependents to drop to zero
5360 ensureNoStateDependencies();
5361
5362 if (!mData->mAccessible)
5363 {
5364 // inaccessible maschines can only be unregistered; uninitialize ourselves
5365 // here because currently there may be no unregistered that are inaccessible
5366 // (this state combination is not supported). Note releasing the caller and
5367 // leaving the lock before calling uninit()
5368 alock.release();
5369 autoCaller.release();
5370
5371 uninit();
5372
5373 mParent->unregisterMachine(this, id);
5374 // calls VirtualBox::saveSettings()
5375
5376 return S_OK;
5377 }
5378
5379 HRESULT rc = S_OK;
5380
5381 // discard saved state
5382 if (mData->mMachineState == MachineState_Saved)
5383 {
5384 // add the saved state file to the list of files the caller should delete
5385 Assert(!mSSData->strStateFilePath.isEmpty());
5386 mData->llFilesToDelete.push_back(mSSData->strStateFilePath);
5387
5388 mSSData->strStateFilePath.setNull();
5389
5390 // unconditionally set the machine state to powered off, we now
5391 // know no session has locked the machine
5392 mData->mMachineState = MachineState_PoweredOff;
5393 }
5394
5395 size_t cSnapshots = 0;
5396 if (mData->mFirstSnapshot)
5397 cSnapshots = mData->mFirstSnapshot->getAllChildrenCount() + 1;
5398 if (cSnapshots && cleanupMode == CleanupMode_UnregisterOnly)
5399 // fail now before we start detaching media
5400 return setError(VBOX_E_INVALID_OBJECT_STATE,
5401 tr("Cannot unregister the machine '%s' because it has %d snapshots"),
5402 mUserData->s.strName.c_str(), cSnapshots);
5403
5404 // This list collects the medium objects from all medium attachments
5405 // which we will detach from the machine and its snapshots, in a specific
5406 // order which allows for closing all media without getting "media in use"
5407 // errors, simply by going through the list from the front to the back:
5408 // 1) first media from machine attachments (these have the "leaf" attachments with snapshots
5409 // and must be closed before the parent media from the snapshots, or closing the parents
5410 // will fail because they still have children);
5411 // 2) media from the youngest snapshots followed by those from the parent snapshots until
5412 // the root ("first") snapshot of the machine.
5413 MediaList llMedia;
5414
5415 if ( !mMediaData.isNull() // can be NULL if machine is inaccessible
5416 && mMediaData->mAttachments.size()
5417 )
5418 {
5419 // we have media attachments: detach them all and add the Medium objects to our list
5420 if (cleanupMode != CleanupMode_UnregisterOnly)
5421 detachAllMedia(alock, NULL /* pSnapshot */, cleanupMode, llMedia);
5422 else
5423 return setError(VBOX_E_INVALID_OBJECT_STATE,
5424 tr("Cannot unregister the machine '%s' because it has %d media attachments"),
5425 mUserData->s.strName.c_str(), mMediaData->mAttachments.size());
5426 }
5427
5428 if (cSnapshots)
5429 {
5430 // autoCleanup must be true here, or we would have failed above
5431
5432 // add the media from the medium attachments of the snapshots to llMedia
5433 // as well, after the "main" machine media; Snapshot::uninitRecursively()
5434 // calls Machine::detachAllMedia() for the snapshot machine, recursing
5435 // into the children first
5436
5437 // Snapshot::beginDeletingSnapshot() asserts if the machine state is not this
5438 MachineState_T oldState = mData->mMachineState;
5439 mData->mMachineState = MachineState_DeletingSnapshot;
5440
5441 // make a copy of the first snapshot so the refcount does not drop to 0
5442 // in beginDeletingSnapshot, which sets pFirstSnapshot to 0 (that hangs
5443 // because of the AutoCaller voodoo)
5444 ComObjPtr<Snapshot> pFirstSnapshot = mData->mFirstSnapshot;
5445
5446 // GO!
5447 pFirstSnapshot->uninitRecursively(alock, cleanupMode, llMedia, mData->llFilesToDelete);
5448
5449 mData->mMachineState = oldState;
5450 }
5451
5452 if (FAILED(rc))
5453 {
5454 rollbackMedia();
5455 return rc;
5456 }
5457
5458 // commit all the media changes made above
5459 commitMedia();
5460
5461 mData->mRegistered = false;
5462
5463 // machine lock no longer needed
5464 alock.release();
5465
5466 // return media to caller
5467 SafeIfaceArray<IMedium> sfaMedia(llMedia);
5468 sfaMedia.detachTo(ComSafeArrayOutArg(aMedia));
5469
5470 mParent->unregisterMachine(this, id);
5471 // calls VirtualBox::saveSettings() and VirtualBox::saveModifiedRegistries()
5472
5473 return S_OK;
5474}
5475
5476struct Machine::DeleteTask
5477{
5478 ComObjPtr<Machine> pMachine;
5479 RTCList<ComPtr<IMedium> > llMediums;
5480 StringsList llFilesToDelete;
5481 ComObjPtr<Progress> pProgress;
5482};
5483
5484STDMETHODIMP Machine::DeleteConfig(ComSafeArrayIn(IMedium*, aMedia), IProgress **aProgress)
5485{
5486 LogFlowFuncEnter();
5487
5488 AutoCaller autoCaller(this);
5489 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5490
5491 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5492
5493 HRESULT rc = checkStateDependency(MutableStateDep);
5494 if (FAILED(rc)) return rc;
5495
5496 if (mData->mRegistered)
5497 return setError(VBOX_E_INVALID_VM_STATE,
5498 tr("Cannot delete settings of a registered machine"));
5499
5500 DeleteTask *pTask = new DeleteTask;
5501 pTask->pMachine = this;
5502 com::SafeIfaceArray<IMedium> sfaMedia(ComSafeArrayInArg(aMedia));
5503
5504 // collect files to delete
5505 pTask->llFilesToDelete = mData->llFilesToDelete; // saved states pushed here by Unregister()
5506
5507 for (size_t i = 0; i < sfaMedia.size(); ++i)
5508 {
5509 IMedium *pIMedium(sfaMedia[i]);
5510 ComObjPtr<Medium> pMedium = static_cast<Medium*>(pIMedium);
5511 if (pMedium.isNull())
5512 return setError(E_INVALIDARG, "The given medium pointer with index %d is invalid", i);
5513 SafeArray<BSTR> ids;
5514 rc = pMedium->COMGETTER(MachineIds)(ComSafeArrayAsOutParam(ids));
5515 if (FAILED(rc)) return rc;
5516 /* At this point the medium should not have any back references
5517 * anymore. If it has it is attached to another VM and *must* not
5518 * deleted. */
5519 if (ids.size() < 1)
5520 pTask->llMediums.append(pMedium);
5521 }
5522 if (mData->pMachineConfigFile->fileExists())
5523 pTask->llFilesToDelete.push_back(mData->m_strConfigFileFull);
5524
5525 pTask->pProgress.createObject();
5526 pTask->pProgress->init(getVirtualBox(),
5527 static_cast<IMachine*>(this) /* aInitiator */,
5528 Bstr(tr("Deleting files")).raw(),
5529 true /* fCancellable */,
5530 pTask->llFilesToDelete.size() + pTask->llMediums.size() + 1, // cOperations
5531 BstrFmt(tr("Deleting '%s'"), pTask->llFilesToDelete.front().c_str()).raw());
5532
5533 int vrc = RTThreadCreate(NULL,
5534 Machine::deleteThread,
5535 (void*)pTask,
5536 0,
5537 RTTHREADTYPE_MAIN_WORKER,
5538 0,
5539 "MachineDelete");
5540
5541 pTask->pProgress.queryInterfaceTo(aProgress);
5542
5543 if (RT_FAILURE(vrc))
5544 {
5545 delete pTask;
5546 return setError(E_FAIL, "Could not create MachineDelete thread (%Rrc)", vrc);
5547 }
5548
5549 LogFlowFuncLeave();
5550
5551 return S_OK;
5552}
5553
5554/**
5555 * Static task wrapper passed to RTThreadCreate() in Machine::Delete() which then
5556 * calls Machine::deleteTaskWorker() on the actual machine object.
5557 * @param Thread
5558 * @param pvUser
5559 * @return
5560 */
5561/*static*/
5562DECLCALLBACK(int) Machine::deleteThread(RTTHREAD Thread, void *pvUser)
5563{
5564 LogFlowFuncEnter();
5565
5566 DeleteTask *pTask = (DeleteTask*)pvUser;
5567 Assert(pTask);
5568 Assert(pTask->pMachine);
5569 Assert(pTask->pProgress);
5570
5571 HRESULT rc = pTask->pMachine->deleteTaskWorker(*pTask);
5572 pTask->pProgress->notifyComplete(rc);
5573
5574 delete pTask;
5575
5576 LogFlowFuncLeave();
5577
5578 NOREF(Thread);
5579
5580 return VINF_SUCCESS;
5581}
5582
5583/**
5584 * Task thread implementation for Machine::Delete(), called from Machine::deleteThread().
5585 * @param task
5586 * @return
5587 */
5588HRESULT Machine::deleteTaskWorker(DeleteTask &task)
5589{
5590 AutoCaller autoCaller(this);
5591 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5592
5593 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5594
5595 HRESULT rc = S_OK;
5596
5597 try
5598 {
5599 ULONG uLogHistoryCount = 3;
5600 ComPtr<ISystemProperties> systemProperties;
5601 rc = mParent->COMGETTER(SystemProperties)(systemProperties.asOutParam());
5602 if (FAILED(rc)) throw rc;
5603
5604 if (!systemProperties.isNull())
5605 {
5606 rc = systemProperties->COMGETTER(LogHistoryCount)(&uLogHistoryCount);
5607 if (FAILED(rc)) throw rc;
5608 }
5609
5610 MachineState_T oldState = mData->mMachineState;
5611 setMachineState(MachineState_SettingUp);
5612 alock.release();
5613 for (size_t i = 0; i < task.llMediums.size(); ++i)
5614 {
5615 ComObjPtr<Medium> pMedium = (Medium*)(IMedium*)task.llMediums.at(i);
5616 {
5617 AutoCaller mac(pMedium);
5618 if (FAILED(mac.rc())) throw mac.rc();
5619 Utf8Str strLocation = pMedium->getLocationFull();
5620 rc = task.pProgress->SetNextOperation(BstrFmt(tr("Deleting '%s'"), strLocation.c_str()).raw(), 1);
5621 if (FAILED(rc)) throw rc;
5622 LogFunc(("Deleting file %s\n", strLocation.c_str()));
5623 }
5624 ComPtr<IProgress> pProgress2;
5625 rc = pMedium->DeleteStorage(pProgress2.asOutParam());
5626 if (FAILED(rc)) throw rc;
5627 rc = task.pProgress->WaitForAsyncProgressCompletion(pProgress2);
5628 if (FAILED(rc)) throw rc;
5629 /* Check the result of the asynchrony process. */
5630 LONG iRc;
5631 rc = pProgress2->COMGETTER(ResultCode)(&iRc);
5632 if (FAILED(rc)) throw rc;
5633 /* If the thread of the progress object has an error, then
5634 * retrieve the error info from there, or it'll be lost. */
5635 if (FAILED(iRc))
5636 throw setError(ProgressErrorInfo(pProgress2));
5637 }
5638 setMachineState(oldState);
5639 alock.acquire();
5640
5641 // delete the files pushed on the task list by Machine::Delete()
5642 // (this includes saved states of the machine and snapshots and
5643 // medium storage files from the IMedium list passed in, and the
5644 // machine XML file)
5645 StringsList::const_iterator it = task.llFilesToDelete.begin();
5646 while (it != task.llFilesToDelete.end())
5647 {
5648 const Utf8Str &strFile = *it;
5649 LogFunc(("Deleting file %s\n", strFile.c_str()));
5650 int vrc = RTFileDelete(strFile.c_str());
5651 if (RT_FAILURE(vrc))
5652 throw setError(VBOX_E_IPRT_ERROR,
5653 tr("Could not delete file '%s' (%Rrc)"), strFile.c_str(), vrc);
5654
5655 ++it;
5656 if (it == task.llFilesToDelete.end())
5657 {
5658 rc = task.pProgress->SetNextOperation(Bstr(tr("Cleaning up machine directory")).raw(), 1);
5659 if (FAILED(rc)) throw rc;
5660 break;
5661 }
5662
5663 rc = task.pProgress->SetNextOperation(BstrFmt(tr("Deleting '%s'"), it->c_str()).raw(), 1);
5664 if (FAILED(rc)) throw rc;
5665 }
5666
5667 /* delete the settings only when the file actually exists */
5668 if (mData->pMachineConfigFile->fileExists())
5669 {
5670 /* Delete any backup or uncommitted XML files. Ignore failures.
5671 See the fSafe parameter of xml::XmlFileWriter::write for details. */
5672 /** @todo Find a way to avoid referring directly to iprt/xml.h here. */
5673 Utf8Str otherXml = Utf8StrFmt("%s%s", mData->m_strConfigFileFull.c_str(), xml::XmlFileWriter::s_pszTmpSuff);
5674 RTFileDelete(otherXml.c_str());
5675 otherXml = Utf8StrFmt("%s%s", mData->m_strConfigFileFull.c_str(), xml::XmlFileWriter::s_pszPrevSuff);
5676 RTFileDelete(otherXml.c_str());
5677
5678 /* delete the Logs folder, nothing important should be left
5679 * there (we don't check for errors because the user might have
5680 * some private files there that we don't want to delete) */
5681 Utf8Str logFolder;
5682 getLogFolder(logFolder);
5683 Assert(logFolder.length());
5684 if (RTDirExists(logFolder.c_str()))
5685 {
5686 /* Delete all VBox.log[.N] files from the Logs folder
5687 * (this must be in sync with the rotation logic in
5688 * Console::powerUpThread()). Also, delete the VBox.png[.N]
5689 * files that may have been created by the GUI. */
5690 Utf8Str log = Utf8StrFmt("%s%cVBox.log",
5691 logFolder.c_str(), RTPATH_DELIMITER);
5692 RTFileDelete(log.c_str());
5693 log = Utf8StrFmt("%s%cVBox.png",
5694 logFolder.c_str(), RTPATH_DELIMITER);
5695 RTFileDelete(log.c_str());
5696 for (int i = uLogHistoryCount; i > 0; i--)
5697 {
5698 log = Utf8StrFmt("%s%cVBox.log.%d",
5699 logFolder.c_str(), RTPATH_DELIMITER, i);
5700 RTFileDelete(log.c_str());
5701 log = Utf8StrFmt("%s%cVBox.png.%d",
5702 logFolder.c_str(), RTPATH_DELIMITER, i);
5703 RTFileDelete(log.c_str());
5704 }
5705
5706 RTDirRemove(logFolder.c_str());
5707 }
5708
5709 /* delete the Snapshots folder, nothing important should be left
5710 * there (we don't check for errors because the user might have
5711 * some private files there that we don't want to delete) */
5712 Utf8Str strFullSnapshotFolder;
5713 calculateFullPath(mUserData->s.strSnapshotFolder, strFullSnapshotFolder);
5714 Assert(!strFullSnapshotFolder.isEmpty());
5715 if (RTDirExists(strFullSnapshotFolder.c_str()))
5716 RTDirRemove(strFullSnapshotFolder.c_str());
5717
5718 // delete the directory that contains the settings file, but only
5719 // if it matches the VM name
5720 Utf8Str settingsDir;
5721 if (isInOwnDir(&settingsDir))
5722 RTDirRemove(settingsDir.c_str());
5723 }
5724
5725 alock.release();
5726
5727 mParent->saveModifiedRegistries();
5728 }
5729 catch (HRESULT aRC) { rc = aRC; }
5730
5731 return rc;
5732}
5733
5734STDMETHODIMP Machine::FindSnapshot(IN_BSTR aNameOrId, ISnapshot **aSnapshot)
5735{
5736 CheckComArgOutPointerValid(aSnapshot);
5737
5738 AutoCaller autoCaller(this);
5739 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5740
5741 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5742
5743 ComObjPtr<Snapshot> pSnapshot;
5744 HRESULT rc;
5745
5746 if (!aNameOrId || !*aNameOrId)
5747 // null case (caller wants root snapshot): findSnapshotById() handles this
5748 rc = findSnapshotById(Guid(), pSnapshot, true /* aSetError */);
5749 else
5750 {
5751 Guid uuid(aNameOrId);
5752 if (uuid.isValid())
5753 rc = findSnapshotById(uuid, pSnapshot, true /* aSetError */);
5754 else
5755 rc = findSnapshotByName(Utf8Str(aNameOrId), pSnapshot, true /* aSetError */);
5756 }
5757 pSnapshot.queryInterfaceTo(aSnapshot);
5758
5759 return rc;
5760}
5761
5762STDMETHODIMP Machine::CreateSharedFolder(IN_BSTR aName, IN_BSTR aHostPath, BOOL aWritable, BOOL aAutoMount)
5763{
5764 CheckComArgStrNotEmptyOrNull(aName);
5765 CheckComArgStrNotEmptyOrNull(aHostPath);
5766
5767 AutoCaller autoCaller(this);
5768 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5769
5770 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5771
5772 HRESULT rc = checkStateDependency(MutableStateDep);
5773 if (FAILED(rc)) return rc;
5774
5775 Utf8Str strName(aName);
5776
5777 ComObjPtr<SharedFolder> sharedFolder;
5778 rc = findSharedFolder(strName, sharedFolder, false /* aSetError */);
5779 if (SUCCEEDED(rc))
5780 return setError(VBOX_E_OBJECT_IN_USE,
5781 tr("Shared folder named '%s' already exists"),
5782 strName.c_str());
5783
5784 sharedFolder.createObject();
5785 rc = sharedFolder->init(getMachine(),
5786 strName,
5787 aHostPath,
5788 !!aWritable,
5789 !!aAutoMount,
5790 true /* fFailOnError */);
5791 if (FAILED(rc)) return rc;
5792
5793 setModified(IsModified_SharedFolders);
5794 mHWData.backup();
5795 mHWData->mSharedFolders.push_back(sharedFolder);
5796
5797 /* inform the direct session if any */
5798 alock.release();
5799 onSharedFolderChange();
5800
5801 return S_OK;
5802}
5803
5804STDMETHODIMP Machine::RemoveSharedFolder(IN_BSTR aName)
5805{
5806 CheckComArgStrNotEmptyOrNull(aName);
5807
5808 AutoCaller autoCaller(this);
5809 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5810
5811 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5812
5813 HRESULT rc = checkStateDependency(MutableStateDep);
5814 if (FAILED(rc)) return rc;
5815
5816 ComObjPtr<SharedFolder> sharedFolder;
5817 rc = findSharedFolder(aName, sharedFolder, true /* aSetError */);
5818 if (FAILED(rc)) return rc;
5819
5820 setModified(IsModified_SharedFolders);
5821 mHWData.backup();
5822 mHWData->mSharedFolders.remove(sharedFolder);
5823
5824 /* inform the direct session if any */
5825 alock.release();
5826 onSharedFolderChange();
5827
5828 return S_OK;
5829}
5830
5831STDMETHODIMP Machine::CanShowConsoleWindow(BOOL *aCanShow)
5832{
5833 CheckComArgOutPointerValid(aCanShow);
5834
5835 /* start with No */
5836 *aCanShow = FALSE;
5837
5838 AutoCaller autoCaller(this);
5839 AssertComRCReturnRC(autoCaller.rc());
5840
5841 ComPtr<IInternalSessionControl> directControl;
5842 {
5843 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5844
5845 if (mData->mSession.mState != SessionState_Locked)
5846 return setError(VBOX_E_INVALID_VM_STATE,
5847 tr("Machine is not locked for session (session state: %s)"),
5848 Global::stringifySessionState(mData->mSession.mState));
5849
5850 directControl = mData->mSession.mDirectControl;
5851 }
5852
5853 /* ignore calls made after #OnSessionEnd() is called */
5854 if (!directControl)
5855 return S_OK;
5856
5857 LONG64 dummy;
5858 return directControl->OnShowWindow(TRUE /* aCheck */, aCanShow, &dummy);
5859}
5860
5861STDMETHODIMP Machine::ShowConsoleWindow(LONG64 *aWinId)
5862{
5863 CheckComArgOutPointerValid(aWinId);
5864
5865 AutoCaller autoCaller(this);
5866 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
5867
5868 ComPtr<IInternalSessionControl> directControl;
5869 {
5870 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5871
5872 if (mData->mSession.mState != SessionState_Locked)
5873 return setError(E_FAIL,
5874 tr("Machine is not locked for session (session state: %s)"),
5875 Global::stringifySessionState(mData->mSession.mState));
5876
5877 directControl = mData->mSession.mDirectControl;
5878 }
5879
5880 /* ignore calls made after #OnSessionEnd() is called */
5881 if (!directControl)
5882 return S_OK;
5883
5884 BOOL dummy;
5885 return directControl->OnShowWindow(FALSE /* aCheck */, &dummy, aWinId);
5886}
5887
5888#ifdef VBOX_WITH_GUEST_PROPS
5889/**
5890 * Look up a guest property in VBoxSVC's internal structures.
5891 */
5892HRESULT Machine::getGuestPropertyFromService(IN_BSTR aName,
5893 BSTR *aValue,
5894 LONG64 *aTimestamp,
5895 BSTR *aFlags) const
5896{
5897 using namespace guestProp;
5898
5899 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
5900 Utf8Str strName(aName);
5901 HWData::GuestPropertyMap::const_iterator it = mHWData->mGuestProperties.find(strName);
5902
5903 if (it != mHWData->mGuestProperties.end())
5904 {
5905 char szFlags[MAX_FLAGS_LEN + 1];
5906 it->second.strValue.cloneTo(aValue);
5907 *aTimestamp = it->second.mTimestamp;
5908 writeFlags(it->second.mFlags, szFlags);
5909 Bstr(szFlags).cloneTo(aFlags);
5910 }
5911
5912 return S_OK;
5913}
5914
5915/**
5916 * Query the VM that a guest property belongs to for the property.
5917 * @returns E_ACCESSDENIED if the VM process is not available or not
5918 * currently handling queries and the lookup should then be done in
5919 * VBoxSVC.
5920 */
5921HRESULT Machine::getGuestPropertyFromVM(IN_BSTR aName,
5922 BSTR *aValue,
5923 LONG64 *aTimestamp,
5924 BSTR *aFlags) const
5925{
5926 HRESULT rc;
5927 ComPtr<IInternalSessionControl> directControl;
5928 directControl = mData->mSession.mDirectControl;
5929
5930 /* fail if we were called after #OnSessionEnd() is called. This is a
5931 * silly race condition. */
5932
5933 /** @todo This code is bothering API clients (like python script clients) with
5934 * the AccessGuestProperty call, creating unncessary IPC. Need to
5935 * have a way of figuring out which kind of direct session it is... */
5936 if (!directControl)
5937 rc = E_ACCESSDENIED;
5938 else
5939 rc = directControl->AccessGuestProperty(aName, NULL, NULL,
5940 false /* isSetter */,
5941 aValue, aTimestamp, aFlags);
5942 return rc;
5943}
5944#endif // VBOX_WITH_GUEST_PROPS
5945
5946STDMETHODIMP Machine::GetGuestProperty(IN_BSTR aName,
5947 BSTR *aValue,
5948 LONG64 *aTimestamp,
5949 BSTR *aFlags)
5950{
5951#ifndef VBOX_WITH_GUEST_PROPS
5952 ReturnComNotImplemented();
5953#else // VBOX_WITH_GUEST_PROPS
5954 CheckComArgStrNotEmptyOrNull(aName);
5955 CheckComArgOutPointerValid(aValue);
5956 CheckComArgOutPointerValid(aTimestamp);
5957 CheckComArgOutPointerValid(aFlags);
5958
5959 AutoCaller autoCaller(this);
5960 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5961
5962 HRESULT rc = getGuestPropertyFromVM(aName, aValue, aTimestamp, aFlags);
5963 if (rc == E_ACCESSDENIED)
5964 /* The VM is not running or the service is not (yet) accessible */
5965 rc = getGuestPropertyFromService(aName, aValue, aTimestamp, aFlags);
5966 return rc;
5967#endif // VBOX_WITH_GUEST_PROPS
5968}
5969
5970STDMETHODIMP Machine::GetGuestPropertyValue(IN_BSTR aName, BSTR *aValue)
5971{
5972 LONG64 dummyTimestamp;
5973 Bstr dummyFlags;
5974 return GetGuestProperty(aName, aValue, &dummyTimestamp, dummyFlags.asOutParam());
5975}
5976
5977STDMETHODIMP Machine::GetGuestPropertyTimestamp(IN_BSTR aName, LONG64 *aTimestamp)
5978{
5979 Bstr dummyValue;
5980 Bstr dummyFlags;
5981 return GetGuestProperty(aName, dummyValue.asOutParam(), aTimestamp, dummyFlags.asOutParam());
5982}
5983
5984#ifdef VBOX_WITH_GUEST_PROPS
5985/**
5986 * Set a guest property in VBoxSVC's internal structures.
5987 */
5988HRESULT Machine::setGuestPropertyToService(IN_BSTR aName, IN_BSTR aValue,
5989 IN_BSTR aFlags)
5990{
5991 using namespace guestProp;
5992
5993 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5994 HRESULT rc = S_OK;
5995
5996 rc = checkStateDependency(MutableStateDep);
5997 if (FAILED(rc)) return rc;
5998
5999 try
6000 {
6001 Utf8Str utf8Name(aName);
6002 Utf8Str utf8Flags(aFlags);
6003 uint32_t fFlags = NILFLAG;
6004 if ( aFlags != NULL
6005 && RT_FAILURE(validateFlags(utf8Flags.c_str(), &fFlags)))
6006 return setError(E_INVALIDARG,
6007 tr("Invalid guest property flag values: '%ls'"),
6008 aFlags);
6009
6010 bool fDelete = !RT_VALID_PTR(aValue) || *(aValue) == '\0';
6011 HWData::GuestPropertyMap::iterator it = mHWData->mGuestProperties.find(utf8Name);
6012 if (it == mHWData->mGuestProperties.end())
6013 {
6014 if (!fDelete)
6015 {
6016 setModified(IsModified_MachineData);
6017 mHWData.backupEx();
6018
6019 RTTIMESPEC time;
6020 HWData::GuestProperty prop;
6021 prop.strValue = aValue;
6022 prop.mTimestamp = RTTimeSpecGetNano(RTTimeNow(&time));
6023 prop.mFlags = fFlags;
6024 mHWData->mGuestProperties[Utf8Str(aName)] = prop;
6025 }
6026 }
6027 else
6028 {
6029 if (it->second.mFlags & (RDONLYHOST))
6030 {
6031 rc = setError(E_ACCESSDENIED,
6032 tr("The property '%ls' cannot be changed by the host"),
6033 aName);
6034 }
6035 else
6036 {
6037 setModified(IsModified_MachineData);
6038 mHWData.backupEx();
6039
6040 /* The backupEx() operation invalidates our iterator,
6041 * so get a new one. */
6042 it = mHWData->mGuestProperties.find(utf8Name);
6043 Assert(it != mHWData->mGuestProperties.end());
6044
6045 if (!fDelete)
6046 {
6047 RTTIMESPEC time;
6048 it->second.strValue = aValue;
6049 it->second.mTimestamp = RTTimeSpecGetNano(RTTimeNow(&time));
6050 it->second.mFlags = fFlags;
6051 }
6052 else
6053 mHWData->mGuestProperties.erase(it);
6054 }
6055 }
6056
6057 if ( SUCCEEDED(rc)
6058 && ( mHWData->mGuestPropertyNotificationPatterns.isEmpty()
6059 || RTStrSimplePatternMultiMatch(mHWData->mGuestPropertyNotificationPatterns.c_str(),
6060 RTSTR_MAX,
6061 utf8Name.c_str(),
6062 RTSTR_MAX,
6063 NULL)
6064 )
6065 )
6066 {
6067 alock.release();
6068
6069 mParent->onGuestPropertyChange(mData->mUuid, aName,
6070 aValue ? aValue : Bstr("").raw(),
6071 aFlags ? aFlags : Bstr("").raw());
6072 }
6073 }
6074 catch (std::bad_alloc &)
6075 {
6076 rc = E_OUTOFMEMORY;
6077 }
6078
6079 return rc;
6080}
6081
6082/**
6083 * Set a property on the VM that that property belongs to.
6084 * @returns E_ACCESSDENIED if the VM process is not available or not
6085 * currently handling queries and the setting should then be done in
6086 * VBoxSVC.
6087 */
6088HRESULT Machine::setGuestPropertyToVM(IN_BSTR aName, IN_BSTR aValue,
6089 IN_BSTR aFlags)
6090{
6091 HRESULT rc;
6092
6093 try
6094 {
6095 ComPtr<IInternalSessionControl> directControl = mData->mSession.mDirectControl;
6096
6097 BSTR dummy = NULL; /* will not be changed (setter) */
6098 LONG64 dummy64;
6099 if (!directControl)
6100 rc = E_ACCESSDENIED;
6101 else
6102 /** @todo Fix when adding DeleteGuestProperty(), see defect. */
6103 rc = directControl->AccessGuestProperty(aName, aValue, aFlags,
6104 true /* isSetter */,
6105 &dummy, &dummy64, &dummy);
6106 }
6107 catch (std::bad_alloc &)
6108 {
6109 rc = E_OUTOFMEMORY;
6110 }
6111
6112 return rc;
6113}
6114#endif // VBOX_WITH_GUEST_PROPS
6115
6116STDMETHODIMP Machine::SetGuestProperty(IN_BSTR aName, IN_BSTR aValue,
6117 IN_BSTR aFlags)
6118{
6119#ifndef VBOX_WITH_GUEST_PROPS
6120 ReturnComNotImplemented();
6121#else // VBOX_WITH_GUEST_PROPS
6122 CheckComArgStrNotEmptyOrNull(aName);
6123 CheckComArgMaybeNull(aFlags);
6124 CheckComArgMaybeNull(aValue);
6125
6126 AutoCaller autoCaller(this);
6127 if (FAILED(autoCaller.rc()))
6128 return autoCaller.rc();
6129
6130 HRESULT rc = setGuestPropertyToVM(aName, aValue, aFlags);
6131 if (rc == E_ACCESSDENIED)
6132 /* The VM is not running or the service is not (yet) accessible */
6133 rc = setGuestPropertyToService(aName, aValue, aFlags);
6134 return rc;
6135#endif // VBOX_WITH_GUEST_PROPS
6136}
6137
6138STDMETHODIMP Machine::SetGuestPropertyValue(IN_BSTR aName, IN_BSTR aValue)
6139{
6140 return SetGuestProperty(aName, aValue, NULL);
6141}
6142
6143STDMETHODIMP Machine::DeleteGuestProperty(IN_BSTR aName)
6144{
6145 return SetGuestProperty(aName, NULL, NULL);
6146}
6147
6148#ifdef VBOX_WITH_GUEST_PROPS
6149/**
6150 * Enumerate the guest properties in VBoxSVC's internal structures.
6151 */
6152HRESULT Machine::enumerateGuestPropertiesInService
6153 (IN_BSTR aPatterns, ComSafeArrayOut(BSTR, aNames),
6154 ComSafeArrayOut(BSTR, aValues),
6155 ComSafeArrayOut(LONG64, aTimestamps),
6156 ComSafeArrayOut(BSTR, aFlags))
6157{
6158 using namespace guestProp;
6159
6160 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6161 Utf8Str strPatterns(aPatterns);
6162
6163 HWData::GuestPropertyMap propMap;
6164
6165 /*
6166 * Look for matching patterns and build up a list.
6167 */
6168 HWData::GuestPropertyMap::const_iterator it = mHWData->mGuestProperties.begin();
6169 while (it != mHWData->mGuestProperties.end())
6170 {
6171 if ( strPatterns.isEmpty()
6172 || RTStrSimplePatternMultiMatch(strPatterns.c_str(),
6173 RTSTR_MAX,
6174 it->first.c_str(),
6175 RTSTR_MAX,
6176 NULL)
6177 )
6178 {
6179 propMap.insert(*it);
6180 }
6181
6182 it++;
6183 }
6184
6185 alock.release();
6186
6187 /*
6188 * And build up the arrays for returning the property information.
6189 */
6190 size_t cEntries = propMap.size();
6191 SafeArray<BSTR> names(cEntries);
6192 SafeArray<BSTR> values(cEntries);
6193 SafeArray<LONG64> timestamps(cEntries);
6194 SafeArray<BSTR> flags(cEntries);
6195 size_t iProp = 0;
6196
6197 it = propMap.begin();
6198 while (it != propMap.end())
6199 {
6200 char szFlags[MAX_FLAGS_LEN + 1];
6201 it->first.cloneTo(&names[iProp]);
6202 it->second.strValue.cloneTo(&values[iProp]);
6203 timestamps[iProp] = it->second.mTimestamp;
6204 writeFlags(it->second.mFlags, szFlags);
6205 Bstr(szFlags).cloneTo(&flags[iProp++]);
6206 it++;
6207 }
6208 names.detachTo(ComSafeArrayOutArg(aNames));
6209 values.detachTo(ComSafeArrayOutArg(aValues));
6210 timestamps.detachTo(ComSafeArrayOutArg(aTimestamps));
6211 flags.detachTo(ComSafeArrayOutArg(aFlags));
6212 return S_OK;
6213}
6214
6215/**
6216 * Enumerate the properties managed by a VM.
6217 * @returns E_ACCESSDENIED if the VM process is not available or not
6218 * currently handling queries and the setting should then be done in
6219 * VBoxSVC.
6220 */
6221HRESULT Machine::enumerateGuestPropertiesOnVM
6222 (IN_BSTR aPatterns, ComSafeArrayOut(BSTR, aNames),
6223 ComSafeArrayOut(BSTR, aValues),
6224 ComSafeArrayOut(LONG64, aTimestamps),
6225 ComSafeArrayOut(BSTR, aFlags))
6226{
6227 HRESULT rc;
6228 ComPtr<IInternalSessionControl> directControl;
6229 directControl = mData->mSession.mDirectControl;
6230
6231 if (!directControl)
6232 rc = E_ACCESSDENIED;
6233 else
6234 rc = directControl->EnumerateGuestProperties
6235 (aPatterns, ComSafeArrayOutArg(aNames),
6236 ComSafeArrayOutArg(aValues),
6237 ComSafeArrayOutArg(aTimestamps),
6238 ComSafeArrayOutArg(aFlags));
6239 return rc;
6240}
6241#endif // VBOX_WITH_GUEST_PROPS
6242
6243STDMETHODIMP Machine::EnumerateGuestProperties(IN_BSTR aPatterns,
6244 ComSafeArrayOut(BSTR, aNames),
6245 ComSafeArrayOut(BSTR, aValues),
6246 ComSafeArrayOut(LONG64, aTimestamps),
6247 ComSafeArrayOut(BSTR, aFlags))
6248{
6249#ifndef VBOX_WITH_GUEST_PROPS
6250 ReturnComNotImplemented();
6251#else // VBOX_WITH_GUEST_PROPS
6252 CheckComArgMaybeNull(aPatterns);
6253 CheckComArgOutSafeArrayPointerValid(aNames);
6254 CheckComArgOutSafeArrayPointerValid(aValues);
6255 CheckComArgOutSafeArrayPointerValid(aTimestamps);
6256 CheckComArgOutSafeArrayPointerValid(aFlags);
6257
6258 AutoCaller autoCaller(this);
6259 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6260
6261 HRESULT rc = enumerateGuestPropertiesOnVM
6262 (aPatterns, ComSafeArrayOutArg(aNames),
6263 ComSafeArrayOutArg(aValues),
6264 ComSafeArrayOutArg(aTimestamps),
6265 ComSafeArrayOutArg(aFlags));
6266 if (rc == E_ACCESSDENIED)
6267 /* The VM is not running or the service is not (yet) accessible */
6268 rc = enumerateGuestPropertiesInService
6269 (aPatterns, ComSafeArrayOutArg(aNames),
6270 ComSafeArrayOutArg(aValues),
6271 ComSafeArrayOutArg(aTimestamps),
6272 ComSafeArrayOutArg(aFlags));
6273 return rc;
6274#endif // VBOX_WITH_GUEST_PROPS
6275}
6276
6277STDMETHODIMP Machine::GetMediumAttachmentsOfController(IN_BSTR aName,
6278 ComSafeArrayOut(IMediumAttachment*, aAttachments))
6279{
6280 MediaData::AttachmentList atts;
6281
6282 HRESULT rc = getMediumAttachmentsOfController(aName, atts);
6283 if (FAILED(rc)) return rc;
6284
6285 SafeIfaceArray<IMediumAttachment> attachments(atts);
6286 attachments.detachTo(ComSafeArrayOutArg(aAttachments));
6287
6288 return S_OK;
6289}
6290
6291STDMETHODIMP Machine::GetMediumAttachment(IN_BSTR aControllerName,
6292 LONG aControllerPort,
6293 LONG aDevice,
6294 IMediumAttachment **aAttachment)
6295{
6296 LogFlowThisFunc(("aControllerName=\"%ls\" aControllerPort=%d aDevice=%d\n",
6297 aControllerName, aControllerPort, aDevice));
6298
6299 CheckComArgStrNotEmptyOrNull(aControllerName);
6300 CheckComArgOutPointerValid(aAttachment);
6301
6302 AutoCaller autoCaller(this);
6303 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6304
6305 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6306
6307 *aAttachment = NULL;
6308
6309 ComObjPtr<MediumAttachment> pAttach = findAttachment(mMediaData->mAttachments,
6310 aControllerName,
6311 aControllerPort,
6312 aDevice);
6313 if (pAttach.isNull())
6314 return setError(VBOX_E_OBJECT_NOT_FOUND,
6315 tr("No storage device attached to device slot %d on port %d of controller '%ls'"),
6316 aDevice, aControllerPort, aControllerName);
6317
6318 pAttach.queryInterfaceTo(aAttachment);
6319
6320 return S_OK;
6321}
6322
6323STDMETHODIMP Machine::AddStorageController(IN_BSTR aName,
6324 StorageBus_T aConnectionType,
6325 IStorageController **controller)
6326{
6327 CheckComArgStrNotEmptyOrNull(aName);
6328
6329 if ( (aConnectionType <= StorageBus_Null)
6330 || (aConnectionType > StorageBus_SAS))
6331 return setError(E_INVALIDARG,
6332 tr("Invalid connection type: %d"),
6333 aConnectionType);
6334
6335 AutoCaller autoCaller(this);
6336 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6337
6338 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6339
6340 HRESULT rc = checkStateDependency(MutableStateDep);
6341 if (FAILED(rc)) return rc;
6342
6343 /* try to find one with the name first. */
6344 ComObjPtr<StorageController> ctrl;
6345
6346 rc = getStorageControllerByName(aName, ctrl, false /* aSetError */);
6347 if (SUCCEEDED(rc))
6348 return setError(VBOX_E_OBJECT_IN_USE,
6349 tr("Storage controller named '%ls' already exists"),
6350 aName);
6351
6352 ctrl.createObject();
6353
6354 /* get a new instance number for the storage controller */
6355 ULONG ulInstance = 0;
6356 bool fBootable = true;
6357 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
6358 it != mStorageControllers->end();
6359 ++it)
6360 {
6361 if ((*it)->getStorageBus() == aConnectionType)
6362 {
6363 ULONG ulCurInst = (*it)->getInstance();
6364
6365 if (ulCurInst >= ulInstance)
6366 ulInstance = ulCurInst + 1;
6367
6368 /* Only one controller of each type can be marked as bootable. */
6369 if ((*it)->getBootable())
6370 fBootable = false;
6371 }
6372 }
6373
6374 rc = ctrl->init(this, aName, aConnectionType, ulInstance, fBootable);
6375 if (FAILED(rc)) return rc;
6376
6377 setModified(IsModified_Storage);
6378 mStorageControllers.backup();
6379 mStorageControllers->push_back(ctrl);
6380
6381 ctrl.queryInterfaceTo(controller);
6382
6383 /* inform the direct session if any */
6384 alock.release();
6385 onStorageControllerChange();
6386
6387 return S_OK;
6388}
6389
6390STDMETHODIMP Machine::GetStorageControllerByName(IN_BSTR aName,
6391 IStorageController **aStorageController)
6392{
6393 CheckComArgStrNotEmptyOrNull(aName);
6394
6395 AutoCaller autoCaller(this);
6396 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6397
6398 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6399
6400 ComObjPtr<StorageController> ctrl;
6401
6402 HRESULT rc = getStorageControllerByName(aName, ctrl, true /* aSetError */);
6403 if (SUCCEEDED(rc))
6404 ctrl.queryInterfaceTo(aStorageController);
6405
6406 return rc;
6407}
6408
6409STDMETHODIMP Machine::GetStorageControllerByInstance(ULONG aInstance,
6410 IStorageController **aStorageController)
6411{
6412 AutoCaller autoCaller(this);
6413 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6414
6415 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6416
6417 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
6418 it != mStorageControllers->end();
6419 ++it)
6420 {
6421 if ((*it)->getInstance() == aInstance)
6422 {
6423 (*it).queryInterfaceTo(aStorageController);
6424 return S_OK;
6425 }
6426 }
6427
6428 return setError(VBOX_E_OBJECT_NOT_FOUND,
6429 tr("Could not find a storage controller with instance number '%lu'"),
6430 aInstance);
6431}
6432
6433STDMETHODIMP Machine::SetStorageControllerBootable(IN_BSTR aName, BOOL fBootable)
6434{
6435 AutoCaller autoCaller(this);
6436 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6437
6438 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6439
6440 HRESULT rc = checkStateDependency(MutableStateDep);
6441 if (FAILED(rc)) return rc;
6442
6443 ComObjPtr<StorageController> ctrl;
6444
6445 rc = getStorageControllerByName(aName, ctrl, true /* aSetError */);
6446 if (SUCCEEDED(rc))
6447 {
6448 /* Ensure that only one controller of each type is marked as bootable. */
6449 if (fBootable == TRUE)
6450 {
6451 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
6452 it != mStorageControllers->end();
6453 ++it)
6454 {
6455 ComObjPtr<StorageController> aCtrl = (*it);
6456
6457 if ( (aCtrl->getName() != Utf8Str(aName))
6458 && aCtrl->getBootable() == TRUE
6459 && aCtrl->getStorageBus() == ctrl->getStorageBus()
6460 && aCtrl->getControllerType() == ctrl->getControllerType())
6461 {
6462 aCtrl->setBootable(FALSE);
6463 break;
6464 }
6465 }
6466 }
6467
6468 if (SUCCEEDED(rc))
6469 {
6470 ctrl->setBootable(fBootable);
6471 setModified(IsModified_Storage);
6472 }
6473 }
6474
6475 if (SUCCEEDED(rc))
6476 {
6477 /* inform the direct session if any */
6478 alock.release();
6479 onStorageControllerChange();
6480 }
6481
6482 return rc;
6483}
6484
6485STDMETHODIMP Machine::RemoveStorageController(IN_BSTR aName)
6486{
6487 CheckComArgStrNotEmptyOrNull(aName);
6488
6489 AutoCaller autoCaller(this);
6490 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6491
6492 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6493
6494 HRESULT rc = checkStateDependency(MutableStateDep);
6495 if (FAILED(rc)) return rc;
6496
6497 ComObjPtr<StorageController> ctrl;
6498 rc = getStorageControllerByName(aName, ctrl, true /* aSetError */);
6499 if (FAILED(rc)) return rc;
6500
6501 {
6502 /* find all attached devices to the appropriate storage controller and detach them all */
6503 // make a temporary list because detachDevice invalidates iterators into
6504 // mMediaData->mAttachments
6505 MediaData::AttachmentList llAttachments2 = mMediaData->mAttachments;
6506
6507 for (MediaData::AttachmentList::iterator it = llAttachments2.begin();
6508 it != llAttachments2.end();
6509 ++it)
6510 {
6511 MediumAttachment *pAttachTemp = *it;
6512
6513 AutoCaller localAutoCaller(pAttachTemp);
6514 if (FAILED(localAutoCaller.rc())) return localAutoCaller.rc();
6515
6516 AutoReadLock local_alock(pAttachTemp COMMA_LOCKVAL_SRC_POS);
6517
6518 if (pAttachTemp->getControllerName() == aName)
6519 {
6520 rc = detachDevice(pAttachTemp, alock, NULL);
6521 if (FAILED(rc)) return rc;
6522 }
6523 }
6524 }
6525
6526 /* We can remove it now. */
6527 setModified(IsModified_Storage);
6528 mStorageControllers.backup();
6529
6530 ctrl->unshare();
6531
6532 mStorageControllers->remove(ctrl);
6533
6534 /* inform the direct session if any */
6535 alock.release();
6536 onStorageControllerChange();
6537
6538 return S_OK;
6539}
6540
6541STDMETHODIMP Machine::AddUSBController(IN_BSTR aName, USBControllerType_T aType,
6542 IUSBController **controller)
6543{
6544 if ( (aType <= USBControllerType_Null)
6545 || (aType >= USBControllerType_Last))
6546 return setError(E_INVALIDARG,
6547 tr("Invalid USB controller type: %d"),
6548 aType);
6549
6550 AutoCaller autoCaller(this);
6551 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6552
6553 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6554
6555 HRESULT rc = checkStateDependency(MutableStateDep);
6556 if (FAILED(rc)) return rc;
6557
6558 /* try to find one with the same type first. */
6559 ComObjPtr<USBController> ctrl;
6560
6561 rc = getUSBControllerByName(aName, ctrl, false /* aSetError */);
6562 if (SUCCEEDED(rc))
6563 return setError(VBOX_E_OBJECT_IN_USE,
6564 tr("USB controller named '%ls' already exists"),
6565 aName);
6566
6567 /* Check that we don't exceed the maximum number of USB controllers for the given type. */
6568 ULONG maxInstances;
6569 rc = mParent->getSystemProperties()->GetMaxInstancesOfUSBControllerType(mHWData->mChipsetType, aType, &maxInstances);
6570 if (FAILED(rc))
6571 return rc;
6572
6573 ULONG cInstances = getUSBControllerCountByType(aType);
6574 if (cInstances >= maxInstances)
6575 return setError(E_INVALIDARG,
6576 tr("Too many USB controllers of this type"));
6577
6578 ctrl.createObject();
6579
6580 rc = ctrl->init(this, aName, aType);
6581 if (FAILED(rc)) return rc;
6582
6583 setModified(IsModified_USB);
6584 mUSBControllers.backup();
6585 mUSBControllers->push_back(ctrl);
6586
6587 ctrl.queryInterfaceTo(controller);
6588
6589 /* inform the direct session if any */
6590 alock.release();
6591 onUSBControllerChange();
6592
6593 return S_OK;
6594}
6595
6596STDMETHODIMP Machine::GetUSBControllerByName(IN_BSTR aName, IUSBController **aUSBController)
6597{
6598 CheckComArgStrNotEmptyOrNull(aName);
6599
6600 AutoCaller autoCaller(this);
6601 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6602
6603 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6604
6605 ComObjPtr<USBController> ctrl;
6606
6607 HRESULT rc = getUSBControllerByName(aName, ctrl, true /* aSetError */);
6608 if (SUCCEEDED(rc))
6609 ctrl.queryInterfaceTo(aUSBController);
6610
6611 return rc;
6612}
6613
6614STDMETHODIMP Machine::GetUSBControllerCountByType(USBControllerType_T aType,
6615 ULONG *aControllers)
6616{
6617 CheckComArgOutPointerValid(aControllers);
6618
6619 if ( (aType <= USBControllerType_Null)
6620 || (aType >= USBControllerType_Last))
6621 return setError(E_INVALIDARG,
6622 tr("Invalid USB controller type: %d"),
6623 aType);
6624
6625 AutoCaller autoCaller(this);
6626 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6627
6628 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6629
6630 ComObjPtr<USBController> ctrl;
6631
6632 *aControllers = getUSBControllerCountByType(aType);
6633
6634 return S_OK;
6635}
6636
6637STDMETHODIMP Machine::RemoveUSBController(IN_BSTR aName)
6638{
6639 CheckComArgStrNotEmptyOrNull(aName);
6640
6641 AutoCaller autoCaller(this);
6642 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6643
6644 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6645
6646 HRESULT rc = checkStateDependency(MutableStateDep);
6647 if (FAILED(rc)) return rc;
6648
6649 ComObjPtr<USBController> ctrl;
6650 rc = getUSBControllerByName(aName, ctrl, true /* aSetError */);
6651 if (FAILED(rc)) return rc;
6652
6653 setModified(IsModified_USB);
6654 mUSBControllers.backup();
6655
6656 ctrl->unshare();
6657
6658 mUSBControllers->remove(ctrl);
6659
6660 /* inform the direct session if any */
6661 alock.release();
6662 onUSBControllerChange();
6663
6664 return S_OK;
6665}
6666
6667STDMETHODIMP Machine::QuerySavedGuestScreenInfo(ULONG uScreenId,
6668 ULONG *puOriginX,
6669 ULONG *puOriginY,
6670 ULONG *puWidth,
6671 ULONG *puHeight,
6672 BOOL *pfEnabled)
6673{
6674 LogFlowThisFunc(("\n"));
6675
6676 CheckComArgNotNull(puOriginX);
6677 CheckComArgNotNull(puOriginY);
6678 CheckComArgNotNull(puWidth);
6679 CheckComArgNotNull(puHeight);
6680 CheckComArgNotNull(pfEnabled);
6681
6682 uint32_t u32OriginX= 0;
6683 uint32_t u32OriginY= 0;
6684 uint32_t u32Width = 0;
6685 uint32_t u32Height = 0;
6686 uint16_t u16Flags = 0;
6687
6688 int vrc = readSavedGuestScreenInfo(mSSData->strStateFilePath, uScreenId,
6689 &u32OriginX, &u32OriginY, &u32Width, &u32Height, &u16Flags);
6690 if (RT_FAILURE(vrc))
6691 {
6692#ifdef RT_OS_WINDOWS
6693 /* HACK: GUI sets *pfEnabled to 'true' and expects it to stay so if the API fails.
6694 * This works with XPCOM. But Windows COM sets all output parameters to zero.
6695 * So just assign fEnable to TRUE again.
6696 * The right fix would be to change GUI API wrappers to make sure that parameters
6697 * are changed only if API succeeds.
6698 */
6699 *pfEnabled = TRUE;
6700#endif
6701 return setError(VBOX_E_IPRT_ERROR,
6702 tr("Saved guest size is not available (%Rrc)"),
6703 vrc);
6704 }
6705
6706 *puOriginX = u32OriginX;
6707 *puOriginY = u32OriginY;
6708 *puWidth = u32Width;
6709 *puHeight = u32Height;
6710 *pfEnabled = (u16Flags & VBVA_SCREEN_F_DISABLED) == 0;
6711
6712 return S_OK;
6713}
6714
6715STDMETHODIMP Machine::QuerySavedThumbnailSize(ULONG aScreenId, ULONG *aSize, ULONG *aWidth, ULONG *aHeight)
6716{
6717 LogFlowThisFunc(("\n"));
6718
6719 CheckComArgNotNull(aSize);
6720 CheckComArgNotNull(aWidth);
6721 CheckComArgNotNull(aHeight);
6722
6723 if (aScreenId != 0)
6724 return E_NOTIMPL;
6725
6726 AutoCaller autoCaller(this);
6727 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6728
6729 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6730
6731 uint8_t *pu8Data = NULL;
6732 uint32_t cbData = 0;
6733 uint32_t u32Width = 0;
6734 uint32_t u32Height = 0;
6735
6736 int vrc = readSavedDisplayScreenshot(mSSData->strStateFilePath, 0 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
6737
6738 if (RT_FAILURE(vrc))
6739 return setError(VBOX_E_IPRT_ERROR,
6740 tr("Saved screenshot data is not available (%Rrc)"),
6741 vrc);
6742
6743 *aSize = cbData;
6744 *aWidth = u32Width;
6745 *aHeight = u32Height;
6746
6747 freeSavedDisplayScreenshot(pu8Data);
6748
6749 return S_OK;
6750}
6751
6752STDMETHODIMP Machine::ReadSavedThumbnailToArray(ULONG aScreenId, BOOL aBGR, ULONG *aWidth, ULONG *aHeight, ComSafeArrayOut(BYTE, aData))
6753{
6754 LogFlowThisFunc(("\n"));
6755
6756 CheckComArgNotNull(aWidth);
6757 CheckComArgNotNull(aHeight);
6758 CheckComArgOutSafeArrayPointerValid(aData);
6759
6760 if (aScreenId != 0)
6761 return E_NOTIMPL;
6762
6763 AutoCaller autoCaller(this);
6764 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6765
6766 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6767
6768 uint8_t *pu8Data = NULL;
6769 uint32_t cbData = 0;
6770 uint32_t u32Width = 0;
6771 uint32_t u32Height = 0;
6772
6773 int vrc = readSavedDisplayScreenshot(mSSData->strStateFilePath, 0 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
6774
6775 if (RT_FAILURE(vrc))
6776 return setError(VBOX_E_IPRT_ERROR,
6777 tr("Saved screenshot data is not available (%Rrc)"),
6778 vrc);
6779
6780 *aWidth = u32Width;
6781 *aHeight = u32Height;
6782
6783 com::SafeArray<BYTE> bitmap(cbData);
6784 /* Convert pixels to format expected by the API caller. */
6785 if (aBGR)
6786 {
6787 /* [0] B, [1] G, [2] R, [3] A. */
6788 for (unsigned i = 0; i < cbData; i += 4)
6789 {
6790 bitmap[i] = pu8Data[i];
6791 bitmap[i + 1] = pu8Data[i + 1];
6792 bitmap[i + 2] = pu8Data[i + 2];
6793 bitmap[i + 3] = 0xff;
6794 }
6795 }
6796 else
6797 {
6798 /* [0] R, [1] G, [2] B, [3] A. */
6799 for (unsigned i = 0; i < cbData; i += 4)
6800 {
6801 bitmap[i] = pu8Data[i + 2];
6802 bitmap[i + 1] = pu8Data[i + 1];
6803 bitmap[i + 2] = pu8Data[i];
6804 bitmap[i + 3] = 0xff;
6805 }
6806 }
6807 bitmap.detachTo(ComSafeArrayOutArg(aData));
6808
6809 freeSavedDisplayScreenshot(pu8Data);
6810
6811 return S_OK;
6812}
6813
6814
6815STDMETHODIMP Machine::ReadSavedThumbnailPNGToArray(ULONG aScreenId, ULONG *aWidth, ULONG *aHeight, ComSafeArrayOut(BYTE, aData))
6816{
6817 LogFlowThisFunc(("\n"));
6818
6819 CheckComArgNotNull(aWidth);
6820 CheckComArgNotNull(aHeight);
6821 CheckComArgOutSafeArrayPointerValid(aData);
6822
6823 if (aScreenId != 0)
6824 return E_NOTIMPL;
6825
6826 AutoCaller autoCaller(this);
6827 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6828
6829 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6830
6831 uint8_t *pu8Data = NULL;
6832 uint32_t cbData = 0;
6833 uint32_t u32Width = 0;
6834 uint32_t u32Height = 0;
6835
6836 int vrc = readSavedDisplayScreenshot(mSSData->strStateFilePath, 0 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
6837
6838 if (RT_FAILURE(vrc))
6839 return setError(VBOX_E_IPRT_ERROR,
6840 tr("Saved screenshot data is not available (%Rrc)"),
6841 vrc);
6842
6843 *aWidth = u32Width;
6844 *aHeight = u32Height;
6845
6846 HRESULT rc = S_OK;
6847 uint8_t *pu8PNG = NULL;
6848 uint32_t cbPNG = 0;
6849 uint32_t cxPNG = 0;
6850 uint32_t cyPNG = 0;
6851
6852 vrc = DisplayMakePNG(pu8Data, u32Width, u32Height, &pu8PNG, &cbPNG, &cxPNG, &cyPNG, 0);
6853
6854 if (RT_SUCCESS(vrc))
6855 {
6856 com::SafeArray<BYTE> screenData(cbPNG);
6857 screenData.initFrom(pu8PNG, cbPNG);
6858 if (pu8PNG)
6859 RTMemFree(pu8PNG);
6860 screenData.detachTo(ComSafeArrayOutArg(aData));
6861 }
6862 else
6863 {
6864 if (pu8PNG)
6865 RTMemFree(pu8PNG);
6866 return setError(VBOX_E_IPRT_ERROR,
6867 tr("Could not convert screenshot to PNG (%Rrc)"),
6868 vrc);
6869 }
6870
6871 freeSavedDisplayScreenshot(pu8Data);
6872
6873 return rc;
6874}
6875
6876STDMETHODIMP Machine::QuerySavedScreenshotPNGSize(ULONG aScreenId, ULONG *aSize, ULONG *aWidth, ULONG *aHeight)
6877{
6878 LogFlowThisFunc(("\n"));
6879
6880 CheckComArgNotNull(aSize);
6881 CheckComArgNotNull(aWidth);
6882 CheckComArgNotNull(aHeight);
6883
6884 if (aScreenId != 0)
6885 return E_NOTIMPL;
6886
6887 AutoCaller autoCaller(this);
6888 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6889
6890 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6891
6892 uint8_t *pu8Data = NULL;
6893 uint32_t cbData = 0;
6894 uint32_t u32Width = 0;
6895 uint32_t u32Height = 0;
6896
6897 int vrc = readSavedDisplayScreenshot(mSSData->strStateFilePath, 1 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
6898
6899 if (RT_FAILURE(vrc))
6900 return setError(VBOX_E_IPRT_ERROR,
6901 tr("Saved screenshot data is not available (%Rrc)"),
6902 vrc);
6903
6904 *aSize = cbData;
6905 *aWidth = u32Width;
6906 *aHeight = u32Height;
6907
6908 freeSavedDisplayScreenshot(pu8Data);
6909
6910 return S_OK;
6911}
6912
6913STDMETHODIMP Machine::ReadSavedScreenshotPNGToArray(ULONG aScreenId, ULONG *aWidth, ULONG *aHeight, ComSafeArrayOut(BYTE, aData))
6914{
6915 LogFlowThisFunc(("\n"));
6916
6917 CheckComArgNotNull(aWidth);
6918 CheckComArgNotNull(aHeight);
6919 CheckComArgOutSafeArrayPointerValid(aData);
6920
6921 if (aScreenId != 0)
6922 return E_NOTIMPL;
6923
6924 AutoCaller autoCaller(this);
6925 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6926
6927 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
6928
6929 uint8_t *pu8Data = NULL;
6930 uint32_t cbData = 0;
6931 uint32_t u32Width = 0;
6932 uint32_t u32Height = 0;
6933
6934 int vrc = readSavedDisplayScreenshot(mSSData->strStateFilePath, 1 /* u32Type */, &pu8Data, &cbData, &u32Width, &u32Height);
6935
6936 if (RT_FAILURE(vrc))
6937 return setError(VBOX_E_IPRT_ERROR,
6938 tr("Saved screenshot thumbnail data is not available (%Rrc)"),
6939 vrc);
6940
6941 *aWidth = u32Width;
6942 *aHeight = u32Height;
6943
6944 com::SafeArray<BYTE> png(cbData);
6945 png.initFrom(pu8Data, cbData);
6946 png.detachTo(ComSafeArrayOutArg(aData));
6947
6948 freeSavedDisplayScreenshot(pu8Data);
6949
6950 return S_OK;
6951}
6952
6953STDMETHODIMP Machine::HotPlugCPU(ULONG aCpu)
6954{
6955 HRESULT rc = S_OK;
6956 LogFlowThisFunc(("\n"));
6957
6958 AutoCaller autoCaller(this);
6959 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6960
6961 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6962
6963 if (!mHWData->mCPUHotPlugEnabled)
6964 return setError(E_INVALIDARG, tr("CPU hotplug is not enabled"));
6965
6966 if (aCpu >= mHWData->mCPUCount)
6967 return setError(E_INVALIDARG, tr("CPU id exceeds number of possible CPUs [0:%lu]"), mHWData->mCPUCount-1);
6968
6969 if (mHWData->mCPUAttached[aCpu])
6970 return setError(VBOX_E_OBJECT_IN_USE, tr("CPU %lu is already attached"), aCpu);
6971
6972 alock.release();
6973 rc = onCPUChange(aCpu, false);
6974 alock.acquire();
6975 if (FAILED(rc)) return rc;
6976
6977 setModified(IsModified_MachineData);
6978 mHWData.backup();
6979 mHWData->mCPUAttached[aCpu] = true;
6980
6981 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
6982 if (Global::IsOnline(mData->mMachineState))
6983 saveSettings(NULL);
6984
6985 return S_OK;
6986}
6987
6988STDMETHODIMP Machine::HotUnplugCPU(ULONG aCpu)
6989{
6990 HRESULT rc = S_OK;
6991 LogFlowThisFunc(("\n"));
6992
6993 AutoCaller autoCaller(this);
6994 if (FAILED(autoCaller.rc())) return autoCaller.rc();
6995
6996 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6997
6998 if (!mHWData->mCPUHotPlugEnabled)
6999 return setError(E_INVALIDARG, tr("CPU hotplug is not enabled"));
7000
7001 if (aCpu >= SchemaDefs::MaxCPUCount)
7002 return setError(E_INVALIDARG,
7003 tr("CPU index exceeds maximum CPU count (must be in range [0:%lu])"),
7004 SchemaDefs::MaxCPUCount);
7005
7006 if (!mHWData->mCPUAttached[aCpu])
7007 return setError(VBOX_E_OBJECT_NOT_FOUND, tr("CPU %lu is not attached"), aCpu);
7008
7009 /* CPU 0 can't be detached */
7010 if (aCpu == 0)
7011 return setError(E_INVALIDARG, tr("It is not possible to detach CPU 0"));
7012
7013 alock.release();
7014 rc = onCPUChange(aCpu, true);
7015 alock.acquire();
7016 if (FAILED(rc)) return rc;
7017
7018 setModified(IsModified_MachineData);
7019 mHWData.backup();
7020 mHWData->mCPUAttached[aCpu] = false;
7021
7022 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
7023 if (Global::IsOnline(mData->mMachineState))
7024 saveSettings(NULL);
7025
7026 return S_OK;
7027}
7028
7029STDMETHODIMP Machine::GetCPUStatus(ULONG aCpu, BOOL *aCpuAttached)
7030{
7031 LogFlowThisFunc(("\n"));
7032
7033 CheckComArgNotNull(aCpuAttached);
7034
7035 *aCpuAttached = false;
7036
7037 AutoCaller autoCaller(this);
7038 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7039
7040 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7041
7042 /* If hotplug is enabled the CPU is always enabled. */
7043 if (!mHWData->mCPUHotPlugEnabled)
7044 {
7045 if (aCpu < mHWData->mCPUCount)
7046 *aCpuAttached = true;
7047 }
7048 else
7049 {
7050 if (aCpu < SchemaDefs::MaxCPUCount)
7051 *aCpuAttached = mHWData->mCPUAttached[aCpu];
7052 }
7053
7054 return S_OK;
7055}
7056
7057STDMETHODIMP Machine::QueryLogFilename(ULONG aIdx, BSTR *aName)
7058{
7059 CheckComArgOutPointerValid(aName);
7060
7061 AutoCaller autoCaller(this);
7062 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7063
7064 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7065
7066 Utf8Str log = queryLogFilename(aIdx);
7067 if (!RTFileExists(log.c_str()))
7068 log.setNull();
7069 log.cloneTo(aName);
7070
7071 return S_OK;
7072}
7073
7074STDMETHODIMP Machine::ReadLog(ULONG aIdx, LONG64 aOffset, LONG64 aSize, ComSafeArrayOut(BYTE, aData))
7075{
7076 LogFlowThisFunc(("\n"));
7077 CheckComArgOutSafeArrayPointerValid(aData);
7078 if (aSize < 0)
7079 return setError(E_INVALIDARG, tr("The size argument (%lld) is negative"), aSize);
7080
7081 AutoCaller autoCaller(this);
7082 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7083
7084 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7085
7086 HRESULT rc = S_OK;
7087 Utf8Str log = queryLogFilename(aIdx);
7088
7089 /* do not unnecessarily hold the lock while doing something which does
7090 * not need the lock and potentially takes a long time. */
7091 alock.release();
7092
7093 /* Limit the chunk size to 32K for now, as that gives better performance
7094 * over (XP)COM, and keeps the SOAP reply size under 1M for the webservice.
7095 * One byte expands to approx. 25 bytes of breathtaking XML. */
7096 size_t cbData = (size_t)RT_MIN(aSize, 32768);
7097 com::SafeArray<BYTE> logData(cbData);
7098
7099 RTFILE LogFile;
7100 int vrc = RTFileOpen(&LogFile, log.c_str(),
7101 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE);
7102 if (RT_SUCCESS(vrc))
7103 {
7104 vrc = RTFileReadAt(LogFile, aOffset, logData.raw(), cbData, &cbData);
7105 if (RT_SUCCESS(vrc))
7106 logData.resize(cbData);
7107 else
7108 rc = setError(VBOX_E_IPRT_ERROR,
7109 tr("Could not read log file '%s' (%Rrc)"),
7110 log.c_str(), vrc);
7111 RTFileClose(LogFile);
7112 }
7113 else
7114 rc = setError(VBOX_E_IPRT_ERROR,
7115 tr("Could not open log file '%s' (%Rrc)"),
7116 log.c_str(), vrc);
7117
7118 if (FAILED(rc))
7119 logData.resize(0);
7120 logData.detachTo(ComSafeArrayOutArg(aData));
7121
7122 return rc;
7123}
7124
7125
7126/**
7127 * Currently this method doesn't attach device to the running VM,
7128 * just makes sure it's plugged on next VM start.
7129 */
7130STDMETHODIMP Machine::AttachHostPCIDevice(LONG hostAddress, LONG desiredGuestAddress, BOOL /*tryToUnbind*/)
7131{
7132 AutoCaller autoCaller(this);
7133 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7134
7135 // lock scope
7136 {
7137 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7138
7139 HRESULT rc = checkStateDependency(MutableStateDep);
7140 if (FAILED(rc)) return rc;
7141
7142 ChipsetType_T aChipset = ChipsetType_PIIX3;
7143 COMGETTER(ChipsetType)(&aChipset);
7144
7145 if (aChipset != ChipsetType_ICH9)
7146 {
7147 return setError(E_INVALIDARG,
7148 tr("Host PCI attachment only supported with ICH9 chipset"));
7149 }
7150
7151 // check if device with this host PCI address already attached
7152 for (HWData::PCIDeviceAssignmentList::iterator it = mHWData->mPCIDeviceAssignments.begin();
7153 it != mHWData->mPCIDeviceAssignments.end();
7154 ++it)
7155 {
7156 LONG iHostAddress = -1;
7157 ComPtr<PCIDeviceAttachment> pAttach;
7158 pAttach = *it;
7159 pAttach->COMGETTER(HostAddress)(&iHostAddress);
7160 if (iHostAddress == hostAddress)
7161 return setError(E_INVALIDARG,
7162 tr("Device with host PCI address already attached to this VM"));
7163 }
7164
7165 ComObjPtr<PCIDeviceAttachment> pda;
7166 char name[32];
7167
7168 RTStrPrintf(name, sizeof(name), "host%02x:%02x.%x", (hostAddress>>8) & 0xff, (hostAddress & 0xf8) >> 3, hostAddress & 7);
7169 Bstr bname(name);
7170 pda.createObject();
7171 pda->init(this, bname, hostAddress, desiredGuestAddress, TRUE);
7172 setModified(IsModified_MachineData);
7173 mHWData.backup();
7174 mHWData->mPCIDeviceAssignments.push_back(pda);
7175 }
7176
7177 return S_OK;
7178}
7179
7180/**
7181 * Currently this method doesn't detach device from the running VM,
7182 * just makes sure it's not plugged on next VM start.
7183 */
7184STDMETHODIMP Machine::DetachHostPCIDevice(LONG hostAddress)
7185{
7186 AutoCaller autoCaller(this);
7187 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7188
7189 ComObjPtr<PCIDeviceAttachment> pAttach;
7190 bool fRemoved = false;
7191 HRESULT rc;
7192
7193 // lock scope
7194 {
7195 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7196
7197 rc = checkStateDependency(MutableStateDep);
7198 if (FAILED(rc)) return rc;
7199
7200 for (HWData::PCIDeviceAssignmentList::iterator it = mHWData->mPCIDeviceAssignments.begin();
7201 it != mHWData->mPCIDeviceAssignments.end();
7202 ++it)
7203 {
7204 LONG iHostAddress = -1;
7205 pAttach = *it;
7206 pAttach->COMGETTER(HostAddress)(&iHostAddress);
7207 if (iHostAddress != -1 && iHostAddress == hostAddress)
7208 {
7209 setModified(IsModified_MachineData);
7210 mHWData.backup();
7211 mHWData->mPCIDeviceAssignments.remove(pAttach);
7212 fRemoved = true;
7213 break;
7214 }
7215 }
7216 }
7217
7218
7219 /* Fire event outside of the lock */
7220 if (fRemoved)
7221 {
7222 Assert(!pAttach.isNull());
7223 ComPtr<IEventSource> es;
7224 rc = mParent->COMGETTER(EventSource)(es.asOutParam());
7225 Assert(SUCCEEDED(rc));
7226 Bstr mid;
7227 rc = this->COMGETTER(Id)(mid.asOutParam());
7228 Assert(SUCCEEDED(rc));
7229 fireHostPCIDevicePlugEvent(es, mid.raw(), false /* unplugged */, true /* success */, pAttach, NULL);
7230 }
7231
7232 return fRemoved ? S_OK : setError(VBOX_E_OBJECT_NOT_FOUND,
7233 tr("No host PCI device %08x attached"),
7234 hostAddress
7235 );
7236}
7237
7238STDMETHODIMP Machine::COMGETTER(PCIDeviceAssignments)(ComSafeArrayOut(IPCIDeviceAttachment *, aAssignments))
7239{
7240 CheckComArgOutSafeArrayPointerValid(aAssignments);
7241
7242 AutoCaller autoCaller(this);
7243 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7244
7245 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7246
7247 SafeIfaceArray<IPCIDeviceAttachment> assignments(mHWData->mPCIDeviceAssignments);
7248 assignments.detachTo(ComSafeArrayOutArg(aAssignments));
7249
7250 return S_OK;
7251}
7252
7253STDMETHODIMP Machine::COMGETTER(BandwidthControl)(IBandwidthControl **aBandwidthControl)
7254{
7255 CheckComArgOutPointerValid(aBandwidthControl);
7256
7257 AutoCaller autoCaller(this);
7258 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7259
7260 mBandwidthControl.queryInterfaceTo(aBandwidthControl);
7261
7262 return S_OK;
7263}
7264
7265STDMETHODIMP Machine::COMGETTER(TracingEnabled)(BOOL *pfEnabled)
7266{
7267 CheckComArgOutPointerValid(pfEnabled);
7268 AutoCaller autoCaller(this);
7269 HRESULT hrc = autoCaller.rc();
7270 if (SUCCEEDED(hrc))
7271 {
7272 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7273 *pfEnabled = mHWData->mDebugging.fTracingEnabled;
7274 }
7275 return hrc;
7276}
7277
7278STDMETHODIMP Machine::COMSETTER(TracingEnabled)(BOOL fEnabled)
7279{
7280 AutoCaller autoCaller(this);
7281 HRESULT hrc = autoCaller.rc();
7282 if (SUCCEEDED(hrc))
7283 {
7284 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7285 hrc = checkStateDependency(MutableStateDep);
7286 if (SUCCEEDED(hrc))
7287 {
7288 hrc = mHWData.backupEx();
7289 if (SUCCEEDED(hrc))
7290 {
7291 setModified(IsModified_MachineData);
7292 mHWData->mDebugging.fTracingEnabled = fEnabled != FALSE;
7293 }
7294 }
7295 }
7296 return hrc;
7297}
7298
7299STDMETHODIMP Machine::COMGETTER(TracingConfig)(BSTR *pbstrConfig)
7300{
7301 CheckComArgOutPointerValid(pbstrConfig);
7302 AutoCaller autoCaller(this);
7303 HRESULT hrc = autoCaller.rc();
7304 if (SUCCEEDED(hrc))
7305 {
7306 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7307 hrc = mHWData->mDebugging.strTracingConfig.cloneToEx(pbstrConfig);
7308 }
7309 return hrc;
7310}
7311
7312STDMETHODIMP Machine::COMSETTER(TracingConfig)(IN_BSTR bstrConfig)
7313{
7314 CheckComArgStr(bstrConfig);
7315 AutoCaller autoCaller(this);
7316 HRESULT hrc = autoCaller.rc();
7317 if (SUCCEEDED(hrc))
7318 {
7319 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7320 hrc = checkStateDependency(MutableStateDep);
7321 if (SUCCEEDED(hrc))
7322 {
7323 hrc = mHWData.backupEx();
7324 if (SUCCEEDED(hrc))
7325 {
7326 hrc = mHWData->mDebugging.strTracingConfig.cloneEx(bstrConfig);
7327 if (SUCCEEDED(hrc))
7328 setModified(IsModified_MachineData);
7329 }
7330 }
7331 }
7332 return hrc;
7333
7334}
7335
7336STDMETHODIMP Machine::COMGETTER(AllowTracingToAccessVM)(BOOL *pfAllow)
7337{
7338 CheckComArgOutPointerValid(pfAllow);
7339 AutoCaller autoCaller(this);
7340 HRESULT hrc = autoCaller.rc();
7341 if (SUCCEEDED(hrc))
7342 {
7343 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7344 *pfAllow = mHWData->mDebugging.fAllowTracingToAccessVM;
7345 }
7346 return hrc;
7347}
7348
7349STDMETHODIMP Machine::COMSETTER(AllowTracingToAccessVM)(BOOL fAllow)
7350{
7351 AutoCaller autoCaller(this);
7352 HRESULT hrc = autoCaller.rc();
7353 if (SUCCEEDED(hrc))
7354 {
7355 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7356 hrc = checkStateDependency(MutableStateDep);
7357 if (SUCCEEDED(hrc))
7358 {
7359 hrc = mHWData.backupEx();
7360 if (SUCCEEDED(hrc))
7361 {
7362 setModified(IsModified_MachineData);
7363 mHWData->mDebugging.fAllowTracingToAccessVM = fAllow != FALSE;
7364 }
7365 }
7366 }
7367 return hrc;
7368}
7369
7370STDMETHODIMP Machine::COMGETTER(AutostartEnabled)(BOOL *pfEnabled)
7371{
7372 CheckComArgOutPointerValid(pfEnabled);
7373 AutoCaller autoCaller(this);
7374 HRESULT hrc = autoCaller.rc();
7375 if (SUCCEEDED(hrc))
7376 {
7377 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7378 *pfEnabled = mHWData->mAutostart.fAutostartEnabled;
7379 }
7380 return hrc;
7381}
7382
7383STDMETHODIMP Machine::COMSETTER(AutostartEnabled)(BOOL fEnabled)
7384{
7385 AutoCaller autoCaller(this);
7386 HRESULT hrc = autoCaller.rc();
7387 if (SUCCEEDED(hrc))
7388 {
7389 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7390 hrc = checkStateDependency(MutableStateDep);
7391 if ( SUCCEEDED(hrc)
7392 && mHWData->mAutostart.fAutostartEnabled != !!fEnabled)
7393 {
7394 AutostartDb *autostartDb = mParent->getAutostartDb();
7395 int vrc;
7396
7397 if (fEnabled)
7398 vrc = autostartDb->addAutostartVM(mUserData->s.strName.c_str());
7399 else
7400 vrc = autostartDb->removeAutostartVM(mUserData->s.strName.c_str());
7401
7402 if (RT_SUCCESS(vrc))
7403 {
7404 hrc = mHWData.backupEx();
7405 if (SUCCEEDED(hrc))
7406 {
7407 setModified(IsModified_MachineData);
7408 mHWData->mAutostart.fAutostartEnabled = fEnabled != FALSE;
7409 }
7410 }
7411 else if (vrc == VERR_NOT_SUPPORTED)
7412 hrc = setError(VBOX_E_NOT_SUPPORTED,
7413 tr("The VM autostart feature is not supported on this platform"));
7414 else if (vrc == VERR_PATH_NOT_FOUND)
7415 hrc = setError(E_FAIL,
7416 tr("The path to the autostart database is not set"));
7417 else
7418 hrc = setError(E_UNEXPECTED,
7419 tr("%s machine '%s' to the autostart database failed with %Rrc"),
7420 fEnabled ? "Adding" : "Removing",
7421 mUserData->s.strName.c_str(), vrc);
7422 }
7423 }
7424 return hrc;
7425}
7426
7427STDMETHODIMP Machine::COMGETTER(AutostartDelay)(ULONG *puDelay)
7428{
7429 CheckComArgOutPointerValid(puDelay);
7430 AutoCaller autoCaller(this);
7431 HRESULT hrc = autoCaller.rc();
7432 if (SUCCEEDED(hrc))
7433 {
7434 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7435 *puDelay = mHWData->mAutostart.uAutostartDelay;
7436 }
7437 return hrc;
7438}
7439
7440STDMETHODIMP Machine::COMSETTER(AutostartDelay)(ULONG uDelay)
7441{
7442 AutoCaller autoCaller(this);
7443 HRESULT hrc = autoCaller.rc();
7444 if (SUCCEEDED(hrc))
7445 {
7446 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7447 hrc = checkStateDependency(MutableStateDep);
7448 if (SUCCEEDED(hrc))
7449 {
7450 hrc = mHWData.backupEx();
7451 if (SUCCEEDED(hrc))
7452 {
7453 setModified(IsModified_MachineData);
7454 mHWData->mAutostart.uAutostartDelay = uDelay;
7455 }
7456 }
7457 }
7458 return hrc;
7459}
7460
7461STDMETHODIMP Machine::COMGETTER(AutostopType)(AutostopType_T *penmAutostopType)
7462{
7463 CheckComArgOutPointerValid(penmAutostopType);
7464 AutoCaller autoCaller(this);
7465 HRESULT hrc = autoCaller.rc();
7466 if (SUCCEEDED(hrc))
7467 {
7468 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7469 *penmAutostopType = mHWData->mAutostart.enmAutostopType;
7470 }
7471 return hrc;
7472}
7473
7474STDMETHODIMP Machine::COMSETTER(AutostopType)(AutostopType_T enmAutostopType)
7475{
7476 AutoCaller autoCaller(this);
7477 HRESULT hrc = autoCaller.rc();
7478 if (SUCCEEDED(hrc))
7479 {
7480 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7481 hrc = checkStateDependency(MutableStateDep);
7482 if ( SUCCEEDED(hrc)
7483 && mHWData->mAutostart.enmAutostopType != enmAutostopType)
7484 {
7485 AutostartDb *autostartDb = mParent->getAutostartDb();
7486 int vrc;
7487
7488 if (enmAutostopType != AutostopType_Disabled)
7489 vrc = autostartDb->addAutostopVM(mUserData->s.strName.c_str());
7490 else
7491 vrc = autostartDb->removeAutostopVM(mUserData->s.strName.c_str());
7492
7493 if (RT_SUCCESS(vrc))
7494 {
7495 hrc = mHWData.backupEx();
7496 if (SUCCEEDED(hrc))
7497 {
7498 setModified(IsModified_MachineData);
7499 mHWData->mAutostart.enmAutostopType = enmAutostopType;
7500 }
7501 }
7502 else if (vrc == VERR_NOT_SUPPORTED)
7503 hrc = setError(VBOX_E_NOT_SUPPORTED,
7504 tr("The VM autostop feature is not supported on this platform"));
7505 else if (vrc == VERR_PATH_NOT_FOUND)
7506 hrc = setError(E_FAIL,
7507 tr("The path to the autostart database is not set"));
7508 else
7509 hrc = setError(E_UNEXPECTED,
7510 tr("%s machine '%s' to the autostop database failed with %Rrc"),
7511 enmAutostopType != AutostopType_Disabled ? "Adding" : "Removing",
7512 mUserData->s.strName.c_str(), vrc);
7513 }
7514 }
7515 return hrc;
7516}
7517
7518STDMETHODIMP Machine::COMGETTER(DefaultFrontend)(BSTR *aDefaultFrontend)
7519{
7520 CheckComArgOutPointerValid(aDefaultFrontend);
7521 AutoCaller autoCaller(this);
7522 HRESULT hrc = autoCaller.rc();
7523 if (SUCCEEDED(hrc))
7524 {
7525 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7526 mHWData->mDefaultFrontend.cloneTo(aDefaultFrontend);
7527 }
7528 return hrc;
7529}
7530
7531STDMETHODIMP Machine::COMSETTER(DefaultFrontend)(IN_BSTR aDefaultFrontend)
7532{
7533 CheckComArgStr(aDefaultFrontend);
7534 AutoCaller autoCaller(this);
7535 HRESULT hrc = autoCaller.rc();
7536 if (SUCCEEDED(hrc))
7537 {
7538 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7539 hrc = checkStateDependency(MutableOrSavedStateDep);
7540 if (SUCCEEDED(hrc))
7541 {
7542 hrc = mHWData.backupEx();
7543 if (SUCCEEDED(hrc))
7544 {
7545 setModified(IsModified_MachineData);
7546 mHWData->mDefaultFrontend = aDefaultFrontend;
7547 }
7548 }
7549 }
7550 return hrc;
7551}
7552
7553STDMETHODIMP Machine::COMGETTER(Icon)(ComSafeArrayOut(BYTE, aIcon))
7554{
7555 CheckComArgSafeArrayNotNull(aIcon);
7556 CheckComArgOutSafeArrayPointerValid(aIcon);
7557 AutoCaller autoCaller(this);
7558 HRESULT hrc = autoCaller.rc();
7559 if (SUCCEEDED(hrc))
7560 {
7561 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7562 com::SafeArray<BYTE> icon(mUserData->mIcon.size());
7563 memcpy(icon.raw(), &mUserData->mIcon[0], mUserData->mIcon.size());
7564 icon.detachTo(ComSafeArrayOutArg(aIcon));
7565 }
7566 return hrc;
7567}
7568
7569STDMETHODIMP Machine::COMSETTER(Icon)(ComSafeArrayIn(BYTE, aIcon))
7570{
7571 CheckComArgSafeArrayNotNull(aIcon);
7572 AutoCaller autoCaller(this);
7573 HRESULT hrc = autoCaller.rc();
7574 if (SUCCEEDED(hrc))
7575 {
7576 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7577 hrc = checkStateDependency(MutableOrSavedStateDep);
7578 if (SUCCEEDED(hrc))
7579 {
7580 setModified(IsModified_MachineData);
7581 mUserData.backup();
7582 com::SafeArray<BYTE> icon(ComSafeArrayInArg(aIcon));
7583 mUserData->mIcon.resize(icon.size());
7584 memcpy(&mUserData->mIcon[0], icon.raw(), mUserData->mIcon.size());
7585 }
7586 }
7587 return hrc;
7588}
7589
7590STDMETHODIMP Machine::COMGETTER(USBProxyAvailable)(BOOL *aAvailable)
7591{
7592 CheckComArgOutPointerValid(aAvailable);
7593
7594 AutoCaller autoCaller(this);
7595 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7596
7597#ifdef VBOX_WITH_USB
7598 *aAvailable = true;
7599#else
7600 *aAvailable = false;
7601#endif
7602 return S_OK;
7603}
7604
7605STDMETHODIMP Machine::CloneTo(IMachine *pTarget, CloneMode_T mode, ComSafeArrayIn(CloneOptions_T, options), IProgress **pProgress)
7606{
7607 LogFlowFuncEnter();
7608
7609 CheckComArgNotNull(pTarget);
7610 CheckComArgOutPointerValid(pProgress);
7611
7612 /* Convert the options. */
7613 RTCList<CloneOptions_T> optList;
7614 if (options != NULL)
7615 optList = com::SafeArray<CloneOptions_T>(ComSafeArrayInArg(options)).toList();
7616
7617 if (optList.contains(CloneOptions_Link))
7618 {
7619 if (!isSnapshotMachine())
7620 return setError(E_INVALIDARG,
7621 tr("Linked clone can only be created from a snapshot"));
7622 if (mode != CloneMode_MachineState)
7623 return setError(E_INVALIDARG,
7624 tr("Linked clone can only be created for a single machine state"));
7625 }
7626 AssertReturn(!(optList.contains(CloneOptions_KeepAllMACs) && optList.contains(CloneOptions_KeepNATMACs)), E_INVALIDARG);
7627
7628 AutoCaller autoCaller(this);
7629 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7630
7631
7632 MachineCloneVM *pWorker = new MachineCloneVM(this, static_cast<Machine*>(pTarget), mode, optList);
7633
7634 HRESULT rc = pWorker->start(pProgress);
7635
7636 LogFlowFuncLeave();
7637
7638 return rc;
7639}
7640
7641// public methods for internal purposes
7642/////////////////////////////////////////////////////////////////////////////
7643
7644/**
7645 * Adds the given IsModified_* flag to the dirty flags of the machine.
7646 * This must be called either during loadSettings or under the machine write lock.
7647 * @param fl
7648 */
7649void Machine::setModified(uint32_t fl, bool fAllowStateModification /* = true */)
7650{
7651 mData->flModifications |= fl;
7652 if (fAllowStateModification && isStateModificationAllowed())
7653 mData->mCurrentStateModified = true;
7654}
7655
7656/**
7657 * Adds the given IsModified_* flag to the dirty flags of the machine, taking
7658 * care of the write locking.
7659 *
7660 * @param fModifications The flag to add.
7661 */
7662void Machine::setModifiedLock(uint32_t fModification, bool fAllowStateModification /* = true */)
7663{
7664 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7665 setModified(fModification, fAllowStateModification);
7666}
7667
7668/**
7669 * Saves the registry entry of this machine to the given configuration node.
7670 *
7671 * @param aEntryNode Node to save the registry entry to.
7672 *
7673 * @note locks this object for reading.
7674 */
7675HRESULT Machine::saveRegistryEntry(settings::MachineRegistryEntry &data)
7676{
7677 AutoLimitedCaller autoCaller(this);
7678 AssertComRCReturnRC(autoCaller.rc());
7679
7680 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7681
7682 data.uuid = mData->mUuid;
7683 data.strSettingsFile = mData->m_strConfigFile;
7684
7685 return S_OK;
7686}
7687
7688/**
7689 * Calculates the absolute path of the given path taking the directory of the
7690 * machine settings file as the current directory.
7691 *
7692 * @param aPath Path to calculate the absolute path for.
7693 * @param aResult Where to put the result (used only on success, can be the
7694 * same Utf8Str instance as passed in @a aPath).
7695 * @return IPRT result.
7696 *
7697 * @note Locks this object for reading.
7698 */
7699int Machine::calculateFullPath(const Utf8Str &strPath, Utf8Str &aResult)
7700{
7701 AutoCaller autoCaller(this);
7702 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
7703
7704 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7705
7706 AssertReturn(!mData->m_strConfigFileFull.isEmpty(), VERR_GENERAL_FAILURE);
7707
7708 Utf8Str strSettingsDir = mData->m_strConfigFileFull;
7709
7710 strSettingsDir.stripFilename();
7711 char folder[RTPATH_MAX];
7712 int vrc = RTPathAbsEx(strSettingsDir.c_str(), strPath.c_str(), folder, sizeof(folder));
7713 if (RT_SUCCESS(vrc))
7714 aResult = folder;
7715
7716 return vrc;
7717}
7718
7719/**
7720 * Copies strSource to strTarget, making it relative to the machine folder
7721 * if it is a subdirectory thereof, or simply copying it otherwise.
7722 *
7723 * @param strSource Path to evaluate and copy.
7724 * @param strTarget Buffer to receive target path.
7725 *
7726 * @note Locks this object for reading.
7727 */
7728void Machine::copyPathRelativeToMachine(const Utf8Str &strSource,
7729 Utf8Str &strTarget)
7730{
7731 AutoCaller autoCaller(this);
7732 AssertComRCReturn(autoCaller.rc(), (void)0);
7733
7734 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7735
7736 AssertReturnVoid(!mData->m_strConfigFileFull.isEmpty());
7737 // use strTarget as a temporary buffer to hold the machine settings dir
7738 strTarget = mData->m_strConfigFileFull;
7739 strTarget.stripFilename();
7740 if (RTPathStartsWith(strSource.c_str(), strTarget.c_str()))
7741 {
7742 // is relative: then append what's left
7743 strTarget = strSource.substr(strTarget.length() + 1); // skip '/'
7744 // for empty paths (only possible for subdirs) use "." to avoid
7745 // triggering default settings for not present config attributes.
7746 if (strTarget.isEmpty())
7747 strTarget = ".";
7748 }
7749 else
7750 // is not relative: then overwrite
7751 strTarget = strSource;
7752}
7753
7754/**
7755 * Returns the full path to the machine's log folder in the
7756 * \a aLogFolder argument.
7757 */
7758void Machine::getLogFolder(Utf8Str &aLogFolder)
7759{
7760 AutoCaller autoCaller(this);
7761 AssertComRCReturnVoid(autoCaller.rc());
7762
7763 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7764
7765 char szTmp[RTPATH_MAX];
7766 int vrc = RTEnvGetEx(RTENV_DEFAULT, "VBOX_USER_VMLOGDIR", szTmp, sizeof(szTmp), NULL);
7767 if (RT_SUCCESS(vrc))
7768 {
7769 if (szTmp[0] && !mUserData.isNull())
7770 {
7771 char szTmp2[RTPATH_MAX];
7772 vrc = RTPathAbs(szTmp, szTmp2, sizeof(szTmp2));
7773 if (RT_SUCCESS(vrc))
7774 aLogFolder = BstrFmt("%s%c%s",
7775 szTmp2,
7776 RTPATH_DELIMITER,
7777 mUserData->s.strName.c_str()); // path/to/logfolder/vmname
7778 }
7779 else
7780 vrc = VERR_PATH_IS_RELATIVE;
7781 }
7782
7783 if (RT_FAILURE(vrc))
7784 {
7785 // fallback if VBOX_USER_LOGHOME is not set or invalid
7786 aLogFolder = mData->m_strConfigFileFull; // path/to/machinesfolder/vmname/vmname.vbox
7787 aLogFolder.stripFilename(); // path/to/machinesfolder/vmname
7788 aLogFolder.append(RTPATH_DELIMITER);
7789 aLogFolder.append("Logs"); // path/to/machinesfolder/vmname/Logs
7790 }
7791}
7792
7793/**
7794 * Returns the full path to the machine's log file for an given index.
7795 */
7796Utf8Str Machine::queryLogFilename(ULONG idx)
7797{
7798 Utf8Str logFolder;
7799 getLogFolder(logFolder);
7800 Assert(logFolder.length());
7801 Utf8Str log;
7802 if (idx == 0)
7803 log = Utf8StrFmt("%s%cVBox.log",
7804 logFolder.c_str(), RTPATH_DELIMITER);
7805 else
7806 log = Utf8StrFmt("%s%cVBox.log.%d",
7807 logFolder.c_str(), RTPATH_DELIMITER, idx);
7808 return log;
7809}
7810
7811/**
7812 * Composes a unique saved state filename based on the current system time. The filename is
7813 * granular to the second so this will work so long as no more than one snapshot is taken on
7814 * a machine per second.
7815 *
7816 * Before version 4.1, we used this formula for saved state files:
7817 * Utf8StrFmt("%s%c{%RTuuid}.sav", strFullSnapshotFolder.c_str(), RTPATH_DELIMITER, mData->mUuid.raw())
7818 * which no longer works because saved state files can now be shared between the saved state of the
7819 * "saved" machine and an online snapshot, and the following would cause problems:
7820 * 1) save machine
7821 * 2) create online snapshot from that machine state --> reusing saved state file
7822 * 3) save machine again --> filename would be reused, breaking the online snapshot
7823 *
7824 * So instead we now use a timestamp.
7825 *
7826 * @param str
7827 */
7828void Machine::composeSavedStateFilename(Utf8Str &strStateFilePath)
7829{
7830 AutoCaller autoCaller(this);
7831 AssertComRCReturnVoid(autoCaller.rc());
7832
7833 {
7834 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7835 calculateFullPath(mUserData->s.strSnapshotFolder, strStateFilePath);
7836 }
7837
7838 RTTIMESPEC ts;
7839 RTTimeNow(&ts);
7840 RTTIME time;
7841 RTTimeExplode(&time, &ts);
7842
7843 strStateFilePath += RTPATH_DELIMITER;
7844 strStateFilePath += Utf8StrFmt("%04d-%02u-%02uT%02u-%02u-%02u-%09uZ.sav",
7845 time.i32Year, time.u8Month, time.u8MonthDay,
7846 time.u8Hour, time.u8Minute, time.u8Second, time.u32Nanosecond);
7847}
7848
7849/**
7850 * Returns the full path to the default video capture file.
7851 */
7852void Machine::getDefaultVideoCaptureFile(Utf8Str &strFile)
7853{
7854 AutoCaller autoCaller(this);
7855 AssertComRCReturnVoid(autoCaller.rc());
7856
7857 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7858
7859 strFile = mData->m_strConfigFileFull; // path/to/machinesfolder/vmname/vmname.vbox
7860 strFile.stripExt(); // path/to/machinesfolder/vmname/vmname
7861 strFile.append(".webm"); // path/to/machinesfolder/vmname/vmname.webm
7862}
7863
7864/**
7865 * Returns whether at least one USB controller is present for the VM.
7866 */
7867bool Machine::isUSBControllerPresent()
7868{
7869 AutoCaller autoCaller(this);
7870 AssertComRCReturn(autoCaller.rc(), false);
7871
7872 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
7873
7874 return (mUSBControllers->size() > 0);
7875}
7876
7877/**
7878 * @note Locks this object for writing, calls the client process
7879 * (inside the lock).
7880 */
7881HRESULT Machine::launchVMProcess(IInternalSessionControl *aControl,
7882 const Utf8Str &strFrontend,
7883 const Utf8Str &strEnvironment,
7884 ProgressProxy *aProgress)
7885{
7886 LogFlowThisFuncEnter();
7887
7888 AssertReturn(aControl, E_FAIL);
7889 AssertReturn(aProgress, E_FAIL);
7890 AssertReturn(!strFrontend.isEmpty(), E_FAIL);
7891
7892 AutoCaller autoCaller(this);
7893 if (FAILED(autoCaller.rc())) return autoCaller.rc();
7894
7895 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
7896
7897 if (!mData->mRegistered)
7898 return setError(E_UNEXPECTED,
7899 tr("The machine '%s' is not registered"),
7900 mUserData->s.strName.c_str());
7901
7902 LogFlowThisFunc(("mSession.mState=%s\n", Global::stringifySessionState(mData->mSession.mState)));
7903
7904 if ( mData->mSession.mState == SessionState_Locked
7905 || mData->mSession.mState == SessionState_Spawning
7906 || mData->mSession.mState == SessionState_Unlocking)
7907 return setError(VBOX_E_INVALID_OBJECT_STATE,
7908 tr("The machine '%s' is already locked by a session (or being locked or unlocked)"),
7909 mUserData->s.strName.c_str());
7910
7911 /* may not be busy */
7912 AssertReturn(!Global::IsOnlineOrTransient(mData->mMachineState), E_FAIL);
7913
7914 /* get the path to the executable */
7915 char szPath[RTPATH_MAX];
7916 RTPathAppPrivateArch(szPath, sizeof(szPath) - 1);
7917 size_t sz = strlen(szPath);
7918 szPath[sz++] = RTPATH_DELIMITER;
7919 szPath[sz] = 0;
7920 char *cmd = szPath + sz;
7921 sz = sizeof(szPath) - sz;
7922
7923 int vrc = VINF_SUCCESS;
7924 RTPROCESS pid = NIL_RTPROCESS;
7925
7926 RTENV env = RTENV_DEFAULT;
7927
7928 if (!strEnvironment.isEmpty())
7929 {
7930 char *newEnvStr = NULL;
7931
7932 do
7933 {
7934 /* clone the current environment */
7935 int vrc2 = RTEnvClone(&env, RTENV_DEFAULT);
7936 AssertRCBreakStmt(vrc2, vrc = vrc2);
7937
7938 newEnvStr = RTStrDup(strEnvironment.c_str());
7939 AssertPtrBreakStmt(newEnvStr, vrc = vrc2);
7940
7941 /* put new variables to the environment
7942 * (ignore empty variable names here since RTEnv API
7943 * intentionally doesn't do that) */
7944 char *var = newEnvStr;
7945 for (char *p = newEnvStr; *p; ++p)
7946 {
7947 if (*p == '\n' && (p == newEnvStr || *(p - 1) != '\\'))
7948 {
7949 *p = '\0';
7950 if (*var)
7951 {
7952 char *val = strchr(var, '=');
7953 if (val)
7954 {
7955 *val++ = '\0';
7956 vrc2 = RTEnvSetEx(env, var, val);
7957 }
7958 else
7959 vrc2 = RTEnvUnsetEx(env, var);
7960 if (RT_FAILURE(vrc2))
7961 break;
7962 }
7963 var = p + 1;
7964 }
7965 }
7966 if (RT_SUCCESS(vrc2) && *var)
7967 vrc2 = RTEnvPutEx(env, var);
7968
7969 AssertRCBreakStmt(vrc2, vrc = vrc2);
7970 }
7971 while (0);
7972
7973 if (newEnvStr != NULL)
7974 RTStrFree(newEnvStr);
7975 }
7976
7977#ifdef VBOX_WITH_QTGUI
7978 if (strFrontend == "gui" || strFrontend == "GUI/Qt")
7979 {
7980# ifdef RT_OS_DARWIN /* Avoid Launch Services confusing this with the selector by using a helper app. */
7981 /* Modify the base path so that we don't need to use ".." below. */
7982 RTPathStripTrailingSlash(szPath);
7983 RTPathStripFilename(szPath);
7984 sz = strlen(szPath);
7985 cmd = szPath + sz;
7986 sz = sizeof(szPath) - sz;
7987
7988#define OSX_APP_NAME "VirtualBoxVM"
7989#define OSX_APP_PATH_FMT "/Resources/%s.app/Contents/MacOS/VirtualBoxVM"
7990
7991 Utf8Str strAppOverride = getExtraData(Utf8Str("VBoxInternal2/VirtualBoxVMAppOverride"));
7992 if ( strAppOverride.contains(".")
7993 || strAppOverride.contains("/")
7994 || strAppOverride.contains("\\")
7995 || strAppOverride.contains(":"))
7996 strAppOverride.setNull();
7997 Utf8Str strAppPath;
7998 if (!strAppOverride.isEmpty())
7999 {
8000 strAppPath = Utf8StrFmt(OSX_APP_PATH_FMT, strAppOverride.c_str());
8001 Utf8Str strFullPath(szPath);
8002 strFullPath.append(strAppPath);
8003 /* there is a race, but people using this deserve the failure */
8004 if (!RTFileExists(strFullPath.c_str()))
8005 strAppOverride.setNull();
8006 }
8007 if (strAppOverride.isEmpty())
8008 strAppPath = Utf8StrFmt(OSX_APP_PATH_FMT, OSX_APP_NAME);
8009 const char *VirtualBox_exe = strAppPath.c_str();
8010 AssertReturn(sz >= strlen(VirtualBox_exe), E_UNEXPECTED);
8011# else
8012 const char VirtualBox_exe[] = "VirtualBox" HOSTSUFF_EXE;
8013 Assert(sz >= sizeof(VirtualBox_exe));
8014# endif
8015 strcpy(cmd, VirtualBox_exe);
8016
8017 Utf8Str idStr = mData->mUuid.toString();
8018 const char * args[] = {szPath, "--comment", mUserData->s.strName.c_str(), "--startvm", idStr.c_str(), "--no-startvm-errormsgbox", 0 };
8019 vrc = RTProcCreate(szPath, args, env, 0, &pid);
8020 }
8021#else /* !VBOX_WITH_QTGUI */
8022 if (0)
8023 ;
8024#endif /* VBOX_WITH_QTGUI */
8025
8026 else
8027
8028#ifdef VBOX_WITH_VBOXSDL
8029 if (strFrontend == "sdl" || strFrontend == "GUI/SDL")
8030 {
8031 const char VBoxSDL_exe[] = "VBoxSDL" HOSTSUFF_EXE;
8032 Assert(sz >= sizeof(VBoxSDL_exe));
8033 strcpy(cmd, VBoxSDL_exe);
8034
8035 Utf8Str idStr = mData->mUuid.toString();
8036 const char * args[] = {szPath, "--comment", mUserData->s.strName.c_str(), "--startvm", idStr.c_str(), 0 };
8037 vrc = RTProcCreate(szPath, args, env, 0, &pid);
8038 }
8039#else /* !VBOX_WITH_VBOXSDL */
8040 if (0)
8041 ;
8042#endif /* !VBOX_WITH_VBOXSDL */
8043
8044 else
8045
8046#ifdef VBOX_WITH_HEADLESS
8047 if ( strFrontend == "headless"
8048 || strFrontend == "capture"
8049 || strFrontend == "vrdp" /* Deprecated. Same as headless. */
8050 )
8051 {
8052 /* On pre-4.0 the "headless" type was used for passing "--vrdp off" to VBoxHeadless to let it work in OSE,
8053 * which did not contain VRDP server. In VBox 4.0 the remote desktop server (VRDE) is optional,
8054 * and a VM works even if the server has not been installed.
8055 * So in 4.0 the "headless" behavior remains the same for default VBox installations.
8056 * Only if a VRDE has been installed and the VM enables it, the "headless" will work
8057 * differently in 4.0 and 3.x.
8058 */
8059 const char VBoxHeadless_exe[] = "VBoxHeadless" HOSTSUFF_EXE;
8060 Assert(sz >= sizeof(VBoxHeadless_exe));
8061 strcpy(cmd, VBoxHeadless_exe);
8062
8063 Utf8Str idStr = mData->mUuid.toString();
8064 /* Leave space for "--capture" arg. */
8065 const char * args[] = {szPath, "--comment", mUserData->s.strName.c_str(),
8066 "--startvm", idStr.c_str(),
8067 "--vrde", "config",
8068 0, /* For "--capture". */
8069 0 };
8070 if (strFrontend == "capture")
8071 {
8072 unsigned pos = RT_ELEMENTS(args) - 2;
8073 args[pos] = "--capture";
8074 }
8075 vrc = RTProcCreate(szPath, args, env,
8076#ifdef RT_OS_WINDOWS
8077 RTPROC_FLAGS_NO_WINDOW
8078#else
8079 0
8080#endif
8081 , &pid);
8082 }
8083#else /* !VBOX_WITH_HEADLESS */
8084 if (0)
8085 ;
8086#endif /* !VBOX_WITH_HEADLESS */
8087 else
8088 {
8089 RTEnvDestroy(env);
8090 return setError(E_INVALIDARG,
8091 tr("Invalid frontend name: '%s'"),
8092 strFrontend.c_str());
8093 }
8094
8095 RTEnvDestroy(env);
8096
8097 if (RT_FAILURE(vrc))
8098 return setError(VBOX_E_IPRT_ERROR,
8099 tr("Could not launch a process for the machine '%s' (%Rrc)"),
8100 mUserData->s.strName.c_str(), vrc);
8101
8102 LogFlowThisFunc(("launched.pid=%d(0x%x)\n", pid, pid));
8103
8104 /*
8105 * Note that we don't release the lock here before calling the client,
8106 * because it doesn't need to call us back if called with a NULL argument.
8107 * Releasing the lock here is dangerous because we didn't prepare the
8108 * launch data yet, but the client we've just started may happen to be
8109 * too fast and call LockMachine() that will fail (because of PID, etc.),
8110 * so that the Machine will never get out of the Spawning session state.
8111 */
8112
8113 /* inform the session that it will be a remote one */
8114 LogFlowThisFunc(("Calling AssignMachine (NULL)...\n"));
8115#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
8116 HRESULT rc = aControl->AssignMachine(NULL, LockType_Write, Bstr::Empty.raw());
8117#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
8118 HRESULT rc = aControl->AssignMachine(NULL, LockType_Write, NULL);
8119#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
8120 LogFlowThisFunc(("AssignMachine (NULL) returned %08X\n", rc));
8121
8122 if (FAILED(rc))
8123 {
8124 /* restore the session state */
8125 mData->mSession.mState = SessionState_Unlocked;
8126 /* The failure may occur w/o any error info (from RPC), so provide one */
8127 return setError(VBOX_E_VM_ERROR,
8128 tr("Failed to assign the machine to the session (%Rhrc)"), rc);
8129 }
8130
8131 /* attach launch data to the machine */
8132 Assert(mData->mSession.mPID == NIL_RTPROCESS);
8133 mData->mSession.mRemoteControls.push_back(aControl);
8134 mData->mSession.mProgress = aProgress;
8135 mData->mSession.mPID = pid;
8136 mData->mSession.mState = SessionState_Spawning;
8137 mData->mSession.mType = strFrontend;
8138
8139 LogFlowThisFuncLeave();
8140 return S_OK;
8141}
8142
8143/**
8144 * Returns @c true if the given session machine instance has an open direct
8145 * session (and optionally also for direct sessions which are closing) and
8146 * returns the session control machine instance if so.
8147 *
8148 * Note that when the method returns @c false, the arguments remain unchanged.
8149 *
8150 * @param aMachine Session machine object.
8151 * @param aControl Direct session control object (optional).
8152 * @param aAllowClosing If true then additionally a session which is currently
8153 * being closed will also be allowed.
8154 *
8155 * @note locks this object for reading.
8156 */
8157bool Machine::isSessionOpen(ComObjPtr<SessionMachine> &aMachine,
8158 ComPtr<IInternalSessionControl> *aControl /*= NULL*/,
8159 bool aAllowClosing /*= false*/)
8160{
8161 AutoLimitedCaller autoCaller(this);
8162 AssertComRCReturn(autoCaller.rc(), false);
8163
8164 /* just return false for inaccessible machines */
8165 if (autoCaller.state() != Ready)
8166 return false;
8167
8168 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
8169
8170 if ( mData->mSession.mState == SessionState_Locked
8171 || (aAllowClosing && mData->mSession.mState == SessionState_Unlocking)
8172 )
8173 {
8174 AssertReturn(!mData->mSession.mMachine.isNull(), false);
8175
8176 aMachine = mData->mSession.mMachine;
8177
8178 if (aControl != NULL)
8179 *aControl = mData->mSession.mDirectControl;
8180
8181 return true;
8182 }
8183
8184 return false;
8185}
8186
8187/**
8188 * Returns @c true if the given machine has an spawning direct session.
8189 *
8190 * @note locks this object for reading.
8191 */
8192bool Machine::isSessionSpawning()
8193{
8194 AutoLimitedCaller autoCaller(this);
8195 AssertComRCReturn(autoCaller.rc(), false);
8196
8197 /* just return false for inaccessible machines */
8198 if (autoCaller.state() != Ready)
8199 return false;
8200
8201 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
8202
8203 if (mData->mSession.mState == SessionState_Spawning)
8204 return true;
8205
8206 return false;
8207}
8208
8209#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
8210/**
8211 * Called from the client watcher thread to check for unexpected client process
8212 * death during Session_Spawning state (e.g. before it successfully opened a
8213 * direct session).
8214 *
8215 * On Win32 and on OS/2, this method is called only when we've got the
8216 * direct client's process termination notification, so it always returns @c
8217 * true.
8218 *
8219 * On other platforms, this method returns @c true if the client process is
8220 * terminated and @c false if it's still alive.
8221 *
8222 * @note Locks this object for writing.
8223 */
8224bool Machine::checkForSpawnFailure()
8225{
8226 AutoCaller autoCaller(this);
8227 if (!autoCaller.isOk())
8228 {
8229 /* nothing to do */
8230 LogFlowThisFunc(("Already uninitialized!\n"));
8231 return true;
8232 }
8233
8234 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8235
8236 if (mData->mSession.mState != SessionState_Spawning)
8237 {
8238 /* nothing to do */
8239 LogFlowThisFunc(("Not spawning any more!\n"));
8240 return true;
8241 }
8242
8243 HRESULT rc = S_OK;
8244
8245 /* PID not yet initialized, skip check. */
8246 if (mData->mSession.mPID == NIL_RTPROCESS)
8247 return false;
8248
8249 RTPROCSTATUS status;
8250 int vrc = RTProcWait(mData->mSession.mPID, RTPROCWAIT_FLAGS_NOBLOCK, &status);
8251
8252 if (vrc != VERR_PROCESS_RUNNING)
8253 {
8254 if (RT_SUCCESS(vrc) && status.enmReason == RTPROCEXITREASON_NORMAL)
8255 rc = setError(E_FAIL,
8256 tr("The virtual machine '%s' has terminated unexpectedly during startup with exit code %d"),
8257 getName().c_str(), status.iStatus);
8258 else if (RT_SUCCESS(vrc) && status.enmReason == RTPROCEXITREASON_SIGNAL)
8259 rc = setError(E_FAIL,
8260 tr("The virtual machine '%s' has terminated unexpectedly during startup because of signal %d"),
8261 getName().c_str(), status.iStatus);
8262 else if (RT_SUCCESS(vrc) && status.enmReason == RTPROCEXITREASON_ABEND)
8263 rc = setError(E_FAIL,
8264 tr("The virtual machine '%s' has terminated abnormally"),
8265 getName().c_str(), status.iStatus);
8266 else
8267 rc = setError(E_FAIL,
8268 tr("The virtual machine '%s' has terminated unexpectedly during startup (%Rrc)"),
8269 getName().c_str(), vrc);
8270 }
8271
8272 if (FAILED(rc))
8273 {
8274 /* Close the remote session, remove the remote control from the list
8275 * and reset session state to Closed (@note keep the code in sync with
8276 * the relevant part in LockMachine()). */
8277
8278 Assert(mData->mSession.mRemoteControls.size() == 1);
8279 if (mData->mSession.mRemoteControls.size() == 1)
8280 {
8281 ErrorInfoKeeper eik;
8282 mData->mSession.mRemoteControls.front()->Uninitialize();
8283 }
8284
8285 mData->mSession.mRemoteControls.clear();
8286 mData->mSession.mState = SessionState_Unlocked;
8287
8288 /* finalize the progress after setting the state */
8289 if (!mData->mSession.mProgress.isNull())
8290 {
8291 mData->mSession.mProgress->notifyComplete(rc);
8292 mData->mSession.mProgress.setNull();
8293 }
8294
8295 mData->mSession.mPID = NIL_RTPROCESS;
8296
8297 mParent->onSessionStateChange(mData->mUuid, SessionState_Unlocked);
8298 return true;
8299 }
8300
8301 return false;
8302}
8303#endif /* !VBOX_WITH_GENERIC_SESSION_WATCHER */
8304
8305/**
8306 * Checks whether the machine can be registered. If so, commits and saves
8307 * all settings.
8308 *
8309 * @note Must be called from mParent's write lock. Locks this object and
8310 * children for writing.
8311 */
8312HRESULT Machine::prepareRegister()
8313{
8314 AssertReturn(mParent->isWriteLockOnCurrentThread(), E_FAIL);
8315
8316 AutoLimitedCaller autoCaller(this);
8317 AssertComRCReturnRC(autoCaller.rc());
8318
8319 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8320
8321 /* wait for state dependents to drop to zero */
8322 ensureNoStateDependencies();
8323
8324 if (!mData->mAccessible)
8325 return setError(VBOX_E_INVALID_OBJECT_STATE,
8326 tr("The machine '%s' with UUID {%s} is inaccessible and cannot be registered"),
8327 mUserData->s.strName.c_str(),
8328 mData->mUuid.toString().c_str());
8329
8330 AssertReturn(autoCaller.state() == Ready, E_FAIL);
8331
8332 if (mData->mRegistered)
8333 return setError(VBOX_E_INVALID_OBJECT_STATE,
8334 tr("The machine '%s' with UUID {%s} is already registered"),
8335 mUserData->s.strName.c_str(),
8336 mData->mUuid.toString().c_str());
8337
8338 HRESULT rc = S_OK;
8339
8340 // Ensure the settings are saved. If we are going to be registered and
8341 // no config file exists yet, create it by calling saveSettings() too.
8342 if ( (mData->flModifications)
8343 || (!mData->pMachineConfigFile->fileExists())
8344 )
8345 {
8346 rc = saveSettings(NULL);
8347 // no need to check whether VirtualBox.xml needs saving too since
8348 // we can't have a machine XML file rename pending
8349 if (FAILED(rc)) return rc;
8350 }
8351
8352 /* more config checking goes here */
8353
8354 if (SUCCEEDED(rc))
8355 {
8356 /* we may have had implicit modifications we want to fix on success */
8357 commit();
8358
8359 mData->mRegistered = true;
8360 }
8361 else
8362 {
8363 /* we may have had implicit modifications we want to cancel on failure*/
8364 rollback(false /* aNotify */);
8365 }
8366
8367 return rc;
8368}
8369
8370/**
8371 * Increases the number of objects dependent on the machine state or on the
8372 * registered state. Guarantees that these two states will not change at least
8373 * until #releaseStateDependency() is called.
8374 *
8375 * Depending on the @a aDepType value, additional state checks may be made.
8376 * These checks will set extended error info on failure. See
8377 * #checkStateDependency() for more info.
8378 *
8379 * If this method returns a failure, the dependency is not added and the caller
8380 * is not allowed to rely on any particular machine state or registration state
8381 * value and may return the failed result code to the upper level.
8382 *
8383 * @param aDepType Dependency type to add.
8384 * @param aState Current machine state (NULL if not interested).
8385 * @param aRegistered Current registered state (NULL if not interested).
8386 *
8387 * @note Locks this object for writing.
8388 */
8389HRESULT Machine::addStateDependency(StateDependency aDepType /* = AnyStateDep */,
8390 MachineState_T *aState /* = NULL */,
8391 BOOL *aRegistered /* = NULL */)
8392{
8393 AutoCaller autoCaller(this);
8394 AssertComRCReturnRC(autoCaller.rc());
8395
8396 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8397
8398 HRESULT rc = checkStateDependency(aDepType);
8399 if (FAILED(rc)) return rc;
8400
8401 {
8402 if (mData->mMachineStateChangePending != 0)
8403 {
8404 /* ensureNoStateDependencies() is waiting for state dependencies to
8405 * drop to zero so don't add more. It may make sense to wait a bit
8406 * and retry before reporting an error (since the pending state
8407 * transition should be really quick) but let's just assert for
8408 * now to see if it ever happens on practice. */
8409
8410 AssertFailed();
8411
8412 return setError(E_ACCESSDENIED,
8413 tr("Machine state change is in progress. Please retry the operation later."));
8414 }
8415
8416 ++mData->mMachineStateDeps;
8417 Assert(mData->mMachineStateDeps != 0 /* overflow */);
8418 }
8419
8420 if (aState)
8421 *aState = mData->mMachineState;
8422 if (aRegistered)
8423 *aRegistered = mData->mRegistered;
8424
8425 return S_OK;
8426}
8427
8428/**
8429 * Decreases the number of objects dependent on the machine state.
8430 * Must always complete the #addStateDependency() call after the state
8431 * dependency is no more necessary.
8432 */
8433void Machine::releaseStateDependency()
8434{
8435 AutoCaller autoCaller(this);
8436 AssertComRCReturnVoid(autoCaller.rc());
8437
8438 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8439
8440 /* releaseStateDependency() w/o addStateDependency()? */
8441 AssertReturnVoid(mData->mMachineStateDeps != 0);
8442 -- mData->mMachineStateDeps;
8443
8444 if (mData->mMachineStateDeps == 0)
8445 {
8446 /* inform ensureNoStateDependencies() that there are no more deps */
8447 if (mData->mMachineStateChangePending != 0)
8448 {
8449 Assert(mData->mMachineStateDepsSem != NIL_RTSEMEVENTMULTI);
8450 RTSemEventMultiSignal (mData->mMachineStateDepsSem);
8451 }
8452 }
8453}
8454
8455Utf8Str Machine::getExtraData(const Utf8Str &strKey)
8456{
8457 /* start with nothing found */
8458 Utf8Str strResult("");
8459
8460 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
8461
8462 settings::StringsMap::const_iterator it = mData->pMachineConfigFile->mapExtraDataItems.find(strKey);
8463 if (it != mData->pMachineConfigFile->mapExtraDataItems.end())
8464 // found:
8465 strResult = it->second; // source is a Utf8Str
8466
8467 return strResult;
8468}
8469
8470// protected methods
8471/////////////////////////////////////////////////////////////////////////////
8472
8473/**
8474 * Performs machine state checks based on the @a aDepType value. If a check
8475 * fails, this method will set extended error info, otherwise it will return
8476 * S_OK. It is supposed, that on failure, the caller will immediately return
8477 * the return value of this method to the upper level.
8478 *
8479 * When @a aDepType is AnyStateDep, this method always returns S_OK.
8480 *
8481 * When @a aDepType is MutableStateDep, this method returns S_OK only if the
8482 * current state of this machine object allows to change settings of the
8483 * machine (i.e. the machine is not registered, or registered but not running
8484 * and not saved). It is useful to call this method from Machine setters
8485 * before performing any change.
8486 *
8487 * When @a aDepType is MutableOrSavedStateDep, this method behaves the same
8488 * as for MutableStateDep except that if the machine is saved, S_OK is also
8489 * returned. This is useful in setters which allow changing machine
8490 * properties when it is in the saved state.
8491 *
8492 * When @a aDepType is OfflineStateDep, this method returns S_OK if the
8493 * state is one of the 4 offline states (PoweredOff, Saved, Teleported,
8494 * Aborted).
8495 *
8496 * @param aDepType Dependency type to check.
8497 *
8498 * @note Non Machine based classes should use #addStateDependency() and
8499 * #releaseStateDependency() methods or the smart AutoStateDependency
8500 * template.
8501 *
8502 * @note This method must be called from under this object's read or write
8503 * lock.
8504 */
8505HRESULT Machine::checkStateDependency(StateDependency aDepType)
8506{
8507 switch (aDepType)
8508 {
8509 case AnyStateDep:
8510 {
8511 break;
8512 }
8513 case MutableStateDep:
8514 {
8515 if ( mData->mRegistered
8516 && ( !isSessionMachine() /** @todo This was just converted raw; Check if Running and Paused should actually be included here... (Live Migration) */
8517 || ( mData->mMachineState != MachineState_Paused
8518 && mData->mMachineState != MachineState_Running
8519 && mData->mMachineState != MachineState_Aborted
8520 && mData->mMachineState != MachineState_Teleported
8521 && mData->mMachineState != MachineState_PoweredOff
8522 )
8523 )
8524 )
8525 return setError(VBOX_E_INVALID_VM_STATE,
8526 tr("The machine is not mutable (state is %s)"),
8527 Global::stringifyMachineState(mData->mMachineState));
8528 break;
8529 }
8530 case MutableOrSavedStateDep:
8531 {
8532 if ( mData->mRegistered
8533 && ( !isSessionMachine() /** @todo This was just converted raw; Check if Running and Paused should actually be included here... (Live Migration) */
8534 || ( mData->mMachineState != MachineState_Paused
8535 && mData->mMachineState != MachineState_Running
8536 && mData->mMachineState != MachineState_Aborted
8537 && mData->mMachineState != MachineState_Teleported
8538 && mData->mMachineState != MachineState_Saved
8539 && mData->mMachineState != MachineState_PoweredOff
8540 )
8541 )
8542 )
8543 return setError(VBOX_E_INVALID_VM_STATE,
8544 tr("The machine is not mutable (state is %s)"),
8545 Global::stringifyMachineState(mData->mMachineState));
8546 break;
8547 }
8548 case OfflineStateDep:
8549 {
8550 if ( mData->mRegistered
8551 && ( !isSessionMachine()
8552 || ( mData->mMachineState != MachineState_PoweredOff
8553 && mData->mMachineState != MachineState_Saved
8554 && mData->mMachineState != MachineState_Aborted
8555 && mData->mMachineState != MachineState_Teleported
8556 )
8557 )
8558 )
8559 return setError(VBOX_E_INVALID_VM_STATE,
8560 tr("The machine is not offline (state is %s)"),
8561 Global::stringifyMachineState(mData->mMachineState));
8562 break;
8563 }
8564 }
8565
8566 return S_OK;
8567}
8568
8569/**
8570 * Helper to initialize all associated child objects and allocate data
8571 * structures.
8572 *
8573 * This method must be called as a part of the object's initialization procedure
8574 * (usually done in the #init() method).
8575 *
8576 * @note Must be called only from #init() or from #registeredInit().
8577 */
8578HRESULT Machine::initDataAndChildObjects()
8579{
8580 AutoCaller autoCaller(this);
8581 AssertComRCReturnRC(autoCaller.rc());
8582 AssertComRCReturn(autoCaller.state() == InInit ||
8583 autoCaller.state() == Limited, E_FAIL);
8584
8585 AssertReturn(!mData->mAccessible, E_FAIL);
8586
8587 /* allocate data structures */
8588 mSSData.allocate();
8589 mUserData.allocate();
8590 mHWData.allocate();
8591 mMediaData.allocate();
8592 mStorageControllers.allocate();
8593 mUSBControllers.allocate();
8594
8595 /* initialize mOSTypeId */
8596 mUserData->s.strOsType = mParent->getUnknownOSType()->id();
8597
8598 /* create associated BIOS settings object */
8599 unconst(mBIOSSettings).createObject();
8600 mBIOSSettings->init(this);
8601
8602 /* create an associated VRDE object (default is disabled) */
8603 unconst(mVRDEServer).createObject();
8604 mVRDEServer->init(this);
8605
8606 /* create associated serial port objects */
8607 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
8608 {
8609 unconst(mSerialPorts[slot]).createObject();
8610 mSerialPorts[slot]->init(this, slot);
8611 }
8612
8613 /* create associated parallel port objects */
8614 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
8615 {
8616 unconst(mParallelPorts[slot]).createObject();
8617 mParallelPorts[slot]->init(this, slot);
8618 }
8619
8620 /* create the audio adapter object (always present, default is disabled) */
8621 unconst(mAudioAdapter).createObject();
8622 mAudioAdapter->init(this);
8623
8624 /* create the USB device filters object (always present) */
8625 unconst(mUSBDeviceFilters).createObject();
8626 mUSBDeviceFilters->init(this);
8627
8628 /* create associated network adapter objects */
8629 mNetworkAdapters.resize(Global::getMaxNetworkAdapters(mHWData->mChipsetType));
8630 for (ULONG slot = 0; slot < mNetworkAdapters.size(); slot++)
8631 {
8632 unconst(mNetworkAdapters[slot]).createObject();
8633 mNetworkAdapters[slot]->init(this, slot);
8634 }
8635
8636 /* create the bandwidth control */
8637 unconst(mBandwidthControl).createObject();
8638 mBandwidthControl->init(this);
8639
8640 return S_OK;
8641}
8642
8643/**
8644 * Helper to uninitialize all associated child objects and to free all data
8645 * structures.
8646 *
8647 * This method must be called as a part of the object's uninitialization
8648 * procedure (usually done in the #uninit() method).
8649 *
8650 * @note Must be called only from #uninit() or from #registeredInit().
8651 */
8652void Machine::uninitDataAndChildObjects()
8653{
8654 AutoCaller autoCaller(this);
8655 AssertComRCReturnVoid(autoCaller.rc());
8656 AssertComRCReturnVoid( autoCaller.state() == InUninit
8657 || autoCaller.state() == Limited);
8658
8659 /* tell all our other child objects we've been uninitialized */
8660 if (mBandwidthControl)
8661 {
8662 mBandwidthControl->uninit();
8663 unconst(mBandwidthControl).setNull();
8664 }
8665
8666 for (ULONG slot = 0; slot < mNetworkAdapters.size(); slot++)
8667 {
8668 if (mNetworkAdapters[slot])
8669 {
8670 mNetworkAdapters[slot]->uninit();
8671 unconst(mNetworkAdapters[slot]).setNull();
8672 }
8673 }
8674
8675 if (mUSBDeviceFilters)
8676 {
8677 mUSBDeviceFilters->uninit();
8678 unconst(mUSBDeviceFilters).setNull();
8679 }
8680
8681 if (mAudioAdapter)
8682 {
8683 mAudioAdapter->uninit();
8684 unconst(mAudioAdapter).setNull();
8685 }
8686
8687 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
8688 {
8689 if (mParallelPorts[slot])
8690 {
8691 mParallelPorts[slot]->uninit();
8692 unconst(mParallelPorts[slot]).setNull();
8693 }
8694 }
8695
8696 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
8697 {
8698 if (mSerialPorts[slot])
8699 {
8700 mSerialPorts[slot]->uninit();
8701 unconst(mSerialPorts[slot]).setNull();
8702 }
8703 }
8704
8705 if (mVRDEServer)
8706 {
8707 mVRDEServer->uninit();
8708 unconst(mVRDEServer).setNull();
8709 }
8710
8711 if (mBIOSSettings)
8712 {
8713 mBIOSSettings->uninit();
8714 unconst(mBIOSSettings).setNull();
8715 }
8716
8717 /* Deassociate media (only when a real Machine or a SnapshotMachine
8718 * instance is uninitialized; SessionMachine instances refer to real
8719 * Machine media). This is necessary for a clean re-initialization of
8720 * the VM after successfully re-checking the accessibility state. Note
8721 * that in case of normal Machine or SnapshotMachine uninitialization (as
8722 * a result of unregistering or deleting the snapshot), outdated media
8723 * attachments will already be uninitialized and deleted, so this
8724 * code will not affect them. */
8725 if ( !!mMediaData
8726 && (!isSessionMachine())
8727 )
8728 {
8729 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
8730 it != mMediaData->mAttachments.end();
8731 ++it)
8732 {
8733 ComObjPtr<Medium> pMedium = (*it)->getMedium();
8734 if (pMedium.isNull())
8735 continue;
8736 HRESULT rc = pMedium->removeBackReference(mData->mUuid, getSnapshotId());
8737 AssertComRC(rc);
8738 }
8739 }
8740
8741 if (!isSessionMachine() && !isSnapshotMachine())
8742 {
8743 // clean up the snapshots list (Snapshot::uninit() will handle the snapshot's children recursively)
8744 if (mData->mFirstSnapshot)
8745 {
8746 // snapshots tree is protected by machine write lock; strictly
8747 // this isn't necessary here since we're deleting the entire
8748 // machine, but otherwise we assert in Snapshot::uninit()
8749 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8750 mData->mFirstSnapshot->uninit();
8751 mData->mFirstSnapshot.setNull();
8752 }
8753
8754 mData->mCurrentSnapshot.setNull();
8755 }
8756
8757 /* free data structures (the essential mData structure is not freed here
8758 * since it may be still in use) */
8759 mMediaData.free();
8760 mStorageControllers.free();
8761 mUSBControllers.free();
8762 mHWData.free();
8763 mUserData.free();
8764 mSSData.free();
8765}
8766
8767/**
8768 * Returns a pointer to the Machine object for this machine that acts like a
8769 * parent for complex machine data objects such as shared folders, etc.
8770 *
8771 * For primary Machine objects and for SnapshotMachine objects, returns this
8772 * object's pointer itself. For SessionMachine objects, returns the peer
8773 * (primary) machine pointer.
8774 */
8775Machine* Machine::getMachine()
8776{
8777 if (isSessionMachine())
8778 return (Machine*)mPeer;
8779 return this;
8780}
8781
8782/**
8783 * Makes sure that there are no machine state dependents. If necessary, waits
8784 * for the number of dependents to drop to zero.
8785 *
8786 * Make sure this method is called from under this object's write lock to
8787 * guarantee that no new dependents may be added when this method returns
8788 * control to the caller.
8789 *
8790 * @note Locks this object for writing. The lock will be released while waiting
8791 * (if necessary).
8792 *
8793 * @warning To be used only in methods that change the machine state!
8794 */
8795void Machine::ensureNoStateDependencies()
8796{
8797 AssertReturnVoid(isWriteLockOnCurrentThread());
8798
8799 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8800
8801 /* Wait for all state dependents if necessary */
8802 if (mData->mMachineStateDeps != 0)
8803 {
8804 /* lazy semaphore creation */
8805 if (mData->mMachineStateDepsSem == NIL_RTSEMEVENTMULTI)
8806 RTSemEventMultiCreate(&mData->mMachineStateDepsSem);
8807
8808 LogFlowThisFunc(("Waiting for state deps (%d) to drop to zero...\n",
8809 mData->mMachineStateDeps));
8810
8811 ++mData->mMachineStateChangePending;
8812
8813 /* reset the semaphore before waiting, the last dependent will signal
8814 * it */
8815 RTSemEventMultiReset(mData->mMachineStateDepsSem);
8816
8817 alock.release();
8818
8819 RTSemEventMultiWait(mData->mMachineStateDepsSem, RT_INDEFINITE_WAIT);
8820
8821 alock.acquire();
8822
8823 -- mData->mMachineStateChangePending;
8824 }
8825}
8826
8827/**
8828 * Changes the machine state and informs callbacks.
8829 *
8830 * This method is not intended to fail so it either returns S_OK or asserts (and
8831 * returns a failure).
8832 *
8833 * @note Locks this object for writing.
8834 */
8835HRESULT Machine::setMachineState(MachineState_T aMachineState)
8836{
8837 LogFlowThisFuncEnter();
8838 LogFlowThisFunc(("aMachineState=%s\n", Global::stringifyMachineState(aMachineState) ));
8839
8840 AutoCaller autoCaller(this);
8841 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
8842
8843 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
8844
8845 /* wait for state dependents to drop to zero */
8846 ensureNoStateDependencies();
8847
8848 if (mData->mMachineState != aMachineState)
8849 {
8850 mData->mMachineState = aMachineState;
8851
8852 RTTimeNow(&mData->mLastStateChange);
8853
8854 mParent->onMachineStateChange(mData->mUuid, aMachineState);
8855 }
8856
8857 LogFlowThisFuncLeave();
8858 return S_OK;
8859}
8860
8861/**
8862 * Searches for a shared folder with the given logical name
8863 * in the collection of shared folders.
8864 *
8865 * @param aName logical name of the shared folder
8866 * @param aSharedFolder where to return the found object
8867 * @param aSetError whether to set the error info if the folder is
8868 * not found
8869 * @return
8870 * S_OK when found or VBOX_E_OBJECT_NOT_FOUND when not found
8871 *
8872 * @note
8873 * must be called from under the object's lock!
8874 */
8875HRESULT Machine::findSharedFolder(const Utf8Str &aName,
8876 ComObjPtr<SharedFolder> &aSharedFolder,
8877 bool aSetError /* = false */)
8878{
8879 HRESULT rc = VBOX_E_OBJECT_NOT_FOUND;
8880 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
8881 it != mHWData->mSharedFolders.end();
8882 ++it)
8883 {
8884 SharedFolder *pSF = *it;
8885 AutoCaller autoCaller(pSF);
8886 if (pSF->getName() == aName)
8887 {
8888 aSharedFolder = pSF;
8889 rc = S_OK;
8890 break;
8891 }
8892 }
8893
8894 if (aSetError && FAILED(rc))
8895 setError(rc, tr("Could not find a shared folder named '%s'"), aName.c_str());
8896
8897 return rc;
8898}
8899
8900/**
8901 * Initializes all machine instance data from the given settings structures
8902 * from XML. The exception is the machine UUID which needs special handling
8903 * depending on the caller's use case, so the caller needs to set that herself.
8904 *
8905 * This gets called in several contexts during machine initialization:
8906 *
8907 * -- When machine XML exists on disk already and needs to be loaded into memory,
8908 * for example, from registeredInit() to load all registered machines on
8909 * VirtualBox startup. In this case, puuidRegistry is NULL because the media
8910 * attached to the machine should be part of some media registry already.
8911 *
8912 * -- During OVF import, when a machine config has been constructed from an
8913 * OVF file. In this case, puuidRegistry is set to the machine UUID to
8914 * ensure that the media listed as attachments in the config (which have
8915 * been imported from the OVF) receive the correct registry ID.
8916 *
8917 * -- During VM cloning.
8918 *
8919 * @param config Machine settings from XML.
8920 * @param puuidRegistry If != NULL, Medium::setRegistryIdIfFirst() gets called with this registry ID for each attached medium in the config.
8921 * @return
8922 */
8923HRESULT Machine::loadMachineDataFromSettings(const settings::MachineConfigFile &config,
8924 const Guid *puuidRegistry)
8925{
8926 // copy name, description, OS type, teleporter, UTC etc.
8927 mUserData->s = config.machineUserData;
8928
8929 // Decode the Icon overide data from config userdata and set onto Machine.
8930 #define DECODE_STR_MAX _1M
8931 const char* pszStr = config.machineUserData.ovIcon.c_str();
8932 ssize_t cbOut = RTBase64DecodedSize(pszStr, NULL);
8933 if (cbOut > DECODE_STR_MAX)
8934 return setError(E_FAIL,
8935 tr("Icon Data too long.'%d' > '%d'"),
8936 cbOut,
8937 DECODE_STR_MAX);
8938 com::SafeArray<BYTE> iconByte(cbOut);
8939 HRESULT rc = RTBase64Decode(pszStr, iconByte.raw(), cbOut, NULL, NULL);
8940 if (FAILED(rc))
8941 return setError(E_FAIL,
8942 tr("Failure to Decode Icon Data. '%s' (%d)"),
8943 pszStr,
8944 rc);
8945 mUserData->mIcon.resize(iconByte.size());
8946 memcpy(&mUserData->mIcon[0], iconByte.raw(), mUserData->mIcon.size());
8947
8948 // look up the object by Id to check it is valid
8949 ComPtr<IGuestOSType> guestOSType;
8950 rc = mParent->GetGuestOSType(Bstr(mUserData->s.strOsType).raw(),
8951 guestOSType.asOutParam());
8952 if (FAILED(rc)) return rc;
8953
8954 // stateFile (optional)
8955 if (config.strStateFile.isEmpty())
8956 mSSData->strStateFilePath.setNull();
8957 else
8958 {
8959 Utf8Str stateFilePathFull(config.strStateFile);
8960 int vrc = calculateFullPath(stateFilePathFull, stateFilePathFull);
8961 if (RT_FAILURE(vrc))
8962 return setError(E_FAIL,
8963 tr("Invalid saved state file path '%s' (%Rrc)"),
8964 config.strStateFile.c_str(),
8965 vrc);
8966 mSSData->strStateFilePath = stateFilePathFull;
8967 }
8968
8969 // snapshot folder needs special processing so set it again
8970 rc = COMSETTER(SnapshotFolder)(Bstr(config.machineUserData.strSnapshotFolder).raw());
8971 if (FAILED(rc)) return rc;
8972
8973 /* Copy the extra data items (Not in any case config is already the same as
8974 * mData->pMachineConfigFile, like when the xml files are read from disk. So
8975 * make sure the extra data map is copied). */
8976 mData->pMachineConfigFile->mapExtraDataItems = config.mapExtraDataItems;
8977
8978 /* currentStateModified (optional, default is true) */
8979 mData->mCurrentStateModified = config.fCurrentStateModified;
8980
8981 mData->mLastStateChange = config.timeLastStateChange;
8982
8983 /*
8984 * note: all mUserData members must be assigned prior this point because
8985 * we need to commit changes in order to let mUserData be shared by all
8986 * snapshot machine instances.
8987 */
8988 mUserData.commitCopy();
8989
8990 // machine registry, if present (must be loaded before snapshots)
8991 if (config.canHaveOwnMediaRegistry())
8992 {
8993 // determine machine folder
8994 Utf8Str strMachineFolder = getSettingsFileFull();
8995 strMachineFolder.stripFilename();
8996 rc = mParent->initMedia(getId(), // media registry ID == machine UUID
8997 config.mediaRegistry,
8998 strMachineFolder);
8999 if (FAILED(rc)) return rc;
9000 }
9001
9002 /* Snapshot node (optional) */
9003 size_t cRootSnapshots;
9004 if ((cRootSnapshots = config.llFirstSnapshot.size()))
9005 {
9006 // there must be only one root snapshot
9007 Assert(cRootSnapshots == 1);
9008
9009 const settings::Snapshot &snap = config.llFirstSnapshot.front();
9010
9011 rc = loadSnapshot(snap,
9012 config.uuidCurrentSnapshot,
9013 NULL); // no parent == first snapshot
9014 if (FAILED(rc)) return rc;
9015 }
9016
9017 // hardware data
9018 rc = loadHardware(config.hardwareMachine, &config.debugging, &config.autostart);
9019 if (FAILED(rc)) return rc;
9020
9021 // load storage controllers
9022 rc = loadStorageControllers(config.storageMachine,
9023 puuidRegistry,
9024 NULL /* puuidSnapshot */);
9025 if (FAILED(rc)) return rc;
9026
9027 /*
9028 * NOTE: the assignment below must be the last thing to do,
9029 * otherwise it will be not possible to change the settings
9030 * somewhere in the code above because all setters will be
9031 * blocked by checkStateDependency(MutableStateDep).
9032 */
9033
9034 /* set the machine state to Aborted or Saved when appropriate */
9035 if (config.fAborted)
9036 {
9037 mSSData->strStateFilePath.setNull();
9038
9039 /* no need to use setMachineState() during init() */
9040 mData->mMachineState = MachineState_Aborted;
9041 }
9042 else if (!mSSData->strStateFilePath.isEmpty())
9043 {
9044 /* no need to use setMachineState() during init() */
9045 mData->mMachineState = MachineState_Saved;
9046 }
9047
9048 // after loading settings, we are no longer different from the XML on disk
9049 mData->flModifications = 0;
9050
9051 return S_OK;
9052}
9053
9054/**
9055 * Recursively loads all snapshots starting from the given.
9056 *
9057 * @param aNode <Snapshot> node.
9058 * @param aCurSnapshotId Current snapshot ID from the settings file.
9059 * @param aParentSnapshot Parent snapshot.
9060 */
9061HRESULT Machine::loadSnapshot(const settings::Snapshot &data,
9062 const Guid &aCurSnapshotId,
9063 Snapshot *aParentSnapshot)
9064{
9065 AssertReturn(!isSnapshotMachine(), E_FAIL);
9066 AssertReturn(!isSessionMachine(), E_FAIL);
9067
9068 HRESULT rc = S_OK;
9069
9070 Utf8Str strStateFile;
9071 if (!data.strStateFile.isEmpty())
9072 {
9073 /* optional */
9074 strStateFile = data.strStateFile;
9075 int vrc = calculateFullPath(strStateFile, strStateFile);
9076 if (RT_FAILURE(vrc))
9077 return setError(E_FAIL,
9078 tr("Invalid saved state file path '%s' (%Rrc)"),
9079 strStateFile.c_str(),
9080 vrc);
9081 }
9082
9083 /* create a snapshot machine object */
9084 ComObjPtr<SnapshotMachine> pSnapshotMachine;
9085 pSnapshotMachine.createObject();
9086 rc = pSnapshotMachine->initFromSettings(this,
9087 data.hardware,
9088 &data.debugging,
9089 &data.autostart,
9090 data.storage,
9091 data.uuid.ref(),
9092 strStateFile);
9093 if (FAILED(rc)) return rc;
9094
9095 /* create a snapshot object */
9096 ComObjPtr<Snapshot> pSnapshot;
9097 pSnapshot.createObject();
9098 /* initialize the snapshot */
9099 rc = pSnapshot->init(mParent, // VirtualBox object
9100 data.uuid,
9101 data.strName,
9102 data.strDescription,
9103 data.timestamp,
9104 pSnapshotMachine,
9105 aParentSnapshot);
9106 if (FAILED(rc)) return rc;
9107
9108 /* memorize the first snapshot if necessary */
9109 if (!mData->mFirstSnapshot)
9110 mData->mFirstSnapshot = pSnapshot;
9111
9112 /* memorize the current snapshot when appropriate */
9113 if ( !mData->mCurrentSnapshot
9114 && pSnapshot->getId() == aCurSnapshotId
9115 )
9116 mData->mCurrentSnapshot = pSnapshot;
9117
9118 // now create the children
9119 for (settings::SnapshotsList::const_iterator it = data.llChildSnapshots.begin();
9120 it != data.llChildSnapshots.end();
9121 ++it)
9122 {
9123 const settings::Snapshot &childData = *it;
9124 // recurse
9125 rc = loadSnapshot(childData,
9126 aCurSnapshotId,
9127 pSnapshot); // parent = the one we created above
9128 if (FAILED(rc)) return rc;
9129 }
9130
9131 return rc;
9132}
9133
9134/**
9135 * Loads settings into mHWData.
9136 *
9137 * @param data Reference to the hardware settings.
9138 * @param pDbg Pointer to the debugging settings.
9139 * @param pAutostart Pointer to the autostart settings.
9140 */
9141HRESULT Machine::loadHardware(const settings::Hardware &data, const settings::Debugging *pDbg,
9142 const settings::Autostart *pAutostart)
9143{
9144 AssertReturn(!isSessionMachine(), E_FAIL);
9145
9146 HRESULT rc = S_OK;
9147
9148 try
9149 {
9150 /* The hardware version attribute (optional). */
9151 mHWData->mHWVersion = data.strVersion;
9152 mHWData->mHardwareUUID = data.uuid;
9153
9154 mHWData->mHWVirtExEnabled = data.fHardwareVirt;
9155 mHWData->mHWVirtExNestedPagingEnabled = data.fNestedPaging;
9156 mHWData->mHWVirtExLargePagesEnabled = data.fLargePages;
9157 mHWData->mHWVirtExVPIDEnabled = data.fVPID;
9158 mHWData->mHWVirtExUXEnabled = data.fUnrestrictedExecution;
9159 mHWData->mHWVirtExForceEnabled = data.fHardwareVirtForce;
9160 mHWData->mPAEEnabled = data.fPAE;
9161 mHWData->mSyntheticCpu = data.fSyntheticCpu;
9162 mHWData->mLongMode = data.enmLongMode;
9163 mHWData->mCPUCount = data.cCPUs;
9164 mHWData->mCPUHotPlugEnabled = data.fCpuHotPlug;
9165 mHWData->mCpuExecutionCap = data.ulCpuExecutionCap;
9166
9167 // cpu
9168 if (mHWData->mCPUHotPlugEnabled)
9169 {
9170 for (settings::CpuList::const_iterator it = data.llCpus.begin();
9171 it != data.llCpus.end();
9172 ++it)
9173 {
9174 const settings::Cpu &cpu = *it;
9175
9176 mHWData->mCPUAttached[cpu.ulId] = true;
9177 }
9178 }
9179
9180 // cpuid leafs
9181 for (settings::CpuIdLeafsList::const_iterator it = data.llCpuIdLeafs.begin();
9182 it != data.llCpuIdLeafs.end();
9183 ++it)
9184 {
9185 const settings::CpuIdLeaf &leaf = *it;
9186
9187 switch (leaf.ulId)
9188 {
9189 case 0x0:
9190 case 0x1:
9191 case 0x2:
9192 case 0x3:
9193 case 0x4:
9194 case 0x5:
9195 case 0x6:
9196 case 0x7:
9197 case 0x8:
9198 case 0x9:
9199 case 0xA:
9200 mHWData->mCpuIdStdLeafs[leaf.ulId] = leaf;
9201 break;
9202
9203 case 0x80000000:
9204 case 0x80000001:
9205 case 0x80000002:
9206 case 0x80000003:
9207 case 0x80000004:
9208 case 0x80000005:
9209 case 0x80000006:
9210 case 0x80000007:
9211 case 0x80000008:
9212 case 0x80000009:
9213 case 0x8000000A:
9214 mHWData->mCpuIdExtLeafs[leaf.ulId - 0x80000000] = leaf;
9215 break;
9216
9217 default:
9218 /* just ignore */
9219 break;
9220 }
9221 }
9222
9223 mHWData->mMemorySize = data.ulMemorySizeMB;
9224 mHWData->mPageFusionEnabled = data.fPageFusionEnabled;
9225
9226 // boot order
9227 for (size_t i = 0;
9228 i < RT_ELEMENTS(mHWData->mBootOrder);
9229 i++)
9230 {
9231 settings::BootOrderMap::const_iterator it = data.mapBootOrder.find(i);
9232 if (it == data.mapBootOrder.end())
9233 mHWData->mBootOrder[i] = DeviceType_Null;
9234 else
9235 mHWData->mBootOrder[i] = it->second;
9236 }
9237
9238 mHWData->mGraphicsControllerType = data.graphicsControllerType;
9239 mHWData->mVRAMSize = data.ulVRAMSizeMB;
9240 mHWData->mMonitorCount = data.cMonitors;
9241 mHWData->mAccelerate3DEnabled = data.fAccelerate3D;
9242 mHWData->mAccelerate2DVideoEnabled = data.fAccelerate2DVideo;
9243 mHWData->mVideoCaptureWidth = data.ulVideoCaptureHorzRes;
9244 mHWData->mVideoCaptureHeight = data.ulVideoCaptureVertRes;
9245 mHWData->mVideoCaptureEnabled = data.fVideoCaptureEnabled;
9246 for (unsigned i = 0; i < RT_ELEMENTS(mHWData->maVideoCaptureScreens); i++)
9247 mHWData->maVideoCaptureScreens[i] = ASMBitTest(&data.u64VideoCaptureScreens, i);
9248 AssertCompile(RT_ELEMENTS(mHWData->maVideoCaptureScreens) == sizeof(data.u64VideoCaptureScreens) * 8);
9249 mHWData->mVideoCaptureRate = data.ulVideoCaptureRate;
9250 mHWData->mVideoCaptureFPS = data.ulVideoCaptureFPS;
9251 if (!data.strVideoCaptureFile.isEmpty())
9252 calculateFullPath(data.strVideoCaptureFile, mHWData->mVideoCaptureFile);
9253 else
9254 mHWData->mVideoCaptureFile.setNull();
9255 mHWData->mFirmwareType = data.firmwareType;
9256 mHWData->mPointingHIDType = data.pointingHIDType;
9257 mHWData->mKeyboardHIDType = data.keyboardHIDType;
9258 mHWData->mChipsetType = data.chipsetType;
9259 mHWData->mEmulatedUSBCardReaderEnabled = data.fEmulatedUSBCardReader;
9260 mHWData->mHPETEnabled = data.fHPETEnabled;
9261
9262 /* VRDEServer */
9263 rc = mVRDEServer->loadSettings(data.vrdeSettings);
9264 if (FAILED(rc)) return rc;
9265
9266 /* BIOS */
9267 rc = mBIOSSettings->loadSettings(data.biosSettings);
9268 if (FAILED(rc)) return rc;
9269
9270 // Bandwidth control (must come before network adapters)
9271 rc = mBandwidthControl->loadSettings(data.ioSettings);
9272 if (FAILED(rc)) return rc;
9273
9274 /* Shared folders */
9275 for (settings::USBControllerList::const_iterator it = data.usbSettings.llUSBControllers.begin();
9276 it != data.usbSettings.llUSBControllers.end();
9277 ++it)
9278 {
9279 const settings::USBController &settingsCtrl = *it;
9280 ComObjPtr<USBController> newCtrl;
9281
9282 newCtrl.createObject();
9283 newCtrl->init(this, settingsCtrl.strName, settingsCtrl.enmType);
9284 mUSBControllers->push_back(newCtrl);
9285 }
9286
9287 /* USB device filters */
9288 rc = mUSBDeviceFilters->loadSettings(data.usbSettings);
9289 if (FAILED(rc)) return rc;
9290
9291 // network adapters
9292 uint32_t newCount = Global::getMaxNetworkAdapters(mHWData->mChipsetType);
9293 uint32_t oldCount = mNetworkAdapters.size();
9294 if (newCount > oldCount)
9295 {
9296 mNetworkAdapters.resize(newCount);
9297 for (ULONG slot = oldCount; slot < mNetworkAdapters.size(); slot++)
9298 {
9299 unconst(mNetworkAdapters[slot]).createObject();
9300 mNetworkAdapters[slot]->init(this, slot);
9301 }
9302 }
9303 else if (newCount < oldCount)
9304 mNetworkAdapters.resize(newCount);
9305 for (settings::NetworkAdaptersList::const_iterator it = data.llNetworkAdapters.begin();
9306 it != data.llNetworkAdapters.end();
9307 ++it)
9308 {
9309 const settings::NetworkAdapter &nic = *it;
9310
9311 /* slot unicity is guaranteed by XML Schema */
9312 AssertBreak(nic.ulSlot < mNetworkAdapters.size());
9313 rc = mNetworkAdapters[nic.ulSlot]->loadSettings(mBandwidthControl, nic);
9314 if (FAILED(rc)) return rc;
9315 }
9316
9317 // serial ports
9318 for (settings::SerialPortsList::const_iterator it = data.llSerialPorts.begin();
9319 it != data.llSerialPorts.end();
9320 ++it)
9321 {
9322 const settings::SerialPort &s = *it;
9323
9324 AssertBreak(s.ulSlot < RT_ELEMENTS(mSerialPorts));
9325 rc = mSerialPorts[s.ulSlot]->loadSettings(s);
9326 if (FAILED(rc)) return rc;
9327 }
9328
9329 // parallel ports (optional)
9330 for (settings::ParallelPortsList::const_iterator it = data.llParallelPorts.begin();
9331 it != data.llParallelPorts.end();
9332 ++it)
9333 {
9334 const settings::ParallelPort &p = *it;
9335
9336 AssertBreak(p.ulSlot < RT_ELEMENTS(mParallelPorts));
9337 rc = mParallelPorts[p.ulSlot]->loadSettings(p);
9338 if (FAILED(rc)) return rc;
9339 }
9340
9341 /* AudioAdapter */
9342 rc = mAudioAdapter->loadSettings(data.audioAdapter);
9343 if (FAILED(rc)) return rc;
9344
9345 /* Shared folders */
9346 for (settings::SharedFoldersList::const_iterator it = data.llSharedFolders.begin();
9347 it != data.llSharedFolders.end();
9348 ++it)
9349 {
9350 const settings::SharedFolder &sf = *it;
9351
9352 ComObjPtr<SharedFolder> sharedFolder;
9353 /* Check for double entries. Not allowed! */
9354 rc = findSharedFolder(sf.strName, sharedFolder, false /* aSetError */);
9355 if (SUCCEEDED(rc))
9356 return setError(VBOX_E_OBJECT_IN_USE,
9357 tr("Shared folder named '%s' already exists"),
9358 sf.strName.c_str());
9359
9360 /* Create the new shared folder. Don't break on error. This will be
9361 * reported when the machine starts. */
9362 sharedFolder.createObject();
9363 rc = sharedFolder->init(getMachine(),
9364 sf.strName,
9365 sf.strHostPath,
9366 RT_BOOL(sf.fWritable),
9367 RT_BOOL(sf.fAutoMount),
9368 false /* fFailOnError */);
9369 if (FAILED(rc)) return rc;
9370 mHWData->mSharedFolders.push_back(sharedFolder);
9371 }
9372
9373 // Clipboard
9374 mHWData->mClipboardMode = data.clipboardMode;
9375
9376 // drag'n'drop
9377 mHWData->mDragAndDropMode = data.dragAndDropMode;
9378
9379 // guest settings
9380 mHWData->mMemoryBalloonSize = data.ulMemoryBalloonSize;
9381
9382 // IO settings
9383 mHWData->mIOCacheEnabled = data.ioSettings.fIOCacheEnabled;
9384 mHWData->mIOCacheSize = data.ioSettings.ulIOCacheSize;
9385
9386 // Host PCI devices
9387 for (settings::HostPCIDeviceAttachmentList::const_iterator it = data.pciAttachments.begin();
9388 it != data.pciAttachments.end();
9389 ++it)
9390 {
9391 const settings::HostPCIDeviceAttachment &hpda = *it;
9392 ComObjPtr<PCIDeviceAttachment> pda;
9393
9394 pda.createObject();
9395 pda->loadSettings(this, hpda);
9396 mHWData->mPCIDeviceAssignments.push_back(pda);
9397 }
9398
9399 /*
9400 * (The following isn't really real hardware, but it lives in HWData
9401 * for reasons of convenience.)
9402 */
9403
9404#ifdef VBOX_WITH_GUEST_PROPS
9405 /* Guest properties (optional) */
9406 for (settings::GuestPropertiesList::const_iterator it = data.llGuestProperties.begin();
9407 it != data.llGuestProperties.end();
9408 ++it)
9409 {
9410 const settings::GuestProperty &prop = *it;
9411 uint32_t fFlags = guestProp::NILFLAG;
9412 guestProp::validateFlags(prop.strFlags.c_str(), &fFlags);
9413 HWData::GuestProperty property = { prop.strValue, (LONG64) prop.timestamp, fFlags };
9414 mHWData->mGuestProperties[prop.strName] = property;
9415 }
9416
9417 mHWData->mGuestPropertyNotificationPatterns = data.strNotificationPatterns;
9418#endif /* VBOX_WITH_GUEST_PROPS defined */
9419
9420 rc = loadDebugging(pDbg);
9421 if (FAILED(rc))
9422 return rc;
9423
9424 mHWData->mAutostart = *pAutostart;
9425
9426 /* default frontend */
9427 mHWData->mDefaultFrontend = data.strDefaultFrontend;
9428 }
9429 catch(std::bad_alloc &)
9430 {
9431 return E_OUTOFMEMORY;
9432 }
9433
9434 AssertComRC(rc);
9435 return rc;
9436}
9437
9438/**
9439 * Called from Machine::loadHardware() to load the debugging settings of the
9440 * machine.
9441 *
9442 * @param pDbg Pointer to the settings.
9443 */
9444HRESULT Machine::loadDebugging(const settings::Debugging *pDbg)
9445{
9446 mHWData->mDebugging = *pDbg;
9447 /* no more processing currently required, this will probably change. */
9448 return S_OK;
9449}
9450
9451/**
9452 * Called from loadMachineDataFromSettings() for the storage controller data, including media.
9453 *
9454 * @param data
9455 * @param puuidRegistry media registry ID to set media to or NULL; see Machine::loadMachineDataFromSettings()
9456 * @param puuidSnapshot
9457 * @return
9458 */
9459HRESULT Machine::loadStorageControllers(const settings::Storage &data,
9460 const Guid *puuidRegistry,
9461 const Guid *puuidSnapshot)
9462{
9463 AssertReturn(!isSessionMachine(), E_FAIL);
9464
9465 HRESULT rc = S_OK;
9466
9467 for (settings::StorageControllersList::const_iterator it = data.llStorageControllers.begin();
9468 it != data.llStorageControllers.end();
9469 ++it)
9470 {
9471 const settings::StorageController &ctlData = *it;
9472
9473 ComObjPtr<StorageController> pCtl;
9474 /* Try to find one with the name first. */
9475 rc = getStorageControllerByName(ctlData.strName, pCtl, false /* aSetError */);
9476 if (SUCCEEDED(rc))
9477 return setError(VBOX_E_OBJECT_IN_USE,
9478 tr("Storage controller named '%s' already exists"),
9479 ctlData.strName.c_str());
9480
9481 pCtl.createObject();
9482 rc = pCtl->init(this,
9483 ctlData.strName,
9484 ctlData.storageBus,
9485 ctlData.ulInstance,
9486 ctlData.fBootable);
9487 if (FAILED(rc)) return rc;
9488
9489 mStorageControllers->push_back(pCtl);
9490
9491 rc = pCtl->COMSETTER(ControllerType)(ctlData.controllerType);
9492 if (FAILED(rc)) return rc;
9493
9494 rc = pCtl->COMSETTER(PortCount)(ctlData.ulPortCount);
9495 if (FAILED(rc)) return rc;
9496
9497 rc = pCtl->COMSETTER(UseHostIOCache)(ctlData.fUseHostIOCache);
9498 if (FAILED(rc)) return rc;
9499
9500 /* Set IDE emulation settings (only for AHCI controller). */
9501 if (ctlData.controllerType == StorageControllerType_IntelAhci)
9502 {
9503 if ( (FAILED(rc = pCtl->setIDEEmulationPort(0, ctlData.lIDE0MasterEmulationPort)))
9504 || (FAILED(rc = pCtl->setIDEEmulationPort(1, ctlData.lIDE0SlaveEmulationPort)))
9505 || (FAILED(rc = pCtl->setIDEEmulationPort(2, ctlData.lIDE1MasterEmulationPort)))
9506 || (FAILED(rc = pCtl->setIDEEmulationPort(3, ctlData.lIDE1SlaveEmulationPort)))
9507 )
9508 return rc;
9509 }
9510
9511 /* Load the attached devices now. */
9512 rc = loadStorageDevices(pCtl,
9513 ctlData,
9514 puuidRegistry,
9515 puuidSnapshot);
9516 if (FAILED(rc)) return rc;
9517 }
9518
9519 return S_OK;
9520}
9521
9522/**
9523 * Called from loadStorageControllers for a controller's devices.
9524 *
9525 * @param aStorageController
9526 * @param data
9527 * @param puuidRegistry media registry ID to set media to or NULL; see Machine::loadMachineDataFromSettings()
9528 * @param aSnapshotId pointer to the snapshot ID if this is a snapshot machine
9529 * @return
9530 */
9531HRESULT Machine::loadStorageDevices(StorageController *aStorageController,
9532 const settings::StorageController &data,
9533 const Guid *puuidRegistry,
9534 const Guid *puuidSnapshot)
9535{
9536 HRESULT rc = S_OK;
9537
9538 /* paranoia: detect duplicate attachments */
9539 for (settings::AttachedDevicesList::const_iterator it = data.llAttachedDevices.begin();
9540 it != data.llAttachedDevices.end();
9541 ++it)
9542 {
9543 const settings::AttachedDevice &ad = *it;
9544
9545 for (settings::AttachedDevicesList::const_iterator it2 = it;
9546 it2 != data.llAttachedDevices.end();
9547 ++it2)
9548 {
9549 if (it == it2)
9550 continue;
9551
9552 const settings::AttachedDevice &ad2 = *it2;
9553
9554 if ( ad.lPort == ad2.lPort
9555 && ad.lDevice == ad2.lDevice)
9556 {
9557 return setError(E_FAIL,
9558 tr("Duplicate attachments for storage controller '%s', port %d, device %d of the virtual machine '%s'"),
9559 aStorageController->getName().c_str(),
9560 ad.lPort,
9561 ad.lDevice,
9562 mUserData->s.strName.c_str());
9563 }
9564 }
9565 }
9566
9567 for (settings::AttachedDevicesList::const_iterator it = data.llAttachedDevices.begin();
9568 it != data.llAttachedDevices.end();
9569 ++it)
9570 {
9571 const settings::AttachedDevice &dev = *it;
9572 ComObjPtr<Medium> medium;
9573
9574 switch (dev.deviceType)
9575 {
9576 case DeviceType_Floppy:
9577 case DeviceType_DVD:
9578 if (dev.strHostDriveSrc.isNotEmpty())
9579 rc = mParent->host()->findHostDriveByName(dev.deviceType, dev.strHostDriveSrc, false /* fRefresh */, medium);
9580 else
9581 rc = mParent->findRemoveableMedium(dev.deviceType,
9582 dev.uuid,
9583 false /* fRefresh */,
9584 false /* aSetError */,
9585 medium);
9586 if (rc == VBOX_E_OBJECT_NOT_FOUND)
9587 // This is not an error. The host drive or UUID might have vanished, so just go ahead without this removeable medium attachment
9588 rc = S_OK;
9589 break;
9590
9591 case DeviceType_HardDisk:
9592 {
9593 /* find a hard disk by UUID */
9594 rc = mParent->findHardDiskById(dev.uuid, true /* aDoSetError */, &medium);
9595 if (FAILED(rc))
9596 {
9597 if (isSnapshotMachine())
9598 {
9599 // wrap another error message around the "cannot find hard disk" set by findHardDisk
9600 // so the user knows that the bad disk is in a snapshot somewhere
9601 com::ErrorInfo info;
9602 return setError(E_FAIL,
9603 tr("A differencing image of snapshot {%RTuuid} could not be found. %ls"),
9604 puuidSnapshot->raw(),
9605 info.getText().raw());
9606 }
9607 else
9608 return rc;
9609 }
9610
9611 AutoWriteLock hdLock(medium COMMA_LOCKVAL_SRC_POS);
9612
9613 if (medium->getType() == MediumType_Immutable)
9614 {
9615 if (isSnapshotMachine())
9616 return setError(E_FAIL,
9617 tr("Immutable hard disk '%s' with UUID {%RTuuid} cannot be directly attached to snapshot with UUID {%RTuuid} "
9618 "of the virtual machine '%s' ('%s')"),
9619 medium->getLocationFull().c_str(),
9620 dev.uuid.raw(),
9621 puuidSnapshot->raw(),
9622 mUserData->s.strName.c_str(),
9623 mData->m_strConfigFileFull.c_str());
9624
9625 return setError(E_FAIL,
9626 tr("Immutable hard disk '%s' with UUID {%RTuuid} cannot be directly attached to the virtual machine '%s' ('%s')"),
9627 medium->getLocationFull().c_str(),
9628 dev.uuid.raw(),
9629 mUserData->s.strName.c_str(),
9630 mData->m_strConfigFileFull.c_str());
9631 }
9632
9633 if (medium->getType() == MediumType_MultiAttach)
9634 {
9635 if (isSnapshotMachine())
9636 return setError(E_FAIL,
9637 tr("Multi-attach hard disk '%s' with UUID {%RTuuid} cannot be directly attached to snapshot with UUID {%RTuuid} "
9638 "of the virtual machine '%s' ('%s')"),
9639 medium->getLocationFull().c_str(),
9640 dev.uuid.raw(),
9641 puuidSnapshot->raw(),
9642 mUserData->s.strName.c_str(),
9643 mData->m_strConfigFileFull.c_str());
9644
9645 return setError(E_FAIL,
9646 tr("Multi-attach hard disk '%s' with UUID {%RTuuid} cannot be directly attached to the virtual machine '%s' ('%s')"),
9647 medium->getLocationFull().c_str(),
9648 dev.uuid.raw(),
9649 mUserData->s.strName.c_str(),
9650 mData->m_strConfigFileFull.c_str());
9651 }
9652
9653 if ( !isSnapshotMachine()
9654 && medium->getChildren().size() != 0
9655 )
9656 return setError(E_FAIL,
9657 tr("Hard disk '%s' with UUID {%RTuuid} cannot be directly attached to the virtual machine '%s' ('%s') "
9658 "because it has %d differencing child hard disks"),
9659 medium->getLocationFull().c_str(),
9660 dev.uuid.raw(),
9661 mUserData->s.strName.c_str(),
9662 mData->m_strConfigFileFull.c_str(),
9663 medium->getChildren().size());
9664
9665 if (findAttachment(mMediaData->mAttachments,
9666 medium))
9667 return setError(E_FAIL,
9668 tr("Hard disk '%s' with UUID {%RTuuid} is already attached to the virtual machine '%s' ('%s')"),
9669 medium->getLocationFull().c_str(),
9670 dev.uuid.raw(),
9671 mUserData->s.strName.c_str(),
9672 mData->m_strConfigFileFull.c_str());
9673
9674 break;
9675 }
9676
9677 default:
9678 return setError(E_FAIL,
9679 tr("Device '%s' with unknown type is attached to the virtual machine '%s' ('%s')"),
9680 medium->getLocationFull().c_str(),
9681 mUserData->s.strName.c_str(),
9682 mData->m_strConfigFileFull.c_str());
9683 }
9684
9685 if (FAILED(rc))
9686 break;
9687
9688 /* Bandwidth groups are loaded at this point. */
9689 ComObjPtr<BandwidthGroup> pBwGroup;
9690
9691 if (!dev.strBwGroup.isEmpty())
9692 {
9693 rc = mBandwidthControl->getBandwidthGroupByName(dev.strBwGroup, pBwGroup, false /* aSetError */);
9694 if (FAILED(rc))
9695 return setError(E_FAIL,
9696 tr("Device '%s' with unknown bandwidth group '%s' is attached to the virtual machine '%s' ('%s')"),
9697 medium->getLocationFull().c_str(),
9698 dev.strBwGroup.c_str(),
9699 mUserData->s.strName.c_str(),
9700 mData->m_strConfigFileFull.c_str());
9701 pBwGroup->reference();
9702 }
9703
9704 const Bstr controllerName = aStorageController->getName();
9705 ComObjPtr<MediumAttachment> pAttachment;
9706 pAttachment.createObject();
9707 rc = pAttachment->init(this,
9708 medium,
9709 controllerName,
9710 dev.lPort,
9711 dev.lDevice,
9712 dev.deviceType,
9713 false,
9714 dev.fPassThrough,
9715 dev.fTempEject,
9716 dev.fNonRotational,
9717 dev.fDiscard,
9718 pBwGroup.isNull() ? Utf8Str::Empty : pBwGroup->getName());
9719 if (FAILED(rc)) break;
9720
9721 /* associate the medium with this machine and snapshot */
9722 if (!medium.isNull())
9723 {
9724 AutoCaller medCaller(medium);
9725 if (FAILED(medCaller.rc())) return medCaller.rc();
9726 AutoWriteLock mlock(medium COMMA_LOCKVAL_SRC_POS);
9727
9728 if (isSnapshotMachine())
9729 rc = medium->addBackReference(mData->mUuid, *puuidSnapshot);
9730 else
9731 rc = medium->addBackReference(mData->mUuid);
9732 /* If the medium->addBackReference fails it sets an appropriate
9733 * error message, so no need to do any guesswork here. */
9734
9735 if (puuidRegistry)
9736 // caller wants registry ID to be set on all attached media (OVF import case)
9737 medium->addRegistry(*puuidRegistry, false /* fRecurse */);
9738 }
9739
9740 if (FAILED(rc))
9741 break;
9742
9743 /* back up mMediaData to let registeredInit() properly rollback on failure
9744 * (= limited accessibility) */
9745 setModified(IsModified_Storage);
9746 mMediaData.backup();
9747 mMediaData->mAttachments.push_back(pAttachment);
9748 }
9749
9750 return rc;
9751}
9752
9753/**
9754 * Returns the snapshot with the given UUID or fails of no such snapshot exists.
9755 *
9756 * @param aId snapshot UUID to find (empty UUID refers the first snapshot)
9757 * @param aSnapshot where to return the found snapshot
9758 * @param aSetError true to set extended error info on failure
9759 */
9760HRESULT Machine::findSnapshotById(const Guid &aId,
9761 ComObjPtr<Snapshot> &aSnapshot,
9762 bool aSetError /* = false */)
9763{
9764 AutoReadLock chlock(this COMMA_LOCKVAL_SRC_POS);
9765
9766 if (!mData->mFirstSnapshot)
9767 {
9768 if (aSetError)
9769 return setError(E_FAIL, tr("This machine does not have any snapshots"));
9770 return E_FAIL;
9771 }
9772
9773 if (aId.isZero())
9774 aSnapshot = mData->mFirstSnapshot;
9775 else
9776 aSnapshot = mData->mFirstSnapshot->findChildOrSelf(aId.ref());
9777
9778 if (!aSnapshot)
9779 {
9780 if (aSetError)
9781 return setError(E_FAIL,
9782 tr("Could not find a snapshot with UUID {%s}"),
9783 aId.toString().c_str());
9784 return E_FAIL;
9785 }
9786
9787 return S_OK;
9788}
9789
9790/**
9791 * Returns the snapshot with the given name or fails of no such snapshot.
9792 *
9793 * @param aName snapshot name to find
9794 * @param aSnapshot where to return the found snapshot
9795 * @param aSetError true to set extended error info on failure
9796 */
9797HRESULT Machine::findSnapshotByName(const Utf8Str &strName,
9798 ComObjPtr<Snapshot> &aSnapshot,
9799 bool aSetError /* = false */)
9800{
9801 AssertReturn(!strName.isEmpty(), E_INVALIDARG);
9802
9803 AutoReadLock chlock(this COMMA_LOCKVAL_SRC_POS);
9804
9805 if (!mData->mFirstSnapshot)
9806 {
9807 if (aSetError)
9808 return setError(VBOX_E_OBJECT_NOT_FOUND,
9809 tr("This machine does not have any snapshots"));
9810 return VBOX_E_OBJECT_NOT_FOUND;
9811 }
9812
9813 aSnapshot = mData->mFirstSnapshot->findChildOrSelf(strName);
9814
9815 if (!aSnapshot)
9816 {
9817 if (aSetError)
9818 return setError(VBOX_E_OBJECT_NOT_FOUND,
9819 tr("Could not find a snapshot named '%s'"), strName.c_str());
9820 return VBOX_E_OBJECT_NOT_FOUND;
9821 }
9822
9823 return S_OK;
9824}
9825
9826/**
9827 * Returns a storage controller object with the given name.
9828 *
9829 * @param aName storage controller name to find
9830 * @param aStorageController where to return the found storage controller
9831 * @param aSetError true to set extended error info on failure
9832 */
9833HRESULT Machine::getStorageControllerByName(const Utf8Str &aName,
9834 ComObjPtr<StorageController> &aStorageController,
9835 bool aSetError /* = false */)
9836{
9837 AssertReturn(!aName.isEmpty(), E_INVALIDARG);
9838
9839 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
9840 it != mStorageControllers->end();
9841 ++it)
9842 {
9843 if ((*it)->getName() == aName)
9844 {
9845 aStorageController = (*it);
9846 return S_OK;
9847 }
9848 }
9849
9850 if (aSetError)
9851 return setError(VBOX_E_OBJECT_NOT_FOUND,
9852 tr("Could not find a storage controller named '%s'"),
9853 aName.c_str());
9854 return VBOX_E_OBJECT_NOT_FOUND;
9855}
9856
9857/**
9858 * Returns a USB controller object with the given name.
9859 *
9860 * @param aName USB controller name to find
9861 * @param aUSBController where to return the found USB controller
9862 * @param aSetError true to set extended error info on failure
9863 */
9864HRESULT Machine::getUSBControllerByName(const Utf8Str &aName,
9865 ComObjPtr<USBController> &aUSBController,
9866 bool aSetError /* = false */)
9867{
9868 AssertReturn(!aName.isEmpty(), E_INVALIDARG);
9869
9870 for (USBControllerList::const_iterator it = mUSBControllers->begin();
9871 it != mUSBControllers->end();
9872 ++it)
9873 {
9874 if ((*it)->getName() == aName)
9875 {
9876 aUSBController = (*it);
9877 return S_OK;
9878 }
9879 }
9880
9881 if (aSetError)
9882 return setError(VBOX_E_OBJECT_NOT_FOUND,
9883 tr("Could not find a storage controller named '%s'"),
9884 aName.c_str());
9885 return VBOX_E_OBJECT_NOT_FOUND;
9886}
9887
9888/**
9889 * Returns the number of USB controller instance of the given type.
9890 *
9891 * @param enmType USB controller type.
9892 */
9893ULONG Machine::getUSBControllerCountByType(USBControllerType_T enmType)
9894{
9895 ULONG cCtrls = 0;
9896
9897 for (USBControllerList::const_iterator it = mUSBControllers->begin();
9898 it != mUSBControllers->end();
9899 ++it)
9900 {
9901 if ((*it)->getControllerType() == enmType)
9902 cCtrls++;
9903 }
9904
9905 return cCtrls;
9906}
9907
9908HRESULT Machine::getMediumAttachmentsOfController(CBSTR aName,
9909 MediaData::AttachmentList &atts)
9910{
9911 AutoCaller autoCaller(this);
9912 if (FAILED(autoCaller.rc())) return autoCaller.rc();
9913
9914 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
9915
9916 for (MediaData::AttachmentList::iterator it = mMediaData->mAttachments.begin();
9917 it != mMediaData->mAttachments.end();
9918 ++it)
9919 {
9920 const ComObjPtr<MediumAttachment> &pAtt = *it;
9921
9922 // should never happen, but deal with NULL pointers in the list.
9923 AssertStmt(!pAtt.isNull(), continue);
9924
9925 // getControllerName() needs caller+read lock
9926 AutoCaller autoAttCaller(pAtt);
9927 if (FAILED(autoAttCaller.rc()))
9928 {
9929 atts.clear();
9930 return autoAttCaller.rc();
9931 }
9932 AutoReadLock attLock(pAtt COMMA_LOCKVAL_SRC_POS);
9933
9934 if (pAtt->getControllerName() == aName)
9935 atts.push_back(pAtt);
9936 }
9937
9938 return S_OK;
9939}
9940
9941/**
9942 * Helper for #saveSettings. Cares about renaming the settings directory and
9943 * file if the machine name was changed and about creating a new settings file
9944 * if this is a new machine.
9945 *
9946 * @note Must be never called directly but only from #saveSettings().
9947 */
9948HRESULT Machine::prepareSaveSettings(bool *pfNeedsGlobalSaveSettings)
9949{
9950 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
9951
9952 HRESULT rc = S_OK;
9953
9954 bool fSettingsFileIsNew = !mData->pMachineConfigFile->fileExists();
9955
9956 /// @todo need to handle primary group change, too
9957
9958 /* attempt to rename the settings file if machine name is changed */
9959 if ( mUserData->s.fNameSync
9960 && mUserData.isBackedUp()
9961 && ( mUserData.backedUpData()->s.strName != mUserData->s.strName
9962 || mUserData.backedUpData()->s.llGroups.front() != mUserData->s.llGroups.front())
9963 )
9964 {
9965 bool dirRenamed = false;
9966 bool fileRenamed = false;
9967
9968 Utf8Str configFile, newConfigFile;
9969 Utf8Str configFilePrev, newConfigFilePrev;
9970 Utf8Str configDir, newConfigDir;
9971
9972 do
9973 {
9974 int vrc = VINF_SUCCESS;
9975
9976 Utf8Str name = mUserData.backedUpData()->s.strName;
9977 Utf8Str newName = mUserData->s.strName;
9978 Utf8Str group = mUserData.backedUpData()->s.llGroups.front();
9979 if (group == "/")
9980 group.setNull();
9981 Utf8Str newGroup = mUserData->s.llGroups.front();
9982 if (newGroup == "/")
9983 newGroup.setNull();
9984
9985 configFile = mData->m_strConfigFileFull;
9986
9987 /* first, rename the directory if it matches the group and machine name */
9988 Utf8Str groupPlusName = Utf8StrFmt("%s%c%s",
9989 group.c_str(), RTPATH_DELIMITER, name.c_str());
9990 /** @todo hack, make somehow use of ComposeMachineFilename */
9991 if (mUserData->s.fDirectoryIncludesUUID)
9992 groupPlusName += Utf8StrFmt(" (%RTuuid)", mData->mUuid.raw());
9993 Utf8Str newGroupPlusName = Utf8StrFmt("%s%c%s",
9994 newGroup.c_str(), RTPATH_DELIMITER, newName.c_str());
9995 /** @todo hack, make somehow use of ComposeMachineFilename */
9996 if (mUserData->s.fDirectoryIncludesUUID)
9997 newGroupPlusName += Utf8StrFmt(" (%RTuuid)", mData->mUuid.raw());
9998 configDir = configFile;
9999 configDir.stripFilename();
10000 newConfigDir = configDir;
10001 if ( configDir.length() >= groupPlusName.length()
10002 && !RTPathCompare(configDir.substr(configDir.length() - groupPlusName.length(), groupPlusName.length()).c_str(), groupPlusName.c_str()))
10003 {
10004 newConfigDir = newConfigDir.substr(0, configDir.length() - groupPlusName.length());
10005 Utf8Str newConfigBaseDir(newConfigDir);
10006 newConfigDir.append(newGroupPlusName);
10007 /* consistency: use \ if appropriate on the platform */
10008 RTPathChangeToDosSlashes(newConfigDir.mutableRaw(), false);
10009 /* new dir and old dir cannot be equal here because of 'if'
10010 * above and because name != newName */
10011 Assert(configDir != newConfigDir);
10012 if (!fSettingsFileIsNew)
10013 {
10014 /* perform real rename only if the machine is not new */
10015 vrc = RTPathRename(configDir.c_str(), newConfigDir.c_str(), 0);
10016 if ( vrc == VERR_FILE_NOT_FOUND
10017 || vrc == VERR_PATH_NOT_FOUND)
10018 {
10019 /* create the parent directory, then retry renaming */
10020 Utf8Str parent(newConfigDir);
10021 parent.stripFilename();
10022 (void)RTDirCreateFullPath(parent.c_str(), 0700);
10023 vrc = RTPathRename(configDir.c_str(), newConfigDir.c_str(), 0);
10024 }
10025 if (RT_FAILURE(vrc))
10026 {
10027 rc = setError(E_FAIL,
10028 tr("Could not rename the directory '%s' to '%s' to save the settings file (%Rrc)"),
10029 configDir.c_str(),
10030 newConfigDir.c_str(),
10031 vrc);
10032 break;
10033 }
10034 /* delete subdirectories which are no longer needed */
10035 Utf8Str dir(configDir);
10036 dir.stripFilename();
10037 while (dir != newConfigBaseDir && dir != ".")
10038 {
10039 vrc = RTDirRemove(dir.c_str());
10040 if (RT_FAILURE(vrc))
10041 break;
10042 dir.stripFilename();
10043 }
10044 dirRenamed = true;
10045 }
10046 }
10047
10048 newConfigFile = Utf8StrFmt("%s%c%s.vbox",
10049 newConfigDir.c_str(), RTPATH_DELIMITER, newName.c_str());
10050
10051 /* then try to rename the settings file itself */
10052 if (newConfigFile != configFile)
10053 {
10054 /* get the path to old settings file in renamed directory */
10055 configFile = Utf8StrFmt("%s%c%s",
10056 newConfigDir.c_str(),
10057 RTPATH_DELIMITER,
10058 RTPathFilename(configFile.c_str()));
10059 if (!fSettingsFileIsNew)
10060 {
10061 /* perform real rename only if the machine is not new */
10062 vrc = RTFileRename(configFile.c_str(), newConfigFile.c_str(), 0);
10063 if (RT_FAILURE(vrc))
10064 {
10065 rc = setError(E_FAIL,
10066 tr("Could not rename the settings file '%s' to '%s' (%Rrc)"),
10067 configFile.c_str(),
10068 newConfigFile.c_str(),
10069 vrc);
10070 break;
10071 }
10072 fileRenamed = true;
10073 configFilePrev = configFile;
10074 configFilePrev += "-prev";
10075 newConfigFilePrev = newConfigFile;
10076 newConfigFilePrev += "-prev";
10077 RTFileRename(configFilePrev.c_str(), newConfigFilePrev.c_str(), 0);
10078 }
10079 }
10080
10081 // update m_strConfigFileFull amd mConfigFile
10082 mData->m_strConfigFileFull = newConfigFile;
10083 // compute the relative path too
10084 mParent->copyPathRelativeToConfig(newConfigFile, mData->m_strConfigFile);
10085
10086 // store the old and new so that VirtualBox::saveSettings() can update
10087 // the media registry
10088 if ( mData->mRegistered
10089 && configDir != newConfigDir)
10090 {
10091 mParent->rememberMachineNameChangeForMedia(configDir, newConfigDir);
10092
10093 if (pfNeedsGlobalSaveSettings)
10094 *pfNeedsGlobalSaveSettings = true;
10095 }
10096
10097 // in the saved state file path, replace the old directory with the new directory
10098 if (RTPathStartsWith(mSSData->strStateFilePath.c_str(), configDir.c_str()))
10099 mSSData->strStateFilePath = newConfigDir.append(mSSData->strStateFilePath.c_str() + configDir.length());
10100
10101 // and do the same thing for the saved state file paths of all the online snapshots
10102 if (mData->mFirstSnapshot)
10103 mData->mFirstSnapshot->updateSavedStatePaths(configDir.c_str(),
10104 newConfigDir.c_str());
10105 }
10106 while (0);
10107
10108 if (FAILED(rc))
10109 {
10110 /* silently try to rename everything back */
10111 if (fileRenamed)
10112 {
10113 RTFileRename(newConfigFilePrev.c_str(), configFilePrev.c_str(), 0);
10114 RTFileRename(newConfigFile.c_str(), configFile.c_str(), 0);
10115 }
10116 if (dirRenamed)
10117 RTPathRename(newConfigDir.c_str(), configDir.c_str(), 0);
10118 }
10119
10120 if (FAILED(rc)) return rc;
10121 }
10122
10123 if (fSettingsFileIsNew)
10124 {
10125 /* create a virgin config file */
10126 int vrc = VINF_SUCCESS;
10127
10128 /* ensure the settings directory exists */
10129 Utf8Str path(mData->m_strConfigFileFull);
10130 path.stripFilename();
10131 if (!RTDirExists(path.c_str()))
10132 {
10133 vrc = RTDirCreateFullPath(path.c_str(), 0700);
10134 if (RT_FAILURE(vrc))
10135 {
10136 return setError(E_FAIL,
10137 tr("Could not create a directory '%s' to save the settings file (%Rrc)"),
10138 path.c_str(),
10139 vrc);
10140 }
10141 }
10142
10143 /* Note: open flags must correlate with RTFileOpen() in lockConfig() */
10144 path = Utf8Str(mData->m_strConfigFileFull);
10145 RTFILE f = NIL_RTFILE;
10146 vrc = RTFileOpen(&f, path.c_str(),
10147 RTFILE_O_READWRITE | RTFILE_O_CREATE | RTFILE_O_DENY_WRITE);
10148 if (RT_FAILURE(vrc))
10149 return setError(E_FAIL,
10150 tr("Could not create the settings file '%s' (%Rrc)"),
10151 path.c_str(),
10152 vrc);
10153 RTFileClose(f);
10154 }
10155
10156 return rc;
10157}
10158
10159/**
10160 * Saves and commits machine data, user data and hardware data.
10161 *
10162 * Note that on failure, the data remains uncommitted.
10163 *
10164 * @a aFlags may combine the following flags:
10165 *
10166 * - SaveS_ResetCurStateModified: Resets mData->mCurrentStateModified to FALSE.
10167 * Used when saving settings after an operation that makes them 100%
10168 * correspond to the settings from the current snapshot.
10169 * - SaveS_InformCallbacksAnyway: Callbacks will be informed even if
10170 * #isReallyModified() returns false. This is necessary for cases when we
10171 * change machine data directly, not through the backup()/commit() mechanism.
10172 * - SaveS_Force: settings will be saved without doing a deep compare of the
10173 * settings structures. This is used when this is called because snapshots
10174 * have changed to avoid the overhead of the deep compare.
10175 *
10176 * @note Must be called from under this object's write lock. Locks children for
10177 * writing.
10178 *
10179 * @param pfNeedsGlobalSaveSettings Optional pointer to a bool that must have been
10180 * initialized to false and that will be set to true by this function if
10181 * the caller must invoke VirtualBox::saveSettings() because the global
10182 * settings have changed. This will happen if a machine rename has been
10183 * saved and the global machine and media registries will therefore need
10184 * updating.
10185 */
10186HRESULT Machine::saveSettings(bool *pfNeedsGlobalSaveSettings,
10187 int aFlags /*= 0*/)
10188{
10189 LogFlowThisFuncEnter();
10190
10191 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
10192
10193 /* make sure child objects are unable to modify the settings while we are
10194 * saving them */
10195 ensureNoStateDependencies();
10196
10197 AssertReturn(!isSnapshotMachine(),
10198 E_FAIL);
10199
10200 HRESULT rc = S_OK;
10201 bool fNeedsWrite = false;
10202
10203 /* First, prepare to save settings. It will care about renaming the
10204 * settings directory and file if the machine name was changed and about
10205 * creating a new settings file if this is a new machine. */
10206 rc = prepareSaveSettings(pfNeedsGlobalSaveSettings);
10207 if (FAILED(rc)) return rc;
10208
10209 // keep a pointer to the current settings structures
10210 settings::MachineConfigFile *pOldConfig = mData->pMachineConfigFile;
10211 settings::MachineConfigFile *pNewConfig = NULL;
10212
10213 try
10214 {
10215 // make a fresh one to have everyone write stuff into
10216 pNewConfig = new settings::MachineConfigFile(NULL);
10217 pNewConfig->copyBaseFrom(*mData->pMachineConfigFile);
10218
10219 // now go and copy all the settings data from COM to the settings structures
10220 // (this calles saveSettings() on all the COM objects in the machine)
10221 copyMachineDataToSettings(*pNewConfig);
10222
10223 if (aFlags & SaveS_ResetCurStateModified)
10224 {
10225 // this gets set by takeSnapshot() (if offline snapshot) and restoreSnapshot()
10226 mData->mCurrentStateModified = FALSE;
10227 fNeedsWrite = true; // always, no need to compare
10228 }
10229 else if (aFlags & SaveS_Force)
10230 {
10231 fNeedsWrite = true; // always, no need to compare
10232 }
10233 else
10234 {
10235 if (!mData->mCurrentStateModified)
10236 {
10237 // do a deep compare of the settings that we just saved with the settings
10238 // previously stored in the config file; this invokes MachineConfigFile::operator==
10239 // which does a deep compare of all the settings, which is expensive but less expensive
10240 // than writing out XML in vain
10241 bool fAnySettingsChanged = !(*pNewConfig == *pOldConfig);
10242
10243 // could still be modified if any settings changed
10244 mData->mCurrentStateModified = fAnySettingsChanged;
10245
10246 fNeedsWrite = fAnySettingsChanged;
10247 }
10248 else
10249 fNeedsWrite = true;
10250 }
10251
10252 pNewConfig->fCurrentStateModified = !!mData->mCurrentStateModified;
10253
10254 if (fNeedsWrite)
10255 // now spit it all out!
10256 pNewConfig->write(mData->m_strConfigFileFull);
10257
10258 mData->pMachineConfigFile = pNewConfig;
10259 delete pOldConfig;
10260 commit();
10261
10262 // after saving settings, we are no longer different from the XML on disk
10263 mData->flModifications = 0;
10264 }
10265 catch (HRESULT err)
10266 {
10267 // we assume that error info is set by the thrower
10268 rc = err;
10269
10270 // restore old config
10271 delete pNewConfig;
10272 mData->pMachineConfigFile = pOldConfig;
10273 }
10274 catch (...)
10275 {
10276 rc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
10277 }
10278
10279 if (fNeedsWrite || (aFlags & SaveS_InformCallbacksAnyway))
10280 {
10281 /* Fire the data change event, even on failure (since we've already
10282 * committed all data). This is done only for SessionMachines because
10283 * mutable Machine instances are always not registered (i.e. private
10284 * to the client process that creates them) and thus don't need to
10285 * inform callbacks. */
10286 if (isSessionMachine())
10287 mParent->onMachineDataChange(mData->mUuid);
10288 }
10289
10290 LogFlowThisFunc(("rc=%08X\n", rc));
10291 LogFlowThisFuncLeave();
10292 return rc;
10293}
10294
10295/**
10296 * Implementation for saving the machine settings into the given
10297 * settings::MachineConfigFile instance. This copies machine extradata
10298 * from the previous machine config file in the instance data, if any.
10299 *
10300 * This gets called from two locations:
10301 *
10302 * -- Machine::saveSettings(), during the regular XML writing;
10303 *
10304 * -- Appliance::buildXMLForOneVirtualSystem(), when a machine gets
10305 * exported to OVF and we write the VirtualBox proprietary XML
10306 * into a <vbox:Machine> tag.
10307 *
10308 * This routine fills all the fields in there, including snapshots, *except*
10309 * for the following:
10310 *
10311 * -- fCurrentStateModified. There is some special logic associated with that.
10312 *
10313 * The caller can then call MachineConfigFile::write() or do something else
10314 * with it.
10315 *
10316 * Caller must hold the machine lock!
10317 *
10318 * This throws XML errors and HRESULT, so the caller must have a catch block!
10319 */
10320void Machine::copyMachineDataToSettings(settings::MachineConfigFile &config)
10321{
10322 // deep copy extradata
10323 config.mapExtraDataItems = mData->pMachineConfigFile->mapExtraDataItems;
10324
10325 config.uuid = mData->mUuid;
10326
10327 // copy name, description, OS type, teleport, UTC etc.
10328 config.machineUserData = mUserData->s;
10329
10330 // Encode the Icon Override data from Machine and store on config userdata.
10331 com::SafeArray<BYTE> iconByte;
10332 COMGETTER(Icon)(ComSafeArrayAsOutParam(iconByte));
10333 ssize_t cbData = iconByte.size();
10334 if (cbData > 0)
10335 {
10336 ssize_t cchOut = RTBase64EncodedLength(cbData);
10337 Utf8Str strIconData;
10338 strIconData.reserve(cchOut+1);
10339 int vrc = RTBase64Encode(iconByte.raw(), cbData,
10340 strIconData.mutableRaw(), strIconData.capacity(),
10341 NULL);
10342 if (RT_FAILURE(vrc))
10343 throw setError(E_FAIL, tr("Failure to Encode Icon Data. '%s' (%Rrc)"), strIconData.mutableRaw(), vrc);
10344 strIconData.jolt();
10345 config.machineUserData.ovIcon = strIconData;
10346 }
10347 else
10348 config.machineUserData.ovIcon.setNull();
10349
10350 if ( mData->mMachineState == MachineState_Saved
10351 || mData->mMachineState == MachineState_Restoring
10352 // when deleting a snapshot we may or may not have a saved state in the current state,
10353 // so let's not assert here please
10354 || ( ( mData->mMachineState == MachineState_DeletingSnapshot
10355 || mData->mMachineState == MachineState_DeletingSnapshotOnline
10356 || mData->mMachineState == MachineState_DeletingSnapshotPaused)
10357 && (!mSSData->strStateFilePath.isEmpty())
10358 )
10359 )
10360 {
10361 Assert(!mSSData->strStateFilePath.isEmpty());
10362 /* try to make the file name relative to the settings file dir */
10363 copyPathRelativeToMachine(mSSData->strStateFilePath, config.strStateFile);
10364 }
10365 else
10366 {
10367 Assert(mSSData->strStateFilePath.isEmpty() || mData->mMachineState == MachineState_Saving);
10368 config.strStateFile.setNull();
10369 }
10370
10371 if (mData->mCurrentSnapshot)
10372 config.uuidCurrentSnapshot = mData->mCurrentSnapshot->getId();
10373 else
10374 config.uuidCurrentSnapshot.clear();
10375
10376 config.timeLastStateChange = mData->mLastStateChange;
10377 config.fAborted = (mData->mMachineState == MachineState_Aborted);
10378 /// @todo Live Migration: config.fTeleported = (mData->mMachineState == MachineState_Teleported);
10379
10380 HRESULT rc = saveHardware(config.hardwareMachine, &config.debugging, &config.autostart);
10381 if (FAILED(rc)) throw rc;
10382
10383 rc = saveStorageControllers(config.storageMachine);
10384 if (FAILED(rc)) throw rc;
10385
10386 // save machine's media registry if this is VirtualBox 4.0 or later
10387 if (config.canHaveOwnMediaRegistry())
10388 {
10389 // determine machine folder
10390 Utf8Str strMachineFolder = getSettingsFileFull();
10391 strMachineFolder.stripFilename();
10392 mParent->saveMediaRegistry(config.mediaRegistry,
10393 getId(), // only media with registry ID == machine UUID
10394 strMachineFolder);
10395 // this throws HRESULT
10396 }
10397
10398 // save snapshots
10399 rc = saveAllSnapshots(config);
10400 if (FAILED(rc)) throw rc;
10401}
10402
10403/**
10404 * Saves all snapshots of the machine into the given machine config file. Called
10405 * from Machine::buildMachineXML() and SessionMachine::deleteSnapshotHandler().
10406 * @param config
10407 * @return
10408 */
10409HRESULT Machine::saveAllSnapshots(settings::MachineConfigFile &config)
10410{
10411 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
10412
10413 HRESULT rc = S_OK;
10414
10415 try
10416 {
10417 config.llFirstSnapshot.clear();
10418
10419 if (mData->mFirstSnapshot)
10420 {
10421 settings::Snapshot snapNew;
10422 config.llFirstSnapshot.push_back(snapNew);
10423
10424 // get reference to the fresh copy of the snapshot on the list and
10425 // work on that copy directly to avoid excessive copying later
10426 settings::Snapshot &snap = config.llFirstSnapshot.front();
10427
10428 rc = mData->mFirstSnapshot->saveSnapshot(snap, false /*aAttrsOnly*/);
10429 if (FAILED(rc)) throw rc;
10430 }
10431
10432// if (mType == IsSessionMachine)
10433// mParent->onMachineDataChange(mData->mUuid); @todo is this necessary?
10434
10435 }
10436 catch (HRESULT err)
10437 {
10438 /* we assume that error info is set by the thrower */
10439 rc = err;
10440 }
10441 catch (...)
10442 {
10443 rc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
10444 }
10445
10446 return rc;
10447}
10448
10449/**
10450 * Saves the VM hardware configuration. It is assumed that the
10451 * given node is empty.
10452 *
10453 * @param data Reference to the settings object for the hardware config.
10454 * @param pDbg Pointer to the settings object for the debugging config
10455 * which happens to live in mHWData.
10456 * @param pAutostart Pointer to the settings object for the autostart config
10457 * which happens to live in mHWData.
10458 */
10459HRESULT Machine::saveHardware(settings::Hardware &data, settings::Debugging *pDbg,
10460 settings::Autostart *pAutostart)
10461{
10462 HRESULT rc = S_OK;
10463
10464 try
10465 {
10466 /* The hardware version attribute (optional).
10467 Automatically upgrade from 1 to 2 when there is no saved state. (ugly!) */
10468 if ( mHWData->mHWVersion == "1"
10469 && mSSData->strStateFilePath.isEmpty()
10470 )
10471 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. */
10472
10473 data.strVersion = mHWData->mHWVersion;
10474 data.uuid = mHWData->mHardwareUUID;
10475
10476 // CPU
10477 data.fHardwareVirt = !!mHWData->mHWVirtExEnabled;
10478 data.fNestedPaging = !!mHWData->mHWVirtExNestedPagingEnabled;
10479 data.fLargePages = !!mHWData->mHWVirtExLargePagesEnabled;
10480 data.fVPID = !!mHWData->mHWVirtExVPIDEnabled;
10481 data.fUnrestrictedExecution = !!mHWData->mHWVirtExUXEnabled;
10482 data.fHardwareVirtForce = !!mHWData->mHWVirtExForceEnabled;
10483 data.fPAE = !!mHWData->mPAEEnabled;
10484 data.enmLongMode = mHWData->mLongMode;
10485 data.fSyntheticCpu = !!mHWData->mSyntheticCpu;
10486
10487 /* Standard and Extended CPUID leafs. */
10488 data.llCpuIdLeafs.clear();
10489 for (unsigned idx = 0; idx < RT_ELEMENTS(mHWData->mCpuIdStdLeafs); idx++)
10490 {
10491 if (mHWData->mCpuIdStdLeafs[idx].ulId != UINT32_MAX)
10492 data.llCpuIdLeafs.push_back(mHWData->mCpuIdStdLeafs[idx]);
10493 }
10494 for (unsigned idx = 0; idx < RT_ELEMENTS(mHWData->mCpuIdExtLeafs); idx++)
10495 {
10496 if (mHWData->mCpuIdExtLeafs[idx].ulId != UINT32_MAX)
10497 data.llCpuIdLeafs.push_back(mHWData->mCpuIdExtLeafs[idx]);
10498 }
10499
10500 data.cCPUs = mHWData->mCPUCount;
10501 data.fCpuHotPlug = !!mHWData->mCPUHotPlugEnabled;
10502 data.ulCpuExecutionCap = mHWData->mCpuExecutionCap;
10503
10504 data.llCpus.clear();
10505 if (data.fCpuHotPlug)
10506 {
10507 for (unsigned idx = 0; idx < data.cCPUs; idx++)
10508 {
10509 if (mHWData->mCPUAttached[idx])
10510 {
10511 settings::Cpu cpu;
10512 cpu.ulId = idx;
10513 data.llCpus.push_back(cpu);
10514 }
10515 }
10516 }
10517
10518 // memory
10519 data.ulMemorySizeMB = mHWData->mMemorySize;
10520 data.fPageFusionEnabled = !!mHWData->mPageFusionEnabled;
10521
10522 // firmware
10523 data.firmwareType = mHWData->mFirmwareType;
10524
10525 // HID
10526 data.pointingHIDType = mHWData->mPointingHIDType;
10527 data.keyboardHIDType = mHWData->mKeyboardHIDType;
10528
10529 // chipset
10530 data.chipsetType = mHWData->mChipsetType;
10531
10532 data.fEmulatedUSBCardReader = !!mHWData->mEmulatedUSBCardReaderEnabled;
10533
10534 // HPET
10535 data.fHPETEnabled = !!mHWData->mHPETEnabled;
10536
10537 // boot order
10538 data.mapBootOrder.clear();
10539 for (size_t i = 0;
10540 i < RT_ELEMENTS(mHWData->mBootOrder);
10541 ++i)
10542 data.mapBootOrder[i] = mHWData->mBootOrder[i];
10543
10544 // display
10545 data.graphicsControllerType = mHWData->mGraphicsControllerType;
10546 data.ulVRAMSizeMB = mHWData->mVRAMSize;
10547 data.cMonitors = mHWData->mMonitorCount;
10548 data.fAccelerate3D = !!mHWData->mAccelerate3DEnabled;
10549 data.fAccelerate2DVideo = !!mHWData->mAccelerate2DVideoEnabled;
10550 data.ulVideoCaptureHorzRes = mHWData->mVideoCaptureWidth;
10551 data.ulVideoCaptureVertRes = mHWData->mVideoCaptureHeight;
10552 data.ulVideoCaptureRate = mHWData->mVideoCaptureRate;
10553 data.ulVideoCaptureFPS = mHWData->mVideoCaptureFPS;
10554 data.fVideoCaptureEnabled = !!mHWData->mVideoCaptureEnabled;
10555 for (unsigned i = 0; i < sizeof(data.u64VideoCaptureScreens) * 8; i++)
10556 {
10557 if (mHWData->maVideoCaptureScreens[i])
10558 ASMBitSet(&data.u64VideoCaptureScreens, i);
10559 else
10560 ASMBitClear(&data.u64VideoCaptureScreens, i);
10561 }
10562 /* store relative video capture file if possible */
10563 copyPathRelativeToMachine(mHWData->mVideoCaptureFile, data.strVideoCaptureFile);
10564
10565 /* VRDEServer settings (optional) */
10566 rc = mVRDEServer->saveSettings(data.vrdeSettings);
10567 if (FAILED(rc)) throw rc;
10568
10569 /* BIOS (required) */
10570 rc = mBIOSSettings->saveSettings(data.biosSettings);
10571 if (FAILED(rc)) throw rc;
10572
10573 /* USB Controller (required) */
10574 for (USBControllerList::const_iterator it = mUSBControllers->begin();
10575 it != mUSBControllers->end();
10576 ++it)
10577 {
10578 ComObjPtr<USBController> ctrl = *it;
10579 settings::USBController settingsCtrl;
10580
10581 settingsCtrl.strName = ctrl->getName();
10582 settingsCtrl.enmType = ctrl->getControllerType();
10583
10584 data.usbSettings.llUSBControllers.push_back(settingsCtrl);
10585 }
10586
10587 /* USB device filters (required) */
10588 rc = mUSBDeviceFilters->saveSettings(data.usbSettings);
10589 if (FAILED(rc)) throw rc;
10590
10591 /* Network adapters (required) */
10592 uint32_t uMaxNICs = RT_MIN(Global::getMaxNetworkAdapters(mHWData->mChipsetType), mNetworkAdapters.size());
10593 data.llNetworkAdapters.clear();
10594 /* Write out only the nominal number of network adapters for this
10595 * chipset type. Since Machine::commit() hasn't been called there
10596 * may be extra NIC settings in the vector. */
10597 for (ULONG slot = 0; slot < uMaxNICs; ++slot)
10598 {
10599 settings::NetworkAdapter nic;
10600 nic.ulSlot = slot;
10601 /* paranoia check... must not be NULL, but must not crash either. */
10602 if (mNetworkAdapters[slot])
10603 {
10604 rc = mNetworkAdapters[slot]->saveSettings(nic);
10605 if (FAILED(rc)) throw rc;
10606
10607 data.llNetworkAdapters.push_back(nic);
10608 }
10609 }
10610
10611 /* Serial ports */
10612 data.llSerialPorts.clear();
10613 for (ULONG slot = 0;
10614 slot < RT_ELEMENTS(mSerialPorts);
10615 ++slot)
10616 {
10617 settings::SerialPort s;
10618 s.ulSlot = slot;
10619 rc = mSerialPorts[slot]->saveSettings(s);
10620 if (FAILED(rc)) return rc;
10621
10622 data.llSerialPorts.push_back(s);
10623 }
10624
10625 /* Parallel ports */
10626 data.llParallelPorts.clear();
10627 for (ULONG slot = 0;
10628 slot < RT_ELEMENTS(mParallelPorts);
10629 ++slot)
10630 {
10631 settings::ParallelPort p;
10632 p.ulSlot = slot;
10633 rc = mParallelPorts[slot]->saveSettings(p);
10634 if (FAILED(rc)) return rc;
10635
10636 data.llParallelPorts.push_back(p);
10637 }
10638
10639 /* Audio adapter */
10640 rc = mAudioAdapter->saveSettings(data.audioAdapter);
10641 if (FAILED(rc)) return rc;
10642
10643 /* Shared folders */
10644 data.llSharedFolders.clear();
10645 for (HWData::SharedFolderList::const_iterator it = mHWData->mSharedFolders.begin();
10646 it != mHWData->mSharedFolders.end();
10647 ++it)
10648 {
10649 SharedFolder *pSF = *it;
10650 AutoCaller sfCaller(pSF);
10651 AutoReadLock sfLock(pSF COMMA_LOCKVAL_SRC_POS);
10652 settings::SharedFolder sf;
10653 sf.strName = pSF->getName();
10654 sf.strHostPath = pSF->getHostPath();
10655 sf.fWritable = !!pSF->isWritable();
10656 sf.fAutoMount = !!pSF->isAutoMounted();
10657
10658 data.llSharedFolders.push_back(sf);
10659 }
10660
10661 // clipboard
10662 data.clipboardMode = mHWData->mClipboardMode;
10663
10664 // drag'n'drop
10665 data.dragAndDropMode = mHWData->mDragAndDropMode;
10666
10667 /* Guest */
10668 data.ulMemoryBalloonSize = mHWData->mMemoryBalloonSize;
10669
10670 // IO settings
10671 data.ioSettings.fIOCacheEnabled = !!mHWData->mIOCacheEnabled;
10672 data.ioSettings.ulIOCacheSize = mHWData->mIOCacheSize;
10673
10674 /* BandwidthControl (required) */
10675 rc = mBandwidthControl->saveSettings(data.ioSettings);
10676 if (FAILED(rc)) throw rc;
10677
10678 /* Host PCI devices */
10679 for (HWData::PCIDeviceAssignmentList::const_iterator it = mHWData->mPCIDeviceAssignments.begin();
10680 it != mHWData->mPCIDeviceAssignments.end();
10681 ++it)
10682 {
10683 ComObjPtr<PCIDeviceAttachment> pda = *it;
10684 settings::HostPCIDeviceAttachment hpda;
10685
10686 rc = pda->saveSettings(hpda);
10687 if (FAILED(rc)) throw rc;
10688
10689 data.pciAttachments.push_back(hpda);
10690 }
10691
10692
10693 // guest properties
10694 data.llGuestProperties.clear();
10695#ifdef VBOX_WITH_GUEST_PROPS
10696 for (HWData::GuestPropertyMap::const_iterator it = mHWData->mGuestProperties.begin();
10697 it != mHWData->mGuestProperties.end();
10698 ++it)
10699 {
10700 HWData::GuestProperty property = it->second;
10701
10702 /* Remove transient guest properties at shutdown unless we
10703 * are saving state */
10704 if ( ( mData->mMachineState == MachineState_PoweredOff
10705 || mData->mMachineState == MachineState_Aborted
10706 || mData->mMachineState == MachineState_Teleported)
10707 && ( property.mFlags & guestProp::TRANSIENT
10708 || property.mFlags & guestProp::TRANSRESET))
10709 continue;
10710 settings::GuestProperty prop;
10711 prop.strName = it->first;
10712 prop.strValue = property.strValue;
10713 prop.timestamp = property.mTimestamp;
10714 char szFlags[guestProp::MAX_FLAGS_LEN + 1];
10715 guestProp::writeFlags(property.mFlags, szFlags);
10716 prop.strFlags = szFlags;
10717
10718 data.llGuestProperties.push_back(prop);
10719 }
10720
10721 data.strNotificationPatterns = mHWData->mGuestPropertyNotificationPatterns;
10722 /* I presume this doesn't require a backup(). */
10723 mData->mGuestPropertiesModified = FALSE;
10724#endif /* VBOX_WITH_GUEST_PROPS defined */
10725
10726 *pDbg = mHWData->mDebugging;
10727 *pAutostart = mHWData->mAutostart;
10728
10729 data.strDefaultFrontend = mHWData->mDefaultFrontend;
10730 }
10731 catch(std::bad_alloc &)
10732 {
10733 return E_OUTOFMEMORY;
10734 }
10735
10736 AssertComRC(rc);
10737 return rc;
10738}
10739
10740/**
10741 * Saves the storage controller configuration.
10742 *
10743 * @param aNode <StorageControllers> node to save the VM hardware configuration to.
10744 */
10745HRESULT Machine::saveStorageControllers(settings::Storage &data)
10746{
10747 data.llStorageControllers.clear();
10748
10749 for (StorageControllerList::const_iterator it = mStorageControllers->begin();
10750 it != mStorageControllers->end();
10751 ++it)
10752 {
10753 HRESULT rc;
10754 ComObjPtr<StorageController> pCtl = *it;
10755
10756 settings::StorageController ctl;
10757 ctl.strName = pCtl->getName();
10758 ctl.controllerType = pCtl->getControllerType();
10759 ctl.storageBus = pCtl->getStorageBus();
10760 ctl.ulInstance = pCtl->getInstance();
10761 ctl.fBootable = pCtl->getBootable();
10762
10763 /* Save the port count. */
10764 ULONG portCount;
10765 rc = pCtl->COMGETTER(PortCount)(&portCount);
10766 ComAssertComRCRet(rc, rc);
10767 ctl.ulPortCount = portCount;
10768
10769 /* Save fUseHostIOCache */
10770 BOOL fUseHostIOCache;
10771 rc = pCtl->COMGETTER(UseHostIOCache)(&fUseHostIOCache);
10772 ComAssertComRCRet(rc, rc);
10773 ctl.fUseHostIOCache = !!fUseHostIOCache;
10774
10775 /* Save IDE emulation settings. */
10776 if (ctl.controllerType == StorageControllerType_IntelAhci)
10777 {
10778 if ( (FAILED(rc = pCtl->getIDEEmulationPort(0, (LONG*)&ctl.lIDE0MasterEmulationPort)))
10779 || (FAILED(rc = pCtl->getIDEEmulationPort(1, (LONG*)&ctl.lIDE0SlaveEmulationPort)))
10780 || (FAILED(rc = pCtl->getIDEEmulationPort(2, (LONG*)&ctl.lIDE1MasterEmulationPort)))
10781 || (FAILED(rc = pCtl->getIDEEmulationPort(3, (LONG*)&ctl.lIDE1SlaveEmulationPort)))
10782 )
10783 ComAssertComRCRet(rc, rc);
10784 }
10785
10786 /* save the devices now. */
10787 rc = saveStorageDevices(pCtl, ctl);
10788 ComAssertComRCRet(rc, rc);
10789
10790 data.llStorageControllers.push_back(ctl);
10791 }
10792
10793 return S_OK;
10794}
10795
10796/**
10797 * Saves the hard disk configuration.
10798 */
10799HRESULT Machine::saveStorageDevices(ComObjPtr<StorageController> aStorageController,
10800 settings::StorageController &data)
10801{
10802 MediaData::AttachmentList atts;
10803
10804 HRESULT rc = getMediumAttachmentsOfController(Bstr(aStorageController->getName()).raw(), atts);
10805 if (FAILED(rc)) return rc;
10806
10807 data.llAttachedDevices.clear();
10808 for (MediaData::AttachmentList::const_iterator it = atts.begin();
10809 it != atts.end();
10810 ++it)
10811 {
10812 settings::AttachedDevice dev;
10813
10814 MediumAttachment *pAttach = *it;
10815 Medium *pMedium = pAttach->getMedium();
10816
10817 dev.deviceType = pAttach->getType();
10818 dev.lPort = pAttach->getPort();
10819 dev.lDevice = pAttach->getDevice();
10820 if (pMedium)
10821 {
10822 if (pMedium->isHostDrive())
10823 dev.strHostDriveSrc = pMedium->getLocationFull();
10824 else
10825 dev.uuid = pMedium->getId();
10826 dev.fPassThrough = pAttach->getPassthrough();
10827 dev.fTempEject = pAttach->getTempEject();
10828 dev.fNonRotational = pAttach->getNonRotational();
10829 dev.fDiscard = pAttach->getDiscard();
10830 }
10831
10832 dev.strBwGroup = pAttach->getBandwidthGroup();
10833
10834 data.llAttachedDevices.push_back(dev);
10835 }
10836
10837 return S_OK;
10838}
10839
10840/**
10841 * Saves machine state settings as defined by aFlags
10842 * (SaveSTS_* values).
10843 *
10844 * @param aFlags Combination of SaveSTS_* flags.
10845 *
10846 * @note Locks objects for writing.
10847 */
10848HRESULT Machine::saveStateSettings(int aFlags)
10849{
10850 if (aFlags == 0)
10851 return S_OK;
10852
10853 AutoCaller autoCaller(this);
10854 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
10855
10856 /* This object's write lock is also necessary to serialize file access
10857 * (prevent concurrent reads and writes) */
10858 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
10859
10860 HRESULT rc = S_OK;
10861
10862 Assert(mData->pMachineConfigFile);
10863
10864 try
10865 {
10866 if (aFlags & SaveSTS_CurStateModified)
10867 mData->pMachineConfigFile->fCurrentStateModified = true;
10868
10869 if (aFlags & SaveSTS_StateFilePath)
10870 {
10871 if (!mSSData->strStateFilePath.isEmpty())
10872 /* try to make the file name relative to the settings file dir */
10873 copyPathRelativeToMachine(mSSData->strStateFilePath, mData->pMachineConfigFile->strStateFile);
10874 else
10875 mData->pMachineConfigFile->strStateFile.setNull();
10876 }
10877
10878 if (aFlags & SaveSTS_StateTimeStamp)
10879 {
10880 Assert( mData->mMachineState != MachineState_Aborted
10881 || mSSData->strStateFilePath.isEmpty());
10882
10883 mData->pMachineConfigFile->timeLastStateChange = mData->mLastStateChange;
10884
10885 mData->pMachineConfigFile->fAborted = (mData->mMachineState == MachineState_Aborted);
10886//@todo live migration mData->pMachineConfigFile->fTeleported = (mData->mMachineState == MachineState_Teleported);
10887 }
10888
10889 mData->pMachineConfigFile->write(mData->m_strConfigFileFull);
10890 }
10891 catch (...)
10892 {
10893 rc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
10894 }
10895
10896 return rc;
10897}
10898
10899/**
10900 * Ensures that the given medium is added to a media registry. If this machine
10901 * was created with 4.0 or later, then the machine registry is used. Otherwise
10902 * the global VirtualBox media registry is used.
10903 *
10904 * Caller must NOT hold machine lock, media tree or any medium locks!
10905 *
10906 * @param pMedium
10907 */
10908void Machine::addMediumToRegistry(ComObjPtr<Medium> &pMedium)
10909{
10910 /* Paranoia checks: do not hold machine or media tree locks. */
10911 AssertReturnVoid(!isWriteLockOnCurrentThread());
10912 AssertReturnVoid(!mParent->getMediaTreeLockHandle().isWriteLockOnCurrentThread());
10913
10914 ComObjPtr<Medium> pBase;
10915 {
10916 AutoReadLock treeLock(&mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
10917 pBase = pMedium->getBase();
10918 }
10919
10920 /* Paranoia checks: do not hold medium locks. */
10921 AssertReturnVoid(!pMedium->isWriteLockOnCurrentThread());
10922 AssertReturnVoid(!pBase->isWriteLockOnCurrentThread());
10923
10924 // decide which medium registry to use now that the medium is attached:
10925 Guid uuid;
10926 if (mData->pMachineConfigFile->canHaveOwnMediaRegistry())
10927 // machine XML is VirtualBox 4.0 or higher:
10928 uuid = getId(); // machine UUID
10929 else
10930 uuid = mParent->getGlobalRegistryId(); // VirtualBox global registry UUID
10931
10932 if (pMedium->addRegistry(uuid, false /* fRecurse */))
10933 mParent->markRegistryModified(uuid);
10934
10935 /* For more complex hard disk structures it can happen that the base
10936 * medium isn't yet associated with any medium registry. Do that now. */
10937 if (pMedium != pBase)
10938 {
10939 if (pBase->addRegistry(uuid, true /* fRecurse */))
10940 mParent->markRegistryModified(uuid);
10941 }
10942}
10943
10944/**
10945 * Creates differencing hard disks for all normal hard disks attached to this
10946 * machine and a new set of attachments to refer to created disks.
10947 *
10948 * Used when taking a snapshot or when deleting the current state. Gets called
10949 * from SessionMachine::BeginTakingSnapshot() and SessionMachine::restoreSnapshotHandler().
10950 *
10951 * This method assumes that mMediaData contains the original hard disk attachments
10952 * it needs to create diffs for. On success, these attachments will be replaced
10953 * with the created diffs. On failure, #deleteImplicitDiffs() is implicitly
10954 * called to delete created diffs which will also rollback mMediaData and restore
10955 * whatever was backed up before calling this method.
10956 *
10957 * Attachments with non-normal hard disks are left as is.
10958 *
10959 * If @a aOnline is @c false then the original hard disks that require implicit
10960 * diffs will be locked for reading. Otherwise it is assumed that they are
10961 * already locked for writing (when the VM was started). Note that in the latter
10962 * case it is responsibility of the caller to lock the newly created diffs for
10963 * writing if this method succeeds.
10964 *
10965 * @param aProgress Progress object to run (must contain at least as
10966 * many operations left as the number of hard disks
10967 * attached).
10968 * @param aOnline Whether the VM was online prior to this operation.
10969 *
10970 * @note The progress object is not marked as completed, neither on success nor
10971 * on failure. This is a responsibility of the caller.
10972 *
10973 * @note Locks this object and the media tree for writing.
10974 */
10975HRESULT Machine::createImplicitDiffs(IProgress *aProgress,
10976 ULONG aWeight,
10977 bool aOnline)
10978{
10979 LogFlowThisFunc(("aOnline=%d\n", aOnline));
10980
10981 AutoCaller autoCaller(this);
10982 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
10983
10984 AutoMultiWriteLock2 alock(this->lockHandle(),
10985 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
10986
10987 /* must be in a protective state because we release the lock below */
10988 AssertReturn( mData->mMachineState == MachineState_Saving
10989 || mData->mMachineState == MachineState_LiveSnapshotting
10990 || mData->mMachineState == MachineState_RestoringSnapshot
10991 || mData->mMachineState == MachineState_DeletingSnapshot
10992 , E_FAIL);
10993
10994 HRESULT rc = S_OK;
10995
10996 // use appropriate locked media map (online or offline)
10997 MediumLockListMap lockedMediaOffline;
10998 MediumLockListMap *lockedMediaMap;
10999 if (aOnline)
11000 lockedMediaMap = &mData->mSession.mLockedMedia;
11001 else
11002 lockedMediaMap = &lockedMediaOffline;
11003
11004 try
11005 {
11006 if (!aOnline)
11007 {
11008 /* lock all attached hard disks early to detect "in use"
11009 * situations before creating actual diffs */
11010 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11011 it != mMediaData->mAttachments.end();
11012 ++it)
11013 {
11014 MediumAttachment* pAtt = *it;
11015 if (pAtt->getType() == DeviceType_HardDisk)
11016 {
11017 Medium* pMedium = pAtt->getMedium();
11018 Assert(pMedium);
11019
11020 MediumLockList *pMediumLockList(new MediumLockList());
11021 alock.release();
11022 rc = pMedium->createMediumLockList(true /* fFailIfInaccessible */,
11023 false /* fMediumLockWrite */,
11024 NULL,
11025 *pMediumLockList);
11026 alock.acquire();
11027 if (FAILED(rc))
11028 {
11029 delete pMediumLockList;
11030 throw rc;
11031 }
11032 rc = lockedMediaMap->Insert(pAtt, pMediumLockList);
11033 if (FAILED(rc))
11034 {
11035 throw setError(rc,
11036 tr("Collecting locking information for all attached media failed"));
11037 }
11038 }
11039 }
11040
11041 /* Now lock all media. If this fails, nothing is locked. */
11042 alock.release();
11043 rc = lockedMediaMap->Lock();
11044 alock.acquire();
11045 if (FAILED(rc))
11046 {
11047 throw setError(rc,
11048 tr("Locking of attached media failed"));
11049 }
11050 }
11051
11052 /* remember the current list (note that we don't use backup() since
11053 * mMediaData may be already backed up) */
11054 MediaData::AttachmentList atts = mMediaData->mAttachments;
11055
11056 /* start from scratch */
11057 mMediaData->mAttachments.clear();
11058
11059 /* go through remembered attachments and create diffs for normal hard
11060 * disks and attach them */
11061 for (MediaData::AttachmentList::const_iterator it = atts.begin();
11062 it != atts.end();
11063 ++it)
11064 {
11065 MediumAttachment* pAtt = *it;
11066
11067 DeviceType_T devType = pAtt->getType();
11068 Medium* pMedium = pAtt->getMedium();
11069
11070 if ( devType != DeviceType_HardDisk
11071 || pMedium == NULL
11072 || pMedium->getType() != MediumType_Normal)
11073 {
11074 /* copy the attachment as is */
11075
11076 /** @todo the progress object created in Console::TakeSnaphot
11077 * only expects operations for hard disks. Later other
11078 * device types need to show up in the progress as well. */
11079 if (devType == DeviceType_HardDisk)
11080 {
11081 if (pMedium == NULL)
11082 aProgress->SetNextOperation(Bstr(tr("Skipping attachment without medium")).raw(),
11083 aWeight); // weight
11084 else
11085 aProgress->SetNextOperation(BstrFmt(tr("Skipping medium '%s'"),
11086 pMedium->getBase()->getName().c_str()).raw(),
11087 aWeight); // weight
11088 }
11089
11090 mMediaData->mAttachments.push_back(pAtt);
11091 continue;
11092 }
11093
11094 /* need a diff */
11095 aProgress->SetNextOperation(BstrFmt(tr("Creating differencing hard disk for '%s'"),
11096 pMedium->getBase()->getName().c_str()).raw(),
11097 aWeight); // weight
11098
11099 Utf8Str strFullSnapshotFolder;
11100 calculateFullPath(mUserData->s.strSnapshotFolder, strFullSnapshotFolder);
11101
11102 ComObjPtr<Medium> diff;
11103 diff.createObject();
11104 // store the diff in the same registry as the parent
11105 // (this cannot fail here because we can't create implicit diffs for
11106 // unregistered images)
11107 Guid uuidRegistryParent;
11108 bool fInRegistry = pMedium->getFirstRegistryMachineId(uuidRegistryParent);
11109 Assert(fInRegistry); NOREF(fInRegistry);
11110 rc = diff->init(mParent,
11111 pMedium->getPreferredDiffFormat(),
11112 strFullSnapshotFolder.append(RTPATH_SLASH_STR),
11113 uuidRegistryParent);
11114 if (FAILED(rc)) throw rc;
11115
11116 /** @todo r=bird: How is the locking and diff image cleaned up if we fail before
11117 * the push_back? Looks like we're going to release medium with the
11118 * wrong kind of lock (general issue with if we fail anywhere at all)
11119 * and an orphaned VDI in the snapshots folder. */
11120
11121 /* update the appropriate lock list */
11122 MediumLockList *pMediumLockList;
11123 rc = lockedMediaMap->Get(pAtt, pMediumLockList);
11124 AssertComRCThrowRC(rc);
11125 if (aOnline)
11126 {
11127 alock.release();
11128 /* The currently attached medium will be read-only, change
11129 * the lock type to read. */
11130 rc = pMediumLockList->Update(pMedium, false);
11131 alock.acquire();
11132 AssertComRCThrowRC(rc);
11133 }
11134
11135 /* release the locks before the potentially lengthy operation */
11136 alock.release();
11137 rc = pMedium->createDiffStorage(diff, MediumVariant_Standard,
11138 pMediumLockList,
11139 NULL /* aProgress */,
11140 true /* aWait */);
11141 alock.acquire();
11142 if (FAILED(rc)) throw rc;
11143
11144 /* actual lock list update is done in Medium::commitMedia */
11145
11146 rc = diff->addBackReference(mData->mUuid);
11147 AssertComRCThrowRC(rc);
11148
11149 /* add a new attachment */
11150 ComObjPtr<MediumAttachment> attachment;
11151 attachment.createObject();
11152 rc = attachment->init(this,
11153 diff,
11154 pAtt->getControllerName(),
11155 pAtt->getPort(),
11156 pAtt->getDevice(),
11157 DeviceType_HardDisk,
11158 true /* aImplicit */,
11159 false /* aPassthrough */,
11160 false /* aTempEject */,
11161 pAtt->getNonRotational(),
11162 pAtt->getDiscard(),
11163 pAtt->getBandwidthGroup());
11164 if (FAILED(rc)) throw rc;
11165
11166 rc = lockedMediaMap->ReplaceKey(pAtt, attachment);
11167 AssertComRCThrowRC(rc);
11168 mMediaData->mAttachments.push_back(attachment);
11169 }
11170 }
11171 catch (HRESULT aRC) { rc = aRC; }
11172
11173 /* unlock all hard disks we locked when there is no VM */
11174 if (!aOnline)
11175 {
11176 ErrorInfoKeeper eik;
11177
11178 HRESULT rc1 = lockedMediaMap->Clear();
11179 AssertComRC(rc1);
11180 }
11181
11182 return rc;
11183}
11184
11185/**
11186 * Deletes implicit differencing hard disks created either by
11187 * #createImplicitDiffs() or by #AttachDevice() and rolls back mMediaData.
11188 *
11189 * Note that to delete hard disks created by #AttachDevice() this method is
11190 * called from #fixupMedia() when the changes are rolled back.
11191 *
11192 * @note Locks this object and the media tree for writing.
11193 */
11194HRESULT Machine::deleteImplicitDiffs(bool aOnline)
11195{
11196 LogFlowThisFunc(("aOnline=%d\n", aOnline));
11197
11198 AutoCaller autoCaller(this);
11199 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
11200
11201 AutoMultiWriteLock2 alock(this->lockHandle(),
11202 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
11203
11204 /* We absolutely must have backed up state. */
11205 AssertReturn(mMediaData.isBackedUp(), E_FAIL);
11206
11207 /* Check if there are any implicitly created diff images. */
11208 bool fImplicitDiffs = false;
11209 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11210 it != mMediaData->mAttachments.end();
11211 ++it)
11212 {
11213 const ComObjPtr<MediumAttachment> &pAtt = *it;
11214 if (pAtt->isImplicit())
11215 {
11216 fImplicitDiffs = true;
11217 break;
11218 }
11219 }
11220 /* If there is nothing to do, leave early. This saves lots of image locking
11221 * effort. It also avoids a MachineStateChanged event without real reason.
11222 * This is important e.g. when loading a VM config, because there should be
11223 * no events. Otherwise API clients can become thoroughly confused for
11224 * inaccessible VMs (the code for loading VM configs uses this method for
11225 * cleanup if the config makes no sense), as they take such events as an
11226 * indication that the VM is alive, and they would force the VM config to
11227 * be reread, leading to an endless loop. */
11228 if (!fImplicitDiffs)
11229 return S_OK;
11230
11231 HRESULT rc = S_OK;
11232 MachineState_T oldState = mData->mMachineState;
11233
11234 /* will release the lock before the potentially lengthy operation,
11235 * so protect with the special state (unless already protected) */
11236 if ( oldState != MachineState_Saving
11237 && oldState != MachineState_LiveSnapshotting
11238 && oldState != MachineState_RestoringSnapshot
11239 && oldState != MachineState_DeletingSnapshot
11240 && oldState != MachineState_DeletingSnapshotOnline
11241 && oldState != MachineState_DeletingSnapshotPaused
11242 )
11243 setMachineState(MachineState_SettingUp);
11244
11245 // use appropriate locked media map (online or offline)
11246 MediumLockListMap lockedMediaOffline;
11247 MediumLockListMap *lockedMediaMap;
11248 if (aOnline)
11249 lockedMediaMap = &mData->mSession.mLockedMedia;
11250 else
11251 lockedMediaMap = &lockedMediaOffline;
11252
11253 try
11254 {
11255 if (!aOnline)
11256 {
11257 /* lock all attached hard disks early to detect "in use"
11258 * situations before deleting actual diffs */
11259 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11260 it != mMediaData->mAttachments.end();
11261 ++it)
11262 {
11263 MediumAttachment* pAtt = *it;
11264 if (pAtt->getType() == DeviceType_HardDisk)
11265 {
11266 Medium* pMedium = pAtt->getMedium();
11267 Assert(pMedium);
11268
11269 MediumLockList *pMediumLockList(new MediumLockList());
11270 alock.release();
11271 rc = pMedium->createMediumLockList(true /* fFailIfInaccessible */,
11272 false /* fMediumLockWrite */,
11273 NULL,
11274 *pMediumLockList);
11275 alock.acquire();
11276
11277 if (FAILED(rc))
11278 {
11279 delete pMediumLockList;
11280 throw rc;
11281 }
11282
11283 rc = lockedMediaMap->Insert(pAtt, pMediumLockList);
11284 if (FAILED(rc))
11285 throw rc;
11286 }
11287 }
11288
11289 if (FAILED(rc))
11290 throw rc;
11291 } // end of offline
11292
11293 /* Lock lists are now up to date and include implicitly created media */
11294
11295 /* Go through remembered attachments and delete all implicitly created
11296 * diffs and fix up the attachment information */
11297 const MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
11298 MediaData::AttachmentList implicitAtts;
11299 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11300 it != mMediaData->mAttachments.end();
11301 ++it)
11302 {
11303 ComObjPtr<MediumAttachment> pAtt = *it;
11304 ComObjPtr<Medium> pMedium = pAtt->getMedium();
11305 if (pMedium.isNull())
11306 continue;
11307
11308 // Implicit attachments go on the list for deletion and back references are removed.
11309 if (pAtt->isImplicit())
11310 {
11311 /* Deassociate and mark for deletion */
11312 LogFlowThisFunc(("Detaching '%s', pending deletion\n", pAtt->getLogName()));
11313 rc = pMedium->removeBackReference(mData->mUuid);
11314 if (FAILED(rc))
11315 throw rc;
11316 implicitAtts.push_back(pAtt);
11317 continue;
11318 }
11319
11320 /* Was this medium attached before? */
11321 if (!findAttachment(oldAtts, pMedium))
11322 {
11323 /* no: de-associate */
11324 LogFlowThisFunc(("Detaching '%s', no deletion\n", pAtt->getLogName()));
11325 rc = pMedium->removeBackReference(mData->mUuid);
11326 if (FAILED(rc))
11327 throw rc;
11328 continue;
11329 }
11330 LogFlowThisFunc(("Not detaching '%s'\n", pAtt->getLogName()));
11331 }
11332
11333 /* If there are implicit attachments to delete, throw away the lock
11334 * map contents (which will unlock all media) since the medium
11335 * attachments will be rolled back. Below we need to completely
11336 * recreate the lock map anyway since it is infinitely complex to
11337 * do this incrementally (would need reconstructing each attachment
11338 * change, which would be extremely hairy). */
11339 if (implicitAtts.size() != 0)
11340 {
11341 ErrorInfoKeeper eik;
11342
11343 HRESULT rc1 = lockedMediaMap->Clear();
11344 AssertComRC(rc1);
11345 }
11346
11347 /* rollback hard disk changes */
11348 mMediaData.rollback();
11349
11350 MultiResult mrc(S_OK);
11351
11352 // Delete unused implicit diffs.
11353 if (implicitAtts.size() != 0)
11354 {
11355 alock.release();
11356
11357 for (MediaData::AttachmentList::const_iterator it = implicitAtts.begin();
11358 it != implicitAtts.end();
11359 ++it)
11360 {
11361 // Remove medium associated with this attachment.
11362 ComObjPtr<MediumAttachment> pAtt = *it;
11363 Assert(pAtt);
11364 LogFlowThisFunc(("Deleting '%s'\n", pAtt->getLogName()));
11365 ComObjPtr<Medium> pMedium = pAtt->getMedium();
11366 Assert(pMedium);
11367
11368 rc = pMedium->deleteStorage(NULL /*aProgress*/, true /*aWait*/);
11369 // continue on delete failure, just collect error messages
11370 AssertMsg(SUCCEEDED(rc), ("rc=%Rhrc it=%s hd=%s\n", rc, pAtt->getLogName(), pMedium->getLocationFull().c_str() ));
11371 mrc = rc;
11372 }
11373
11374 alock.acquire();
11375
11376 /* if there is a VM recreate media lock map as mentioned above,
11377 * otherwise it is a waste of time and we leave things unlocked */
11378 if (aOnline)
11379 {
11380 const ComObjPtr<SessionMachine> pMachine = mData->mSession.mMachine;
11381 /* must never be NULL, but better safe than sorry */
11382 if (!pMachine.isNull())
11383 {
11384 alock.release();
11385 rc = mData->mSession.mMachine->lockMedia();
11386 alock.acquire();
11387 if (FAILED(rc))
11388 throw rc;
11389 }
11390 }
11391 }
11392 }
11393 catch (HRESULT aRC) {rc = aRC;}
11394
11395 if (mData->mMachineState == MachineState_SettingUp)
11396 setMachineState(oldState);
11397
11398 /* unlock all hard disks we locked when there is no VM */
11399 if (!aOnline)
11400 {
11401 ErrorInfoKeeper eik;
11402
11403 HRESULT rc1 = lockedMediaMap->Clear();
11404 AssertComRC(rc1);
11405 }
11406
11407 return rc;
11408}
11409
11410
11411/**
11412 * Looks through the given list of media attachments for one with the given parameters
11413 * and returns it, or NULL if not found. The list is a parameter so that backup lists
11414 * can be searched as well if needed.
11415 *
11416 * @param list
11417 * @param aControllerName
11418 * @param aControllerPort
11419 * @param aDevice
11420 * @return
11421 */
11422MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
11423 IN_BSTR aControllerName,
11424 LONG aControllerPort,
11425 LONG aDevice)
11426{
11427 for (MediaData::AttachmentList::const_iterator it = ll.begin();
11428 it != ll.end();
11429 ++it)
11430 {
11431 MediumAttachment *pAttach = *it;
11432 if (pAttach->matches(aControllerName, aControllerPort, aDevice))
11433 return pAttach;
11434 }
11435
11436 return NULL;
11437}
11438
11439/**
11440 * Looks through the given list of media attachments for one with the given parameters
11441 * and returns it, or NULL if not found. The list is a parameter so that backup lists
11442 * can be searched as well if needed.
11443 *
11444 * @param list
11445 * @param aControllerName
11446 * @param aControllerPort
11447 * @param aDevice
11448 * @return
11449 */
11450MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
11451 ComObjPtr<Medium> pMedium)
11452{
11453 for (MediaData::AttachmentList::const_iterator it = ll.begin();
11454 it != ll.end();
11455 ++it)
11456 {
11457 MediumAttachment *pAttach = *it;
11458 ComObjPtr<Medium> pMediumThis = pAttach->getMedium();
11459 if (pMediumThis == pMedium)
11460 return pAttach;
11461 }
11462
11463 return NULL;
11464}
11465
11466/**
11467 * Looks through the given list of media attachments for one with the given parameters
11468 * and returns it, or NULL if not found. The list is a parameter so that backup lists
11469 * can be searched as well if needed.
11470 *
11471 * @param list
11472 * @param aControllerName
11473 * @param aControllerPort
11474 * @param aDevice
11475 * @return
11476 */
11477MediumAttachment* Machine::findAttachment(const MediaData::AttachmentList &ll,
11478 Guid &id)
11479{
11480 for (MediaData::AttachmentList::const_iterator it = ll.begin();
11481 it != ll.end();
11482 ++it)
11483 {
11484 MediumAttachment *pAttach = *it;
11485 ComObjPtr<Medium> pMediumThis = pAttach->getMedium();
11486 if (pMediumThis->getId() == id)
11487 return pAttach;
11488 }
11489
11490 return NULL;
11491}
11492
11493/**
11494 * Main implementation for Machine::DetachDevice. This also gets called
11495 * from Machine::prepareUnregister() so it has been taken out for simplicity.
11496 *
11497 * @param pAttach Medium attachment to detach.
11498 * @param writeLock Machine write lock which the caller must have locked once. This may be released temporarily in here.
11499 * @param pSnapshot If NULL, then the detachment is for the current machine. Otherwise this is for a SnapshotMachine, and this must be its snapshot.
11500 * @return
11501 */
11502HRESULT Machine::detachDevice(MediumAttachment *pAttach,
11503 AutoWriteLock &writeLock,
11504 Snapshot *pSnapshot)
11505{
11506 ComObjPtr<Medium> oldmedium = pAttach->getMedium();
11507 DeviceType_T mediumType = pAttach->getType();
11508
11509 LogFlowThisFunc(("Entering, medium of attachment is %s\n", oldmedium ? oldmedium->getLocationFull().c_str() : "NULL"));
11510
11511 if (pAttach->isImplicit())
11512 {
11513 /* attempt to implicitly delete the implicitly created diff */
11514
11515 /// @todo move the implicit flag from MediumAttachment to Medium
11516 /// and forbid any hard disk operation when it is implicit. Or maybe
11517 /// a special media state for it to make it even more simple.
11518
11519 Assert(mMediaData.isBackedUp());
11520
11521 /* will release the lock before the potentially lengthy operation, so
11522 * protect with the special state */
11523 MachineState_T oldState = mData->mMachineState;
11524 setMachineState(MachineState_SettingUp);
11525
11526 writeLock.release();
11527
11528 HRESULT rc = oldmedium->deleteStorage(NULL /*aProgress*/,
11529 true /*aWait*/);
11530
11531 writeLock.acquire();
11532
11533 setMachineState(oldState);
11534
11535 if (FAILED(rc)) return rc;
11536 }
11537
11538 setModified(IsModified_Storage);
11539 mMediaData.backup();
11540 mMediaData->mAttachments.remove(pAttach);
11541
11542 if (!oldmedium.isNull())
11543 {
11544 // if this is from a snapshot, do not defer detachment to commitMedia()
11545 if (pSnapshot)
11546 oldmedium->removeBackReference(mData->mUuid, pSnapshot->getId());
11547 // else if non-hard disk media, do not defer detachment to commitMedia() either
11548 else if (mediumType != DeviceType_HardDisk)
11549 oldmedium->removeBackReference(mData->mUuid);
11550 }
11551
11552 return S_OK;
11553}
11554
11555/**
11556 * Goes thru all media of the given list and
11557 *
11558 * 1) calls detachDevice() on each of them for this machine and
11559 * 2) adds all Medium objects found in the process to the given list,
11560 * depending on cleanupMode.
11561 *
11562 * If cleanupMode is CleanupMode_DetachAllReturnHardDisksOnly, this only
11563 * adds hard disks to the list. If it is CleanupMode_Full, this adds all
11564 * media to the list.
11565 *
11566 * This gets called from Machine::Unregister, both for the actual Machine and
11567 * the SnapshotMachine objects that might be found in the snapshots.
11568 *
11569 * Requires caller and locking. The machine lock must be passed in because it
11570 * will be passed on to detachDevice which needs it for temporary unlocking.
11571 *
11572 * @param writeLock Machine lock from top-level caller; this gets passed to detachDevice.
11573 * @param pSnapshot Must be NULL when called for a "real" Machine or a snapshot object if called for a SnapshotMachine.
11574 * @param cleanupMode If DetachAllReturnHardDisksOnly, only hard disk media get added to llMedia; if Full, then all media get added;
11575 * otherwise no media get added.
11576 * @param llMedia Caller's list to receive Medium objects which got detached so caller can close() them, depending on cleanupMode.
11577 * @return
11578 */
11579HRESULT Machine::detachAllMedia(AutoWriteLock &writeLock,
11580 Snapshot *pSnapshot,
11581 CleanupMode_T cleanupMode,
11582 MediaList &llMedia)
11583{
11584 Assert(isWriteLockOnCurrentThread());
11585
11586 HRESULT rc;
11587
11588 // make a temporary list because detachDevice invalidates iterators into
11589 // mMediaData->mAttachments
11590 MediaData::AttachmentList llAttachments2 = mMediaData->mAttachments;
11591
11592 for (MediaData::AttachmentList::iterator it = llAttachments2.begin();
11593 it != llAttachments2.end();
11594 ++it)
11595 {
11596 ComObjPtr<MediumAttachment> &pAttach = *it;
11597 ComObjPtr<Medium> pMedium = pAttach->getMedium();
11598
11599 if (!pMedium.isNull())
11600 {
11601 AutoCaller mac(pMedium);
11602 if (FAILED(mac.rc())) return mac.rc();
11603 AutoReadLock lock(pMedium COMMA_LOCKVAL_SRC_POS);
11604 DeviceType_T devType = pMedium->getDeviceType();
11605 if ( ( cleanupMode == CleanupMode_DetachAllReturnHardDisksOnly
11606 && devType == DeviceType_HardDisk)
11607 || (cleanupMode == CleanupMode_Full)
11608 )
11609 {
11610 llMedia.push_back(pMedium);
11611 ComObjPtr<Medium> pParent = pMedium->getParent();
11612 /*
11613 * Search for medias which are not attached to any machine, but
11614 * in the chain to an attached disk. Mediums are only consided
11615 * if they are:
11616 * - have only one child
11617 * - no references to any machines
11618 * - are of normal medium type
11619 */
11620 while (!pParent.isNull())
11621 {
11622 AutoCaller mac1(pParent);
11623 if (FAILED(mac1.rc())) return mac1.rc();
11624 AutoReadLock lock1(pParent COMMA_LOCKVAL_SRC_POS);
11625 if (pParent->getChildren().size() == 1)
11626 {
11627 if ( pParent->getMachineBackRefCount() == 0
11628 && pParent->getType() == MediumType_Normal
11629 && find(llMedia.begin(), llMedia.end(), pParent) == llMedia.end())
11630 llMedia.push_back(pParent);
11631 }
11632 else
11633 break;
11634 pParent = pParent->getParent();
11635 }
11636 }
11637 }
11638
11639 // real machine: then we need to use the proper method
11640 rc = detachDevice(pAttach, writeLock, pSnapshot);
11641
11642 if (FAILED(rc))
11643 return rc;
11644 }
11645
11646 return S_OK;
11647}
11648
11649/**
11650 * Perform deferred hard disk detachments.
11651 *
11652 * Does nothing if the hard disk attachment data (mMediaData) is not changed (not
11653 * backed up).
11654 *
11655 * If @a aOnline is @c true then this method will also unlock the old hard disks
11656 * for which the new implicit diffs were created and will lock these new diffs for
11657 * writing.
11658 *
11659 * @param aOnline Whether the VM was online prior to this operation.
11660 *
11661 * @note Locks this object for writing!
11662 */
11663void Machine::commitMedia(bool aOnline /*= false*/)
11664{
11665 AutoCaller autoCaller(this);
11666 AssertComRCReturnVoid(autoCaller.rc());
11667
11668 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
11669
11670 LogFlowThisFunc(("Entering, aOnline=%d\n", aOnline));
11671
11672 HRESULT rc = S_OK;
11673
11674 /* no attach/detach operations -- nothing to do */
11675 if (!mMediaData.isBackedUp())
11676 return;
11677
11678 MediaData::AttachmentList &oldAtts = mMediaData.backedUpData()->mAttachments;
11679 bool fMediaNeedsLocking = false;
11680
11681 /* enumerate new attachments */
11682 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11683 it != mMediaData->mAttachments.end();
11684 ++it)
11685 {
11686 MediumAttachment *pAttach = *it;
11687
11688 pAttach->commit();
11689
11690 Medium* pMedium = pAttach->getMedium();
11691 bool fImplicit = pAttach->isImplicit();
11692
11693 LogFlowThisFunc(("Examining current medium '%s' (implicit: %d)\n",
11694 (pMedium) ? pMedium->getName().c_str() : "NULL",
11695 fImplicit));
11696
11697 /** @todo convert all this Machine-based voodoo to MediumAttachment
11698 * based commit logic. */
11699 if (fImplicit)
11700 {
11701 /* convert implicit attachment to normal */
11702 pAttach->setImplicit(false);
11703
11704 if ( aOnline
11705 && pMedium
11706 && pAttach->getType() == DeviceType_HardDisk
11707 )
11708 {
11709 ComObjPtr<Medium> parent = pMedium->getParent();
11710 AutoWriteLock parentLock(parent COMMA_LOCKVAL_SRC_POS);
11711
11712 /* update the appropriate lock list */
11713 MediumLockList *pMediumLockList;
11714 rc = mData->mSession.mLockedMedia.Get(pAttach, pMediumLockList);
11715 AssertComRC(rc);
11716 if (pMediumLockList)
11717 {
11718 /* unlock if there's a need to change the locking */
11719 if (!fMediaNeedsLocking)
11720 {
11721 rc = mData->mSession.mLockedMedia.Unlock();
11722 AssertComRC(rc);
11723 fMediaNeedsLocking = true;
11724 }
11725 rc = pMediumLockList->Update(parent, false);
11726 AssertComRC(rc);
11727 rc = pMediumLockList->Append(pMedium, true);
11728 AssertComRC(rc);
11729 }
11730 }
11731
11732 continue;
11733 }
11734
11735 if (pMedium)
11736 {
11737 /* was this medium attached before? */
11738 for (MediaData::AttachmentList::iterator oldIt = oldAtts.begin();
11739 oldIt != oldAtts.end();
11740 ++oldIt)
11741 {
11742 MediumAttachment *pOldAttach = *oldIt;
11743 if (pOldAttach->getMedium() == pMedium)
11744 {
11745 LogFlowThisFunc(("--> medium '%s' was attached before, will not remove\n", pMedium->getName().c_str()));
11746
11747 /* yes: remove from old to avoid de-association */
11748 oldAtts.erase(oldIt);
11749 break;
11750 }
11751 }
11752 }
11753 }
11754
11755 /* enumerate remaining old attachments and de-associate from the
11756 * current machine state */
11757 for (MediaData::AttachmentList::const_iterator it = oldAtts.begin();
11758 it != oldAtts.end();
11759 ++it)
11760 {
11761 MediumAttachment *pAttach = *it;
11762 Medium* pMedium = pAttach->getMedium();
11763
11764 /* Detach only hard disks, since DVD/floppy media is detached
11765 * instantly in MountMedium. */
11766 if (pAttach->getType() == DeviceType_HardDisk && pMedium)
11767 {
11768 LogFlowThisFunc(("detaching medium '%s' from machine\n", pMedium->getName().c_str()));
11769
11770 /* now de-associate from the current machine state */
11771 rc = pMedium->removeBackReference(mData->mUuid);
11772 AssertComRC(rc);
11773
11774 if (aOnline)
11775 {
11776 /* unlock since medium is not used anymore */
11777 MediumLockList *pMediumLockList;
11778 rc = mData->mSession.mLockedMedia.Get(pAttach, pMediumLockList);
11779 if (RT_UNLIKELY(rc == VBOX_E_INVALID_OBJECT_STATE))
11780 {
11781 /* this happens for online snapshots, there the attachment
11782 * is changing, but only to a diff image created under
11783 * the old one, so there is no separate lock list */
11784 Assert(!pMediumLockList);
11785 }
11786 else
11787 {
11788 AssertComRC(rc);
11789 if (pMediumLockList)
11790 {
11791 rc = mData->mSession.mLockedMedia.Remove(pAttach);
11792 AssertComRC(rc);
11793 }
11794 }
11795 }
11796 }
11797 }
11798
11799 /* take media locks again so that the locking state is consistent */
11800 if (fMediaNeedsLocking)
11801 {
11802 Assert(aOnline);
11803 rc = mData->mSession.mLockedMedia.Lock();
11804 AssertComRC(rc);
11805 }
11806
11807 /* commit the hard disk changes */
11808 mMediaData.commit();
11809
11810 if (isSessionMachine())
11811 {
11812 /*
11813 * Update the parent machine to point to the new owner.
11814 * This is necessary because the stored parent will point to the
11815 * session machine otherwise and cause crashes or errors later
11816 * when the session machine gets invalid.
11817 */
11818 /** @todo Change the MediumAttachment class to behave like any other
11819 * class in this regard by creating peer MediumAttachment
11820 * objects for session machines and share the data with the peer
11821 * machine.
11822 */
11823 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11824 it != mMediaData->mAttachments.end();
11825 ++it)
11826 {
11827 (*it)->updateParentMachine(mPeer);
11828 }
11829
11830 /* attach new data to the primary machine and reshare it */
11831 mPeer->mMediaData.attach(mMediaData);
11832 }
11833
11834 return;
11835}
11836
11837/**
11838 * Perform deferred deletion of implicitly created diffs.
11839 *
11840 * Does nothing if the hard disk attachment data (mMediaData) is not changed (not
11841 * backed up).
11842 *
11843 * @note Locks this object for writing!
11844 */
11845void Machine::rollbackMedia()
11846{
11847 AutoCaller autoCaller(this);
11848 AssertComRCReturnVoid(autoCaller.rc());
11849
11850 // AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
11851 LogFlowThisFunc(("Entering rollbackMedia\n"));
11852
11853 HRESULT rc = S_OK;
11854
11855 /* no attach/detach operations -- nothing to do */
11856 if (!mMediaData.isBackedUp())
11857 return;
11858
11859 /* enumerate new attachments */
11860 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
11861 it != mMediaData->mAttachments.end();
11862 ++it)
11863 {
11864 MediumAttachment *pAttach = *it;
11865 /* Fix up the backrefs for DVD/floppy media. */
11866 if (pAttach->getType() != DeviceType_HardDisk)
11867 {
11868 Medium* pMedium = pAttach->getMedium();
11869 if (pMedium)
11870 {
11871 rc = pMedium->removeBackReference(mData->mUuid);
11872 AssertComRC(rc);
11873 }
11874 }
11875
11876 (*it)->rollback();
11877
11878 pAttach = *it;
11879 /* Fix up the backrefs for DVD/floppy media. */
11880 if (pAttach->getType() != DeviceType_HardDisk)
11881 {
11882 Medium* pMedium = pAttach->getMedium();
11883 if (pMedium)
11884 {
11885 rc = pMedium->addBackReference(mData->mUuid);
11886 AssertComRC(rc);
11887 }
11888 }
11889 }
11890
11891 /** @todo convert all this Machine-based voodoo to MediumAttachment
11892 * based rollback logic. */
11893 deleteImplicitDiffs(Global::IsOnline(mData->mMachineState));
11894
11895 return;
11896}
11897
11898/**
11899 * Returns true if the settings file is located in the directory named exactly
11900 * as the machine; this means, among other things, that the machine directory
11901 * should be auto-renamed.
11902 *
11903 * @param aSettingsDir if not NULL, the full machine settings file directory
11904 * name will be assigned there.
11905 *
11906 * @note Doesn't lock anything.
11907 * @note Not thread safe (must be called from this object's lock).
11908 */
11909bool Machine::isInOwnDir(Utf8Str *aSettingsDir /* = NULL */) const
11910{
11911 Utf8Str strMachineDirName(mData->m_strConfigFileFull); // path/to/machinesfolder/vmname/vmname.vbox
11912 strMachineDirName.stripFilename(); // path/to/machinesfolder/vmname
11913 if (aSettingsDir)
11914 *aSettingsDir = strMachineDirName;
11915 strMachineDirName.stripPath(); // vmname
11916 Utf8Str strConfigFileOnly(mData->m_strConfigFileFull); // path/to/machinesfolder/vmname/vmname.vbox
11917 strConfigFileOnly.stripPath() // vmname.vbox
11918 .stripExt(); // vmname
11919 /** @todo hack, make somehow use of ComposeMachineFilename */
11920 if (mUserData->s.fDirectoryIncludesUUID)
11921 strConfigFileOnly += Utf8StrFmt(" (%RTuuid)", mData->mUuid.raw());
11922
11923 AssertReturn(!strMachineDirName.isEmpty(), false);
11924 AssertReturn(!strConfigFileOnly.isEmpty(), false);
11925
11926 return strMachineDirName == strConfigFileOnly;
11927}
11928
11929/**
11930 * Discards all changes to machine settings.
11931 *
11932 * @param aNotify Whether to notify the direct session about changes or not.
11933 *
11934 * @note Locks objects for writing!
11935 */
11936void Machine::rollback(bool aNotify)
11937{
11938 AutoCaller autoCaller(this);
11939 AssertComRCReturn(autoCaller.rc(), (void)0);
11940
11941 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
11942
11943 if (!mStorageControllers.isNull())
11944 {
11945 if (mStorageControllers.isBackedUp())
11946 {
11947 /* unitialize all new devices (absent in the backed up list). */
11948 StorageControllerList::const_iterator it = mStorageControllers->begin();
11949 StorageControllerList *backedList = mStorageControllers.backedUpData();
11950 while (it != mStorageControllers->end())
11951 {
11952 if ( std::find(backedList->begin(), backedList->end(), *it)
11953 == backedList->end()
11954 )
11955 {
11956 (*it)->uninit();
11957 }
11958 ++it;
11959 }
11960
11961 /* restore the list */
11962 mStorageControllers.rollback();
11963 }
11964
11965 /* rollback any changes to devices after restoring the list */
11966 if (mData->flModifications & IsModified_Storage)
11967 {
11968 StorageControllerList::const_iterator it = mStorageControllers->begin();
11969 while (it != mStorageControllers->end())
11970 {
11971 (*it)->rollback();
11972 ++it;
11973 }
11974 }
11975 }
11976
11977 if (!mUSBControllers.isNull())
11978 {
11979 if (mUSBControllers.isBackedUp())
11980 {
11981 /* unitialize all new devices (absent in the backed up list). */
11982 USBControllerList::const_iterator it = mUSBControllers->begin();
11983 USBControllerList *backedList = mUSBControllers.backedUpData();
11984 while (it != mUSBControllers->end())
11985 {
11986 if ( std::find(backedList->begin(), backedList->end(), *it)
11987 == backedList->end()
11988 )
11989 {
11990 (*it)->uninit();
11991 }
11992 ++it;
11993 }
11994
11995 /* restore the list */
11996 mUSBControllers.rollback();
11997 }
11998
11999 /* rollback any changes to devices after restoring the list */
12000 if (mData->flModifications & IsModified_USB)
12001 {
12002 USBControllerList::const_iterator it = mUSBControllers->begin();
12003 while (it != mUSBControllers->end())
12004 {
12005 (*it)->rollback();
12006 ++it;
12007 }
12008 }
12009 }
12010
12011 mUserData.rollback();
12012
12013 mHWData.rollback();
12014
12015 if (mData->flModifications & IsModified_Storage)
12016 rollbackMedia();
12017
12018 if (mBIOSSettings)
12019 mBIOSSettings->rollback();
12020
12021 if (mVRDEServer && (mData->flModifications & IsModified_VRDEServer))
12022 mVRDEServer->rollback();
12023
12024 if (mAudioAdapter)
12025 mAudioAdapter->rollback();
12026
12027 if (mUSBDeviceFilters && (mData->flModifications & IsModified_USB))
12028 mUSBDeviceFilters->rollback();
12029
12030 if (mBandwidthControl && (mData->flModifications & IsModified_BandwidthControl))
12031 mBandwidthControl->rollback();
12032
12033 if (!mHWData.isNull())
12034 mNetworkAdapters.resize(Global::getMaxNetworkAdapters(mHWData->mChipsetType));
12035 NetworkAdapterVector networkAdapters(mNetworkAdapters.size());
12036 ComPtr<ISerialPort> serialPorts[RT_ELEMENTS(mSerialPorts)];
12037 ComPtr<IParallelPort> parallelPorts[RT_ELEMENTS(mParallelPorts)];
12038
12039 if (mData->flModifications & IsModified_NetworkAdapters)
12040 for (ULONG slot = 0; slot < mNetworkAdapters.size(); slot++)
12041 if ( mNetworkAdapters[slot]
12042 && mNetworkAdapters[slot]->isModified())
12043 {
12044 mNetworkAdapters[slot]->rollback();
12045 networkAdapters[slot] = mNetworkAdapters[slot];
12046 }
12047
12048 if (mData->flModifications & IsModified_SerialPorts)
12049 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
12050 if ( mSerialPorts[slot]
12051 && mSerialPorts[slot]->isModified())
12052 {
12053 mSerialPorts[slot]->rollback();
12054 serialPorts[slot] = mSerialPorts[slot];
12055 }
12056
12057 if (mData->flModifications & IsModified_ParallelPorts)
12058 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
12059 if ( mParallelPorts[slot]
12060 && mParallelPorts[slot]->isModified())
12061 {
12062 mParallelPorts[slot]->rollback();
12063 parallelPorts[slot] = mParallelPorts[slot];
12064 }
12065
12066 if (aNotify)
12067 {
12068 /* inform the direct session about changes */
12069
12070 ComObjPtr<Machine> that = this;
12071 uint32_t flModifications = mData->flModifications;
12072 alock.release();
12073
12074 if (flModifications & IsModified_SharedFolders)
12075 that->onSharedFolderChange();
12076
12077 if (flModifications & IsModified_VRDEServer)
12078 that->onVRDEServerChange(/* aRestart */ TRUE);
12079 if (flModifications & IsModified_USB)
12080 that->onUSBControllerChange();
12081
12082 for (ULONG slot = 0; slot < networkAdapters.size(); slot++)
12083 if (networkAdapters[slot])
12084 that->onNetworkAdapterChange(networkAdapters[slot], FALSE);
12085 for (ULONG slot = 0; slot < RT_ELEMENTS(serialPorts); slot++)
12086 if (serialPorts[slot])
12087 that->onSerialPortChange(serialPorts[slot]);
12088 for (ULONG slot = 0; slot < RT_ELEMENTS(parallelPorts); slot++)
12089 if (parallelPorts[slot])
12090 that->onParallelPortChange(parallelPorts[slot]);
12091
12092 if (flModifications & IsModified_Storage)
12093 that->onStorageControllerChange();
12094
12095#if 0
12096 if (flModifications & IsModified_BandwidthControl)
12097 that->onBandwidthControlChange();
12098#endif
12099 }
12100}
12101
12102/**
12103 * Commits all the changes to machine settings.
12104 *
12105 * Note that this operation is supposed to never fail.
12106 *
12107 * @note Locks this object and children for writing.
12108 */
12109void Machine::commit()
12110{
12111 AutoCaller autoCaller(this);
12112 AssertComRCReturnVoid(autoCaller.rc());
12113
12114 AutoCaller peerCaller(mPeer);
12115 AssertComRCReturnVoid(peerCaller.rc());
12116
12117 AutoMultiWriteLock2 alock(mPeer, this COMMA_LOCKVAL_SRC_POS);
12118
12119 /*
12120 * use safe commit to ensure Snapshot machines (that share mUserData)
12121 * will still refer to a valid memory location
12122 */
12123 mUserData.commitCopy();
12124
12125 mHWData.commit();
12126
12127 if (mMediaData.isBackedUp())
12128 commitMedia(Global::IsOnline(mData->mMachineState));
12129
12130 mBIOSSettings->commit();
12131 mVRDEServer->commit();
12132 mAudioAdapter->commit();
12133 mUSBDeviceFilters->commit();
12134 mBandwidthControl->commit();
12135
12136 /* Since mNetworkAdapters is a list which might have been changed (resized)
12137 * without using the Backupable<> template we need to handle the copying
12138 * of the list entries manually, including the creation of peers for the
12139 * new objects. */
12140 bool commitNetworkAdapters = false;
12141 size_t newSize = Global::getMaxNetworkAdapters(mHWData->mChipsetType);
12142 if (mPeer)
12143 {
12144 /* commit everything, even the ones which will go away */
12145 for (size_t slot = 0; slot < mNetworkAdapters.size(); slot++)
12146 mNetworkAdapters[slot]->commit();
12147 /* copy over the new entries, creating a peer and uninit the original */
12148 mPeer->mNetworkAdapters.resize(RT_MAX(newSize, mPeer->mNetworkAdapters.size()));
12149 for (size_t slot = 0; slot < newSize; slot++)
12150 {
12151 /* look if this adapter has a peer device */
12152 ComObjPtr<NetworkAdapter> peer = mNetworkAdapters[slot]->getPeer();
12153 if (!peer)
12154 {
12155 /* no peer means the adapter is a newly created one;
12156 * create a peer owning data this data share it with */
12157 peer.createObject();
12158 peer->init(mPeer, mNetworkAdapters[slot], true /* aReshare */);
12159 }
12160 mPeer->mNetworkAdapters[slot] = peer;
12161 }
12162 /* uninit any no longer needed network adapters */
12163 for (size_t slot = newSize; slot < mNetworkAdapters.size(); slot++)
12164 mNetworkAdapters[slot]->uninit();
12165 for (size_t slot = newSize; slot < mPeer->mNetworkAdapters.size(); slot++)
12166 {
12167 if (mPeer->mNetworkAdapters[slot])
12168 mPeer->mNetworkAdapters[slot]->uninit();
12169 }
12170 /* Keep the original network adapter count until this point, so that
12171 * discarding a chipset type change will not lose settings. */
12172 mNetworkAdapters.resize(newSize);
12173 mPeer->mNetworkAdapters.resize(newSize);
12174 }
12175 else
12176 {
12177 /* we have no peer (our parent is the newly created machine);
12178 * just commit changes to the network adapters */
12179 commitNetworkAdapters = true;
12180 }
12181 if (commitNetworkAdapters)
12182 {
12183 for (size_t slot = 0; slot < mNetworkAdapters.size(); slot++)
12184 mNetworkAdapters[slot]->commit();
12185 }
12186
12187 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
12188 mSerialPorts[slot]->commit();
12189 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
12190 mParallelPorts[slot]->commit();
12191
12192 bool commitStorageControllers = false;
12193
12194 if (mStorageControllers.isBackedUp())
12195 {
12196 mStorageControllers.commit();
12197
12198 if (mPeer)
12199 {
12200 /* Commit all changes to new controllers (this will reshare data with
12201 * peers for those who have peers) */
12202 StorageControllerList *newList = new StorageControllerList();
12203 StorageControllerList::const_iterator it = mStorageControllers->begin();
12204 while (it != mStorageControllers->end())
12205 {
12206 (*it)->commit();
12207
12208 /* look if this controller has a peer device */
12209 ComObjPtr<StorageController> peer = (*it)->getPeer();
12210 if (!peer)
12211 {
12212 /* no peer means the device is a newly created one;
12213 * create a peer owning data this device share it with */
12214 peer.createObject();
12215 peer->init(mPeer, *it, true /* aReshare */);
12216 }
12217 else
12218 {
12219 /* remove peer from the old list */
12220 mPeer->mStorageControllers->remove(peer);
12221 }
12222 /* and add it to the new list */
12223 newList->push_back(peer);
12224
12225 ++it;
12226 }
12227
12228 /* uninit old peer's controllers that are left */
12229 it = mPeer->mStorageControllers->begin();
12230 while (it != mPeer->mStorageControllers->end())
12231 {
12232 (*it)->uninit();
12233 ++it;
12234 }
12235
12236 /* attach new list of controllers to our peer */
12237 mPeer->mStorageControllers.attach(newList);
12238 }
12239 else
12240 {
12241 /* we have no peer (our parent is the newly created machine);
12242 * just commit changes to devices */
12243 commitStorageControllers = true;
12244 }
12245 }
12246 else
12247 {
12248 /* the list of controllers itself is not changed,
12249 * just commit changes to controllers themselves */
12250 commitStorageControllers = true;
12251 }
12252
12253 if (commitStorageControllers)
12254 {
12255 StorageControllerList::const_iterator it = mStorageControllers->begin();
12256 while (it != mStorageControllers->end())
12257 {
12258 (*it)->commit();
12259 ++it;
12260 }
12261 }
12262
12263 bool commitUSBControllers = false;
12264
12265 if (mUSBControllers.isBackedUp())
12266 {
12267 mUSBControllers.commit();
12268
12269 if (mPeer)
12270 {
12271 /* Commit all changes to new controllers (this will reshare data with
12272 * peers for those who have peers) */
12273 USBControllerList *newList = new USBControllerList();
12274 USBControllerList::const_iterator it = mUSBControllers->begin();
12275 while (it != mUSBControllers->end())
12276 {
12277 (*it)->commit();
12278
12279 /* look if this controller has a peer device */
12280 ComObjPtr<USBController> peer = (*it)->getPeer();
12281 if (!peer)
12282 {
12283 /* no peer means the device is a newly created one;
12284 * create a peer owning data this device share it with */
12285 peer.createObject();
12286 peer->init(mPeer, *it, true /* aReshare */);
12287 }
12288 else
12289 {
12290 /* remove peer from the old list */
12291 mPeer->mUSBControllers->remove(peer);
12292 }
12293 /* and add it to the new list */
12294 newList->push_back(peer);
12295
12296 ++it;
12297 }
12298
12299 /* uninit old peer's controllers that are left */
12300 it = mPeer->mUSBControllers->begin();
12301 while (it != mPeer->mUSBControllers->end())
12302 {
12303 (*it)->uninit();
12304 ++it;
12305 }
12306
12307 /* attach new list of controllers to our peer */
12308 mPeer->mUSBControllers.attach(newList);
12309 }
12310 else
12311 {
12312 /* we have no peer (our parent is the newly created machine);
12313 * just commit changes to devices */
12314 commitUSBControllers = true;
12315 }
12316 }
12317 else
12318 {
12319 /* the list of controllers itself is not changed,
12320 * just commit changes to controllers themselves */
12321 commitUSBControllers = true;
12322 }
12323
12324 if (commitUSBControllers)
12325 {
12326 USBControllerList::const_iterator it = mUSBControllers->begin();
12327 while (it != mUSBControllers->end())
12328 {
12329 (*it)->commit();
12330 ++it;
12331 }
12332 }
12333
12334 if (isSessionMachine())
12335 {
12336 /* attach new data to the primary machine and reshare it */
12337 mPeer->mUserData.attach(mUserData);
12338 mPeer->mHWData.attach(mHWData);
12339 /* mMediaData is reshared by fixupMedia */
12340 // mPeer->mMediaData.attach(mMediaData);
12341 Assert(mPeer->mMediaData.data() == mMediaData.data());
12342 }
12343}
12344
12345/**
12346 * Copies all the hardware data from the given machine.
12347 *
12348 * Currently, only called when the VM is being restored from a snapshot. In
12349 * particular, this implies that the VM is not running during this method's
12350 * call.
12351 *
12352 * @note This method must be called from under this object's lock.
12353 *
12354 * @note This method doesn't call #commit(), so all data remains backed up and
12355 * unsaved.
12356 */
12357void Machine::copyFrom(Machine *aThat)
12358{
12359 AssertReturnVoid(!isSnapshotMachine());
12360 AssertReturnVoid(aThat->isSnapshotMachine());
12361
12362 AssertReturnVoid(!Global::IsOnline(mData->mMachineState));
12363
12364 mHWData.assignCopy(aThat->mHWData);
12365
12366 // create copies of all shared folders (mHWData after attaching a copy
12367 // contains just references to original objects)
12368 for (HWData::SharedFolderList::iterator it = mHWData->mSharedFolders.begin();
12369 it != mHWData->mSharedFolders.end();
12370 ++it)
12371 {
12372 ComObjPtr<SharedFolder> folder;
12373 folder.createObject();
12374 HRESULT rc = folder->initCopy(getMachine(), *it);
12375 AssertComRC(rc);
12376 *it = folder;
12377 }
12378
12379 mBIOSSettings->copyFrom(aThat->mBIOSSettings);
12380 mVRDEServer->copyFrom(aThat->mVRDEServer);
12381 mAudioAdapter->copyFrom(aThat->mAudioAdapter);
12382 mUSBDeviceFilters->copyFrom(aThat->mUSBDeviceFilters);
12383 mBandwidthControl->copyFrom(aThat->mBandwidthControl);
12384
12385 /* create private copies of all controllers */
12386 mStorageControllers.backup();
12387 mStorageControllers->clear();
12388 for (StorageControllerList::iterator it = aThat->mStorageControllers->begin();
12389 it != aThat->mStorageControllers->end();
12390 ++it)
12391 {
12392 ComObjPtr<StorageController> ctrl;
12393 ctrl.createObject();
12394 ctrl->initCopy(this, *it);
12395 mStorageControllers->push_back(ctrl);
12396 }
12397
12398 /* create private copies of all USB controllers */
12399 mUSBControllers.backup();
12400 mUSBControllers->clear();
12401 for (USBControllerList::iterator it = aThat->mUSBControllers->begin();
12402 it != aThat->mUSBControllers->end();
12403 ++it)
12404 {
12405 ComObjPtr<USBController> ctrl;
12406 ctrl.createObject();
12407 ctrl->initCopy(this, *it);
12408 mUSBControllers->push_back(ctrl);
12409 }
12410
12411 mNetworkAdapters.resize(aThat->mNetworkAdapters.size());
12412 for (ULONG slot = 0; slot < mNetworkAdapters.size(); slot++)
12413 mNetworkAdapters[slot]->copyFrom(aThat->mNetworkAdapters[slot]);
12414 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
12415 mSerialPorts[slot]->copyFrom(aThat->mSerialPorts[slot]);
12416 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
12417 mParallelPorts[slot]->copyFrom(aThat->mParallelPorts[slot]);
12418}
12419
12420/**
12421 * Returns whether the given storage controller is hotplug capable.
12422 *
12423 * @returns true if the controller supports hotplugging
12424 * false otherwise.
12425 * @param enmCtrlType The controller type to check for.
12426 */
12427bool Machine::isControllerHotplugCapable(StorageControllerType_T enmCtrlType)
12428{
12429 switch (enmCtrlType)
12430 {
12431 case StorageControllerType_IntelAhci:
12432 return true;
12433 case StorageControllerType_LsiLogic:
12434 case StorageControllerType_LsiLogicSas:
12435 case StorageControllerType_BusLogic:
12436 case StorageControllerType_PIIX3:
12437 case StorageControllerType_PIIX4:
12438 case StorageControllerType_ICH6:
12439 case StorageControllerType_I82078:
12440 default:
12441 return false;
12442 }
12443}
12444
12445#ifdef VBOX_WITH_RESOURCE_USAGE_API
12446
12447void Machine::getDiskList(MediaList &list)
12448{
12449 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
12450 it != mMediaData->mAttachments.end();
12451 ++it)
12452 {
12453 MediumAttachment* pAttach = *it;
12454 /* just in case */
12455 AssertStmt(pAttach, continue);
12456
12457 AutoCaller localAutoCallerA(pAttach);
12458 if (FAILED(localAutoCallerA.rc())) continue;
12459
12460 AutoReadLock local_alockA(pAttach COMMA_LOCKVAL_SRC_POS);
12461
12462 if (pAttach->getType() == DeviceType_HardDisk)
12463 list.push_back(pAttach->getMedium());
12464 }
12465}
12466
12467void Machine::registerMetrics(PerformanceCollector *aCollector, Machine *aMachine, RTPROCESS pid)
12468{
12469 AssertReturnVoid(isWriteLockOnCurrentThread());
12470 AssertPtrReturnVoid(aCollector);
12471
12472 pm::CollectorHAL *hal = aCollector->getHAL();
12473 /* Create sub metrics */
12474 pm::SubMetric *cpuLoadUser = new pm::SubMetric("CPU/Load/User",
12475 "Percentage of processor time spent in user mode by the VM process.");
12476 pm::SubMetric *cpuLoadKernel = new pm::SubMetric("CPU/Load/Kernel",
12477 "Percentage of processor time spent in kernel mode by the VM process.");
12478 pm::SubMetric *ramUsageUsed = new pm::SubMetric("RAM/Usage/Used",
12479 "Size of resident portion of VM process in memory.");
12480 pm::SubMetric *diskUsageUsed = new pm::SubMetric("Disk/Usage/Used",
12481 "Actual size of all VM disks combined.");
12482 pm::SubMetric *machineNetRx = new pm::SubMetric("Net/Rate/Rx",
12483 "Network receive rate.");
12484 pm::SubMetric *machineNetTx = new pm::SubMetric("Net/Rate/Tx",
12485 "Network transmit rate.");
12486 /* Create and register base metrics */
12487 pm::BaseMetric *cpuLoad = new pm::MachineCpuLoadRaw(hal, aMachine, pid,
12488 cpuLoadUser, cpuLoadKernel);
12489 aCollector->registerBaseMetric(cpuLoad);
12490 pm::BaseMetric *ramUsage = new pm::MachineRamUsage(hal, aMachine, pid,
12491 ramUsageUsed);
12492 aCollector->registerBaseMetric(ramUsage);
12493 MediaList disks;
12494 getDiskList(disks);
12495 pm::BaseMetric *diskUsage = new pm::MachineDiskUsage(hal, aMachine, disks,
12496 diskUsageUsed);
12497 aCollector->registerBaseMetric(diskUsage);
12498
12499 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser, 0));
12500 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
12501 new pm::AggregateAvg()));
12502 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
12503 new pm::AggregateMin()));
12504 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
12505 new pm::AggregateMax()));
12506 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel, 0));
12507 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
12508 new pm::AggregateAvg()));
12509 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
12510 new pm::AggregateMin()));
12511 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
12512 new pm::AggregateMax()));
12513
12514 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed, 0));
12515 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
12516 new pm::AggregateAvg()));
12517 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
12518 new pm::AggregateMin()));
12519 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
12520 new pm::AggregateMax()));
12521
12522 aCollector->registerMetric(new pm::Metric(diskUsage, diskUsageUsed, 0));
12523 aCollector->registerMetric(new pm::Metric(diskUsage, diskUsageUsed,
12524 new pm::AggregateAvg()));
12525 aCollector->registerMetric(new pm::Metric(diskUsage, diskUsageUsed,
12526 new pm::AggregateMin()));
12527 aCollector->registerMetric(new pm::Metric(diskUsage, diskUsageUsed,
12528 new pm::AggregateMax()));
12529
12530
12531 /* Guest metrics collector */
12532 mCollectorGuest = new pm::CollectorGuest(aMachine, pid);
12533 aCollector->registerGuest(mCollectorGuest);
12534 LogAleksey(("{%p} " LOG_FN_FMT ": mCollectorGuest=%p\n",
12535 this, __PRETTY_FUNCTION__, mCollectorGuest));
12536
12537 /* Create sub metrics */
12538 pm::SubMetric *guestLoadUser = new pm::SubMetric("Guest/CPU/Load/User",
12539 "Percentage of processor time spent in user mode as seen by the guest.");
12540 pm::SubMetric *guestLoadKernel = new pm::SubMetric("Guest/CPU/Load/Kernel",
12541 "Percentage of processor time spent in kernel mode as seen by the guest.");
12542 pm::SubMetric *guestLoadIdle = new pm::SubMetric("Guest/CPU/Load/Idle",
12543 "Percentage of processor time spent idling as seen by the guest.");
12544
12545 /* The total amount of physical ram is fixed now, but we'll support dynamic guest ram configurations in the future. */
12546 pm::SubMetric *guestMemTotal = new pm::SubMetric("Guest/RAM/Usage/Total", "Total amount of physical guest RAM.");
12547 pm::SubMetric *guestMemFree = new pm::SubMetric("Guest/RAM/Usage/Free", "Free amount of physical guest RAM.");
12548 pm::SubMetric *guestMemBalloon = new pm::SubMetric("Guest/RAM/Usage/Balloon", "Amount of ballooned physical guest RAM.");
12549 pm::SubMetric *guestMemShared = new pm::SubMetric("Guest/RAM/Usage/Shared", "Amount of shared physical guest RAM.");
12550 pm::SubMetric *guestMemCache = new pm::SubMetric("Guest/RAM/Usage/Cache", "Total amount of guest (disk) cache memory.");
12551
12552 pm::SubMetric *guestPagedTotal = new pm::SubMetric("Guest/Pagefile/Usage/Total", "Total amount of space in the page file.");
12553
12554 /* Create and register base metrics */
12555 pm::BaseMetric *machineNetRate = new pm::MachineNetRate(mCollectorGuest, aMachine,
12556 machineNetRx, machineNetTx);
12557 aCollector->registerBaseMetric(machineNetRate);
12558
12559 pm::BaseMetric *guestCpuLoad = new pm::GuestCpuLoad(mCollectorGuest, aMachine,
12560 guestLoadUser, guestLoadKernel, guestLoadIdle);
12561 aCollector->registerBaseMetric(guestCpuLoad);
12562
12563 pm::BaseMetric *guestCpuMem = new pm::GuestRamUsage(mCollectorGuest, aMachine,
12564 guestMemTotal, guestMemFree,
12565 guestMemBalloon, guestMemShared,
12566 guestMemCache, guestPagedTotal);
12567 aCollector->registerBaseMetric(guestCpuMem);
12568
12569 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetRx, 0));
12570 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetRx, new pm::AggregateAvg()));
12571 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetRx, new pm::AggregateMin()));
12572 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetRx, new pm::AggregateMax()));
12573
12574 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetTx, 0));
12575 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetTx, new pm::AggregateAvg()));
12576 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetTx, new pm::AggregateMin()));
12577 aCollector->registerMetric(new pm::Metric(machineNetRate, machineNetTx, new pm::AggregateMax()));
12578
12579 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadUser, 0));
12580 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadUser, new pm::AggregateAvg()));
12581 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadUser, new pm::AggregateMin()));
12582 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadUser, new pm::AggregateMax()));
12583
12584 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadKernel, 0));
12585 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadKernel, new pm::AggregateAvg()));
12586 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadKernel, new pm::AggregateMin()));
12587 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadKernel, new pm::AggregateMax()));
12588
12589 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadIdle, 0));
12590 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadIdle, new pm::AggregateAvg()));
12591 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadIdle, new pm::AggregateMin()));
12592 aCollector->registerMetric(new pm::Metric(guestCpuLoad, guestLoadIdle, new pm::AggregateMax()));
12593
12594 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemTotal, 0));
12595 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemTotal, new pm::AggregateAvg()));
12596 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemTotal, new pm::AggregateMin()));
12597 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemTotal, new pm::AggregateMax()));
12598
12599 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemFree, 0));
12600 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemFree, new pm::AggregateAvg()));
12601 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemFree, new pm::AggregateMin()));
12602 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemFree, new pm::AggregateMax()));
12603
12604 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemBalloon, 0));
12605 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemBalloon, new pm::AggregateAvg()));
12606 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemBalloon, new pm::AggregateMin()));
12607 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemBalloon, new pm::AggregateMax()));
12608
12609 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemShared, 0));
12610 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemShared, new pm::AggregateAvg()));
12611 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemShared, new pm::AggregateMin()));
12612 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemShared, new pm::AggregateMax()));
12613
12614 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemCache, 0));
12615 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemCache, new pm::AggregateAvg()));
12616 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemCache, new pm::AggregateMin()));
12617 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestMemCache, new pm::AggregateMax()));
12618
12619 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestPagedTotal, 0));
12620 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestPagedTotal, new pm::AggregateAvg()));
12621 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestPagedTotal, new pm::AggregateMin()));
12622 aCollector->registerMetric(new pm::Metric(guestCpuMem, guestPagedTotal, new pm::AggregateMax()));
12623}
12624
12625void Machine::unregisterMetrics(PerformanceCollector *aCollector, Machine *aMachine)
12626{
12627 AssertReturnVoid(isWriteLockOnCurrentThread());
12628
12629 if (aCollector)
12630 {
12631 aCollector->unregisterMetricsFor(aMachine);
12632 aCollector->unregisterBaseMetricsFor(aMachine);
12633 }
12634}
12635
12636#endif /* VBOX_WITH_RESOURCE_USAGE_API */
12637
12638
12639////////////////////////////////////////////////////////////////////////////////
12640
12641DEFINE_EMPTY_CTOR_DTOR(SessionMachine)
12642
12643HRESULT SessionMachine::FinalConstruct()
12644{
12645 LogFlowThisFunc(("\n"));
12646
12647 mClientToken = NULL;
12648
12649 return BaseFinalConstruct();
12650}
12651
12652void SessionMachine::FinalRelease()
12653{
12654 LogFlowThisFunc(("\n"));
12655
12656 Assert(!mClientToken);
12657 /* paranoia, should not hang around any more */
12658 if (mClientToken)
12659 {
12660 delete mClientToken;
12661 mClientToken = NULL;
12662 }
12663
12664 uninit(Uninit::Unexpected);
12665
12666 BaseFinalRelease();
12667}
12668
12669/**
12670 * @note Must be called only by Machine::LockMachine() from its own write lock.
12671 */
12672HRESULT SessionMachine::init(Machine *aMachine)
12673{
12674 LogFlowThisFuncEnter();
12675 LogFlowThisFunc(("mName={%s}\n", aMachine->mUserData->s.strName.c_str()));
12676
12677 AssertReturn(aMachine, E_INVALIDARG);
12678
12679 AssertReturn(aMachine->lockHandle()->isWriteLockOnCurrentThread(), E_FAIL);
12680
12681 /* Enclose the state transition NotReady->InInit->Ready */
12682 AutoInitSpan autoInitSpan(this);
12683 AssertReturn(autoInitSpan.isOk(), E_FAIL);
12684
12685 HRESULT rc = S_OK;
12686
12687 /* create the machine client token */
12688 try
12689 {
12690 mClientToken = new ClientToken(aMachine, this);
12691 if (!mClientToken->isReady())
12692 {
12693 delete mClientToken;
12694 mClientToken = NULL;
12695 rc = E_FAIL;
12696 }
12697 }
12698 catch (std::bad_alloc &)
12699 {
12700 rc = E_OUTOFMEMORY;
12701 }
12702 if (FAILED(rc))
12703 return rc;
12704
12705 /* memorize the peer Machine */
12706 unconst(mPeer) = aMachine;
12707 /* share the parent pointer */
12708 unconst(mParent) = aMachine->mParent;
12709
12710 /* take the pointers to data to share */
12711 mData.share(aMachine->mData);
12712 mSSData.share(aMachine->mSSData);
12713
12714 mUserData.share(aMachine->mUserData);
12715 mHWData.share(aMachine->mHWData);
12716 mMediaData.share(aMachine->mMediaData);
12717
12718 mStorageControllers.allocate();
12719 for (StorageControllerList::const_iterator it = aMachine->mStorageControllers->begin();
12720 it != aMachine->mStorageControllers->end();
12721 ++it)
12722 {
12723 ComObjPtr<StorageController> ctl;
12724 ctl.createObject();
12725 ctl->init(this, *it);
12726 mStorageControllers->push_back(ctl);
12727 }
12728
12729 mUSBControllers.allocate();
12730 for (USBControllerList::const_iterator it = aMachine->mUSBControllers->begin();
12731 it != aMachine->mUSBControllers->end();
12732 ++it)
12733 {
12734 ComObjPtr<USBController> ctl;
12735 ctl.createObject();
12736 ctl->init(this, *it);
12737 mUSBControllers->push_back(ctl);
12738 }
12739
12740 unconst(mBIOSSettings).createObject();
12741 mBIOSSettings->init(this, aMachine->mBIOSSettings);
12742 /* create another VRDEServer object that will be mutable */
12743 unconst(mVRDEServer).createObject();
12744 mVRDEServer->init(this, aMachine->mVRDEServer);
12745 /* create another audio adapter object that will be mutable */
12746 unconst(mAudioAdapter).createObject();
12747 mAudioAdapter->init(this, aMachine->mAudioAdapter);
12748 /* create a list of serial ports that will be mutable */
12749 for (ULONG slot = 0; slot < RT_ELEMENTS(mSerialPorts); slot++)
12750 {
12751 unconst(mSerialPorts[slot]).createObject();
12752 mSerialPorts[slot]->init(this, aMachine->mSerialPorts[slot]);
12753 }
12754 /* create a list of parallel ports that will be mutable */
12755 for (ULONG slot = 0; slot < RT_ELEMENTS(mParallelPorts); slot++)
12756 {
12757 unconst(mParallelPorts[slot]).createObject();
12758 mParallelPorts[slot]->init(this, aMachine->mParallelPorts[slot]);
12759 }
12760
12761 /* create another USB device filters object that will be mutable */
12762 unconst(mUSBDeviceFilters).createObject();
12763 mUSBDeviceFilters->init(this, aMachine->mUSBDeviceFilters);
12764
12765 /* create a list of network adapters that will be mutable */
12766 mNetworkAdapters.resize(aMachine->mNetworkAdapters.size());
12767 for (ULONG slot = 0; slot < mNetworkAdapters.size(); slot++)
12768 {
12769 unconst(mNetworkAdapters[slot]).createObject();
12770 mNetworkAdapters[slot]->init(this, aMachine->mNetworkAdapters[slot]);
12771 }
12772
12773 /* create another bandwidth control object that will be mutable */
12774 unconst(mBandwidthControl).createObject();
12775 mBandwidthControl->init(this, aMachine->mBandwidthControl);
12776
12777 /* default is to delete saved state on Saved -> PoweredOff transition */
12778 mRemoveSavedState = true;
12779
12780 /* Confirm a successful initialization when it's the case */
12781 autoInitSpan.setSucceeded();
12782
12783 LogFlowThisFuncLeave();
12784 return rc;
12785}
12786
12787/**
12788 * Uninitializes this session object. If the reason is other than
12789 * Uninit::Unexpected, then this method MUST be called from #checkForDeath()
12790 * or the client watcher code.
12791 *
12792 * @param aReason uninitialization reason
12793 *
12794 * @note Locks mParent + this object for writing.
12795 */
12796void SessionMachine::uninit(Uninit::Reason aReason)
12797{
12798 LogFlowThisFuncEnter();
12799 LogFlowThisFunc(("reason=%d\n", aReason));
12800
12801 /*
12802 * Strongly reference ourselves to prevent this object deletion after
12803 * mData->mSession.mMachine.setNull() below (which can release the last
12804 * reference and call the destructor). Important: this must be done before
12805 * accessing any members (and before AutoUninitSpan that does it as well).
12806 * This self reference will be released as the very last step on return.
12807 */
12808 ComObjPtr<SessionMachine> selfRef = this;
12809
12810 /* Enclose the state transition Ready->InUninit->NotReady */
12811 AutoUninitSpan autoUninitSpan(this);
12812 if (autoUninitSpan.uninitDone())
12813 {
12814 LogFlowThisFunc(("Already uninitialized\n"));
12815 LogFlowThisFuncLeave();
12816 return;
12817 }
12818
12819 if (autoUninitSpan.initFailed())
12820 {
12821 /* We've been called by init() because it's failed. It's not really
12822 * necessary (nor it's safe) to perform the regular uninit sequence
12823 * below, the following is enough.
12824 */
12825 LogFlowThisFunc(("Initialization failed.\n"));
12826 /* destroy the machine client token */
12827 if (mClientToken)
12828 {
12829 delete mClientToken;
12830 mClientToken = NULL;
12831 }
12832 uninitDataAndChildObjects();
12833 mData.free();
12834 unconst(mParent) = NULL;
12835 unconst(mPeer) = NULL;
12836 LogFlowThisFuncLeave();
12837 return;
12838 }
12839
12840 MachineState_T lastState;
12841 {
12842 AutoReadLock tempLock(this COMMA_LOCKVAL_SRC_POS);
12843 lastState = mData->mMachineState;
12844 }
12845 NOREF(lastState);
12846
12847#ifdef VBOX_WITH_USB
12848 // release all captured USB devices, but do this before requesting the locks below
12849 if (aReason == Uninit::Abnormal && Global::IsOnline(lastState))
12850 {
12851 /* Console::captureUSBDevices() is called in the VM process only after
12852 * setting the machine state to Starting or Restoring.
12853 * Console::detachAllUSBDevices() will be called upon successful
12854 * termination. So, we need to release USB devices only if there was
12855 * an abnormal termination of a running VM.
12856 *
12857 * This is identical to SessionMachine::DetachAllUSBDevices except
12858 * for the aAbnormal argument. */
12859 HRESULT rc = mUSBDeviceFilters->notifyProxy(false /* aInsertFilters */);
12860 AssertComRC(rc);
12861 NOREF(rc);
12862
12863 USBProxyService *service = mParent->host()->usbProxyService();
12864 if (service)
12865 service->detachAllDevicesFromVM(this, true /* aDone */, true /* aAbnormal */);
12866 }
12867#endif /* VBOX_WITH_USB */
12868
12869 // we need to lock this object in uninit() because the lock is shared
12870 // with mPeer (as well as data we modify below). mParent->addProcessToReap()
12871 // and others need mParent lock, and USB needs host lock.
12872 AutoMultiWriteLock3 multilock(mParent, mParent->host(), this COMMA_LOCKVAL_SRC_POS);
12873
12874#ifdef VBOX_WITH_RESOURCE_USAGE_API
12875 /*
12876 * It is safe to call Machine::unregisterMetrics() here because
12877 * PerformanceCollector::samplerCallback no longer accesses guest methods
12878 * holding the lock.
12879 */
12880 unregisterMetrics(mParent->performanceCollector(), mPeer);
12881 /* The guest must be unregistered after its metrics (@bugref{5949}). */
12882 LogAleksey(("{%p} " LOG_FN_FMT ": mCollectorGuest=%p\n",
12883 this, __PRETTY_FUNCTION__, mCollectorGuest));
12884 if (mCollectorGuest)
12885 {
12886 mParent->performanceCollector()->unregisterGuest(mCollectorGuest);
12887 // delete mCollectorGuest; => CollectorGuestManager::destroyUnregistered()
12888 mCollectorGuest = NULL;
12889 }
12890#endif
12891
12892 if (aReason == Uninit::Abnormal)
12893 {
12894 LogWarningThisFunc(("ABNORMAL client termination! (wasBusy=%d)\n",
12895 Global::IsOnlineOrTransient(lastState)));
12896
12897 /* reset the state to Aborted */
12898 if (mData->mMachineState != MachineState_Aborted)
12899 setMachineState(MachineState_Aborted);
12900 }
12901
12902 // any machine settings modified?
12903 if (mData->flModifications)
12904 {
12905 LogWarningThisFunc(("Discarding unsaved settings changes!\n"));
12906 rollback(false /* aNotify */);
12907 }
12908
12909 Assert( mConsoleTaskData.strStateFilePath.isEmpty()
12910 || !mConsoleTaskData.mSnapshot);
12911 if (!mConsoleTaskData.strStateFilePath.isEmpty())
12912 {
12913 LogWarningThisFunc(("canceling failed save state request!\n"));
12914 endSavingState(E_FAIL, tr("Machine terminated with pending save state!"));
12915 }
12916 else if (!mConsoleTaskData.mSnapshot.isNull())
12917 {
12918 LogWarningThisFunc(("canceling untaken snapshot!\n"));
12919
12920 /* delete all differencing hard disks created (this will also attach
12921 * their parents back by rolling back mMediaData) */
12922 rollbackMedia();
12923
12924 // delete the saved state file (it might have been already created)
12925 // AFTER killing the snapshot so that releaseSavedStateFile() won't
12926 // think it's still in use
12927 Utf8Str strStateFile = mConsoleTaskData.mSnapshot->getStateFilePath();
12928 mConsoleTaskData.mSnapshot->uninit();
12929 releaseSavedStateFile(strStateFile, NULL /* pSnapshotToIgnore */ );
12930 }
12931
12932 if (!mData->mSession.mType.isEmpty())
12933 {
12934 /* mType is not null when this machine's process has been started by
12935 * Machine::LaunchVMProcess(), therefore it is our child. We
12936 * need to queue the PID to reap the process (and avoid zombies on
12937 * Linux). */
12938 Assert(mData->mSession.mPID != NIL_RTPROCESS);
12939 mParent->addProcessToReap(mData->mSession.mPID);
12940 }
12941
12942 mData->mSession.mPID = NIL_RTPROCESS;
12943
12944 if (aReason == Uninit::Unexpected)
12945 {
12946 /* Uninitialization didn't come from #checkForDeath(), so tell the
12947 * client watcher thread to update the set of machines that have open
12948 * sessions. */
12949 mParent->updateClientWatcher();
12950 }
12951
12952 /* uninitialize all remote controls */
12953 if (mData->mSession.mRemoteControls.size())
12954 {
12955 LogFlowThisFunc(("Closing remote sessions (%d):\n",
12956 mData->mSession.mRemoteControls.size()));
12957
12958 Data::Session::RemoteControlList::iterator it =
12959 mData->mSession.mRemoteControls.begin();
12960 while (it != mData->mSession.mRemoteControls.end())
12961 {
12962 LogFlowThisFunc((" Calling remoteControl->Uninitialize()...\n"));
12963 HRESULT rc = (*it)->Uninitialize();
12964 LogFlowThisFunc((" remoteControl->Uninitialize() returned %08X\n", rc));
12965 if (FAILED(rc))
12966 LogWarningThisFunc(("Forgot to close the remote session?\n"));
12967 ++it;
12968 }
12969 mData->mSession.mRemoteControls.clear();
12970 }
12971
12972 /*
12973 * An expected uninitialization can come only from #checkForDeath().
12974 * Otherwise it means that something's gone really wrong (for example,
12975 * the Session implementation has released the VirtualBox reference
12976 * before it triggered #OnSessionEnd(), or before releasing IPC semaphore,
12977 * etc). However, it's also possible, that the client releases the IPC
12978 * semaphore correctly (i.e. before it releases the VirtualBox reference),
12979 * but the VirtualBox release event comes first to the server process.
12980 * This case is practically possible, so we should not assert on an
12981 * unexpected uninit, just log a warning.
12982 */
12983
12984 if ((aReason == Uninit::Unexpected))
12985 LogWarningThisFunc(("Unexpected SessionMachine uninitialization!\n"));
12986
12987 if (aReason != Uninit::Normal)
12988 {
12989 mData->mSession.mDirectControl.setNull();
12990 }
12991 else
12992 {
12993 /* this must be null here (see #OnSessionEnd()) */
12994 Assert(mData->mSession.mDirectControl.isNull());
12995 Assert(mData->mSession.mState == SessionState_Unlocking);
12996 Assert(!mData->mSession.mProgress.isNull());
12997 }
12998 if (mData->mSession.mProgress)
12999 {
13000 if (aReason == Uninit::Normal)
13001 mData->mSession.mProgress->notifyComplete(S_OK);
13002 else
13003 mData->mSession.mProgress->notifyComplete(E_FAIL,
13004 COM_IIDOF(ISession),
13005 getComponentName(),
13006 tr("The VM session was aborted"));
13007 mData->mSession.mProgress.setNull();
13008 }
13009
13010 /* remove the association between the peer machine and this session machine */
13011 Assert( (SessionMachine*)mData->mSession.mMachine == this
13012 || aReason == Uninit::Unexpected);
13013
13014 /* reset the rest of session data */
13015 mData->mSession.mMachine.setNull();
13016 mData->mSession.mState = SessionState_Unlocked;
13017 mData->mSession.mType.setNull();
13018
13019 /* destroy the machine client token before leaving the exclusive lock */
13020 if (mClientToken)
13021 {
13022 delete mClientToken;
13023 mClientToken = NULL;
13024 }
13025
13026 /* fire an event */
13027 mParent->onSessionStateChange(mData->mUuid, SessionState_Unlocked);
13028
13029 uninitDataAndChildObjects();
13030
13031 /* free the essential data structure last */
13032 mData.free();
13033
13034 /* release the exclusive lock before setting the below two to NULL */
13035 multilock.release();
13036
13037 unconst(mParent) = NULL;
13038 unconst(mPeer) = NULL;
13039
13040 LogFlowThisFuncLeave();
13041}
13042
13043// util::Lockable interface
13044////////////////////////////////////////////////////////////////////////////////
13045
13046/**
13047 * Overrides VirtualBoxBase::lockHandle() in order to share the lock handle
13048 * with the primary Machine instance (mPeer).
13049 */
13050RWLockHandle *SessionMachine::lockHandle() const
13051{
13052 AssertReturn(mPeer != NULL, NULL);
13053 return mPeer->lockHandle();
13054}
13055
13056// IInternalMachineControl methods
13057////////////////////////////////////////////////////////////////////////////////
13058
13059/**
13060 * Passes collected guest statistics to performance collector object
13061 */
13062STDMETHODIMP SessionMachine::ReportVmStatistics(ULONG aValidStats, ULONG aCpuUser,
13063 ULONG aCpuKernel, ULONG aCpuIdle,
13064 ULONG aMemTotal, ULONG aMemFree,
13065 ULONG aMemBalloon, ULONG aMemShared,
13066 ULONG aMemCache, ULONG aPageTotal,
13067 ULONG aAllocVMM, ULONG aFreeVMM,
13068 ULONG aBalloonedVMM, ULONG aSharedVMM,
13069 ULONG aVmNetRx, ULONG aVmNetTx)
13070{
13071#ifdef VBOX_WITH_RESOURCE_USAGE_API
13072 if (mCollectorGuest)
13073 mCollectorGuest->updateStats(aValidStats, aCpuUser, aCpuKernel, aCpuIdle,
13074 aMemTotal, aMemFree, aMemBalloon, aMemShared,
13075 aMemCache, aPageTotal, aAllocVMM, aFreeVMM,
13076 aBalloonedVMM, aSharedVMM, aVmNetRx, aVmNetTx);
13077
13078 return S_OK;
13079#else
13080 NOREF(aValidStats);
13081 NOREF(aCpuUser);
13082 NOREF(aCpuKernel);
13083 NOREF(aCpuIdle);
13084 NOREF(aMemTotal);
13085 NOREF(aMemFree);
13086 NOREF(aMemBalloon);
13087 NOREF(aMemShared);
13088 NOREF(aMemCache);
13089 NOREF(aPageTotal);
13090 NOREF(aAllocVMM);
13091 NOREF(aFreeVMM);
13092 NOREF(aBalloonedVMM);
13093 NOREF(aSharedVMM);
13094 NOREF(aVmNetRx);
13095 NOREF(aVmNetTx);
13096 return E_NOTIMPL;
13097#endif
13098}
13099
13100/**
13101 * @note Locks this object for writing.
13102 */
13103STDMETHODIMP SessionMachine::SetRemoveSavedStateFile(BOOL aRemove)
13104{
13105 AutoCaller autoCaller(this);
13106 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13107
13108 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13109
13110 mRemoveSavedState = aRemove;
13111
13112 return S_OK;
13113}
13114
13115/**
13116 * @note Locks the same as #setMachineState() does.
13117 */
13118STDMETHODIMP SessionMachine::UpdateState(MachineState_T aMachineState)
13119{
13120 return setMachineState(aMachineState);
13121}
13122
13123/**
13124 * @note Locks this object for writing.
13125 */
13126STDMETHODIMP SessionMachine::BeginPowerUp(IProgress *aProgress)
13127{
13128 LogFlowThisFunc(("aProgress=%p\n", aProgress));
13129 AutoCaller autoCaller(this);
13130 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13131
13132 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13133
13134 if (mData->mSession.mState != SessionState_Locked)
13135 return VBOX_E_INVALID_OBJECT_STATE;
13136
13137 if (!mData->mSession.mProgress.isNull())
13138 mData->mSession.mProgress->setOtherProgressObject(aProgress);
13139
13140 LogFlowThisFunc(("returns S_OK.\n"));
13141 return S_OK;
13142}
13143
13144/**
13145 * @note Locks this object for writing.
13146 */
13147STDMETHODIMP SessionMachine::EndPowerUp(LONG iResult)
13148{
13149 AutoCaller autoCaller(this);
13150 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13151
13152 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13153
13154 if (mData->mSession.mState != SessionState_Locked)
13155 return VBOX_E_INVALID_OBJECT_STATE;
13156
13157 /* Finalize the LaunchVMProcess progress object. */
13158 if (mData->mSession.mProgress)
13159 {
13160 mData->mSession.mProgress->notifyComplete((HRESULT)iResult);
13161 mData->mSession.mProgress.setNull();
13162 }
13163
13164 if (SUCCEEDED((HRESULT)iResult))
13165 {
13166#ifdef VBOX_WITH_RESOURCE_USAGE_API
13167 /* The VM has been powered up successfully, so it makes sense
13168 * now to offer the performance metrics for a running machine
13169 * object. Doing it earlier wouldn't be safe. */
13170 registerMetrics(mParent->performanceCollector(), mPeer,
13171 mData->mSession.mPID);
13172#endif /* VBOX_WITH_RESOURCE_USAGE_API */
13173 }
13174
13175 return S_OK;
13176}
13177
13178/**
13179 * @note Locks this object for writing.
13180 */
13181STDMETHODIMP SessionMachine::BeginPoweringDown(IProgress **aProgress)
13182{
13183 LogFlowThisFuncEnter();
13184
13185 CheckComArgOutPointerValid(aProgress);
13186
13187 AutoCaller autoCaller(this);
13188 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13189
13190 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13191
13192 AssertReturn(mConsoleTaskData.mLastState == MachineState_Null,
13193 E_FAIL);
13194
13195 /* create a progress object to track operation completion */
13196 ComObjPtr<Progress> pProgress;
13197 pProgress.createObject();
13198 pProgress->init(getVirtualBox(),
13199 static_cast<IMachine *>(this) /* aInitiator */,
13200 Bstr(tr("Stopping the virtual machine")).raw(),
13201 FALSE /* aCancelable */);
13202
13203 /* fill in the console task data */
13204 mConsoleTaskData.mLastState = mData->mMachineState;
13205 mConsoleTaskData.mProgress = pProgress;
13206
13207 /* set the state to Stopping (this is expected by Console::PowerDown()) */
13208 setMachineState(MachineState_Stopping);
13209
13210 pProgress.queryInterfaceTo(aProgress);
13211
13212 return S_OK;
13213}
13214
13215/**
13216 * @note Locks this object for writing.
13217 */
13218STDMETHODIMP SessionMachine::EndPoweringDown(LONG iResult, IN_BSTR aErrMsg)
13219{
13220 LogFlowThisFuncEnter();
13221
13222 AutoCaller autoCaller(this);
13223 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13224
13225 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13226
13227 AssertReturn( ( (SUCCEEDED(iResult) && mData->mMachineState == MachineState_PoweredOff)
13228 || (FAILED(iResult) && mData->mMachineState == MachineState_Stopping))
13229 && mConsoleTaskData.mLastState != MachineState_Null,
13230 E_FAIL);
13231
13232 /*
13233 * On failure, set the state to the state we had when BeginPoweringDown()
13234 * was called (this is expected by Console::PowerDown() and the associated
13235 * task). On success the VM process already changed the state to
13236 * MachineState_PoweredOff, so no need to do anything.
13237 */
13238 if (FAILED(iResult))
13239 setMachineState(mConsoleTaskData.mLastState);
13240
13241 /* notify the progress object about operation completion */
13242 Assert(mConsoleTaskData.mProgress);
13243 if (SUCCEEDED(iResult))
13244 mConsoleTaskData.mProgress->notifyComplete(S_OK);
13245 else
13246 {
13247 Utf8Str strErrMsg(aErrMsg);
13248 if (strErrMsg.length())
13249 mConsoleTaskData.mProgress->notifyComplete(iResult,
13250 COM_IIDOF(ISession),
13251 getComponentName(),
13252 strErrMsg.c_str());
13253 else
13254 mConsoleTaskData.mProgress->notifyComplete(iResult);
13255 }
13256
13257 /* clear out the temporary saved state data */
13258 mConsoleTaskData.mLastState = MachineState_Null;
13259 mConsoleTaskData.mProgress.setNull();
13260
13261 LogFlowThisFuncLeave();
13262 return S_OK;
13263}
13264
13265
13266/**
13267 * Goes through the USB filters of the given machine to see if the given
13268 * device matches any filter or not.
13269 *
13270 * @note Locks the same as USBController::hasMatchingFilter() does.
13271 */
13272STDMETHODIMP SessionMachine::RunUSBDeviceFilters(IUSBDevice *aUSBDevice,
13273 BOOL *aMatched,
13274 ULONG *aMaskedIfs)
13275{
13276 LogFlowThisFunc(("\n"));
13277
13278 CheckComArgNotNull(aUSBDevice);
13279 CheckComArgOutPointerValid(aMatched);
13280
13281 AutoCaller autoCaller(this);
13282 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13283
13284#ifdef VBOX_WITH_USB
13285 *aMatched = mUSBDeviceFilters->hasMatchingFilter(aUSBDevice, aMaskedIfs);
13286#else
13287 NOREF(aUSBDevice);
13288 NOREF(aMaskedIfs);
13289 *aMatched = FALSE;
13290#endif
13291
13292 return S_OK;
13293}
13294
13295/**
13296 * @note Locks the same as Host::captureUSBDevice() does.
13297 */
13298STDMETHODIMP SessionMachine::CaptureUSBDevice(IN_BSTR aId)
13299{
13300 LogFlowThisFunc(("\n"));
13301
13302 AutoCaller autoCaller(this);
13303 AssertComRCReturnRC(autoCaller.rc());
13304
13305#ifdef VBOX_WITH_USB
13306 /* if captureDeviceForVM() fails, it must have set extended error info */
13307 clearError();
13308 MultiResult rc = mParent->host()->checkUSBProxyService();
13309 if (FAILED(rc)) return rc;
13310
13311 USBProxyService *service = mParent->host()->usbProxyService();
13312 AssertReturn(service, E_FAIL);
13313 return service->captureDeviceForVM(this, Guid(aId).ref());
13314#else
13315 NOREF(aId);
13316 return E_NOTIMPL;
13317#endif
13318}
13319
13320/**
13321 * @note Locks the same as Host::detachUSBDevice() does.
13322 */
13323STDMETHODIMP SessionMachine::DetachUSBDevice(IN_BSTR aId, BOOL aDone)
13324{
13325 LogFlowThisFunc(("\n"));
13326
13327 AutoCaller autoCaller(this);
13328 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13329
13330#ifdef VBOX_WITH_USB
13331 USBProxyService *service = mParent->host()->usbProxyService();
13332 AssertReturn(service, E_FAIL);
13333 return service->detachDeviceFromVM(this, Guid(aId).ref(), !!aDone);
13334#else
13335 NOREF(aId);
13336 NOREF(aDone);
13337 return E_NOTIMPL;
13338#endif
13339}
13340
13341/**
13342 * Inserts all machine filters to the USB proxy service and then calls
13343 * Host::autoCaptureUSBDevices().
13344 *
13345 * Called by Console from the VM process upon VM startup.
13346 *
13347 * @note Locks what called methods lock.
13348 */
13349STDMETHODIMP SessionMachine::AutoCaptureUSBDevices()
13350{
13351 LogFlowThisFunc(("\n"));
13352
13353 AutoCaller autoCaller(this);
13354 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13355
13356#ifdef VBOX_WITH_USB
13357 HRESULT rc = mUSBDeviceFilters->notifyProxy(true /* aInsertFilters */);
13358 AssertComRC(rc);
13359 NOREF(rc);
13360
13361 USBProxyService *service = mParent->host()->usbProxyService();
13362 AssertReturn(service, E_FAIL);
13363 return service->autoCaptureDevicesForVM(this);
13364#else
13365 return S_OK;
13366#endif
13367}
13368
13369/**
13370 * Removes all machine filters from the USB proxy service and then calls
13371 * Host::detachAllUSBDevices().
13372 *
13373 * Called by Console from the VM process upon normal VM termination or by
13374 * SessionMachine::uninit() upon abnormal VM termination (from under the
13375 * Machine/SessionMachine lock).
13376 *
13377 * @note Locks what called methods lock.
13378 */
13379STDMETHODIMP SessionMachine::DetachAllUSBDevices(BOOL aDone)
13380{
13381 LogFlowThisFunc(("\n"));
13382
13383 AutoCaller autoCaller(this);
13384 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13385
13386#ifdef VBOX_WITH_USB
13387 HRESULT rc = mUSBDeviceFilters->notifyProxy(false /* aInsertFilters */);
13388 AssertComRC(rc);
13389 NOREF(rc);
13390
13391 USBProxyService *service = mParent->host()->usbProxyService();
13392 AssertReturn(service, E_FAIL);
13393 return service->detachAllDevicesFromVM(this, !!aDone, false /* aAbnormal */);
13394#else
13395 NOREF(aDone);
13396 return S_OK;
13397#endif
13398}
13399
13400/**
13401 * @note Locks this object for writing.
13402 */
13403STDMETHODIMP SessionMachine::OnSessionEnd(ISession *aSession,
13404 IProgress **aProgress)
13405{
13406 LogFlowThisFuncEnter();
13407
13408 AssertReturn(aSession, E_INVALIDARG);
13409 AssertReturn(aProgress, E_INVALIDARG);
13410
13411 AutoCaller autoCaller(this);
13412
13413 LogFlowThisFunc(("callerstate=%d\n", autoCaller.state()));
13414 /*
13415 * We don't assert below because it might happen that a non-direct session
13416 * informs us it is closed right after we've been uninitialized -- it's ok.
13417 */
13418 if (FAILED(autoCaller.rc())) return autoCaller.rc();
13419
13420 /* get IInternalSessionControl interface */
13421 ComPtr<IInternalSessionControl> control(aSession);
13422
13423 ComAssertRet(!control.isNull(), E_INVALIDARG);
13424
13425 /* Creating a Progress object requires the VirtualBox lock, and
13426 * thus locking it here is required by the lock order rules. */
13427 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
13428
13429 if (control == mData->mSession.mDirectControl)
13430 {
13431 ComAssertRet(aProgress, E_POINTER);
13432
13433 /* The direct session is being normally closed by the client process
13434 * ----------------------------------------------------------------- */
13435
13436 /* go to the closing state (essential for all open*Session() calls and
13437 * for #checkForDeath()) */
13438 Assert(mData->mSession.mState == SessionState_Locked);
13439 mData->mSession.mState = SessionState_Unlocking;
13440
13441 /* set direct control to NULL to release the remote instance */
13442 mData->mSession.mDirectControl.setNull();
13443 LogFlowThisFunc(("Direct control is set to NULL\n"));
13444
13445 if (mData->mSession.mProgress)
13446 {
13447 /* finalize the progress, someone might wait if a frontend
13448 * closes the session before powering on the VM. */
13449 mData->mSession.mProgress->notifyComplete(E_FAIL,
13450 COM_IIDOF(ISession),
13451 getComponentName(),
13452 tr("The VM session was closed before any attempt to power it on"));
13453 mData->mSession.mProgress.setNull();
13454 }
13455
13456 /* Create the progress object the client will use to wait until
13457 * #checkForDeath() is called to uninitialize this session object after
13458 * it releases the IPC semaphore.
13459 * Note! Because we're "reusing" mProgress here, this must be a proxy
13460 * object just like for LaunchVMProcess. */
13461 Assert(mData->mSession.mProgress.isNull());
13462 ComObjPtr<ProgressProxy> progress;
13463 progress.createObject();
13464 ComPtr<IUnknown> pPeer(mPeer);
13465 progress->init(mParent, pPeer,
13466 Bstr(tr("Closing session")).raw(),
13467 FALSE /* aCancelable */);
13468 progress.queryInterfaceTo(aProgress);
13469 mData->mSession.mProgress = progress;
13470 }
13471 else
13472 {
13473 /* the remote session is being normally closed */
13474 Data::Session::RemoteControlList::iterator it =
13475 mData->mSession.mRemoteControls.begin();
13476 while (it != mData->mSession.mRemoteControls.end())
13477 {
13478 if (control == *it)
13479 break;
13480 ++it;
13481 }
13482 BOOL found = it != mData->mSession.mRemoteControls.end();
13483 ComAssertMsgRet(found, ("The session is not found in the session list!"),
13484 E_INVALIDARG);
13485 // This MUST be erase(it), not remove(*it) as the latter triggers a
13486 // very nasty use after free due to the place where the value "lives".
13487 mData->mSession.mRemoteControls.erase(it);
13488 }
13489
13490 /* signal the client watcher thread, because the client is going away */
13491 mParent->updateClientWatcher();
13492
13493 LogFlowThisFuncLeave();
13494 return S_OK;
13495}
13496
13497/**
13498 * @note Locks this object for writing.
13499 */
13500STDMETHODIMP SessionMachine::BeginSavingState(IProgress **aProgress, BSTR *aStateFilePath)
13501{
13502 LogFlowThisFuncEnter();
13503
13504 CheckComArgOutPointerValid(aProgress);
13505 CheckComArgOutPointerValid(aStateFilePath);
13506
13507 AutoCaller autoCaller(this);
13508 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13509
13510 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13511
13512 AssertReturn( mData->mMachineState == MachineState_Paused
13513 && mConsoleTaskData.mLastState == MachineState_Null
13514 && mConsoleTaskData.strStateFilePath.isEmpty(),
13515 E_FAIL);
13516
13517 /* create a progress object to track operation completion */
13518 ComObjPtr<Progress> pProgress;
13519 pProgress.createObject();
13520 pProgress->init(getVirtualBox(),
13521 static_cast<IMachine *>(this) /* aInitiator */,
13522 Bstr(tr("Saving the execution state of the virtual machine")).raw(),
13523 FALSE /* aCancelable */);
13524
13525 Utf8Str strStateFilePath;
13526 /* stateFilePath is null when the machine is not running */
13527 if (mData->mMachineState == MachineState_Paused)
13528 composeSavedStateFilename(strStateFilePath);
13529
13530 /* fill in the console task data */
13531 mConsoleTaskData.mLastState = mData->mMachineState;
13532 mConsoleTaskData.strStateFilePath = strStateFilePath;
13533 mConsoleTaskData.mProgress = pProgress;
13534
13535 /* set the state to Saving (this is expected by Console::SaveState()) */
13536 setMachineState(MachineState_Saving);
13537
13538 strStateFilePath.cloneTo(aStateFilePath);
13539 pProgress.queryInterfaceTo(aProgress);
13540
13541 return S_OK;
13542}
13543
13544/**
13545 * @note Locks mParent + this object for writing.
13546 */
13547STDMETHODIMP SessionMachine::EndSavingState(LONG iResult, IN_BSTR aErrMsg)
13548{
13549 LogFlowThisFunc(("\n"));
13550
13551 AutoCaller autoCaller(this);
13552 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13553
13554 /* endSavingState() need mParent lock */
13555 AutoMultiWriteLock2 alock(mParent, this COMMA_LOCKVAL_SRC_POS);
13556
13557 AssertReturn( ( (SUCCEEDED(iResult) && mData->mMachineState == MachineState_Saved)
13558 || (FAILED(iResult) && mData->mMachineState == MachineState_Saving))
13559 && mConsoleTaskData.mLastState != MachineState_Null
13560 && !mConsoleTaskData.strStateFilePath.isEmpty(),
13561 E_FAIL);
13562
13563 /*
13564 * On failure, set the state to the state we had when BeginSavingState()
13565 * was called (this is expected by Console::SaveState() and the associated
13566 * task). On success the VM process already changed the state to
13567 * MachineState_Saved, so no need to do anything.
13568 */
13569 if (FAILED(iResult))
13570 setMachineState(mConsoleTaskData.mLastState);
13571
13572 return endSavingState(iResult, aErrMsg);
13573}
13574
13575/**
13576 * @note Locks this object for writing.
13577 */
13578STDMETHODIMP SessionMachine::AdoptSavedState(IN_BSTR aSavedStateFile)
13579{
13580 LogFlowThisFunc(("\n"));
13581
13582 CheckComArgStrNotEmptyOrNull(aSavedStateFile);
13583
13584 AutoCaller autoCaller(this);
13585 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13586
13587 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13588
13589 AssertReturn( mData->mMachineState == MachineState_PoweredOff
13590 || mData->mMachineState == MachineState_Teleported
13591 || mData->mMachineState == MachineState_Aborted
13592 , E_FAIL); /** @todo setError. */
13593
13594 Utf8Str stateFilePathFull = aSavedStateFile;
13595 int vrc = calculateFullPath(stateFilePathFull, stateFilePathFull);
13596 if (RT_FAILURE(vrc))
13597 return setError(VBOX_E_FILE_ERROR,
13598 tr("Invalid saved state file path '%ls' (%Rrc)"),
13599 aSavedStateFile,
13600 vrc);
13601
13602 mSSData->strStateFilePath = stateFilePathFull;
13603
13604 /* The below setMachineState() will detect the state transition and will
13605 * update the settings file */
13606
13607 return setMachineState(MachineState_Saved);
13608}
13609
13610STDMETHODIMP SessionMachine::PullGuestProperties(ComSafeArrayOut(BSTR, aNames),
13611 ComSafeArrayOut(BSTR, aValues),
13612 ComSafeArrayOut(LONG64, aTimestamps),
13613 ComSafeArrayOut(BSTR, aFlags))
13614{
13615 LogFlowThisFunc(("\n"));
13616
13617#ifdef VBOX_WITH_GUEST_PROPS
13618 using namespace guestProp;
13619
13620 AutoCaller autoCaller(this);
13621 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13622
13623 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
13624
13625 CheckComArgOutSafeArrayPointerValid(aNames);
13626 CheckComArgOutSafeArrayPointerValid(aValues);
13627 CheckComArgOutSafeArrayPointerValid(aTimestamps);
13628 CheckComArgOutSafeArrayPointerValid(aFlags);
13629
13630 size_t cEntries = mHWData->mGuestProperties.size();
13631 com::SafeArray<BSTR> names(cEntries);
13632 com::SafeArray<BSTR> values(cEntries);
13633 com::SafeArray<LONG64> timestamps(cEntries);
13634 com::SafeArray<BSTR> flags(cEntries);
13635 unsigned i = 0;
13636 for (HWData::GuestPropertyMap::iterator it = mHWData->mGuestProperties.begin();
13637 it != mHWData->mGuestProperties.end();
13638 ++it)
13639 {
13640 char szFlags[MAX_FLAGS_LEN + 1];
13641 it->first.cloneTo(&names[i]);
13642 it->second.strValue.cloneTo(&values[i]);
13643 timestamps[i] = it->second.mTimestamp;
13644 /* If it is NULL, keep it NULL. */
13645 if (it->second.mFlags)
13646 {
13647 writeFlags(it->second.mFlags, szFlags);
13648 Bstr(szFlags).cloneTo(&flags[i]);
13649 }
13650 else
13651 flags[i] = NULL;
13652 ++i;
13653 }
13654 names.detachTo(ComSafeArrayOutArg(aNames));
13655 values.detachTo(ComSafeArrayOutArg(aValues));
13656 timestamps.detachTo(ComSafeArrayOutArg(aTimestamps));
13657 flags.detachTo(ComSafeArrayOutArg(aFlags));
13658 return S_OK;
13659#else
13660 ReturnComNotImplemented();
13661#endif
13662}
13663
13664STDMETHODIMP SessionMachine::PushGuestProperty(IN_BSTR aName,
13665 IN_BSTR aValue,
13666 LONG64 aTimestamp,
13667 IN_BSTR aFlags)
13668{
13669 LogFlowThisFunc(("\n"));
13670
13671#ifdef VBOX_WITH_GUEST_PROPS
13672 using namespace guestProp;
13673
13674 CheckComArgStrNotEmptyOrNull(aName);
13675 CheckComArgNotNull(aValue);
13676 CheckComArgNotNull(aFlags);
13677
13678 try
13679 {
13680 /*
13681 * Convert input up front.
13682 */
13683 Utf8Str utf8Name(aName);
13684 uint32_t fFlags = NILFLAG;
13685 if (aFlags)
13686 {
13687 Utf8Str utf8Flags(aFlags);
13688 int vrc = validateFlags(utf8Flags.c_str(), &fFlags);
13689 AssertRCReturn(vrc, E_INVALIDARG);
13690 }
13691
13692 /*
13693 * Now grab the object lock, validate the state and do the update.
13694 */
13695 AutoCaller autoCaller(this);
13696 if (FAILED(autoCaller.rc())) return autoCaller.rc();
13697
13698 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13699
13700 switch (mData->mMachineState)
13701 {
13702 case MachineState_Paused:
13703 case MachineState_Running:
13704 case MachineState_Teleporting:
13705 case MachineState_TeleportingPausedVM:
13706 case MachineState_LiveSnapshotting:
13707 case MachineState_DeletingSnapshotOnline:
13708 case MachineState_DeletingSnapshotPaused:
13709 case MachineState_Saving:
13710 case MachineState_Stopping:
13711 break;
13712
13713 default:
13714 AssertMsgFailedReturn(("%s\n", Global::stringifyMachineState(mData->mMachineState)),
13715 VBOX_E_INVALID_VM_STATE);
13716 }
13717
13718 setModified(IsModified_MachineData);
13719 mHWData.backup();
13720
13721 bool fDelete = !RT_VALID_PTR(aValue) || *(aValue) == '\0';
13722 HWData::GuestPropertyMap::iterator it = mHWData->mGuestProperties.find(utf8Name);
13723 if (it != mHWData->mGuestProperties.end())
13724 {
13725 if (!fDelete)
13726 {
13727 it->second.strValue = aValue;
13728 it->second.mTimestamp = aTimestamp;
13729 it->second.mFlags = fFlags;
13730 }
13731 else
13732 mHWData->mGuestProperties.erase(it);
13733
13734 mData->mGuestPropertiesModified = TRUE;
13735 }
13736 else if (!fDelete)
13737 {
13738 HWData::GuestProperty prop;
13739 prop.strValue = aValue;
13740 prop.mTimestamp = aTimestamp;
13741 prop.mFlags = fFlags;
13742
13743 mHWData->mGuestProperties[utf8Name] = prop;
13744 mData->mGuestPropertiesModified = TRUE;
13745 }
13746
13747 /*
13748 * Send a callback notification if appropriate
13749 */
13750 if ( mHWData->mGuestPropertyNotificationPatterns.isEmpty()
13751 || RTStrSimplePatternMultiMatch(mHWData->mGuestPropertyNotificationPatterns.c_str(),
13752 RTSTR_MAX,
13753 utf8Name.c_str(),
13754 RTSTR_MAX, NULL)
13755 )
13756 {
13757 alock.release();
13758
13759 mParent->onGuestPropertyChange(mData->mUuid,
13760 aName,
13761 aValue,
13762 aFlags);
13763 }
13764 }
13765 catch (...)
13766 {
13767 return VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);
13768 }
13769 return S_OK;
13770#else
13771 ReturnComNotImplemented();
13772#endif
13773}
13774
13775STDMETHODIMP SessionMachine::LockMedia()
13776{
13777 AutoCaller autoCaller(this);
13778 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13779
13780 AutoMultiWriteLock2 alock(this->lockHandle(),
13781 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
13782
13783 AssertReturn( mData->mMachineState == MachineState_Starting
13784 || mData->mMachineState == MachineState_Restoring
13785 || mData->mMachineState == MachineState_TeleportingIn, E_FAIL);
13786
13787 clearError();
13788 alock.release();
13789 return lockMedia();
13790}
13791
13792STDMETHODIMP SessionMachine::UnlockMedia()
13793{
13794 unlockMedia();
13795 return S_OK;
13796}
13797
13798STDMETHODIMP SessionMachine::EjectMedium(IMediumAttachment *aAttachment,
13799 IMediumAttachment **aNewAttachment)
13800{
13801 CheckComArgNotNull(aAttachment);
13802 CheckComArgOutPointerValid(aNewAttachment);
13803
13804 AutoCaller autoCaller(this);
13805 if (FAILED(autoCaller.rc())) return autoCaller.rc();
13806
13807 // request the host lock first, since might be calling Host methods for getting host drives;
13808 // next, protect the media tree all the while we're in here, as well as our member variables
13809 AutoMultiWriteLock3 multiLock(mParent->host()->lockHandle(),
13810 this->lockHandle(),
13811 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
13812
13813 ComObjPtr<MediumAttachment> pAttach = static_cast<MediumAttachment *>(aAttachment);
13814
13815 Bstr ctrlName;
13816 LONG lPort;
13817 LONG lDevice;
13818 bool fTempEject;
13819 {
13820 AutoCaller autoAttachCaller(this);
13821 if (FAILED(autoAttachCaller.rc())) return autoAttachCaller.rc();
13822
13823 AutoReadLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
13824
13825 /* Need to query the details first, as the IMediumAttachment reference
13826 * might be to the original settings, which we are going to change. */
13827 ctrlName = pAttach->getControllerName();
13828 lPort = pAttach->getPort();
13829 lDevice = pAttach->getDevice();
13830 fTempEject = pAttach->getTempEject();
13831 }
13832
13833 if (!fTempEject)
13834 {
13835 /* Remember previously mounted medium. The medium before taking the
13836 * backup is not necessarily the same thing. */
13837 ComObjPtr<Medium> oldmedium;
13838 oldmedium = pAttach->getMedium();
13839
13840 setModified(IsModified_Storage);
13841 mMediaData.backup();
13842
13843 // The backup operation makes the pAttach reference point to the
13844 // old settings. Re-get the correct reference.
13845 pAttach = findAttachment(mMediaData->mAttachments,
13846 ctrlName.raw(),
13847 lPort,
13848 lDevice);
13849
13850 {
13851 AutoCaller autoAttachCaller(this);
13852 if (FAILED(autoAttachCaller.rc())) return autoAttachCaller.rc();
13853
13854 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
13855 if (!oldmedium.isNull())
13856 oldmedium->removeBackReference(mData->mUuid);
13857
13858 pAttach->updateMedium(NULL);
13859 pAttach->updateEjected();
13860 }
13861
13862 setModified(IsModified_Storage);
13863 }
13864 else
13865 {
13866 {
13867 AutoWriteLock attLock(pAttach COMMA_LOCKVAL_SRC_POS);
13868 pAttach->updateEjected();
13869 }
13870 }
13871
13872 pAttach.queryInterfaceTo(aNewAttachment);
13873
13874 return S_OK;
13875}
13876
13877// public methods only for internal purposes
13878/////////////////////////////////////////////////////////////////////////////
13879
13880#ifndef VBOX_WITH_GENERIC_SESSION_WATCHER
13881/**
13882 * Called from the client watcher thread to check for expected or unexpected
13883 * death of the client process that has a direct session to this machine.
13884 *
13885 * On Win32 and on OS/2, this method is called only when we've got the
13886 * mutex (i.e. the client has either died or terminated normally) so it always
13887 * returns @c true (the client is terminated, the session machine is
13888 * uninitialized).
13889 *
13890 * On other platforms, the method returns @c true if the client process has
13891 * terminated normally or abnormally and the session machine was uninitialized,
13892 * and @c false if the client process is still alive.
13893 *
13894 * @note Locks this object for writing.
13895 */
13896bool SessionMachine::checkForDeath()
13897{
13898 Uninit::Reason reason;
13899 bool terminated = false;
13900
13901 /* Enclose autoCaller with a block because calling uninit() from under it
13902 * will deadlock. */
13903 {
13904 AutoCaller autoCaller(this);
13905 if (!autoCaller.isOk())
13906 {
13907 /* return true if not ready, to cause the client watcher to exclude
13908 * the corresponding session from watching */
13909 LogFlowThisFunc(("Already uninitialized!\n"));
13910 return true;
13911 }
13912
13913 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
13914
13915 /* Determine the reason of death: if the session state is Closing here,
13916 * everything is fine. Otherwise it means that the client did not call
13917 * OnSessionEnd() before it released the IPC semaphore. This may happen
13918 * either because the client process has abnormally terminated, or
13919 * because it simply forgot to call ISession::Close() before exiting. We
13920 * threat the latter also as an abnormal termination (see
13921 * Session::uninit() for details). */
13922 reason = mData->mSession.mState == SessionState_Unlocking ?
13923 Uninit::Normal :
13924 Uninit::Abnormal;
13925
13926 if (mClientToken)
13927 terminated = mClientToken->release();
13928 } /* AutoCaller block */
13929
13930 if (terminated)
13931 uninit(reason);
13932
13933 return terminated;
13934}
13935
13936void SessionMachine::getTokenId(Utf8Str &strTokenId)
13937{
13938 LogFlowThisFunc(("\n"));
13939
13940 strTokenId.setNull();
13941
13942 AutoCaller autoCaller(this);
13943 AssertComRCReturnVoid(autoCaller.rc());
13944
13945 Assert(mClientToken);
13946 if (mClientToken)
13947 mClientToken->getId(strTokenId);
13948}
13949#else /* VBOX_WITH_GENERIC_SESSION_WATCHER */
13950IToken *SessionMachine::getToken()
13951{
13952 LogFlowThisFunc(("\n"));
13953
13954 AutoCaller autoCaller(this);
13955 AssertComRCReturn(autoCaller.rc(), NULL);
13956
13957 Assert(mClientToken);
13958 if (mClientToken)
13959 return mClientToken->getToken();
13960 else
13961 return NULL;
13962}
13963#endif /* VBOX_WITH_GENERIC_SESSION_WATCHER */
13964
13965Machine::ClientToken *SessionMachine::getClientToken()
13966{
13967 LogFlowThisFunc(("\n"));
13968
13969 AutoCaller autoCaller(this);
13970 AssertComRCReturn(autoCaller.rc(), NULL);
13971
13972 return mClientToken;
13973}
13974
13975
13976/**
13977 * @note Locks this object for reading.
13978 */
13979HRESULT SessionMachine::onNetworkAdapterChange(INetworkAdapter *networkAdapter, BOOL changeAdapter)
13980{
13981 LogFlowThisFunc(("\n"));
13982
13983 AutoCaller autoCaller(this);
13984 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
13985
13986 ComPtr<IInternalSessionControl> directControl;
13987 {
13988 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
13989 directControl = mData->mSession.mDirectControl;
13990 }
13991
13992 /* ignore notifications sent after #OnSessionEnd() is called */
13993 if (!directControl)
13994 return S_OK;
13995
13996 return directControl->OnNetworkAdapterChange(networkAdapter, changeAdapter);
13997}
13998
13999/**
14000 * @note Locks this object for reading.
14001 */
14002HRESULT SessionMachine::onNATRedirectRuleChange(ULONG ulSlot, BOOL aNatRuleRemove, IN_BSTR aRuleName,
14003 NATProtocol_T aProto, IN_BSTR aHostIp, LONG aHostPort, IN_BSTR aGuestIp, LONG aGuestPort)
14004{
14005 LogFlowThisFunc(("\n"));
14006
14007 AutoCaller autoCaller(this);
14008 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14009
14010 ComPtr<IInternalSessionControl> directControl;
14011 {
14012 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14013 directControl = mData->mSession.mDirectControl;
14014 }
14015
14016 /* ignore notifications sent after #OnSessionEnd() is called */
14017 if (!directControl)
14018 return S_OK;
14019 /*
14020 * instead acting like callback we ask IVirtualBox deliver corresponding event
14021 */
14022
14023 mParent->onNatRedirectChange(getId(), ulSlot, RT_BOOL(aNatRuleRemove), aRuleName, aProto, aHostIp, (uint16_t)aHostPort, aGuestIp, (uint16_t)aGuestPort);
14024 return S_OK;
14025}
14026
14027/**
14028 * @note Locks this object for reading.
14029 */
14030HRESULT SessionMachine::onSerialPortChange(ISerialPort *serialPort)
14031{
14032 LogFlowThisFunc(("\n"));
14033
14034 AutoCaller autoCaller(this);
14035 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14036
14037 ComPtr<IInternalSessionControl> directControl;
14038 {
14039 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14040 directControl = mData->mSession.mDirectControl;
14041 }
14042
14043 /* ignore notifications sent after #OnSessionEnd() is called */
14044 if (!directControl)
14045 return S_OK;
14046
14047 return directControl->OnSerialPortChange(serialPort);
14048}
14049
14050/**
14051 * @note Locks this object for reading.
14052 */
14053HRESULT SessionMachine::onParallelPortChange(IParallelPort *parallelPort)
14054{
14055 LogFlowThisFunc(("\n"));
14056
14057 AutoCaller autoCaller(this);
14058 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14059
14060 ComPtr<IInternalSessionControl> directControl;
14061 {
14062 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14063 directControl = mData->mSession.mDirectControl;
14064 }
14065
14066 /* ignore notifications sent after #OnSessionEnd() is called */
14067 if (!directControl)
14068 return S_OK;
14069
14070 return directControl->OnParallelPortChange(parallelPort);
14071}
14072
14073/**
14074 * @note Locks this object for reading.
14075 */
14076HRESULT SessionMachine::onStorageControllerChange()
14077{
14078 LogFlowThisFunc(("\n"));
14079
14080 AutoCaller autoCaller(this);
14081 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14082
14083 ComPtr<IInternalSessionControl> directControl;
14084 {
14085 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14086 directControl = mData->mSession.mDirectControl;
14087 }
14088
14089 /* ignore notifications sent after #OnSessionEnd() is called */
14090 if (!directControl)
14091 return S_OK;
14092
14093 return directControl->OnStorageControllerChange();
14094}
14095
14096/**
14097 * @note Locks this object for reading.
14098 */
14099HRESULT SessionMachine::onMediumChange(IMediumAttachment *aAttachment, BOOL aForce)
14100{
14101 LogFlowThisFunc(("\n"));
14102
14103 AutoCaller autoCaller(this);
14104 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14105
14106 ComPtr<IInternalSessionControl> directControl;
14107 {
14108 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14109 directControl = mData->mSession.mDirectControl;
14110 }
14111
14112 /* ignore notifications sent after #OnSessionEnd() is called */
14113 if (!directControl)
14114 return S_OK;
14115
14116 return directControl->OnMediumChange(aAttachment, aForce);
14117}
14118
14119/**
14120 * @note Locks this object for reading.
14121 */
14122HRESULT SessionMachine::onCPUChange(ULONG aCPU, BOOL aRemove)
14123{
14124 LogFlowThisFunc(("\n"));
14125
14126 AutoCaller autoCaller(this);
14127 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14128
14129 ComPtr<IInternalSessionControl> directControl;
14130 {
14131 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14132 directControl = mData->mSession.mDirectControl;
14133 }
14134
14135 /* ignore notifications sent after #OnSessionEnd() is called */
14136 if (!directControl)
14137 return S_OK;
14138
14139 return directControl->OnCPUChange(aCPU, aRemove);
14140}
14141
14142HRESULT SessionMachine::onCPUExecutionCapChange(ULONG aExecutionCap)
14143{
14144 LogFlowThisFunc(("\n"));
14145
14146 AutoCaller autoCaller(this);
14147 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14148
14149 ComPtr<IInternalSessionControl> directControl;
14150 {
14151 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14152 directControl = mData->mSession.mDirectControl;
14153 }
14154
14155 /* ignore notifications sent after #OnSessionEnd() is called */
14156 if (!directControl)
14157 return S_OK;
14158
14159 return directControl->OnCPUExecutionCapChange(aExecutionCap);
14160}
14161
14162/**
14163 * @note Locks this object for reading.
14164 */
14165HRESULT SessionMachine::onVRDEServerChange(BOOL aRestart)
14166{
14167 LogFlowThisFunc(("\n"));
14168
14169 AutoCaller autoCaller(this);
14170 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14171
14172 ComPtr<IInternalSessionControl> directControl;
14173 {
14174 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14175 directControl = mData->mSession.mDirectControl;
14176 }
14177
14178 /* ignore notifications sent after #OnSessionEnd() is called */
14179 if (!directControl)
14180 return S_OK;
14181
14182 return directControl->OnVRDEServerChange(aRestart);
14183}
14184
14185/**
14186 * @note Locks this object for reading.
14187 */
14188HRESULT SessionMachine::onVideoCaptureChange()
14189{
14190 LogFlowThisFunc(("\n"));
14191
14192 AutoCaller autoCaller(this);
14193 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14194
14195 ComPtr<IInternalSessionControl> directControl;
14196 {
14197 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14198 directControl = mData->mSession.mDirectControl;
14199 }
14200
14201 /* ignore notifications sent after #OnSessionEnd() is called */
14202 if (!directControl)
14203 return S_OK;
14204
14205 return directControl->OnVideoCaptureChange();
14206}
14207
14208/**
14209 * @note Locks this object for reading.
14210 */
14211HRESULT SessionMachine::onUSBControllerChange()
14212{
14213 LogFlowThisFunc(("\n"));
14214
14215 AutoCaller autoCaller(this);
14216 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14217
14218 ComPtr<IInternalSessionControl> directControl;
14219 {
14220 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14221 directControl = mData->mSession.mDirectControl;
14222 }
14223
14224 /* ignore notifications sent after #OnSessionEnd() is called */
14225 if (!directControl)
14226 return S_OK;
14227
14228 return directControl->OnUSBControllerChange();
14229}
14230
14231/**
14232 * @note Locks this object for reading.
14233 */
14234HRESULT SessionMachine::onSharedFolderChange()
14235{
14236 LogFlowThisFunc(("\n"));
14237
14238 AutoCaller autoCaller(this);
14239 AssertComRCReturnRC(autoCaller.rc());
14240
14241 ComPtr<IInternalSessionControl> directControl;
14242 {
14243 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14244 directControl = mData->mSession.mDirectControl;
14245 }
14246
14247 /* ignore notifications sent after #OnSessionEnd() is called */
14248 if (!directControl)
14249 return S_OK;
14250
14251 return directControl->OnSharedFolderChange(FALSE /* aGlobal */);
14252}
14253
14254/**
14255 * @note Locks this object for reading.
14256 */
14257HRESULT SessionMachine::onClipboardModeChange(ClipboardMode_T aClipboardMode)
14258{
14259 LogFlowThisFunc(("\n"));
14260
14261 AutoCaller autoCaller(this);
14262 AssertComRCReturnRC(autoCaller.rc());
14263
14264 ComPtr<IInternalSessionControl> directControl;
14265 {
14266 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14267 directControl = mData->mSession.mDirectControl;
14268 }
14269
14270 /* ignore notifications sent after #OnSessionEnd() is called */
14271 if (!directControl)
14272 return S_OK;
14273
14274 return directControl->OnClipboardModeChange(aClipboardMode);
14275}
14276
14277/**
14278 * @note Locks this object for reading.
14279 */
14280HRESULT SessionMachine::onDragAndDropModeChange(DragAndDropMode_T aDragAndDropMode)
14281{
14282 LogFlowThisFunc(("\n"));
14283
14284 AutoCaller autoCaller(this);
14285 AssertComRCReturnRC(autoCaller.rc());
14286
14287 ComPtr<IInternalSessionControl> directControl;
14288 {
14289 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14290 directControl = mData->mSession.mDirectControl;
14291 }
14292
14293 /* ignore notifications sent after #OnSessionEnd() is called */
14294 if (!directControl)
14295 return S_OK;
14296
14297 return directControl->OnDragAndDropModeChange(aDragAndDropMode);
14298}
14299
14300/**
14301 * @note Locks this object for reading.
14302 */
14303HRESULT SessionMachine::onBandwidthGroupChange(IBandwidthGroup *aBandwidthGroup)
14304{
14305 LogFlowThisFunc(("\n"));
14306
14307 AutoCaller autoCaller(this);
14308 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14309
14310 ComPtr<IInternalSessionControl> directControl;
14311 {
14312 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14313 directControl = mData->mSession.mDirectControl;
14314 }
14315
14316 /* ignore notifications sent after #OnSessionEnd() is called */
14317 if (!directControl)
14318 return S_OK;
14319
14320 return directControl->OnBandwidthGroupChange(aBandwidthGroup);
14321}
14322
14323/**
14324 * @note Locks this object for reading.
14325 */
14326HRESULT SessionMachine::onStorageDeviceChange(IMediumAttachment *aAttachment, BOOL aRemove, BOOL aSilent)
14327{
14328 LogFlowThisFunc(("\n"));
14329
14330 AutoCaller autoCaller(this);
14331 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14332
14333 ComPtr<IInternalSessionControl> directControl;
14334 {
14335 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14336 directControl = mData->mSession.mDirectControl;
14337 }
14338
14339 /* ignore notifications sent after #OnSessionEnd() is called */
14340 if (!directControl)
14341 return S_OK;
14342
14343 return directControl->OnStorageDeviceChange(aAttachment, aRemove, aSilent);
14344}
14345
14346/**
14347 * Returns @c true if this machine's USB controller reports it has a matching
14348 * filter for the given USB device and @c false otherwise.
14349 *
14350 * @note locks this object for reading.
14351 */
14352bool SessionMachine::hasMatchingUSBFilter(const ComObjPtr<HostUSBDevice> &aDevice, ULONG *aMaskedIfs)
14353{
14354 AutoCaller autoCaller(this);
14355 /* silently return if not ready -- this method may be called after the
14356 * direct machine session has been called */
14357 if (!autoCaller.isOk())
14358 return false;
14359
14360#ifdef VBOX_WITH_USB
14361 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14362
14363 switch (mData->mMachineState)
14364 {
14365 case MachineState_Starting:
14366 case MachineState_Restoring:
14367 case MachineState_TeleportingIn:
14368 case MachineState_Paused:
14369 case MachineState_Running:
14370 /** @todo Live Migration: snapshoting & teleporting. Need to fend things of
14371 * elsewhere... */
14372 alock.release();
14373 return mUSBDeviceFilters->hasMatchingFilter(aDevice, aMaskedIfs);
14374 default: break;
14375 }
14376#else
14377 NOREF(aDevice);
14378 NOREF(aMaskedIfs);
14379#endif
14380 return false;
14381}
14382
14383/**
14384 * @note The calls shall hold no locks. Will temporarily lock this object for reading.
14385 */
14386HRESULT SessionMachine::onUSBDeviceAttach(IUSBDevice *aDevice,
14387 IVirtualBoxErrorInfo *aError,
14388 ULONG aMaskedIfs)
14389{
14390 LogFlowThisFunc(("\n"));
14391
14392 AutoCaller autoCaller(this);
14393
14394 /* This notification may happen after the machine object has been
14395 * uninitialized (the session was closed), so don't assert. */
14396 if (FAILED(autoCaller.rc())) return autoCaller.rc();
14397
14398 ComPtr<IInternalSessionControl> directControl;
14399 {
14400 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14401 directControl = mData->mSession.mDirectControl;
14402 }
14403
14404 /* fail on notifications sent after #OnSessionEnd() is called, it is
14405 * expected by the caller */
14406 if (!directControl)
14407 return E_FAIL;
14408
14409 /* No locks should be held at this point. */
14410 AssertMsg(RTLockValidatorWriteLockGetCount(RTThreadSelf()) == 0, ("%d\n", RTLockValidatorWriteLockGetCount(RTThreadSelf())));
14411 AssertMsg(RTLockValidatorReadLockGetCount(RTThreadSelf()) == 0, ("%d\n", RTLockValidatorReadLockGetCount(RTThreadSelf())));
14412
14413 return directControl->OnUSBDeviceAttach(aDevice, aError, aMaskedIfs);
14414}
14415
14416/**
14417 * @note The calls shall hold no locks. Will temporarily lock this object for reading.
14418 */
14419HRESULT SessionMachine::onUSBDeviceDetach(IN_BSTR aId,
14420 IVirtualBoxErrorInfo *aError)
14421{
14422 LogFlowThisFunc(("\n"));
14423
14424 AutoCaller autoCaller(this);
14425
14426 /* This notification may happen after the machine object has been
14427 * uninitialized (the session was closed), so don't assert. */
14428 if (FAILED(autoCaller.rc())) return autoCaller.rc();
14429
14430 ComPtr<IInternalSessionControl> directControl;
14431 {
14432 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14433 directControl = mData->mSession.mDirectControl;
14434 }
14435
14436 /* fail on notifications sent after #OnSessionEnd() is called, it is
14437 * expected by the caller */
14438 if (!directControl)
14439 return E_FAIL;
14440
14441 /* No locks should be held at this point. */
14442 AssertMsg(RTLockValidatorWriteLockGetCount(RTThreadSelf()) == 0, ("%d\n", RTLockValidatorWriteLockGetCount(RTThreadSelf())));
14443 AssertMsg(RTLockValidatorReadLockGetCount(RTThreadSelf()) == 0, ("%d\n", RTLockValidatorReadLockGetCount(RTThreadSelf())));
14444
14445 return directControl->OnUSBDeviceDetach(aId, aError);
14446}
14447
14448// protected methods
14449/////////////////////////////////////////////////////////////////////////////
14450
14451/**
14452 * Helper method to finalize saving the state.
14453 *
14454 * @note Must be called from under this object's lock.
14455 *
14456 * @param aRc S_OK if the snapshot has been taken successfully
14457 * @param aErrMsg human readable error message for failure
14458 *
14459 * @note Locks mParent + this objects for writing.
14460 */
14461HRESULT SessionMachine::endSavingState(HRESULT aRc, const Utf8Str &aErrMsg)
14462{
14463 LogFlowThisFuncEnter();
14464
14465 AutoCaller autoCaller(this);
14466 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14467
14468 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
14469
14470 HRESULT rc = S_OK;
14471
14472 if (SUCCEEDED(aRc))
14473 {
14474 mSSData->strStateFilePath = mConsoleTaskData.strStateFilePath;
14475
14476 /* save all VM settings */
14477 rc = saveSettings(NULL);
14478 // no need to check whether VirtualBox.xml needs saving also since
14479 // we can't have a name change pending at this point
14480 }
14481 else
14482 {
14483 // delete the saved state file (it might have been already created);
14484 // we need not check whether this is shared with a snapshot here because
14485 // we certainly created this saved state file here anew
14486 RTFileDelete(mConsoleTaskData.strStateFilePath.c_str());
14487 }
14488
14489 /* notify the progress object about operation completion */
14490 Assert(mConsoleTaskData.mProgress);
14491 if (SUCCEEDED(aRc))
14492 mConsoleTaskData.mProgress->notifyComplete(S_OK);
14493 else
14494 {
14495 if (aErrMsg.length())
14496 mConsoleTaskData.mProgress->notifyComplete(aRc,
14497 COM_IIDOF(ISession),
14498 getComponentName(),
14499 aErrMsg.c_str());
14500 else
14501 mConsoleTaskData.mProgress->notifyComplete(aRc);
14502 }
14503
14504 /* clear out the temporary saved state data */
14505 mConsoleTaskData.mLastState = MachineState_Null;
14506 mConsoleTaskData.strStateFilePath.setNull();
14507 mConsoleTaskData.mProgress.setNull();
14508
14509 LogFlowThisFuncLeave();
14510 return rc;
14511}
14512
14513/**
14514 * Deletes the given file if it is no longer in use by either the current machine state
14515 * (if the machine is "saved") or any of the machine's snapshots.
14516 *
14517 * Note: This checks mSSData->strStateFilePath, which is shared by the Machine and SessionMachine
14518 * but is different for each SnapshotMachine. When calling this, the order of calling this
14519 * function on the one hand and changing that variable OR the snapshots tree on the other hand
14520 * is therefore critical. I know, it's all rather messy.
14521 *
14522 * @param strStateFile
14523 * @param pSnapshotToIgnore Passed to Snapshot::sharesSavedStateFile(); this snapshot is ignored in the test for whether the saved state file is in use.
14524 */
14525void SessionMachine::releaseSavedStateFile(const Utf8Str &strStateFile,
14526 Snapshot *pSnapshotToIgnore)
14527{
14528 // it is safe to delete this saved state file if it is not currently in use by the machine ...
14529 if ( (strStateFile.isNotEmpty())
14530 && (strStateFile != mSSData->strStateFilePath) // session machine's saved state
14531 )
14532 // ... and it must also not be shared with other snapshots
14533 if ( !mData->mFirstSnapshot
14534 || !mData->mFirstSnapshot->sharesSavedStateFile(strStateFile, pSnapshotToIgnore)
14535 // this checks the SnapshotMachine's state file paths
14536 )
14537 RTFileDelete(strStateFile.c_str());
14538}
14539
14540/**
14541 * Locks the attached media.
14542 *
14543 * All attached hard disks are locked for writing and DVD/floppy are locked for
14544 * reading. Parents of attached hard disks (if any) are locked for reading.
14545 *
14546 * This method also performs accessibility check of all media it locks: if some
14547 * media is inaccessible, the method will return a failure and a bunch of
14548 * extended error info objects per each inaccessible medium.
14549 *
14550 * Note that this method is atomic: if it returns a success, all media are
14551 * locked as described above; on failure no media is locked at all (all
14552 * succeeded individual locks will be undone).
14553 *
14554 * The caller is responsible for doing the necessary state sanity checks.
14555 *
14556 * The locks made by this method must be undone by calling #unlockMedia() when
14557 * no more needed.
14558 */
14559HRESULT SessionMachine::lockMedia()
14560{
14561 AutoCaller autoCaller(this);
14562 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14563
14564 AutoMultiWriteLock2 alock(this->lockHandle(),
14565 &mParent->getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
14566
14567 /* bail out if trying to lock things with already set up locking */
14568 AssertReturn(mData->mSession.mLockedMedia.IsEmpty(), E_FAIL);
14569
14570 MultiResult mrc(S_OK);
14571
14572 /* Collect locking information for all medium objects attached to the VM. */
14573 for (MediaData::AttachmentList::const_iterator it = mMediaData->mAttachments.begin();
14574 it != mMediaData->mAttachments.end();
14575 ++it)
14576 {
14577 MediumAttachment* pAtt = *it;
14578 DeviceType_T devType = pAtt->getType();
14579 Medium *pMedium = pAtt->getMedium();
14580
14581 MediumLockList *pMediumLockList(new MediumLockList());
14582 // There can be attachments without a medium (floppy/dvd), and thus
14583 // it's impossible to create a medium lock list. It still makes sense
14584 // to have the empty medium lock list in the map in case a medium is
14585 // attached later.
14586 if (pMedium != NULL)
14587 {
14588 MediumType_T mediumType = pMedium->getType();
14589 bool fIsReadOnlyLock = mediumType == MediumType_Readonly
14590 || mediumType == MediumType_Shareable;
14591 bool fIsVitalImage = (devType == DeviceType_HardDisk);
14592
14593 alock.release();
14594 mrc = pMedium->createMediumLockList(fIsVitalImage /* fFailIfInaccessible */,
14595 !fIsReadOnlyLock /* fMediumLockWrite */,
14596 NULL,
14597 *pMediumLockList);
14598 alock.acquire();
14599 if (FAILED(mrc))
14600 {
14601 delete pMediumLockList;
14602 mData->mSession.mLockedMedia.Clear();
14603 break;
14604 }
14605 }
14606
14607 HRESULT rc = mData->mSession.mLockedMedia.Insert(pAtt, pMediumLockList);
14608 if (FAILED(rc))
14609 {
14610 mData->mSession.mLockedMedia.Clear();
14611 mrc = setError(rc,
14612 tr("Collecting locking information for all attached media failed"));
14613 break;
14614 }
14615 }
14616
14617 if (SUCCEEDED(mrc))
14618 {
14619 /* Now lock all media. If this fails, nothing is locked. */
14620 alock.release();
14621 HRESULT rc = mData->mSession.mLockedMedia.Lock();
14622 alock.acquire();
14623 if (FAILED(rc))
14624 {
14625 mrc = setError(rc,
14626 tr("Locking of attached media failed"));
14627 }
14628 }
14629
14630 return mrc;
14631}
14632
14633/**
14634 * Undoes the locks made by by #lockMedia().
14635 */
14636void SessionMachine::unlockMedia()
14637{
14638 AutoCaller autoCaller(this);
14639 AssertComRCReturnVoid(autoCaller.rc());
14640
14641 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
14642
14643 /* we may be holding important error info on the current thread;
14644 * preserve it */
14645 ErrorInfoKeeper eik;
14646
14647 HRESULT rc = mData->mSession.mLockedMedia.Clear();
14648 AssertComRC(rc);
14649}
14650
14651/**
14652 * Helper to change the machine state (reimplementation).
14653 *
14654 * @note Locks this object for writing.
14655 * @note This method must not call saveSettings or SaveSettings, otherwise
14656 * it can cause crashes in random places due to unexpectedly committing
14657 * the current settings. The caller is responsible for that. The call
14658 * to saveStateSettings is fine, because this method does not commit.
14659 */
14660HRESULT SessionMachine::setMachineState(MachineState_T aMachineState)
14661{
14662 LogFlowThisFuncEnter();
14663 LogFlowThisFunc(("aMachineState=%s\n", Global::stringifyMachineState(aMachineState) ));
14664
14665 AutoCaller autoCaller(this);
14666 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14667
14668 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
14669
14670 MachineState_T oldMachineState = mData->mMachineState;
14671
14672 AssertMsgReturn(oldMachineState != aMachineState,
14673 ("oldMachineState=%s, aMachineState=%s\n",
14674 Global::stringifyMachineState(oldMachineState), Global::stringifyMachineState(aMachineState)),
14675 E_FAIL);
14676
14677 HRESULT rc = S_OK;
14678
14679 int stsFlags = 0;
14680 bool deleteSavedState = false;
14681
14682 /* detect some state transitions */
14683
14684 if ( ( oldMachineState == MachineState_Saved
14685 && aMachineState == MachineState_Restoring)
14686 || ( ( oldMachineState == MachineState_PoweredOff
14687 || oldMachineState == MachineState_Teleported
14688 || oldMachineState == MachineState_Aborted
14689 )
14690 && ( aMachineState == MachineState_TeleportingIn
14691 || aMachineState == MachineState_Starting
14692 )
14693 )
14694 )
14695 {
14696 /* The EMT thread is about to start */
14697
14698 /* Nothing to do here for now... */
14699
14700 /// @todo NEWMEDIA don't let mDVDDrive and other children
14701 /// change anything when in the Starting/Restoring state
14702 }
14703 else if ( ( oldMachineState == MachineState_Running
14704 || oldMachineState == MachineState_Paused
14705 || oldMachineState == MachineState_Teleporting
14706 || oldMachineState == MachineState_LiveSnapshotting
14707 || oldMachineState == MachineState_Stuck
14708 || oldMachineState == MachineState_Starting
14709 || oldMachineState == MachineState_Stopping
14710 || oldMachineState == MachineState_Saving
14711 || oldMachineState == MachineState_Restoring
14712 || oldMachineState == MachineState_TeleportingPausedVM
14713 || oldMachineState == MachineState_TeleportingIn
14714 )
14715 && ( aMachineState == MachineState_PoweredOff
14716 || aMachineState == MachineState_Saved
14717 || aMachineState == MachineState_Teleported
14718 || aMachineState == MachineState_Aborted
14719 )
14720 /* ignore PoweredOff->Saving->PoweredOff transition when taking a
14721 * snapshot */
14722 && ( mConsoleTaskData.mSnapshot.isNull()
14723 || mConsoleTaskData.mLastState >= MachineState_Running /** @todo Live Migration: clean up (lazy bird) */
14724 )
14725 )
14726 {
14727 /* The EMT thread has just stopped, unlock attached media. Note that as
14728 * opposed to locking that is done from Console, we do unlocking here
14729 * because the VM process may have aborted before having a chance to
14730 * properly unlock all media it locked. */
14731
14732 unlockMedia();
14733 }
14734
14735 if (oldMachineState == MachineState_Restoring)
14736 {
14737 if (aMachineState != MachineState_Saved)
14738 {
14739 /*
14740 * delete the saved state file once the machine has finished
14741 * restoring from it (note that Console sets the state from
14742 * Restoring to Saved if the VM couldn't restore successfully,
14743 * to give the user an ability to fix an error and retry --
14744 * we keep the saved state file in this case)
14745 */
14746 deleteSavedState = true;
14747 }
14748 }
14749 else if ( oldMachineState == MachineState_Saved
14750 && ( aMachineState == MachineState_PoweredOff
14751 || aMachineState == MachineState_Aborted
14752 || aMachineState == MachineState_Teleported
14753 )
14754 )
14755 {
14756 /*
14757 * delete the saved state after Console::ForgetSavedState() is called
14758 * or if the VM process (owning a direct VM session) crashed while the
14759 * VM was Saved
14760 */
14761
14762 /// @todo (dmik)
14763 // Not sure that deleting the saved state file just because of the
14764 // client death before it attempted to restore the VM is a good
14765 // thing. But when it crashes we need to go to the Aborted state
14766 // which cannot have the saved state file associated... The only
14767 // way to fix this is to make the Aborted condition not a VM state
14768 // but a bool flag: i.e., when a crash occurs, set it to true and
14769 // change the state to PoweredOff or Saved depending on the
14770 // saved state presence.
14771
14772 deleteSavedState = true;
14773 mData->mCurrentStateModified = TRUE;
14774 stsFlags |= SaveSTS_CurStateModified;
14775 }
14776
14777 if ( aMachineState == MachineState_Starting
14778 || aMachineState == MachineState_Restoring
14779 || aMachineState == MachineState_TeleportingIn
14780 )
14781 {
14782 /* set the current state modified flag to indicate that the current
14783 * state is no more identical to the state in the
14784 * current snapshot */
14785 if (!mData->mCurrentSnapshot.isNull())
14786 {
14787 mData->mCurrentStateModified = TRUE;
14788 stsFlags |= SaveSTS_CurStateModified;
14789 }
14790 }
14791
14792 if (deleteSavedState)
14793 {
14794 if (mRemoveSavedState)
14795 {
14796 Assert(!mSSData->strStateFilePath.isEmpty());
14797
14798 // it is safe to delete the saved state file if ...
14799 if ( !mData->mFirstSnapshot // ... we have no snapshots or
14800 || !mData->mFirstSnapshot->sharesSavedStateFile(mSSData->strStateFilePath, NULL /* pSnapshotToIgnore */)
14801 // ... none of the snapshots share the saved state file
14802 )
14803 RTFileDelete(mSSData->strStateFilePath.c_str());
14804 }
14805
14806 mSSData->strStateFilePath.setNull();
14807 stsFlags |= SaveSTS_StateFilePath;
14808 }
14809
14810 /* redirect to the underlying peer machine */
14811 mPeer->setMachineState(aMachineState);
14812
14813 if ( aMachineState == MachineState_PoweredOff
14814 || aMachineState == MachineState_Teleported
14815 || aMachineState == MachineState_Aborted
14816 || aMachineState == MachineState_Saved)
14817 {
14818 /* the machine has stopped execution
14819 * (or the saved state file was adopted) */
14820 stsFlags |= SaveSTS_StateTimeStamp;
14821 }
14822
14823 if ( ( oldMachineState == MachineState_PoweredOff
14824 || oldMachineState == MachineState_Aborted
14825 || oldMachineState == MachineState_Teleported
14826 )
14827 && aMachineState == MachineState_Saved)
14828 {
14829 /* the saved state file was adopted */
14830 Assert(!mSSData->strStateFilePath.isEmpty());
14831 stsFlags |= SaveSTS_StateFilePath;
14832 }
14833
14834#ifdef VBOX_WITH_GUEST_PROPS
14835 if ( aMachineState == MachineState_PoweredOff
14836 || aMachineState == MachineState_Aborted
14837 || aMachineState == MachineState_Teleported)
14838 {
14839 /* Make sure any transient guest properties get removed from the
14840 * property store on shutdown. */
14841
14842 HWData::GuestPropertyMap::const_iterator it;
14843 BOOL fNeedsSaving = mData->mGuestPropertiesModified;
14844 if (!fNeedsSaving)
14845 for (it = mHWData->mGuestProperties.begin();
14846 it != mHWData->mGuestProperties.end(); ++it)
14847 if ( (it->second.mFlags & guestProp::TRANSIENT)
14848 || (it->second.mFlags & guestProp::TRANSRESET))
14849 {
14850 fNeedsSaving = true;
14851 break;
14852 }
14853 if (fNeedsSaving)
14854 {
14855 mData->mCurrentStateModified = TRUE;
14856 stsFlags |= SaveSTS_CurStateModified;
14857 }
14858 }
14859#endif
14860
14861 rc = saveStateSettings(stsFlags);
14862
14863 if ( ( oldMachineState != MachineState_PoweredOff
14864 && oldMachineState != MachineState_Aborted
14865 && oldMachineState != MachineState_Teleported
14866 )
14867 && ( aMachineState == MachineState_PoweredOff
14868 || aMachineState == MachineState_Aborted
14869 || aMachineState == MachineState_Teleported
14870 )
14871 )
14872 {
14873 /* we've been shut down for any reason */
14874 /* no special action so far */
14875 }
14876
14877 LogFlowThisFunc(("rc=%Rhrc [%s]\n", rc, Global::stringifyMachineState(mData->mMachineState) ));
14878 LogFlowThisFuncLeave();
14879 return rc;
14880}
14881
14882/**
14883 * Sends the current machine state value to the VM process.
14884 *
14885 * @note Locks this object for reading, then calls a client process.
14886 */
14887HRESULT SessionMachine::updateMachineStateOnClient()
14888{
14889 AutoCaller autoCaller(this);
14890 AssertComRCReturn(autoCaller.rc(), autoCaller.rc());
14891
14892 ComPtr<IInternalSessionControl> directControl;
14893 {
14894 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
14895 AssertReturn(!!mData, E_FAIL);
14896 directControl = mData->mSession.mDirectControl;
14897
14898 /* directControl may be already set to NULL here in #OnSessionEnd()
14899 * called too early by the direct session process while there is still
14900 * some operation (like deleting the snapshot) in progress. The client
14901 * process in this case is waiting inside Session::close() for the
14902 * "end session" process object to complete, while #uninit() called by
14903 * #checkForDeath() on the Watcher thread is waiting for the pending
14904 * operation to complete. For now, we accept this inconsistent behavior
14905 * and simply do nothing here. */
14906
14907 if (mData->mSession.mState == SessionState_Unlocking)
14908 return S_OK;
14909
14910 AssertReturn(!directControl.isNull(), E_FAIL);
14911 }
14912
14913 return directControl->UpdateMachineState(mData->mMachineState);
14914}
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