VirtualBox

source: vbox/trunk/src/VBox/Main/VirtualBoxImpl.cpp@ 4764

Last change on this file since 4764 was 4475, checked in by vboxsync, 17 years ago

Some adjustments to RTEnv and RTProcCreate. Should work on darwin now.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 143.0 KB
Line 
1/** @file
2 *
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include "VirtualBoxImpl.h"
19#include "MachineImpl.h"
20#include "HardDiskImpl.h"
21#include "DVDImageImpl.h"
22#include "FloppyImageImpl.h"
23#include "SharedFolderImpl.h"
24#include "ProgressImpl.h"
25#include "HostImpl.h"
26#include "USBControllerImpl.h"
27#include "SystemPropertiesImpl.h"
28#include "Logging.h"
29
30#include "GuestOSTypeImpl.h"
31
32#ifdef RT_OS_WINDOWS
33#include "win32/svchlp.h"
34#endif
35
36#include <stdio.h>
37#include <stdlib.h>
38
39#include <iprt/path.h>
40#include <iprt/dir.h>
41#include <iprt/file.h>
42#include <iprt/string.h>
43#include <iprt/uuid.h>
44#include <iprt/thread.h>
45#include <iprt/process.h>
46#include <iprt/env.h>
47#include <iprt/cpputils.h>
48
49#include <VBox/err.h>
50#include <VBox/param.h>
51#include <VBox/VBoxHDD.h>
52#include <VBox/VBoxHDD-new.h>
53#include <VBox/ostypes.h>
54#include <VBox/version.h>
55
56#include <VBox/com/com.h>
57
58#include <algorithm>
59#include <set>
60#include <memory> // for auto_ptr
61
62// defines
63/////////////////////////////////////////////////////////////////////////////
64
65#define VBOX_GLOBAL_SETTINGS_FILE "VirtualBox.xml"
66
67// globals
68/////////////////////////////////////////////////////////////////////////////
69
70static const char DefaultGlobalConfig [] =
71{
72 "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>" RTFILE_LINEFEED
73 "<!-- innotek VirtualBox Global Configuration -->" RTFILE_LINEFEED
74 "<VirtualBox xmlns=\"" VBOX_XML_NAMESPACE "\" "
75 "version=\"" VBOX_XML_VERSION "-" VBOX_XML_PLATFORM "\">" RTFILE_LINEFEED
76 " <Global>"RTFILE_LINEFEED
77 " <MachineRegistry/>"RTFILE_LINEFEED
78 " <DiskRegistry/>"RTFILE_LINEFEED
79 " <USBDeviceFilters/>"RTFILE_LINEFEED
80 " <SystemProperties/>"RTFILE_LINEFEED
81 " </Global>"RTFILE_LINEFEED
82 "</VirtualBox>"RTFILE_LINEFEED
83};
84
85// static
86Bstr VirtualBox::sVersion;
87
88// constructor / destructor
89/////////////////////////////////////////////////////////////////////////////
90
91VirtualBox::VirtualBox()
92 : mAsyncEventThread (NIL_RTTHREAD)
93 , mAsyncEventQ (NULL)
94{}
95
96VirtualBox::~VirtualBox() {}
97
98HRESULT VirtualBox::FinalConstruct()
99{
100 LogFlowThisFunc (("\n"));
101
102 return init();
103}
104
105void VirtualBox::FinalRelease()
106{
107 LogFlowThisFunc (("\n"));
108
109 uninit();
110}
111
112VirtualBox::Data::Data()
113{
114}
115
116// public initializer/uninitializer for internal purposes only
117/////////////////////////////////////////////////////////////////////////////
118
119/**
120 * Initializes the VirtualBox object.
121 *
122 * @return COM result code
123 */
124HRESULT VirtualBox::init()
125{
126 /* Enclose the state transition NotReady->InInit->Ready */
127 AutoInitSpan autoInitSpan (this);
128 AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED);
129
130 LogFlow (("===========================================================\n"));
131 LogFlowThisFuncEnter();
132
133 if (sVersion.isNull())
134 sVersion = VBOX_VERSION_STRING;
135 LogFlowThisFunc (("Version: %ls\n", sVersion.raw()));
136
137 /* Get the VirtualBox home directory. */
138 {
139 char homeDir [RTPATH_MAX];
140 int vrc = com::GetVBoxUserHomeDirectory (homeDir, sizeof (homeDir));
141 if (VBOX_FAILURE (vrc))
142 return setError (E_FAIL,
143 tr ("Could not create the VirtualBox home directory '%s'"
144 "(%Vrc)"),
145 homeDir, vrc);
146
147 unconst (mData.mHomeDir) = homeDir;
148 }
149
150 /* compose the global config file name (always full path) */
151 Utf8StrFmt vboxConfigFile ("%s%c%s", mData.mHomeDir.raw(),
152 RTPATH_DELIMITER, VBOX_GLOBAL_SETTINGS_FILE);
153
154 /* store the config file name */
155 unconst (mData.mCfgFile.mName) = vboxConfigFile;
156
157 /* lock the config file */
158 HRESULT rc = lockConfig();
159 if (SUCCEEDED (rc))
160 {
161 if (!isConfigLocked())
162 {
163 /*
164 * This means the config file not found. This is not fatal --
165 * we just create an empty one.
166 */
167 RTFILE handle = NIL_RTFILE;
168 int vrc = RTFileOpen (&handle, vboxConfigFile,
169 RTFILE_O_READWRITE | RTFILE_O_CREATE |
170 RTFILE_O_DENY_WRITE);
171 if (VBOX_SUCCESS (vrc))
172 vrc = RTFileWrite (handle,
173 (void *) DefaultGlobalConfig,
174 sizeof (DefaultGlobalConfig), NULL);
175 if (VBOX_FAILURE (vrc))
176 {
177 rc = setError (E_FAIL, tr ("Could not create the default settings file "
178 "'%s' (%Vrc)"),
179 vboxConfigFile.raw(), vrc);
180 }
181 else
182 {
183 mData.mCfgFile.mHandle = handle;
184 /* we do not close the file to simulate lockConfig() */
185 }
186 }
187 }
188
189 /* initialize our Xerces XML subsystem */
190 if (SUCCEEDED (rc))
191 {
192 int vrc = CFGLDRInitialize();
193 if (VBOX_FAILURE (vrc))
194 rc = setError (E_FAIL, tr ("Could not initialize the XML parser (%Vrc)"),
195 vrc);
196 }
197
198 if (SUCCEEDED (rc))
199 {
200 CFGHANDLE configLoader = NULL;
201 char *loaderError = NULL;
202
203 /* load the config file */
204 int vrc = CFGLDRLoad (&configLoader, vboxConfigFile, mData.mCfgFile.mHandle,
205 XmlSchemaNS, true, cfgLdrEntityResolver,
206 &loaderError);
207 if (VBOX_SUCCESS (vrc))
208 {
209 CFGNODE global = NULL;
210 CFGLDRGetNode (configLoader, "VirtualBox/Global", 0, &global);
211 Assert (global);
212
213 do
214 {
215 /* create the host object early, machines will need it */
216 unconst (mData.mHost).createObject();
217 rc = mData.mHost->init (this);
218 ComAssertComRCBreak (rc, rc = rc);
219
220 unconst (mData.mSystemProperties).createObject();
221 rc = mData.mSystemProperties->init (this);
222 ComAssertComRCBreak (rc, rc = rc);
223
224 rc = loadDisks (global);
225 CheckComRCBreakRC ((rc));
226
227 /* guest OS type objects, needed by the machines */
228 rc = registerGuestOSTypes();
229 ComAssertComRCBreak (rc, rc = rc);
230
231 /* machines */
232 rc = loadMachines (global);
233 CheckComRCBreakRC ((rc));
234
235 /* host data (USB filters) */
236 rc = mData.mHost->loadSettings (global);
237 CheckComRCBreakRC ((rc));
238
239 rc = mData.mSystemProperties->loadSettings (global);
240 CheckComRCBreakRC ((rc));
241
242 /* check hard disk consistency */
243/// @todo (r=dmik) add IVirtualBox::cleanupHardDisks() instead or similar
244// for (HardDiskList::const_iterator it = mData.mHardDisks.begin();
245// it != mData.mHardDisks.end() && SUCCEEDED (rc);
246// ++ it)
247// {
248// rc = (*it)->checkConsistency();
249// }
250// CheckComRCBreakRC ((rc));
251 }
252 while (0);
253
254 /// @todo (dmik) if successful, check for orphan (unused) diffs
255 // that might be left because of the server crash, and remove them.
256
257 CFGLDRReleaseNode (global);
258 CFGLDRFree(configLoader);
259 }
260 else
261 {
262 rc = setError (E_FAIL,
263 tr ("Could not load the settings file '%ls' (%Vrc)%s%s"),
264 mData.mCfgFile.mName.raw(), vrc,
265 loaderError ? ".\n" : "", loaderError ? loaderError : "");
266 }
267
268 if (loaderError)
269 RTMemTmpFree (loaderError);
270 }
271
272 if (SUCCEEDED (rc))
273 {
274 /* start the client watcher thread */
275#if defined(RT_OS_WINDOWS)
276 unconst (mWatcherData.mUpdateReq) = ::CreateEvent (NULL, FALSE, FALSE, NULL);
277#elif defined(RT_OS_OS2)
278 RTSemEventCreate (&unconst (mWatcherData.mUpdateReq));
279#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
280 RTSemEventCreate (&unconst (mWatcherData.mUpdateReq));
281#else
282# error "Port me!"
283#endif
284 int vrc = RTThreadCreate (&unconst (mWatcherData.mThread),
285 ClientWatcher, (void *) this,
286 0, RTTHREADTYPE_MAIN_WORKER,
287 RTTHREADFLAGS_WAITABLE, "Watcher");
288 ComAssertRC (vrc);
289 if (VBOX_FAILURE (vrc))
290 rc = E_FAIL;
291 }
292
293 if (SUCCEEDED (rc)) do
294 {
295 /* start the async event handler thread */
296 int vrc = RTThreadCreate (&unconst (mAsyncEventThread), AsyncEventHandler,
297 &unconst (mAsyncEventQ),
298 0, RTTHREADTYPE_MAIN_WORKER,
299 RTTHREADFLAGS_WAITABLE, "EventHandler");
300 ComAssertRCBreak (vrc, rc = E_FAIL);
301
302 /* wait until the thread sets mAsyncEventQ */
303 RTThreadUserWait (mAsyncEventThread, RT_INDEFINITE_WAIT);
304 ComAssertBreak (mAsyncEventQ, rc = E_FAIL);
305 }
306 while (0);
307
308 /* Confirm a successful initialization when it's the case */
309 if (SUCCEEDED (rc))
310 autoInitSpan.setSucceeded();
311
312 LogFlowThisFunc (("rc=%08X\n", rc));
313 LogFlowThisFuncLeave();
314 LogFlow (("===========================================================\n"));
315 return rc;
316}
317
318void VirtualBox::uninit()
319{
320 /* Enclose the state transition Ready->InUninit->NotReady */
321 AutoUninitSpan autoUninitSpan (this);
322 if (autoUninitSpan.uninitDone())
323 return;
324
325 LogFlow (("===========================================================\n"));
326 LogFlowThisFuncEnter();
327 LogFlowThisFunc (("initFailed()=%d\n", autoUninitSpan.initFailed()));
328
329 /* tell all our child objects we've been uninitialized */
330
331 LogFlowThisFunc (("Uninitializing machines (%d)...\n", mData.mMachines.size()));
332 if (mData.mMachines.size())
333 {
334 MachineList::iterator it = mData.mMachines.begin();
335 while (it != mData.mMachines.end())
336 (*it++)->uninit();
337 mData.mMachines.clear();
338 }
339
340 if (mData.mSystemProperties)
341 {
342 mData.mSystemProperties->uninit();
343 unconst (mData.mSystemProperties).setNull();
344 }
345
346 if (mData.mHost)
347 {
348 mData.mHost->uninit();
349 unconst (mData.mHost).setNull();
350 }
351
352 /*
353 * Uninit all other children still referenced by clients
354 * (unregistered machines, hard disks, DVD/floppy images,
355 * server-side progress operations).
356 */
357 uninitDependentChildren();
358
359 mData.mFloppyImages.clear();
360 mData.mDVDImages.clear();
361 mData.mHardDisks.clear();
362
363 mData.mHardDiskMap.clear();
364
365 mData.mProgressOperations.clear();
366
367 mData.mGuestOSTypes.clear();
368
369 /* unlock the config file */
370 unlockConfig();
371
372 LogFlowThisFunc (("Releasing callbacks...\n"));
373 if (mData.mCallbacks.size())
374 {
375 /* release all callbacks */
376 LogWarningFunc (("%d unregistered callbacks!\n",
377 mData.mCallbacks.size()));
378 mData.mCallbacks.clear();
379 }
380
381 LogFlowThisFunc (("Terminating the async event handler...\n"));
382 if (mAsyncEventThread != NIL_RTTHREAD)
383 {
384 /* signal to exit the event loop */
385 if (mAsyncEventQ->postEvent (NULL))
386 {
387 /*
388 * Wait for thread termination (only if we've successfully posted
389 * a NULL event!)
390 */
391 int vrc = RTThreadWait (mAsyncEventThread, 60000, NULL);
392 if (VBOX_FAILURE (vrc))
393 LogWarningFunc (("RTThreadWait(%RTthrd) -> %Vrc\n",
394 mAsyncEventThread, vrc));
395 }
396 else
397 {
398 AssertMsgFailed (("postEvent(NULL) failed\n"));
399 RTThreadWait (mAsyncEventThread, 0, NULL);
400 }
401
402 unconst (mAsyncEventThread) = NIL_RTTHREAD;
403 unconst (mAsyncEventQ) = NULL;
404 }
405
406 LogFlowThisFunc (("Terminating the client watcher...\n"));
407 if (mWatcherData.mThread != NIL_RTTHREAD)
408 {
409 /* signal the client watcher thread */
410 updateClientWatcher();
411 /* wait for the termination */
412 RTThreadWait (mWatcherData.mThread, RT_INDEFINITE_WAIT, NULL);
413 unconst (mWatcherData.mThread) = NIL_RTTHREAD;
414 }
415 mWatcherData.mProcesses.clear();
416#if defined(RT_OS_WINDOWS)
417 if (mWatcherData.mUpdateReq != NULL)
418 {
419 ::CloseHandle (mWatcherData.mUpdateReq);
420 unconst (mWatcherData.mUpdateReq) = NULL;
421 }
422#elif defined(RT_OS_OS2)
423 if (mWatcherData.mUpdateReq != NIL_RTSEMEVENT)
424 {
425 RTSemEventDestroy (mWatcherData.mUpdateReq);
426 unconst (mWatcherData.mUpdateReq) = NIL_RTSEMEVENT;
427 }
428#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
429 if (mWatcherData.mUpdateReq != NIL_RTSEMEVENT)
430 {
431 RTSemEventDestroy (mWatcherData.mUpdateReq);
432 unconst (mWatcherData.mUpdateReq) = NIL_RTSEMEVENT;
433 }
434#else
435# error "Port me!"
436#endif
437
438 /* uninitialize the Xerces XML subsystem */
439 CFGLDRShutdown();
440
441 LogFlowThisFuncLeave();
442 LogFlow (("===========================================================\n"));
443}
444
445// IVirtualBox properties
446/////////////////////////////////////////////////////////////////////////////
447
448STDMETHODIMP VirtualBox::COMGETTER(Version) (BSTR *aVersion)
449{
450 if (!aVersion)
451 return E_INVALIDARG;
452
453 AutoCaller autoCaller (this);
454 CheckComRCReturnRC (autoCaller.rc());
455
456 sVersion.cloneTo (aVersion);
457 return S_OK;
458}
459
460STDMETHODIMP VirtualBox::COMGETTER(HomeFolder) (BSTR *aHomeFolder)
461{
462 if (!aHomeFolder)
463 return E_POINTER;
464
465 AutoCaller autoCaller (this);
466 CheckComRCReturnRC (autoCaller.rc());
467
468 mData.mHomeDir.cloneTo (aHomeFolder);
469 return S_OK;
470}
471
472STDMETHODIMP VirtualBox::COMGETTER(Host) (IHost **aHost)
473{
474 if (!aHost)
475 return E_POINTER;
476
477 AutoCaller autoCaller (this);
478 CheckComRCReturnRC (autoCaller.rc());
479
480 mData.mHost.queryInterfaceTo (aHost);
481 return S_OK;
482}
483
484STDMETHODIMP VirtualBox::COMGETTER(SystemProperties) (ISystemProperties **aSystemProperties)
485{
486 if (!aSystemProperties)
487 return E_POINTER;
488
489 AutoCaller autoCaller (this);
490 CheckComRCReturnRC (autoCaller.rc());
491
492 mData.mSystemProperties.queryInterfaceTo (aSystemProperties);
493 return S_OK;
494}
495
496/** @note Locks this object for reading. */
497STDMETHODIMP VirtualBox::COMGETTER(Machines) (IMachineCollection **aMachines)
498{
499 if (!aMachines)
500 return E_POINTER;
501
502 AutoCaller autoCaller (this);
503 CheckComRCReturnRC (autoCaller.rc());
504
505 ComObjPtr <MachineCollection> collection;
506 collection.createObject();
507
508 AutoReaderLock alock (this);
509 collection->init (mData.mMachines);
510 collection.queryInterfaceTo (aMachines);
511
512 return S_OK;
513}
514
515/** @note Locks this object for reading. */
516STDMETHODIMP VirtualBox::COMGETTER(HardDisks) (IHardDiskCollection **aHardDisks)
517{
518 if (!aHardDisks)
519 return E_POINTER;
520
521 AutoCaller autoCaller (this);
522 CheckComRCReturnRC (autoCaller.rc());
523
524 ComObjPtr <HardDiskCollection> collection;
525 collection.createObject();
526
527 AutoReaderLock alock (this);
528 collection->init (mData.mHardDisks);
529 collection.queryInterfaceTo (aHardDisks);
530
531 return S_OK;
532}
533
534/** @note Locks this object for reading. */
535STDMETHODIMP VirtualBox::COMGETTER(DVDImages) (IDVDImageCollection **aDVDImages)
536{
537 if (!aDVDImages)
538 return E_POINTER;
539
540 AutoCaller autoCaller (this);
541 CheckComRCReturnRC (autoCaller.rc());
542
543 ComObjPtr <DVDImageCollection> collection;
544 collection.createObject();
545
546 AutoReaderLock alock (this);
547 collection->init (mData.mDVDImages);
548 collection.queryInterfaceTo (aDVDImages);
549
550 return S_OK;
551}
552
553/** @note Locks this object for reading. */
554STDMETHODIMP VirtualBox::COMGETTER(FloppyImages) (IFloppyImageCollection **aFloppyImages)
555{
556 if (!aFloppyImages)
557 return E_POINTER;
558
559 AutoCaller autoCaller (this);
560 CheckComRCReturnRC (autoCaller.rc());
561
562 ComObjPtr <FloppyImageCollection> collection;
563 collection.createObject();
564
565 AutoReaderLock alock (this);
566 collection->init (mData.mFloppyImages);
567 collection.queryInterfaceTo (aFloppyImages);
568
569 return S_OK;
570}
571
572/** @note Locks this object for reading. */
573STDMETHODIMP VirtualBox::COMGETTER(ProgressOperations) (IProgressCollection **aOperations)
574{
575 if (!aOperations)
576 return E_POINTER;
577
578 AutoCaller autoCaller (this);
579 CheckComRCReturnRC (autoCaller.rc());
580
581 ComObjPtr <ProgressCollection> collection;
582 collection.createObject();
583
584 AutoReaderLock alock (this);
585 collection->init (mData.mProgressOperations);
586 collection.queryInterfaceTo (aOperations);
587
588 return S_OK;
589}
590
591/** @note Locks this object for reading. */
592STDMETHODIMP VirtualBox::COMGETTER(GuestOSTypes) (IGuestOSTypeCollection **aGuestOSTypes)
593{
594 if (!aGuestOSTypes)
595 return E_POINTER;
596
597 AutoCaller autoCaller (this);
598 CheckComRCReturnRC (autoCaller.rc());
599
600 ComObjPtr <GuestOSTypeCollection> collection;
601 collection.createObject();
602
603 AutoReaderLock alock (this);
604 collection->init (mData.mGuestOSTypes);
605 collection.queryInterfaceTo (aGuestOSTypes);
606
607 return S_OK;
608}
609
610STDMETHODIMP
611VirtualBox::COMGETTER(SharedFolders) (ISharedFolderCollection **aSharedFolders)
612{
613 if (!aSharedFolders)
614 return E_POINTER;
615
616 AutoCaller autoCaller (this);
617 CheckComRCReturnRC (autoCaller.rc());
618
619 return setError (E_NOTIMPL, "Not yet implemented");
620}
621
622// IVirtualBox methods
623/////////////////////////////////////////////////////////////////////////////
624
625/** @note Locks mSystemProperties object for reading. */
626STDMETHODIMP VirtualBox::CreateMachine (INPTR BSTR aBaseFolder,
627 INPTR BSTR aName,
628 IMachine **aMachine)
629{
630 LogFlowThisFuncEnter();
631 LogFlowThisFunc (("aBaseFolder='%ls', aName='%ls' aMachine={%p}\n",
632 aBaseFolder, aName, aMachine));
633
634 if (!aName)
635 return E_INVALIDARG;
636 if (!aMachine)
637 return E_POINTER;
638
639 if (!*aName)
640 return setError (E_INVALIDARG,
641 tr ("Machine name cannot be empty"));
642
643 AutoCaller autoCaller (this);
644 CheckComRCReturnRC (autoCaller.rc());
645
646 /* Compose the settings file name using the following scheme:
647 *
648 * <base_folder>/<machine_name>/<machine_name>.xml
649 *
650 * If a non-null and non-empty base folder is specified, the default
651 * machine folder will be used as a base folder.
652 */
653 Bstr settingsFile = aBaseFolder;
654 if (settingsFile.isEmpty())
655 {
656 AutoReaderLock propsLock (systemProperties());
657 /* we use the non-full folder value below to keep the path relative */
658 settingsFile = systemProperties()->defaultMachineFolder();
659 }
660 settingsFile = Utf8StrFmt ("%ls%c%ls%c%ls.xml",
661 settingsFile.raw(), RTPATH_DELIMITER,
662 aName, RTPATH_DELIMITER, aName);
663
664 HRESULT rc = E_FAIL;
665
666 /* create a new object */
667 ComObjPtr <Machine> machine;
668 rc = machine.createObject();
669 if (SUCCEEDED (rc))
670 {
671 /* initialize the machine object */
672 rc = machine->init (this, settingsFile, Machine::Init_New, aName);
673 if (SUCCEEDED (rc))
674 {
675 /* set the return value */
676 rc = machine.queryInterfaceTo (aMachine);
677 ComAssertComRC (rc);
678 }
679 }
680
681 LogFlowThisFunc (("rc=%08X\n", rc));
682 LogFlowThisFuncLeave();
683
684 return rc;
685}
686
687STDMETHODIMP VirtualBox::CreateLegacyMachine (INPTR BSTR aSettingsFile,
688 INPTR BSTR aName,
689 IMachine **aMachine)
690{
691 /* null and empty strings are not allowed as path names */
692 if (!aSettingsFile || !(*aSettingsFile))
693 return E_INVALIDARG;
694
695 if (!aName)
696 return E_INVALIDARG;
697 if (!aMachine)
698 return E_POINTER;
699
700 if (!*aName)
701 return setError (E_INVALIDARG,
702 tr ("Machine name cannot be empty"));
703
704 AutoCaller autoCaller (this);
705 CheckComRCReturnRC (autoCaller.rc());
706
707 HRESULT rc = E_FAIL;
708
709 Utf8Str settingsFile = aSettingsFile;
710 /* append the default extension if none */
711 if (!RTPathHaveExt (settingsFile))
712 settingsFile = Utf8StrFmt ("%s.xml", settingsFile.raw());
713
714 /* create a new object */
715 ComObjPtr<Machine> machine;
716 rc = machine.createObject();
717 if (SUCCEEDED (rc))
718 {
719 /* initialize the machine object */
720 rc = machine->init (this, Bstr (settingsFile), Machine::Init_New,
721 aName, FALSE /* aNameSync */);
722 if (SUCCEEDED (rc))
723 {
724 /* set the return value */
725 rc = machine.queryInterfaceTo (aMachine);
726 ComAssertComRC (rc);
727 }
728 }
729 return rc;
730}
731
732STDMETHODIMP VirtualBox::OpenMachine (INPTR BSTR aSettingsFile,
733 IMachine **aMachine)
734{
735 /* null and empty strings are not allowed as path names */
736 if (!aSettingsFile || !(*aSettingsFile))
737 return E_INVALIDARG;
738
739 if (!aMachine)
740 return E_POINTER;
741
742 AutoCaller autoCaller (this);
743 CheckComRCReturnRC (autoCaller.rc());
744
745 HRESULT rc = E_FAIL;
746
747 /* create a new object */
748 ComObjPtr<Machine> machine;
749 rc = machine.createObject();
750 if (SUCCEEDED (rc))
751 {
752 /* initialize the machine object */
753 rc = machine->init (this, aSettingsFile, Machine::Init_Existing);
754 if (SUCCEEDED (rc))
755 {
756 /* set the return value */
757 rc = machine.queryInterfaceTo (aMachine);
758 ComAssertComRC (rc);
759 }
760 }
761
762 return rc;
763}
764
765/** @note Locks objects! */
766STDMETHODIMP VirtualBox::RegisterMachine (IMachine *aMachine)
767{
768 if (!aMachine)
769 return E_INVALIDARG;
770
771 AutoCaller autoCaller (this);
772 CheckComRCReturnRC (autoCaller.rc());
773
774 HRESULT rc;
775
776 Bstr name;
777 rc = aMachine->COMGETTER(Name) (name.asOutParam());
778 CheckComRCReturnRC (rc);
779
780 /*
781 * we can safely cast child to Machine * here because only Machine
782 * implementations of IMachine can be among our children
783 */
784 Machine *machine = static_cast <Machine *> (getDependentChild (aMachine));
785 if (!machine)
786 {
787 /*
788 * this machine was not created by CreateMachine()
789 * or opened by OpenMachine() or loaded during startup
790 */
791 return setError (E_FAIL,
792 tr ("The machine named '%ls' is not created within this "
793 "VirtualBox instance"), name.raw());
794 }
795
796 /*
797 * Machine::trySetRegistered() will commit and save machine settings, and
798 * will also call #registerMachine() on success.
799 */
800 rc = registerMachine (machine);
801
802 /* fire an event */
803 if (SUCCEEDED (rc))
804 onMachineRegistered (machine->data()->mUuid, TRUE);
805
806 return rc;
807}
808
809/** @note Locks objects! */
810STDMETHODIMP VirtualBox::GetMachine (INPTR GUIDPARAM aId, IMachine **aMachine)
811{
812 if (!aMachine)
813 return E_POINTER;
814
815 AutoCaller autoCaller (this);
816 CheckComRCReturnRC (autoCaller.rc());
817
818 ComObjPtr <Machine> machine;
819 HRESULT rc = findMachine (Guid (aId), true /* setError */, &machine);
820
821 /* the below will set *aMachine to NULL if machine is null */
822 machine.queryInterfaceTo (aMachine);
823
824 return rc;
825}
826
827/** @note Locks this object for reading, then some machine objects for reading. */
828STDMETHODIMP VirtualBox::FindMachine (INPTR BSTR aName, IMachine **aMachine)
829{
830 LogFlowThisFuncEnter();
831 LogFlowThisFunc (("aName=\"%ls\", aMachine={%p}\n", aName, aMachine));
832
833 if (!aName)
834 return E_INVALIDARG;
835 if (!aMachine)
836 return E_POINTER;
837
838 AutoCaller autoCaller (this);
839 CheckComRCReturnRC (autoCaller.rc());
840
841 /* start with not found */
842 ComObjPtr <Machine> machine;
843 MachineList machines;
844 {
845 /* take a copy for safe iteration outside the lock */
846 AutoReaderLock alock (this);
847 machines = mData.mMachines;
848 }
849
850 for (MachineList::iterator it = machines.begin();
851 !machine && it != machines.end();
852 ++ it)
853 {
854 AutoReaderLock machineLock (*it);
855 if ((*it)->userData()->mName == aName)
856 machine = *it;
857 }
858
859 /* this will set (*machine) to NULL if machineObj is null */
860 machine.queryInterfaceTo (aMachine);
861
862 HRESULT rc = machine
863 ? S_OK
864 : setError (E_INVALIDARG,
865 tr ("Could not find a registered machine named '%ls'"), aName);
866
867 LogFlowThisFunc (("rc=%08X\n", rc));
868 LogFlowThisFuncLeave();
869
870 return rc;
871}
872
873/** @note Locks objects! */
874STDMETHODIMP VirtualBox::UnregisterMachine (INPTR GUIDPARAM aId,
875 IMachine **aMachine)
876{
877 Guid id = aId;
878 if (id.isEmpty())
879 return E_INVALIDARG;
880
881 AutoCaller autoCaller (this);
882 CheckComRCReturnRC (autoCaller.rc());
883
884 AutoLock alock (this);
885
886 ComObjPtr <Machine> machine;
887
888 HRESULT rc = findMachine (id, true /* setError */, &machine);
889 CheckComRCReturnRC (rc);
890
891 rc = machine->trySetRegistered (FALSE);
892 CheckComRCReturnRC (rc);
893
894 /* remove from the collection of registered machines */
895 mData.mMachines.remove (machine);
896
897 /* save the global registry */
898 rc = saveConfig();
899
900 /* return the unregistered machine to the caller */
901 machine.queryInterfaceTo (aMachine);
902
903 /* fire an event */
904 onMachineRegistered (id, FALSE);
905
906 return rc;
907}
908
909STDMETHODIMP VirtualBox::CreateHardDisk (HardDiskStorageType_T aStorageType,
910 IHardDisk **aHardDisk)
911{
912 if (!aHardDisk)
913 return E_POINTER;
914
915 AutoCaller autoCaller (this);
916 CheckComRCReturnRC (autoCaller.rc());
917
918 HRESULT rc = E_FAIL;
919
920 ComObjPtr <HardDisk> hardDisk;
921
922 switch (aStorageType)
923 {
924 case HardDiskStorageType_VirtualDiskImage:
925 {
926 ComObjPtr <HVirtualDiskImage> vdi;
927 vdi.createObject();
928 rc = vdi->init (this, NULL, NULL);
929 hardDisk = vdi;
930 break;
931 }
932
933 case HardDiskStorageType_ISCSIHardDisk:
934 {
935 ComObjPtr <HISCSIHardDisk> iscsi;
936 iscsi.createObject();
937 rc = iscsi->init (this);
938 hardDisk = iscsi;
939 break;
940 }
941
942 case HardDiskStorageType_VMDKImage:
943 {
944 ComObjPtr <HVMDKImage> vmdk;
945 vmdk.createObject();
946 rc = vmdk->init (this, NULL, NULL);
947 hardDisk = vmdk;
948 break;
949 }
950 default:
951 AssertFailed();
952 };
953
954 if (SUCCEEDED (rc))
955 hardDisk.queryInterfaceTo (aHardDisk);
956
957 return rc;
958}
959
960/** @note Locks mSystemProperties object for reading. */
961STDMETHODIMP VirtualBox::OpenHardDisk (INPTR BSTR aLocation, IHardDisk **aHardDisk)
962{
963 /* null and empty strings are not allowed locations */
964 if (!aLocation || !(*aLocation))
965 return E_INVALIDARG;
966
967 if (!aHardDisk)
968 return E_POINTER;
969
970 AutoCaller autoCaller (this);
971 CheckComRCReturnRC (autoCaller.rc());
972
973 /* Currently, the location is always a path. So, append the
974 * default path if only a name is given. */
975 Bstr location = aLocation;
976 {
977 Utf8Str loc = aLocation;
978 if (!RTPathHavePath (loc))
979 {
980 AutoLock propsLock (mData.mSystemProperties);
981 location = Utf8StrFmt ("%ls%c%s",
982 mData.mSystemProperties->defaultVDIFolder().raw(),
983 RTPATH_DELIMITER,
984 loc.raw());
985 }
986 }
987
988 ComObjPtr <HardDisk> hardDisk;
989 HRESULT rc = HardDisk::openHardDisk (this, location, hardDisk);
990 if (SUCCEEDED (rc))
991 hardDisk.queryInterfaceTo (aHardDisk);
992
993 return rc;
994}
995
996/** @note Locks mSystemProperties object for reading. */
997STDMETHODIMP VirtualBox::OpenVirtualDiskImage (INPTR BSTR aFilePath,
998 IVirtualDiskImage **aImage)
999{
1000 /* null and empty strings are not allowed as path names here */
1001 if (!aFilePath || !(*aFilePath))
1002 return E_INVALIDARG;
1003
1004 if (!aImage)
1005 return E_POINTER;
1006
1007 AutoCaller autoCaller (this);
1008 CheckComRCReturnRC (autoCaller.rc());
1009
1010 /* append the default path if only a name is given */
1011 Bstr path = aFilePath;
1012 {
1013 Utf8Str fp = aFilePath;
1014 if (!RTPathHavePath (fp))
1015 {
1016 AutoLock propsLock (mData.mSystemProperties);
1017 path = Utf8StrFmt ("%ls%c%s",
1018 mData.mSystemProperties->defaultVDIFolder().raw(),
1019 RTPATH_DELIMITER,
1020 fp.raw());
1021 }
1022 }
1023
1024 ComObjPtr <HVirtualDiskImage> vdi;
1025 vdi.createObject();
1026 HRESULT rc = vdi->init (this, NULL, path);
1027
1028 if (SUCCEEDED (rc))
1029 vdi.queryInterfaceTo (aImage);
1030
1031 return rc;
1032}
1033
1034/** @note Locks objects! */
1035STDMETHODIMP VirtualBox::RegisterHardDisk (IHardDisk *aHardDisk)
1036{
1037 if (!aHardDisk)
1038 return E_POINTER;
1039
1040 AutoCaller autoCaller (this);
1041 CheckComRCReturnRC (autoCaller.rc());
1042
1043 VirtualBoxBase *child = getDependentChild (aHardDisk);
1044 if (!child)
1045 return setError (E_FAIL, tr ("The given hard disk is not created within "
1046 "this VirtualBox instance"));
1047
1048 /*
1049 * we can safely cast child to HardDisk * here because only HardDisk
1050 * implementations of IHardDisk can be among our children
1051 */
1052
1053 return registerHardDisk (static_cast <HardDisk *> (child), RHD_External);
1054}
1055
1056/** @note Locks objects! */
1057STDMETHODIMP VirtualBox::GetHardDisk (INPTR GUIDPARAM aId, IHardDisk **aHardDisk)
1058{
1059 if (!aHardDisk)
1060 return E_POINTER;
1061
1062 AutoCaller autoCaller (this);
1063 CheckComRCReturnRC (autoCaller.rc());
1064
1065 Guid id = aId;
1066 ComObjPtr <HardDisk> hd;
1067 HRESULT rc = findHardDisk (&id, NULL, true /* setError */, &hd);
1068
1069 /* the below will set *aHardDisk to NULL if hd is null */
1070 hd.queryInterfaceTo (aHardDisk);
1071
1072 return rc;
1073}
1074
1075/** @note Locks objects! */
1076STDMETHODIMP VirtualBox::FindHardDisk (INPTR BSTR aLocation,
1077 IHardDisk **aHardDisk)
1078{
1079 if (!aLocation)
1080 return E_INVALIDARG;
1081 if (!aHardDisk)
1082 return E_POINTER;
1083
1084 AutoCaller autoCaller (this);
1085 CheckComRCReturnRC (autoCaller.rc());
1086
1087 Utf8Str location = aLocation;
1088 if (strncmp (location, "iscsi:", 6) == 0)
1089 {
1090 /* nothing special */
1091 }
1092 else
1093 {
1094 /* For locations represented by file paths, append the default path if
1095 * only a name is given, and then get the full path. */
1096 if (!RTPathHavePath (location))
1097 {
1098 AutoLock propsLock (mData.mSystemProperties);
1099 location = Utf8StrFmt ("%ls%c%s",
1100 mData.mSystemProperties->defaultVDIFolder().raw(),
1101 RTPATH_DELIMITER,
1102 location.raw());
1103 }
1104
1105 /* get the full file name */
1106 char buf [RTPATH_MAX];
1107 int vrc = RTPathAbsEx (mData.mHomeDir, location, buf, sizeof (buf));
1108 if (VBOX_FAILURE (vrc))
1109 return setError (E_FAIL, tr ("Invalid hard disk location '%ls' (%Vrc)"),
1110 aLocation, vrc);
1111 location = buf;
1112 }
1113
1114 ComObjPtr <HardDisk> hardDisk;
1115 HRESULT rc = findHardDisk (NULL, Bstr (location), true /* setError */,
1116 &hardDisk);
1117
1118 /* the below will set *aHardDisk to NULL if hardDisk is null */
1119 hardDisk.queryInterfaceTo (aHardDisk);
1120
1121 return rc;
1122}
1123
1124/** @note Locks objects! */
1125STDMETHODIMP VirtualBox::FindVirtualDiskImage (INPTR BSTR aFilePath,
1126 IVirtualDiskImage **aImage)
1127{
1128 if (!aFilePath)
1129 return E_INVALIDARG;
1130 if (!aImage)
1131 return E_POINTER;
1132
1133 AutoCaller autoCaller (this);
1134 CheckComRCReturnRC (autoCaller.rc());
1135
1136 /* append the default path if only a name is given */
1137 Utf8Str path = aFilePath;
1138 {
1139 Utf8Str fp = path;
1140 if (!RTPathHavePath (fp))
1141 {
1142 AutoLock propsLock (mData.mSystemProperties);
1143 path = Utf8StrFmt ("%ls%c%s",
1144 mData.mSystemProperties->defaultVDIFolder().raw(),
1145 RTPATH_DELIMITER,
1146 fp.raw());
1147 }
1148 }
1149
1150 /* get the full file name */
1151 char buf [RTPATH_MAX];
1152 int vrc = RTPathAbsEx (mData.mHomeDir, path, buf, sizeof (buf));
1153 if (VBOX_FAILURE (vrc))
1154 return setError (E_FAIL, tr ("Invalid image file path '%ls' (%Vrc)"),
1155 aFilePath, vrc);
1156
1157 ComObjPtr <HVirtualDiskImage> vdi;
1158 HRESULT rc = findVirtualDiskImage (NULL, Bstr (buf), true /* setError */,
1159 &vdi);
1160
1161 /* the below will set *aImage to NULL if vdi is null */
1162 vdi.queryInterfaceTo (aImage);
1163
1164 return rc;
1165}
1166
1167/** @note Locks objects! */
1168STDMETHODIMP VirtualBox::UnregisterHardDisk (INPTR GUIDPARAM aId, IHardDisk **aHardDisk)
1169{
1170 if (!aHardDisk)
1171 return E_POINTER;
1172
1173 AutoCaller autoCaller (this);
1174 CheckComRCReturnRC (autoCaller.rc());
1175
1176 *aHardDisk = NULL;
1177
1178 Guid id = aId;
1179 ComObjPtr <HardDisk> hd;
1180 HRESULT rc = findHardDisk (&id, NULL, true /* setError */, &hd);
1181 CheckComRCReturnRC (rc);
1182
1183 rc = unregisterHardDisk (hd);
1184 if (SUCCEEDED (rc))
1185 hd.queryInterfaceTo (aHardDisk);
1186
1187 return rc;
1188}
1189
1190/** @note Doesn't lock anything. */
1191STDMETHODIMP VirtualBox::OpenDVDImage (INPTR BSTR aFilePath, INPTR GUIDPARAM aId,
1192 IDVDImage **aDVDImage)
1193{
1194 /* null and empty strings are not allowed as path names */
1195 if (!aFilePath || !(*aFilePath))
1196 return E_INVALIDARG;
1197
1198 if (!aDVDImage)
1199 return E_POINTER;
1200
1201 AutoCaller autoCaller (this);
1202 CheckComRCReturnRC (autoCaller.rc());
1203
1204 HRESULT rc = E_FAIL;
1205
1206 Guid uuid = aId;
1207 /* generate an UUID if not specified */
1208 if (uuid.isEmpty())
1209 uuid.create();
1210
1211 ComObjPtr <DVDImage> dvdImage;
1212 dvdImage.createObject();
1213 rc = dvdImage->init (this, aFilePath, FALSE /* !isRegistered */, uuid);
1214 if (SUCCEEDED (rc))
1215 dvdImage.queryInterfaceTo (aDVDImage);
1216
1217 return rc;
1218}
1219
1220/** @note Locks objects! */
1221STDMETHODIMP VirtualBox::RegisterDVDImage (IDVDImage *aDVDImage)
1222{
1223 if (!aDVDImage)
1224 return E_POINTER;
1225
1226 AutoCaller autoCaller (this);
1227 CheckComRCReturnRC (autoCaller.rc());
1228
1229 VirtualBoxBase *child = getDependentChild (aDVDImage);
1230 if (!child)
1231 return setError (E_FAIL, tr ("The given CD/DVD image is not created within "
1232 "this VirtualBox instance"));
1233
1234 /*
1235 * we can safely cast child to DVDImage * here because only DVDImage
1236 * implementations of IDVDImage can be among our children
1237 */
1238
1239 return registerDVDImage (static_cast <DVDImage *> (child),
1240 FALSE /* aOnStartUp */);
1241}
1242
1243/** @note Locks objects! */
1244STDMETHODIMP VirtualBox::GetDVDImage (INPTR GUIDPARAM aId, IDVDImage **aDVDImage)
1245{
1246 if (!aDVDImage)
1247 return E_POINTER;
1248
1249 AutoCaller autoCaller (this);
1250 CheckComRCReturnRC (autoCaller.rc());
1251
1252 Guid uuid = aId;
1253 ComObjPtr <DVDImage> dvd;
1254 HRESULT rc = findDVDImage (&uuid, NULL, true /* setError */, &dvd);
1255
1256 /* the below will set *aDVDImage to NULL if dvd is null */
1257 dvd.queryInterfaceTo (aDVDImage);
1258
1259 return rc;
1260}
1261
1262/** @note Locks objects! */
1263STDMETHODIMP VirtualBox::FindDVDImage (INPTR BSTR aFilePath, IDVDImage **aDVDImage)
1264{
1265 if (!aFilePath)
1266 return E_INVALIDARG;
1267 if (!aDVDImage)
1268 return E_POINTER;
1269
1270 AutoCaller autoCaller (this);
1271 CheckComRCReturnRC (autoCaller.rc());
1272
1273 /* get the full file name */
1274 char buf [RTPATH_MAX];
1275 int vrc = RTPathAbsEx (mData.mHomeDir, Utf8Str (aFilePath), buf, sizeof (buf));
1276 if (VBOX_FAILURE (vrc))
1277 return setError (E_FAIL, tr ("Invalid image file path: '%ls' (%Vrc)"),
1278 aFilePath, vrc);
1279
1280 ComObjPtr <DVDImage> dvd;
1281 HRESULT rc = findDVDImage (NULL, Bstr (buf), true /* setError */, &dvd);
1282
1283 /* the below will set *dvdImage to NULL if dvd is null */
1284 dvd.queryInterfaceTo (aDVDImage);
1285
1286 return rc;
1287}
1288
1289/** @note Locks objects! */
1290STDMETHODIMP VirtualBox::GetDVDImageUsage (INPTR GUIDPARAM aId,
1291 ResourceUsage_T aUsage,
1292 BSTR *aMachineIDs)
1293{
1294 if (!aMachineIDs)
1295 return E_POINTER;
1296 if (aUsage == ResourceUsage_InvalidUsage)
1297 return E_INVALIDARG;
1298
1299 AutoCaller autoCaller (this);
1300 CheckComRCReturnRC (autoCaller.rc());
1301
1302 AutoReaderLock alock (this);
1303
1304 Guid uuid = Guid (aId);
1305 HRESULT rc = findDVDImage (&uuid, NULL, true /* setError */, NULL);
1306 if (FAILED (rc))
1307 return rc;
1308
1309 Bstr ids;
1310 getDVDImageUsage (uuid, aUsage, &ids);
1311 ids.cloneTo (aMachineIDs);
1312
1313 return S_OK;
1314}
1315
1316/** @note Locks objects! */
1317STDMETHODIMP VirtualBox::UnregisterDVDImage (INPTR GUIDPARAM aId,
1318 IDVDImage **aDVDImage)
1319{
1320 if (!aDVDImage)
1321 return E_POINTER;
1322
1323 AutoCaller autoCaller (this);
1324 CheckComRCReturnRC (autoCaller.rc());
1325
1326 AutoLock alock (this);
1327
1328 *aDVDImage = NULL;
1329
1330 Guid uuid = aId;
1331 ComObjPtr <DVDImage> dvd;
1332 HRESULT rc = findDVDImage (&uuid, NULL, true /* setError */, &dvd);
1333 CheckComRCReturnRC (rc);
1334
1335 if (!getDVDImageUsage (aId, ResourceUsage_AllUsage))
1336 {
1337 /* remove from the collection */
1338 mData.mDVDImages.remove (dvd);
1339
1340 /* save the global config file */
1341 rc = saveConfig();
1342
1343 if (SUCCEEDED (rc))
1344 {
1345 rc = dvd.queryInterfaceTo (aDVDImage);
1346 ComAssertComRC (rc);
1347 }
1348 }
1349 else
1350 rc = setError(E_FAIL,
1351 tr ("The CD/DVD image with the UUID {%s} is currently in use"),
1352 uuid.toString().raw());
1353
1354 return rc;
1355}
1356
1357/** @note Doesn't lock anything. */
1358STDMETHODIMP VirtualBox::OpenFloppyImage (INPTR BSTR aFilePath, INPTR GUIDPARAM aId,
1359 IFloppyImage **aFloppyImage)
1360{
1361 /* null and empty strings are not allowed as path names */
1362 if (!aFilePath || !(*aFilePath))
1363 return E_INVALIDARG;
1364
1365 if (!aFloppyImage)
1366 return E_POINTER;
1367
1368 AutoCaller autoCaller (this);
1369 CheckComRCReturnRC (autoCaller.rc());
1370
1371 HRESULT rc = E_FAIL;
1372
1373 Guid uuid = aId;
1374 /* generate an UUID if not specified */
1375 if (Guid::isEmpty (aId))
1376 uuid.create();
1377
1378 ComObjPtr <FloppyImage> floppyImage;
1379 floppyImage.createObject();
1380 rc = floppyImage->init (this, aFilePath, FALSE /* !isRegistered */, uuid);
1381 if (SUCCEEDED (rc))
1382 floppyImage.queryInterfaceTo (aFloppyImage);
1383
1384 return rc;
1385}
1386
1387/** @note Locks objects! */
1388STDMETHODIMP VirtualBox::RegisterFloppyImage (IFloppyImage *aFloppyImage)
1389{
1390 if (!aFloppyImage)
1391 return E_POINTER;
1392
1393 AutoCaller autoCaller (this);
1394 CheckComRCReturnRC (autoCaller.rc());
1395
1396 VirtualBoxBase *child = getDependentChild (aFloppyImage);
1397 if (!child)
1398 return setError (E_FAIL, tr ("The given floppy image is not created within "
1399 "this VirtualBox instance"));
1400
1401 /*
1402 * we can safely cast child to FloppyImage * here because only FloppyImage
1403 * implementations of IFloppyImage can be among our children
1404 */
1405
1406 return registerFloppyImage (static_cast <FloppyImage *> (child),
1407 FALSE /* aOnStartUp */);
1408}
1409
1410/** @note Locks objects! */
1411STDMETHODIMP VirtualBox::GetFloppyImage (INPTR GUIDPARAM aId,
1412 IFloppyImage **aFloppyImage)
1413{
1414 if (!aFloppyImage)
1415 return E_POINTER;
1416
1417 AutoCaller autoCaller (this);
1418 CheckComRCReturnRC (autoCaller.rc());
1419
1420 Guid uuid = aId;
1421 ComObjPtr <FloppyImage> floppy;
1422 HRESULT rc = findFloppyImage (&uuid, NULL, true /* setError */, &floppy);
1423
1424 /* the below will set *aFloppyImage to NULL if dvd is null */
1425 floppy.queryInterfaceTo (aFloppyImage);
1426
1427 return rc;
1428}
1429
1430/** @note Locks objects! */
1431STDMETHODIMP VirtualBox::FindFloppyImage (INPTR BSTR aFilePath,
1432 IFloppyImage **aFloppyImage)
1433{
1434 if (!aFilePath)
1435 return E_INVALIDARG;
1436 if (!aFloppyImage)
1437 return E_POINTER;
1438
1439 AutoCaller autoCaller (this);
1440 CheckComRCReturnRC (autoCaller.rc());
1441
1442 /* get the full file name */
1443 char buf [RTPATH_MAX];
1444 int vrc = RTPathAbsEx (mData.mHomeDir, Utf8Str (aFilePath), buf, sizeof (buf));
1445 if (VBOX_FAILURE (vrc))
1446 return setError (E_FAIL, tr ("Invalid image file path: '%ls' (%Vrc)"),
1447 aFilePath, vrc);
1448
1449 ComObjPtr <FloppyImage> floppy;
1450 HRESULT rc = findFloppyImage (NULL, Bstr (buf), true /* setError */, &floppy);
1451
1452 /* the below will set *image to NULL if img is null */
1453 floppy.queryInterfaceTo (aFloppyImage);
1454
1455 return rc;
1456}
1457
1458/** @note Locks objects! */
1459STDMETHODIMP VirtualBox::GetFloppyImageUsage (INPTR GUIDPARAM aId,
1460 ResourceUsage_T aUsage,
1461 BSTR *aMachineIDs)
1462{
1463 if (!aMachineIDs)
1464 return E_POINTER;
1465 if (aUsage == ResourceUsage_InvalidUsage)
1466 return E_INVALIDARG;
1467
1468 AutoCaller autoCaller (this);
1469 CheckComRCReturnRC (autoCaller.rc());
1470
1471 AutoReaderLock alock (this);
1472
1473 Guid uuid = Guid (aId);
1474 HRESULT rc = findFloppyImage (&uuid, NULL, true /* setError */, NULL);
1475 if (FAILED (rc))
1476 return rc;
1477
1478 Bstr ids;
1479 getFloppyImageUsage (uuid, aUsage, &ids);
1480 ids.cloneTo (aMachineIDs);
1481
1482 return S_OK;
1483}
1484
1485/** @note Locks objects! */
1486STDMETHODIMP VirtualBox::UnregisterFloppyImage (INPTR GUIDPARAM aId,
1487 IFloppyImage **aFloppyImage)
1488{
1489 if (!aFloppyImage)
1490 return E_INVALIDARG;
1491
1492 AutoCaller autoCaller (this);
1493 CheckComRCReturnRC (autoCaller.rc());
1494
1495 AutoLock alock (this);
1496
1497 *aFloppyImage = NULL;
1498
1499 Guid uuid = aId;
1500 ComObjPtr <FloppyImage> floppy;
1501 HRESULT rc = findFloppyImage (&uuid, NULL, true /* setError */, &floppy);
1502 CheckComRCReturnRC (rc);
1503
1504 if (!getFloppyImageUsage (aId, ResourceUsage_AllUsage))
1505 {
1506 /* remove from the collection */
1507 mData.mFloppyImages.remove (floppy);
1508
1509 /* save the global config file */
1510 rc = saveConfig();
1511 if (SUCCEEDED (rc))
1512 {
1513 rc = floppy.queryInterfaceTo (aFloppyImage);
1514 ComAssertComRC (rc);
1515 }
1516 }
1517 else
1518 rc = setError(E_FAIL,
1519 tr ("A floppy image with UUID {%s} is currently in use"),
1520 uuid.toString().raw());
1521
1522 return rc;
1523}
1524
1525/** @note Locks this object for reading. */
1526STDMETHODIMP VirtualBox::GetGuestOSType (INPTR BSTR aId, IGuestOSType **aType)
1527{
1528 if (!aType)
1529 return E_INVALIDARG;
1530
1531 AutoCaller autoCaller (this);
1532 CheckComRCReturnRC (autoCaller.rc());
1533
1534 *aType = NULL;
1535
1536 AutoReaderLock alock (this);
1537
1538 for (GuestOSTypeList::iterator it = mData.mGuestOSTypes.begin();
1539 it != mData.mGuestOSTypes.end();
1540 ++ it)
1541 {
1542 const Bstr &typeId = (*it)->id();
1543 AssertMsg (!!typeId, ("ID must not be NULL"));
1544 if (typeId == aId)
1545 {
1546 (*it).queryInterfaceTo (aType);
1547 break;
1548 }
1549 }
1550
1551 return (*aType) ? S_OK :
1552 setError (E_INVALIDARG,
1553 tr ("'%ls' is not a valid Guest OS type"),
1554 aId);
1555}
1556
1557STDMETHODIMP
1558VirtualBox::CreateSharedFolder (INPTR BSTR aName, INPTR BSTR aHostPath)
1559{
1560 if (!aName || !aHostPath)
1561 return E_INVALIDARG;
1562
1563 AutoCaller autoCaller (this);
1564 CheckComRCReturnRC (autoCaller.rc());
1565
1566 return setError (E_NOTIMPL, "Not yet implemented");
1567}
1568
1569STDMETHODIMP VirtualBox::RemoveSharedFolder (INPTR BSTR aName)
1570{
1571 if (!aName)
1572 return E_INVALIDARG;
1573
1574 AutoCaller autoCaller (this);
1575 CheckComRCReturnRC (autoCaller.rc());
1576
1577 return setError (E_NOTIMPL, "Not yet implemented");
1578}
1579
1580/** @note Locks this object for reading. */
1581STDMETHODIMP VirtualBox::
1582GetNextExtraDataKey (INPTR BSTR aKey, BSTR *aNextKey, BSTR *aNextValue)
1583{
1584 if (!aNextKey)
1585 return E_POINTER;
1586
1587 AutoCaller autoCaller (this);
1588 CheckComRCReturnRC (autoCaller.rc());
1589
1590 /* start with nothing found */
1591 *aNextKey = NULL;
1592
1593 HRESULT rc = S_OK;
1594
1595 /* serialize file access */
1596 AutoReaderLock alock (this);
1597
1598 CFGHANDLE configLoader;
1599
1600 /* load the config file */
1601 int vrc = CFGLDRLoad (&configLoader, Utf8Str (mData.mCfgFile.mName),
1602 mData.mCfgFile.mHandle,
1603 XmlSchemaNS, true, cfgLdrEntityResolver, NULL);
1604 ComAssertRCRet (vrc, E_FAIL);
1605
1606 CFGNODE extraDataNode;
1607
1608 /* navigate to the right position */
1609 if (VBOX_SUCCESS (CFGLDRGetNode (configLoader,
1610 "VirtualBox/Global/ExtraData", 0,
1611 &extraDataNode)))
1612 {
1613 /* check if it exists */
1614 bool found = false;
1615 unsigned count;
1616 CFGNODE extraDataItemNode;
1617 CFGLDRCountChildren (extraDataNode, "ExtraDataItem", &count);
1618 for (unsigned i = 0; (i < count) && (found == false); i++)
1619 {
1620 Bstr name;
1621 CFGLDRGetChildNode (extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1622 CFGLDRQueryBSTR (extraDataItemNode, "name", name.asOutParam());
1623
1624 /* if we're supposed to return the first one */
1625 if (aKey == NULL)
1626 {
1627 name.cloneTo (aNextKey);
1628 if (aNextValue)
1629 CFGLDRQueryBSTR (extraDataItemNode, "value", aNextValue);
1630 found = true;
1631 }
1632 /* did we find the key we're looking for? */
1633 else if (name == aKey)
1634 {
1635 found = true;
1636 /* is there another item? */
1637 if (i + 1 < count)
1638 {
1639 CFGLDRGetChildNode (extraDataNode, "ExtraDataItem", i + 1,
1640 &extraDataItemNode);
1641 CFGLDRQueryBSTR (extraDataItemNode, "name", name.asOutParam());
1642 name.cloneTo (aNextKey);
1643 if (aNextValue)
1644 CFGLDRQueryBSTR (extraDataItemNode, "value", aNextValue);
1645 found = true;
1646 }
1647 else
1648 {
1649 /* it's the last one */
1650 *aNextKey = NULL;
1651 }
1652 }
1653 CFGLDRReleaseNode (extraDataItemNode);
1654 }
1655
1656 /* if we haven't found the key, it's an error */
1657 if (!found)
1658 rc = setError (E_FAIL,
1659 tr ("Could not find extra data key '%ls'"), aKey);
1660
1661 CFGLDRReleaseNode (extraDataNode);
1662 }
1663
1664 CFGLDRFree (configLoader);
1665
1666 return rc;
1667}
1668
1669/** @note Locks this object for reading. */
1670STDMETHODIMP VirtualBox::GetExtraData (INPTR BSTR aKey, BSTR *aValue)
1671{
1672 if (!aKey || !(*aKey))
1673 return E_INVALIDARG;
1674 if (!aValue)
1675 return E_POINTER;
1676
1677 AutoCaller autoCaller (this);
1678 CheckComRCReturnRC (autoCaller.rc());
1679
1680 /* start with nothing found */
1681 *aValue = NULL;
1682
1683 HRESULT rc = S_OK;
1684
1685 /* serialize file access */
1686 AutoReaderLock alock (this);
1687
1688 CFGHANDLE configLoader;
1689
1690 /* load the config file */
1691 int vrc = CFGLDRLoad (&configLoader, Utf8Str (mData.mCfgFile.mName),
1692 mData.mCfgFile.mHandle,
1693 XmlSchemaNS, true, cfgLdrEntityResolver, NULL);
1694 ComAssertRCRet (vrc, E_FAIL);
1695
1696 CFGNODE extraDataNode;
1697
1698 /* navigate to the right position */
1699 if (VBOX_SUCCESS (CFGLDRGetNode (configLoader,
1700 "VirtualBox/Global/ExtraData", 0,
1701 &extraDataNode)))
1702 {
1703 /* check if it exists */
1704 bool found = false;
1705 unsigned count;
1706 CFGNODE extraDataItemNode;
1707 CFGLDRCountChildren (extraDataNode, "ExtraDataItem", &count);
1708 for (unsigned i = 0; (i < count) && (found == false); i++)
1709 {
1710 Bstr name;
1711 CFGLDRGetChildNode (extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1712 CFGLDRQueryBSTR (extraDataItemNode, "name", name.asOutParam());
1713 if (name == aKey)
1714 {
1715 found = true;
1716 CFGLDRQueryBSTR (extraDataItemNode, "value", aValue);
1717 }
1718 CFGLDRReleaseNode (extraDataItemNode);
1719 }
1720
1721 CFGLDRReleaseNode (extraDataNode);
1722 }
1723
1724 CFGLDRFree (configLoader);
1725
1726 return rc;
1727}
1728
1729/** @note Locks this object for writing. */
1730STDMETHODIMP VirtualBox::SetExtraData(INPTR BSTR aKey, INPTR BSTR aValue)
1731{
1732 if (!aKey)
1733 return E_POINTER;
1734
1735 AutoCaller autoCaller (this);
1736 CheckComRCReturnRC (autoCaller.rc());
1737
1738 Guid emptyGuid;
1739
1740 bool changed = false;
1741 HRESULT rc = S_OK;
1742
1743 /* serialize file access */
1744 AutoLock alock (this);
1745
1746 CFGHANDLE configLoader;
1747
1748 /* load the config file */
1749 int vrc = CFGLDRLoad (&configLoader, Utf8Str (mData.mCfgFile.mName),
1750 mData.mCfgFile.mHandle,
1751 XmlSchemaNS, true, cfgLdrEntityResolver, NULL);
1752 ComAssertRCRet (vrc, E_FAIL);
1753
1754 CFGNODE extraDataNode = 0;
1755
1756 vrc = CFGLDRGetNode (configLoader, "VirtualBox/Global/ExtraData", 0,
1757 &extraDataNode);
1758 if (VBOX_FAILURE (vrc) && aValue)
1759 vrc = CFGLDRCreateNode (configLoader, "VirtualBox/Global/ExtraData",
1760 &extraDataNode);
1761
1762 if (extraDataNode)
1763 {
1764 CFGNODE extraDataItemNode = 0;
1765 Bstr oldVal;
1766
1767 unsigned count;
1768 CFGLDRCountChildren (extraDataNode, "ExtraDataItem", &count);
1769
1770 for (unsigned i = 0; i < count; i++)
1771 {
1772 CFGLDRGetChildNode (extraDataNode, "ExtraDataItem", i, &extraDataItemNode);
1773 Bstr name;
1774 CFGLDRQueryBSTR (extraDataItemNode, "name", name.asOutParam());
1775 if (name == aKey)
1776 {
1777 CFGLDRQueryBSTR (extraDataItemNode, "value", oldVal.asOutParam());
1778 break;
1779 }
1780 CFGLDRReleaseNode (extraDataItemNode);
1781 extraDataItemNode = 0;
1782 }
1783
1784 /*
1785 * When no key is found, oldVal is null. Note:
1786 * 1. when oldVal is null, |oldVal == (BSTR) NULL| is true
1787 * 2. we cannot do |oldVal != value| because it will compare
1788 * BSTR pointers instead of strings (due to type conversion ops)
1789 */
1790 changed = !(oldVal == aValue);
1791
1792 if (changed)
1793 {
1794 /* ask for permission from all listeners */
1795 Bstr error;
1796 if (!onExtraDataCanChange (emptyGuid, aKey, aValue, error))
1797 {
1798 const char *sep = error.isEmpty() ? "" : ": ";
1799 const BSTR err = error.isNull() ? (const BSTR) L"" : error.raw();
1800 LogWarningFunc (("Someone vetoed! Change refused%s%ls\n",
1801 sep, err));
1802 rc = setError (E_ACCESSDENIED,
1803 tr ("Could not set extra data because someone refused "
1804 "the requested change of '%ls' to '%ls'%s%ls"),
1805 aKey, aValue, sep, err);
1806 }
1807 else
1808 {
1809 if (aValue)
1810 {
1811 if (!extraDataItemNode)
1812 {
1813 /* create a new item */
1814 CFGLDRAppendChildNode (extraDataNode, "ExtraDataItem",
1815 &extraDataItemNode);
1816 CFGLDRSetBSTR (extraDataItemNode, "name", aKey);
1817 }
1818 CFGLDRSetBSTR (extraDataItemNode, "value", aValue);
1819 }
1820 else
1821 {
1822 /* an old value does for sure exist here */
1823 CFGLDRDeleteNode (extraDataItemNode);
1824 extraDataItemNode = 0;
1825 }
1826 }
1827 }
1828
1829 if (extraDataItemNode)
1830 CFGLDRReleaseNode (extraDataItemNode);
1831
1832 CFGLDRReleaseNode (extraDataNode);
1833
1834 if (SUCCEEDED (rc) && changed)
1835 {
1836 char *loaderError = NULL;
1837 vrc = CFGLDRSave (configLoader, &loaderError);
1838 if (VBOX_FAILURE (vrc))
1839 {
1840 rc = setError (E_FAIL,
1841 tr ("Could not save the settings file '%ls' (%Vrc)%s%s"),
1842 mData.mCfgFile.mName.raw(), vrc,
1843 loaderError ? ".\n" : "", loaderError ? loaderError : "");
1844 if (loaderError)
1845 RTMemTmpFree (loaderError);
1846 }
1847 }
1848 }
1849
1850 CFGLDRFree (configLoader);
1851
1852 /* notification handling */
1853 if (SUCCEEDED (rc) && changed)
1854 onExtraDataChange (emptyGuid, aKey, aValue);
1855
1856 return rc;
1857}
1858
1859/**
1860 * @note Locks objects!
1861 */
1862STDMETHODIMP VirtualBox::OpenSession (ISession *aSession, INPTR GUIDPARAM aMachineId)
1863{
1864 if (!aSession)
1865 return E_INVALIDARG;
1866
1867 AutoCaller autoCaller (this);
1868 CheckComRCReturnRC (autoCaller.rc());
1869
1870 Guid id = aMachineId;
1871 ComObjPtr <Machine> machine;
1872
1873 HRESULT rc = findMachine (id, true /* setError */, &machine);
1874 CheckComRCReturnRC (rc);
1875
1876 /* check the session state */
1877 SessionState_T state;
1878 rc = aSession->COMGETTER(State) (&state);
1879 CheckComRCReturnRC (rc);
1880
1881 if (state != SessionState_SessionClosed)
1882 return setError (E_INVALIDARG,
1883 tr ("The given session is already open or being opened"));
1884
1885 /* get the IInternalSessionControl interface */
1886 ComPtr <IInternalSessionControl> control = aSession;
1887 ComAssertMsgRet (!!control, ("No IInternalSessionControl interface"),
1888 E_INVALIDARG);
1889
1890 rc = machine->openSession (control);
1891
1892 if (SUCCEEDED (rc))
1893 {
1894 /*
1895 * tell the client watcher thread to update the set of
1896 * machines that have open sessions
1897 */
1898 updateClientWatcher();
1899
1900 /* fire an event */
1901 onSessionStateChange (aMachineId, SessionState_SessionOpen);
1902 }
1903
1904 return rc;
1905}
1906
1907/**
1908 * @note Locks objects!
1909 */
1910STDMETHODIMP VirtualBox::OpenRemoteSession (ISession *aSession,
1911 INPTR GUIDPARAM aMachineId,
1912 INPTR BSTR aType,
1913 INPTR BSTR aEnvironment,
1914 IProgress **aProgress)
1915{
1916 if (!aSession || !aType)
1917 return E_INVALIDARG;
1918 if (!aProgress)
1919 return E_POINTER;
1920
1921 AutoCaller autoCaller (this);
1922 CheckComRCReturnRC (autoCaller.rc());
1923
1924 Guid id = aMachineId;
1925 ComObjPtr <Machine> machine;
1926
1927 HRESULT rc = findMachine (id, true /* setError */, &machine);
1928 CheckComRCReturnRC (rc);
1929
1930 /* check the session state */
1931 SessionState_T state;
1932 rc = aSession->COMGETTER(State) (&state);
1933 CheckComRCReturnRC (rc);
1934
1935 if (state != SessionState_SessionClosed)
1936 return setError (E_INVALIDARG,
1937 tr ("The given session is already open or being opened"));
1938
1939 /* get the IInternalSessionControl interface */
1940 ComPtr <IInternalSessionControl> control = aSession;
1941 ComAssertMsgRet (!!control, ("No IInternalSessionControl interface"),
1942 E_INVALIDARG);
1943
1944 /* create a progress object */
1945 ComObjPtr <Progress> progress;
1946 progress.createObject();
1947 progress->init (this, (IMachine *) machine,
1948 Bstr (tr ("Spawning session")),
1949 FALSE /* aCancelable */);
1950
1951 rc = machine->openRemoteSession (control, aType, aEnvironment, progress);
1952
1953 if (SUCCEEDED (rc))
1954 {
1955 progress.queryInterfaceTo (aProgress);
1956
1957 /* fire an event */
1958 onSessionStateChange (aMachineId, SessionState_SessionSpawning);
1959 }
1960
1961 return rc;
1962}
1963
1964/**
1965 * @note Locks objects!
1966 */
1967STDMETHODIMP VirtualBox::OpenExistingSession (ISession *aSession,
1968 INPTR GUIDPARAM aMachineId)
1969{
1970 if (!aSession)
1971 return E_POINTER;
1972
1973 AutoCaller autoCaller (this);
1974 CheckComRCReturnRC (autoCaller.rc());
1975
1976 Guid id = aMachineId;
1977 ComObjPtr <Machine> machine;
1978
1979 HRESULT rc = findMachine (id, true /* setError */, &machine);
1980 CheckComRCReturnRC (rc);
1981
1982 /* check the session state */
1983 SessionState_T state;
1984 rc = aSession->COMGETTER(State) (&state);
1985 CheckComRCReturnRC (rc);
1986
1987 if (state != SessionState_SessionClosed)
1988 return setError (E_INVALIDARG,
1989 tr ("The given session is already open or being opened"));
1990
1991 /* get the IInternalSessionControl interface */
1992 ComPtr <IInternalSessionControl> control = aSession;
1993 ComAssertMsgRet (!!control, ("No IInternalSessionControl interface"),
1994 E_INVALIDARG);
1995
1996 rc = machine->openExistingSession (control);
1997
1998 return rc;
1999}
2000
2001/**
2002 * Registers a new client callback on this instance. The methods of the
2003 * callback interface will be called by this instance when the appropriate
2004 * event occurs.
2005 *
2006 * @note Locks this object for writing.
2007 */
2008STDMETHODIMP VirtualBox::RegisterCallback (IVirtualBoxCallback *callback)
2009{
2010 LogFlowMember (("VirtualBox::RegisterCallback(): callback=%p\n", callback));
2011
2012 if (!callback)
2013 return E_INVALIDARG;
2014
2015 AutoCaller autoCaller (this);
2016 CheckComRCReturnRC (autoCaller.rc());
2017
2018 AutoLock alock (this);
2019 mData.mCallbacks.push_back (CallbackList::value_type (callback));
2020
2021 return S_OK;
2022}
2023
2024/**
2025 * Unregisters the previously registered client callback.
2026 *
2027 * @note Locks this object for writing.
2028 */
2029STDMETHODIMP VirtualBox::UnregisterCallback (IVirtualBoxCallback *callback)
2030{
2031 if (!callback)
2032 return E_INVALIDARG;
2033
2034 AutoCaller autoCaller (this);
2035 CheckComRCReturnRC (autoCaller.rc());
2036
2037 HRESULT rc = S_OK;
2038
2039 AutoLock alock (this);
2040
2041 CallbackList::iterator it;
2042 it = std::find (mData.mCallbacks.begin(),
2043 mData.mCallbacks.end(),
2044 CallbackList::value_type (callback));
2045 if (it == mData.mCallbacks.end())
2046 rc = E_INVALIDARG;
2047 else
2048 mData.mCallbacks.erase (it);
2049
2050 LogFlowMember (("VirtualBox::UnregisterCallback(): callback=%p, rc=%08X\n",
2051 callback, rc));
2052 return rc;
2053}
2054
2055// public methods only for internal purposes
2056/////////////////////////////////////////////////////////////////////////////
2057
2058/**
2059 * Posts an event to the event queue that is processed asynchronously
2060 * on a dedicated thread.
2061 *
2062 * Posting events to the dedicated event queue is useful to perform secondary
2063 * actions outside any object locks -- for example, to iterate over a list
2064 * of callbacks and inform them about some change caused by some object's
2065 * method call.
2066 *
2067 * @param event event to post
2068 * (must be allocated using |new|, will be deleted automatically
2069 * by the event thread after processing)
2070 *
2071 * @note Doesn't lock any object.
2072 */
2073HRESULT VirtualBox::postEvent (Event *event)
2074{
2075 AutoCaller autoCaller (this);
2076 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
2077
2078 if (autoCaller.state() != Ready)
2079 {
2080 LogWarningFunc (("VirtualBox has been uninitialized (state=%d), "
2081 "the event is discarded!\n",
2082 autoCaller.state()));
2083 return S_OK;
2084 }
2085
2086 AssertReturn (event, E_FAIL);
2087 AssertReturn (mAsyncEventQ, E_FAIL);
2088
2089 AutoLock alock (mAsyncEventQLock);
2090 if (mAsyncEventQ->postEvent (event))
2091 return S_OK;
2092
2093 return E_FAIL;
2094}
2095
2096/**
2097 * Helper method to add a progress to the global collection of pending
2098 * operations.
2099 *
2100 * @param aProgress operation to add to the collection
2101 * @return COM status code
2102 *
2103 * @note Locks this object for writing.
2104 */
2105HRESULT VirtualBox::addProgress (IProgress *aProgress)
2106{
2107 if (!aProgress)
2108 return E_INVALIDARG;
2109
2110 AutoCaller autoCaller (this);
2111 CheckComRCReturnRC (autoCaller.rc());
2112
2113 AutoLock alock (this);
2114 mData.mProgressOperations.push_back (aProgress);
2115 return S_OK;
2116}
2117
2118/**
2119 * Helper method to remove the progress from the global collection of pending
2120 * operations. Usualy gets called upon progress completion.
2121 *
2122 * @param aId UUID of the progress operation to remove
2123 * @return COM status code
2124 *
2125 * @note Locks this object for writing.
2126 */
2127HRESULT VirtualBox::removeProgress (INPTR GUIDPARAM aId)
2128{
2129 AutoCaller autoCaller (this);
2130 CheckComRCReturnRC (autoCaller.rc());
2131
2132 ComPtr <IProgress> progress;
2133
2134 AutoLock alock (this);
2135
2136 for (ProgressList::iterator it = mData.mProgressOperations.begin();
2137 it != mData.mProgressOperations.end();
2138 ++ it)
2139 {
2140 Guid id;
2141 (*it)->COMGETTER(Id) (id.asOutParam());
2142 if (id == aId)
2143 {
2144 mData.mProgressOperations.erase (it);
2145 return S_OK;
2146 }
2147 }
2148
2149 AssertFailed(); /* should never happen */
2150
2151 return E_FAIL;
2152}
2153
2154#ifdef RT_OS_WINDOWS
2155
2156struct StartSVCHelperClientData
2157{
2158 ComObjPtr <VirtualBox> that;
2159 ComObjPtr <Progress> progress;
2160 bool privileged;
2161 VirtualBox::SVCHelperClientFunc func;
2162 void *user;
2163};
2164
2165/**
2166 * Helper method to that starts a worker thread that:
2167 * - creates a pipe communication channel using SVCHlpClient;
2168 * - starts a SVC Helper process that will inherit this channel;
2169 * - executes the supplied function by passing it the created SVCHlpClient
2170 * and opened instance to communicate to the Helper process and the given
2171 * Progress object.
2172 *
2173 * The user function is supposed to communicate to the helper process
2174 * using the \a aClient argument to do the requested job and optionally expose
2175 * the prgress through the \a aProgress object. The user function should never
2176 * call notifyComplete() on it: this will be done automatically using the
2177 * result code returned by the function.
2178 *
2179 * Before the user function is stared, the communication channel passed to in
2180 * the \a aClient argument, is fully set up, the function should start using
2181 * it's write() and read() methods directly.
2182 *
2183 * The \a aVrc parameter of the user function may be used to return an error
2184 * code if it is related to communication errors (for example, returned by
2185 * the SVCHlpClient members when they fail). In this case, the correct error
2186 * message using this value will be reported to the caller. Note that the
2187 * value of \a aVrc is inspected only if the user function itself returns
2188 * a success.
2189 *
2190 * If a failure happens anywhere before the user function would be normally
2191 * called, it will be called anyway in special "cleanup only" mode indicated
2192 * by \a aClient, \a aProgress and \aVrc arguments set to NULL. In this mode,
2193 * all the function is supposed to do is to cleanup its aUser argument if
2194 * necessary (it's assumed that the ownership of this argument is passed to
2195 * the user function once #startSVCHelperClient() returns a success, thus
2196 * making it responsible for the cleanup).
2197 *
2198 * After the user function returns, the thread will send the SVCHlpMsg::Null
2199 * message to indicate a process termination.
2200 *
2201 * @param aPrivileged |true| to start the SVC Hepler process as a privlieged
2202 * user that can perform administrative tasks
2203 * @param aFunc user function to run
2204 * @param aUser argument to the user function
2205 * @param aProgress progress object that will track operation completion
2206 *
2207 * @note aPrivileged is currently ignored (due to some unsolved problems in
2208 * Vista) and the process will be started as a normal (unprivileged)
2209 * process.
2210 *
2211 * @note Doesn't lock anything.
2212 */
2213HRESULT VirtualBox::startSVCHelperClient (bool aPrivileged,
2214 SVCHelperClientFunc aFunc,
2215 void *aUser, Progress *aProgress)
2216{
2217 AssertReturn (aFunc, E_POINTER);
2218 AssertReturn (aProgress, E_POINTER);
2219
2220 AutoCaller autoCaller (this);
2221 CheckComRCReturnRC (autoCaller.rc());
2222
2223 /* create the SVCHelperClientThread() argument */
2224 std::auto_ptr <StartSVCHelperClientData>
2225 d (new StartSVCHelperClientData());
2226 AssertReturn (d.get(), E_OUTOFMEMORY);
2227
2228 d->that = this;
2229 d->progress = aProgress;
2230 d->privileged = aPrivileged;
2231 d->func = aFunc;
2232 d->user = aUser;
2233
2234 RTTHREAD tid = NIL_RTTHREAD;
2235 int vrc = RTThreadCreate (&tid, SVCHelperClientThread,
2236 static_cast <void *> (d.get()),
2237 0, RTTHREADTYPE_MAIN_WORKER,
2238 RTTHREADFLAGS_WAITABLE, "SVCHelper");
2239
2240 ComAssertMsgRCRet (vrc, ("Could not create SVCHelper thread (%Vrc)\n", vrc),
2241 E_FAIL);
2242
2243 /* d is now owned by SVCHelperClientThread(), so release it */
2244 d.release();
2245
2246 return S_OK;
2247}
2248
2249/**
2250 * Worker thread for startSVCHelperClient().
2251 */
2252/* static */
2253DECLCALLBACK(int)
2254VirtualBox::SVCHelperClientThread (RTTHREAD aThread, void *aUser)
2255{
2256 LogFlowFuncEnter();
2257
2258 std::auto_ptr <StartSVCHelperClientData>
2259 d (static_cast <StartSVCHelperClientData *> (aUser));
2260
2261 HRESULT rc = S_OK;
2262 bool userFuncCalled = false;
2263
2264 do
2265 {
2266 AssertBreak (d.get(), rc = E_POINTER);
2267 AssertReturn (!d->progress.isNull(), E_POINTER);
2268
2269 /* protect VirtualBox from uninitialization */
2270 AutoCaller autoCaller (d->that);
2271 if (!autoCaller.isOk())
2272 {
2273 /* it's too late */
2274 rc = autoCaller.rc();
2275 break;
2276 }
2277
2278 int vrc = VINF_SUCCESS;
2279
2280 Guid id;
2281 id.create();
2282 SVCHlpClient client;
2283 vrc = client.create (Utf8StrFmt ("VirtualBox\\SVCHelper\\{%Vuuid}",
2284 id.raw()));
2285 if (VBOX_FAILURE (vrc))
2286 {
2287 rc = setError (E_FAIL,
2288 tr ("Could not create the communication channel (%Vrc)"), vrc);
2289 break;
2290 }
2291
2292 /* get the path to the executable */
2293 char exePathBuf [RTPATH_MAX];
2294 char *exePath = RTProcGetExecutableName (exePathBuf, RTPATH_MAX);
2295 ComAssertBreak (exePath, E_FAIL);
2296
2297 Utf8Str argsStr = Utf8StrFmt ("/Helper %s", client.name().raw());
2298
2299 LogFlowFunc (("Starting '\"%s\" %s'...\n", exePath, argsStr.raw()));
2300
2301 RTPROCESS pid = NIL_RTPROCESS;
2302
2303 if (d->privileged)
2304 {
2305 /* Attempt to start a privileged process using the Run As dialog */
2306
2307 Bstr file = exePath;
2308 Bstr parameters = argsStr;
2309
2310 SHELLEXECUTEINFO shExecInfo;
2311
2312 shExecInfo.cbSize = sizeof (SHELLEXECUTEINFO);
2313
2314 shExecInfo.fMask = NULL;
2315 shExecInfo.hwnd = NULL;
2316 shExecInfo.lpVerb = L"runas";
2317 shExecInfo.lpFile = file;
2318 shExecInfo.lpParameters = parameters;
2319 shExecInfo.lpDirectory = NULL;
2320 shExecInfo.nShow = SW_NORMAL;
2321 shExecInfo.hInstApp = NULL;
2322
2323 if (!ShellExecuteEx (&shExecInfo))
2324 {
2325 int vrc2 = RTErrConvertFromWin32 (GetLastError());
2326 /* hide excessive details in case of a frequent error
2327 * (pressing the Cancel button to close the Run As dialog) */
2328 if (vrc2 == VERR_CANCELLED)
2329 rc = setError (E_FAIL,
2330 tr ("Operatiion cancelled by the user"));
2331 else
2332 rc = setError (E_FAIL,
2333 tr ("Could not launch a privileged process '%s' (%Vrc)"),
2334 exePath, vrc2);
2335 break;
2336 }
2337 }
2338 else
2339 {
2340 const char *args[] = { exePath, "/Helper", client.name(), 0 };
2341 vrc = RTProcCreate (exePath, args, RTENV_DEFAULT, 0, &pid);
2342 if (VBOX_FAILURE (vrc))
2343 {
2344 rc = setError (E_FAIL,
2345 tr ("Could not launch a process '%s' (%Vrc)"), exePath, vrc);
2346 break;
2347 }
2348 }
2349
2350 /* wait for the client to connect */
2351 vrc = client.connect();
2352 if (VBOX_SUCCESS (vrc))
2353 {
2354 /* start the user supplied function */
2355 rc = d->func (&client, d->progress, d->user, &vrc);
2356 userFuncCalled = true;
2357 }
2358
2359 /* send the termination signal to the process anyway */
2360 {
2361 int vrc2 = client.write (SVCHlpMsg::Null);
2362 if (VBOX_SUCCESS (vrc))
2363 vrc = vrc2;
2364 }
2365
2366 if (SUCCEEDED (rc) && VBOX_FAILURE (vrc))
2367 {
2368 rc = setError (E_FAIL,
2369 tr ("Could not operate the communication channel (%Vrc)"), vrc);
2370 break;
2371 }
2372 }
2373 while (0);
2374
2375 if (FAILED (rc) && !userFuncCalled)
2376 {
2377 /* call the user function in the "cleanup only" mode
2378 * to let it free resources passed to in aUser */
2379 d->func (NULL, NULL, d->user, NULL);
2380 }
2381
2382 d->progress->notifyComplete (rc);
2383
2384 LogFlowFuncLeave();
2385 return 0;
2386}
2387
2388#endif /* RT_OS_WINDOWS */
2389
2390/**
2391 * Sends a signal to the client watcher thread to rescan the set of machines
2392 * that have open sessions.
2393 *
2394 * @note Doesn't lock anything.
2395 */
2396void VirtualBox::updateClientWatcher()
2397{
2398 AutoCaller autoCaller (this);
2399 AssertComRCReturn (autoCaller.rc(), (void) 0);
2400
2401 AssertReturn (mWatcherData.mThread != NIL_RTTHREAD, (void) 0);
2402
2403 /* sent an update request */
2404#if defined(RT_OS_WINDOWS)
2405 ::SetEvent (mWatcherData.mUpdateReq);
2406#elif defined(RT_OS_OS2)
2407 RTSemEventSignal (mWatcherData.mUpdateReq);
2408#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
2409 RTSemEventSignal (mWatcherData.mUpdateReq);
2410#else
2411# error "Port me!"
2412#endif
2413}
2414
2415/**
2416 * Adds the given child process ID to the list of processes to be reaped.
2417 * This call should be followed by #updateClientWatcher() to take the effect.
2418 */
2419void VirtualBox::addProcessToReap (RTPROCESS pid)
2420{
2421 AutoCaller autoCaller (this);
2422 AssertComRCReturn (autoCaller.rc(), (void) 0);
2423
2424 /// @todo (dmik) Win32?
2425#ifndef RT_OS_WINDOWS
2426 AutoLock alock (this);
2427 mWatcherData.mProcesses.push_back (pid);
2428#endif
2429}
2430
2431/** Event for onMachineStateChange(), onMachineDataChange(), onMachineRegistered() */
2432struct MachineEvent : public VirtualBox::CallbackEvent
2433{
2434 enum What { DataChanged, StateChanged, Registered };
2435
2436 MachineEvent (VirtualBox *aVB, const Guid &aId)
2437 : CallbackEvent (aVB), what (DataChanged), id (aId)
2438 {}
2439
2440 MachineEvent (VirtualBox *aVB, const Guid &aId, MachineState_T aState)
2441 : CallbackEvent (aVB), what (StateChanged), id (aId)
2442 , state (aState)
2443 {}
2444
2445 MachineEvent (VirtualBox *aVB, const Guid &aId, BOOL aRegistered)
2446 : CallbackEvent (aVB), what (Registered), id (aId)
2447 , registered (aRegistered)
2448 {}
2449
2450 void handleCallback (const ComPtr <IVirtualBoxCallback> &aCallback)
2451 {
2452 switch (what)
2453 {
2454 case DataChanged:
2455 LogFlow (("OnMachineDataChange: id={%Vuuid}\n", id.ptr()));
2456 aCallback->OnMachineDataChange (id);
2457 break;
2458
2459 case StateChanged:
2460 LogFlow (("OnMachineStateChange: id={%Vuuid}, state=%d\n",
2461 id.ptr(), state));
2462 aCallback->OnMachineStateChange (id, state);
2463 break;
2464
2465 case Registered:
2466 LogFlow (("OnMachineRegistered: id={%Vuuid}, registered=%d\n",
2467 id.ptr(), registered));
2468 aCallback->OnMachineRegistered (id, registered);
2469 break;
2470 }
2471 }
2472
2473 const What what;
2474
2475 Guid id;
2476 MachineState_T state;
2477 BOOL registered;
2478};
2479
2480/**
2481 * @note Doesn't lock any object.
2482 */
2483void VirtualBox::onMachineStateChange (const Guid &aId, MachineState_T aState)
2484{
2485 postEvent (new MachineEvent (this, aId, aState));
2486}
2487
2488/**
2489 * @note Doesn't lock any object.
2490 */
2491void VirtualBox::onMachineDataChange (const Guid &aId)
2492{
2493 postEvent (new MachineEvent (this, aId));
2494}
2495
2496/**
2497 * @note Locks this object for reading.
2498 */
2499BOOL VirtualBox::onExtraDataCanChange (const Guid &aId, INPTR BSTR aKey, INPTR BSTR aValue,
2500 Bstr &aError)
2501{
2502 LogFlowThisFunc (("machine={%s} aKey={%ls} aValue={%ls}\n",
2503 aId.toString().raw(), aKey, aValue));
2504
2505 AutoCaller autoCaller (this);
2506 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
2507
2508 CallbackList list;
2509 {
2510 AutoReaderLock alock (this);
2511 list = mData.mCallbacks;
2512 }
2513
2514 BOOL allowChange = TRUE;
2515 CallbackList::iterator it = list.begin();
2516 while ((it != list.end()) && allowChange)
2517 {
2518 HRESULT rc = (*it++)->OnExtraDataCanChange (aId, aKey, aValue,
2519 aError.asOutParam(), &allowChange);
2520 if (FAILED (rc))
2521 {
2522 /* if a call to this method fails for some reason (for ex., because
2523 * the other side is dead), we ensure allowChange stays true
2524 * (MS COM RPC implementation seems to zero all output vars before
2525 * issuing an IPC call or after a failure, so it's essential
2526 * there) */
2527 allowChange = TRUE;
2528 }
2529 }
2530
2531 LogFlowThisFunc (("allowChange=%RTbool\n", allowChange));
2532 return allowChange;
2533}
2534
2535/** Event for onExtraDataChange() */
2536struct ExtraDataEvent : public VirtualBox::CallbackEvent
2537{
2538 ExtraDataEvent (VirtualBox *aVB, const Guid &aMachineId,
2539 INPTR BSTR aKey, INPTR BSTR aVal)
2540 : CallbackEvent (aVB), machineId (aMachineId)
2541 , key (aKey), val (aVal)
2542 {}
2543
2544 void handleCallback (const ComPtr <IVirtualBoxCallback> &aCallback)
2545 {
2546 LogFlow (("OnExtraDataChange: machineId={%Vuuid}, key='%ls', val='%ls'\n",
2547 machineId.ptr(), key.raw(), val.raw()));
2548 aCallback->OnExtraDataChange (machineId, key, val);
2549 }
2550
2551 Guid machineId;
2552 Bstr key, val;
2553};
2554
2555/**
2556 * @note Doesn't lock any object.
2557 */
2558void VirtualBox::onExtraDataChange (const Guid &aId, INPTR BSTR aKey, INPTR BSTR aValue)
2559{
2560 postEvent (new ExtraDataEvent (this, aId, aKey, aValue));
2561}
2562
2563/**
2564 * @note Doesn't lock any object.
2565 */
2566void VirtualBox::onMachineRegistered (const Guid &aId, BOOL aRegistered)
2567{
2568 postEvent (new MachineEvent (this, aId, aRegistered));
2569}
2570
2571/** Event for onSessionStateChange() */
2572struct SessionEvent : public VirtualBox::CallbackEvent
2573{
2574 SessionEvent (VirtualBox *aVB, const Guid &aMachineId, SessionState_T aState)
2575 : CallbackEvent (aVB), machineId (aMachineId), sessionState (aState)
2576 {}
2577
2578 void handleCallback (const ComPtr <IVirtualBoxCallback> &aCallback)
2579 {
2580 LogFlow (("OnSessionStateChange: machineId={%Vuuid}, sessionState=%d\n",
2581 machineId.ptr(), sessionState));
2582 aCallback->OnSessionStateChange (machineId, sessionState);
2583 }
2584
2585 Guid machineId;
2586 SessionState_T sessionState;
2587};
2588
2589/**
2590 * @note Doesn't lock any object.
2591 */
2592void VirtualBox::onSessionStateChange (const Guid &aId, SessionState_T aState)
2593{
2594 postEvent (new SessionEvent (this, aId, aState));
2595}
2596
2597/** Event for onSnapshotTaken(), onSnapshotRemoved() and onSnapshotChange() */
2598struct SnapshotEvent : public VirtualBox::CallbackEvent
2599{
2600 enum What { Taken, Discarded, Changed };
2601
2602 SnapshotEvent (VirtualBox *aVB, const Guid &aMachineId, const Guid &aSnapshotId,
2603 What aWhat)
2604 : CallbackEvent (aVB)
2605 , what (aWhat)
2606 , machineId (aMachineId), snapshotId (aSnapshotId)
2607 {}
2608
2609 void handleCallback (const ComPtr <IVirtualBoxCallback> &aCallback)
2610 {
2611 switch (what)
2612 {
2613 case Taken:
2614 LogFlow (("OnSnapshotTaken: machineId={%Vuuid}, snapshotId={%Vuuid}\n",
2615 machineId.ptr(), snapshotId.ptr()));
2616 aCallback->OnSnapshotTaken (machineId, snapshotId);
2617 break;
2618
2619 case Discarded:
2620 LogFlow (("OnSnapshotDiscarded: machineId={%Vuuid}, snapshotId={%Vuuid}\n",
2621 machineId.ptr(), snapshotId.ptr()));
2622 aCallback->OnSnapshotDiscarded (machineId, snapshotId);
2623 break;
2624
2625 case Changed:
2626 LogFlow (("OnSnapshotChange: machineId={%Vuuid}, snapshotId={%Vuuid}\n",
2627 machineId.ptr(), snapshotId.ptr()));
2628 aCallback->OnSnapshotChange (machineId, snapshotId);
2629 break;
2630 }
2631 }
2632
2633 const What what;
2634
2635 Guid machineId;
2636 Guid snapshotId;
2637};
2638
2639/**
2640 * @note Doesn't lock any object.
2641 */
2642void VirtualBox::onSnapshotTaken (const Guid &aMachineId, const Guid &aSnapshotId)
2643{
2644 postEvent (new SnapshotEvent (this, aMachineId, aSnapshotId, SnapshotEvent::Taken));
2645}
2646
2647/**
2648 * @note Doesn't lock any object.
2649 */
2650void VirtualBox::onSnapshotDiscarded (const Guid &aMachineId, const Guid &aSnapshotId)
2651{
2652 postEvent (new SnapshotEvent (this, aMachineId, aSnapshotId, SnapshotEvent::Discarded));
2653}
2654
2655/**
2656 * @note Doesn't lock any object.
2657 */
2658void VirtualBox::onSnapshotChange (const Guid &aMachineId, const Guid &aSnapshotId)
2659{
2660 postEvent (new SnapshotEvent (this, aMachineId, aSnapshotId, SnapshotEvent::Changed));
2661}
2662
2663/**
2664 * @note Locks this object for reading.
2665 */
2666ComObjPtr <GuestOSType> VirtualBox::getUnknownOSType()
2667{
2668 ComObjPtr <GuestOSType> type;
2669
2670 AutoCaller autoCaller (this);
2671 AssertComRCReturn (autoCaller.rc(), type);
2672
2673 AutoReaderLock alock (this);
2674
2675 /* unknown type must always be the first */
2676 ComAssertRet (mData.mGuestOSTypes.size() > 0, type);
2677
2678 type = mData.mGuestOSTypes.front();
2679 return type;
2680}
2681
2682/**
2683 * Returns the list of opened machines (i.e. machines having direct sessions
2684 * opened by client processes).
2685 *
2686 * @note the returned list contains smart pointers. So, clear it as soon as
2687 * it becomes no more necessary to release instances.
2688 * @note it can be possible that a session machine from the list has been
2689 * already uninitialized, so a) lock the instance and b) chheck for
2690 * instance->isReady() return value before manipulating the object directly
2691 * (i.e. not through COM methods).
2692 *
2693 * @note Locks objects for reading.
2694 */
2695void VirtualBox::getOpenedMachines (SessionMachineVector &aVector)
2696{
2697 AutoCaller autoCaller (this);
2698 AssertComRCReturn (autoCaller.rc(), (void) 0);
2699
2700 std::list <ComObjPtr <SessionMachine> > list;
2701
2702 {
2703 AutoReaderLock alock (this);
2704
2705 for (MachineList::iterator it = mData.mMachines.begin();
2706 it != mData.mMachines.end();
2707 ++ it)
2708 {
2709 ComObjPtr <SessionMachine> sm = (*it)->sessionMachine();
2710 /* SessionMachine is null when there are no open sessions */
2711 if (!sm.isNull())
2712 list.push_back (sm);
2713 }
2714 }
2715
2716 aVector = SessionMachineVector (list.begin(), list.end());
2717 return;
2718}
2719
2720/**
2721 * Helper to find machines that use the given DVD image.
2722 *
2723 * @param machineIDs string where to store the list (can be NULL)
2724 * @return TRUE if at least one machine found and false otherwise
2725 *
2726 * @note For now, we just scan all the machines. We can optimize this later
2727 * if required by adding the corresponding field to DVDImage and requiring all
2728 * IDVDImage instances to be DVDImage objects.
2729 *
2730 * @note Locks objects for reading.
2731 */
2732BOOL VirtualBox::getDVDImageUsage (const Guid &id,
2733 ResourceUsage_T usage,
2734 Bstr *machineIDs)
2735{
2736 AutoCaller autoCaller (this);
2737 AssertComRCReturn (autoCaller.rc(), FALSE);
2738
2739 typedef std::set <Guid> Set;
2740 Set idSet;
2741
2742 {
2743 AutoReaderLock alock (this);
2744
2745 for (MachineList::const_iterator mit = mData.mMachines.begin();
2746 mit != mData.mMachines.end();
2747 ++ mit)
2748 {
2749 /// @todo (dmik) move this part to Machine for better incapsulation
2750
2751 ComObjPtr <Machine> m = *mit;
2752 AutoReaderLock malock (m);
2753
2754 /* take the session machine when appropriate */
2755 if (!m->data()->mSession.mMachine.isNull())
2756 m = m->data()->mSession.mMachine;
2757
2758 const ComObjPtr <DVDDrive> &dvd = m->dvdDrive();
2759 AutoReaderLock dalock (dvd);
2760
2761 /* loop over the backed up (permanent) and current (temporary) dvd data */
2762 DVDDrive::Data *dvdData [2];
2763 if (dvd->data().isBackedUp())
2764 {
2765 dvdData [0] = dvd->data().backedUpData();
2766 dvdData [1] = dvd->data().data();
2767 }
2768 else
2769 {
2770 dvdData [0] = dvd->data().data();
2771 dvdData [1] = NULL;
2772 }
2773
2774 if (!(usage & ResourceUsage_PermanentUsage))
2775 dvdData [0] = NULL;
2776 if (!(usage & ResourceUsage_TemporaryUsage))
2777 dvdData [1] = NULL;
2778
2779 for (unsigned i = 0; i < ELEMENTS (dvdData); i++)
2780 {
2781 if (dvdData [i])
2782 {
2783 if (dvdData [i]->mDriveState == DriveState_ImageMounted)
2784 {
2785 Guid iid;
2786 dvdData [i]->mDVDImage->COMGETTER(Id) (iid.asOutParam());
2787 if (iid == id)
2788 idSet.insert (m->data()->mUuid);
2789 }
2790 }
2791 }
2792 }
2793 }
2794
2795 if (machineIDs)
2796 {
2797 if (!idSet.empty())
2798 {
2799 /* convert to a string of UUIDs */
2800 char *idList = (char *) RTMemTmpAllocZ (RTUUID_STR_LENGTH * idSet.size());
2801 char *idListPtr = idList;
2802 for (Set::iterator it = idSet.begin(); it != idSet.end(); ++ it)
2803 {
2804 RTUuidToStr (*it, idListPtr, RTUUID_STR_LENGTH);
2805 idListPtr += RTUUID_STR_LENGTH - 1;
2806 /* replace EOS with a space char */
2807 *(idListPtr ++) = ' ';
2808 }
2809 Assert (int (idListPtr - idList) == int (RTUUID_STR_LENGTH * idSet.size()));
2810 /* remove the trailing space */
2811 *(-- idListPtr) = 0;
2812 /* copy the string */
2813 *machineIDs = idList;
2814 RTMemTmpFree (idList);
2815 }
2816 else
2817 {
2818 (*machineIDs).setNull();
2819 }
2820 }
2821
2822 return !idSet.empty();
2823}
2824
2825/**
2826 * Helper to find machines that use the given floppy image.
2827 *
2828 * @param machineIDs string where to store the list (can be NULL)
2829 * @return TRUE if at least one machine found and false otherwise
2830 *
2831 * @note For now, we just scan all the machines. We can optimize this later
2832 * if required by adding the corresponding field to FloppyImage and requiring all
2833 * IFloppyImage instances to be FloppyImage objects.
2834 *
2835 * @note Locks objects for reading.
2836 */
2837BOOL VirtualBox::getFloppyImageUsage (const Guid &id,
2838 ResourceUsage_T usage,
2839 Bstr *machineIDs)
2840{
2841 AutoCaller autoCaller (this);
2842 AssertComRCReturn (autoCaller.rc(), FALSE);
2843
2844 typedef std::set <Guid> Set;
2845 Set idSet;
2846
2847 {
2848 AutoReaderLock alock (this);
2849
2850 for (MachineList::const_iterator mit = mData.mMachines.begin();
2851 mit != mData.mMachines.end();
2852 ++ mit)
2853 {
2854 /// @todo (dmik) move this part to Machine for better incapsulation
2855
2856 ComObjPtr <Machine> m = *mit;
2857 AutoReaderLock malock (m);
2858
2859 /* take the session machine when appropriate */
2860 if (!m->data()->mSession.mMachine.isNull())
2861 m = m->data()->mSession.mMachine;
2862
2863 const ComObjPtr <FloppyDrive> &drv = m->floppyDrive();
2864 AutoReaderLock dalock (drv);
2865
2866 /* loop over the backed up (permanent) and current (temporary) floppy data */
2867 FloppyDrive::Data *data [2];
2868 if (drv->data().isBackedUp())
2869 {
2870 data [0] = drv->data().backedUpData();
2871 data [1] = drv->data().data();
2872 }
2873 else
2874 {
2875 data [0] = drv->data().data();
2876 data [1] = NULL;
2877 }
2878
2879 if (!(usage & ResourceUsage_PermanentUsage))
2880 data [0] = NULL;
2881 if (!(usage & ResourceUsage_TemporaryUsage))
2882 data [1] = NULL;
2883
2884 for (unsigned i = 0; i < ELEMENTS (data); i++)
2885 {
2886 if (data [i])
2887 {
2888 if (data [i]->mDriveState == DriveState_ImageMounted)
2889 {
2890 Guid iid;
2891 data [i]->mFloppyImage->COMGETTER(Id) (iid.asOutParam());
2892 if (iid == id)
2893 idSet.insert (m->data()->mUuid);
2894 }
2895 }
2896 }
2897 }
2898 }
2899
2900 if (machineIDs)
2901 {
2902 if (!idSet.empty())
2903 {
2904 /* convert to a string of UUIDs */
2905 char *idList = (char *) RTMemTmpAllocZ (RTUUID_STR_LENGTH * idSet.size());
2906 char *idListPtr = idList;
2907 for (Set::iterator it = idSet.begin(); it != idSet.end(); ++ it)
2908 {
2909 RTUuidToStr (*it, idListPtr, RTUUID_STR_LENGTH);
2910 idListPtr += RTUUID_STR_LENGTH - 1;
2911 /* replace EOS with a space char */
2912 *(idListPtr ++) = ' ';
2913 }
2914 Assert (int (idListPtr - idList) == int (RTUUID_STR_LENGTH * idSet.size()));
2915 /* remove the trailing space */
2916 *(-- idListPtr) = 0;
2917 /* copy the string */
2918 *machineIDs = idList;
2919 RTMemTmpFree (idList);
2920 }
2921 else
2922 {
2923 (*machineIDs).setNull();
2924 }
2925 }
2926
2927 return !idSet.empty();
2928}
2929
2930/**
2931 * Tries to calculate the relative path of the given absolute path using the
2932 * directory of the VirtualBox settings file as the base directory.
2933 *
2934 * @param aPath absolute path to calculate the relative path for
2935 * @param aResult where to put the result (used only when it's possible to
2936 * make a relative path from the given absolute path;
2937 * otherwise left untouched)
2938 *
2939 * @note Doesn't lock any object.
2940 */
2941void VirtualBox::calculateRelativePath (const char *aPath, Utf8Str &aResult)
2942{
2943 AutoCaller autoCaller (this);
2944 AssertComRCReturn (autoCaller.rc(), (void) 0);
2945
2946 /* no need to lock since mHomeDir is const */
2947
2948 Utf8Str settingsDir = mData.mHomeDir;
2949
2950 if (RTPathStartsWith (aPath, settingsDir))
2951 {
2952 /* when assigning, we create a separate Utf8Str instance because both
2953 * aPath and aResult can point to the same memory location when this
2954 * func is called (if we just do aResult = aPath, aResult will be freed
2955 * first, and since its the same as aPath, an attempt to copy garbage
2956 * will be made. */
2957 aResult = Utf8Str (aPath + settingsDir.length() + 1);
2958 }
2959}
2960
2961// private methods
2962/////////////////////////////////////////////////////////////////////////////
2963
2964/**
2965 * Searches for a Machine object with the given ID in the collection
2966 * of registered machines.
2967 *
2968 * @param id
2969 * ID of the machine
2970 * @param doSetError
2971 * if TRUE, the appropriate error info is set in case when the machine
2972 * is not found
2973 * @param machine
2974 * where to store the found machine object (can be NULL)
2975 *
2976 * @return
2977 * S_OK when found or E_INVALIDARG when not found
2978 *
2979 * @note Locks this object for reading.
2980 */
2981HRESULT VirtualBox::findMachine (const Guid &aId, bool aSetError,
2982 ComObjPtr <Machine> *aMachine /* = NULL */)
2983{
2984 AutoCaller autoCaller (this);
2985 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
2986
2987 bool found = false;
2988
2989 {
2990 AutoReaderLock alock (this);
2991
2992 for (MachineList::iterator it = mData.mMachines.begin();
2993 !found && it != mData.mMachines.end();
2994 ++ it)
2995 {
2996 /* mUuid is constant, no need to lock */
2997 found = (*it)->data()->mUuid == aId;
2998 if (found && aMachine)
2999 *aMachine = *it;
3000 }
3001 }
3002
3003 HRESULT rc = found ? S_OK : E_INVALIDARG;
3004
3005 if (aSetError && !found)
3006 {
3007 setError (E_INVALIDARG,
3008 tr ("Could not find a registered machine with UUID {%Vuuid}"),
3009 aId.raw());
3010 }
3011
3012 return rc;
3013}
3014
3015/**
3016 * Searches for a HardDisk object with the given ID or location specification
3017 * in the collection of registered hard disks. If both ID and location are
3018 * specified, the first object that matches either of them (not necessarily
3019 * both) is returned.
3020 *
3021 * @param aId ID of the hard disk (NULL when unused)
3022 * @param aLocation full location specification (NULL when unused)
3023 * @param aSetError if TRUE, the appropriate error info is set in case when
3024 * the disk is not found and only one search criteria (ID
3025 * or file name) is specified.
3026 * @param aHardDisk where to store the found hard disk object (can be NULL)
3027 *
3028 * @return
3029 * S_OK when found or E_INVALIDARG when not found
3030 *
3031 * @note Locks objects for reading!
3032 */
3033HRESULT VirtualBox::
3034findHardDisk (const Guid *aId, const BSTR aLocation,
3035 bool aSetError, ComObjPtr <HardDisk> *aHardDisk /* = NULL */)
3036{
3037 ComAssertRet (aId || aLocation, E_INVALIDARG);
3038
3039 AutoReaderLock alock (this);
3040
3041 /* first lookup the map by UUID if UUID is provided */
3042 if (aId)
3043 {
3044 HardDiskMap::const_iterator it = mData.mHardDiskMap.find (*aId);
3045 if (it != mData.mHardDiskMap.end())
3046 {
3047 if (aHardDisk)
3048 *aHardDisk = (*it).second;
3049 return S_OK;
3050 }
3051 }
3052
3053 /* then iterate and find by location */
3054 bool found = false;
3055 if (aLocation)
3056 {
3057 Utf8Str location = aLocation;
3058
3059 for (HardDiskMap::const_iterator it = mData.mHardDiskMap.begin();
3060 !found && it != mData.mHardDiskMap.end();
3061 ++ it)
3062 {
3063 const ComObjPtr <HardDisk> &hd = (*it).second;
3064 AutoReaderLock hdLock (hd);
3065
3066 if (hd->storageType() == HardDiskStorageType_VirtualDiskImage ||
3067 hd->storageType() == HardDiskStorageType_VMDKImage)
3068 {
3069 /* locations of VDI and VMDK hard disks for now are just
3070 * file paths */
3071 found = RTPathCompare (location,
3072 Utf8Str (hd->toString
3073 (false /* aShort */))) == 0;
3074 }
3075 else
3076 {
3077 found = aLocation == hd->toString (false /* aShort */);
3078 }
3079
3080 if (found && aHardDisk)
3081 *aHardDisk = hd;
3082 }
3083 }
3084
3085 HRESULT rc = found ? S_OK : E_INVALIDARG;
3086
3087 if (aSetError && !found)
3088 {
3089 if (aId && !aLocation)
3090 setError (rc, tr ("Could not find a registered hard disk "
3091 "with UUID {%Vuuid}"), aId->raw());
3092 else if (aLocation && !aId)
3093 setError (rc, tr ("Could not find a registered hard disk "
3094 "with location '%ls'"), aLocation);
3095 }
3096
3097 return rc;
3098}
3099
3100/**
3101 * @deprecated Use #findHardDisk() instead.
3102 *
3103 * Searches for a HVirtualDiskImage object with the given ID or file path in the
3104 * collection of registered hard disks. If both ID and file path are specified,
3105 * the first object that matches either of them (not necessarily both)
3106 * is returned.
3107 *
3108 * @param aId ID of the hard disk (NULL when unused)
3109 * @param filePathFull full path to the image file (NULL when unused)
3110 * @param aSetError if TRUE, the appropriate error info is set in case when
3111 * the disk is not found and only one search criteria (ID
3112 * or file name) is specified.
3113 * @param aHardDisk where to store the found hard disk object (can be NULL)
3114 *
3115 * @return
3116 * S_OK when found or E_INVALIDARG when not found
3117 *
3118 * @note Locks objects for reading!
3119 */
3120HRESULT VirtualBox::
3121findVirtualDiskImage (const Guid *aId, const BSTR aFilePathFull,
3122 bool aSetError, ComObjPtr <HVirtualDiskImage> *aImage /* = NULL */)
3123{
3124 ComAssertRet (aId || aFilePathFull, E_INVALIDARG);
3125
3126 AutoReaderLock alock (this);
3127
3128 /* first lookup the map by UUID if UUID is provided */
3129 if (aId)
3130 {
3131 HardDiskMap::const_iterator it = mData.mHardDiskMap.find (*aId);
3132 if (it != mData.mHardDiskMap.end())
3133 {
3134 AutoReaderLock hdLock ((*it).second);
3135 if ((*it).second->storageType() == HardDiskStorageType_VirtualDiskImage)
3136 {
3137 if (aImage)
3138 *aImage = (*it).second->asVDI();
3139 return S_OK;
3140 }
3141 }
3142 }
3143
3144 /* then iterate and find by name */
3145 bool found = false;
3146 if (aFilePathFull)
3147 {
3148 for (HardDiskMap::const_iterator it = mData.mHardDiskMap.begin();
3149 !found && it != mData.mHardDiskMap.end();
3150 ++ it)
3151 {
3152 const ComObjPtr <HardDisk> &hd = (*it).second;
3153 AutoReaderLock hdLock (hd);
3154 if (hd->storageType() != HardDiskStorageType_VirtualDiskImage)
3155 continue;
3156
3157 found = RTPathCompare (Utf8Str (aFilePathFull),
3158 Utf8Str (hd->asVDI()->filePathFull())) == 0;
3159 if (found && aImage)
3160 *aImage = hd->asVDI();
3161 }
3162 }
3163
3164 HRESULT rc = found ? S_OK : E_INVALIDARG;
3165
3166 if (aSetError && !found)
3167 {
3168 if (aId && !aFilePathFull)
3169 setError (rc, tr ("Could not find a registered VDI hard disk "
3170 "with UUID {%Vuuid}"), aId->raw());
3171 else if (aFilePathFull && !aId)
3172 setError (rc, tr ("Could not find a registered VDI hard disk "
3173 "with the file path '%ls'"), aFilePathFull);
3174 }
3175
3176 return rc;
3177}
3178
3179/**
3180 * Searches for a DVDImage object with the given ID or file path in the
3181 * collection of registered DVD images. If both ID and file path are specified,
3182 * the first object that matches either of them (not necessarily both)
3183 * is returned.
3184 *
3185 * @param aId
3186 * ID of the DVD image (unused when NULL)
3187 * @param aFilePathFull
3188 * full path to the image file (unused when NULL)
3189 * @param aSetError
3190 * if TRUE, the appropriate error info is set in case when the image is not
3191 * found and only one search criteria (ID or file name) is specified.
3192 * @param aImage
3193 * where to store the found DVD image object (can be NULL)
3194 *
3195 * @return
3196 * S_OK when found or E_INVALIDARG when not found
3197 *
3198 * @note Locks this object for reading.
3199 */
3200HRESULT VirtualBox::findDVDImage (const Guid *aId, const BSTR aFilePathFull,
3201 bool aSetError,
3202 ComObjPtr <DVDImage> *aImage /* = NULL */)
3203{
3204 ComAssertRet (aId || aFilePathFull, E_INVALIDARG);
3205
3206 bool found = false;
3207
3208 {
3209 AutoReaderLock alock (this);
3210
3211 for (DVDImageList::const_iterator it = mData.mDVDImages.begin();
3212 !found && it != mData.mDVDImages.end();
3213 ++ it)
3214 {
3215 /* DVDImage fields are constant, so no need to lock */
3216 found = (aId && (*it)->id() == *aId) ||
3217 (aFilePathFull &&
3218 RTPathCompare (Utf8Str (aFilePathFull),
3219 Utf8Str ((*it)->filePathFull())) == 0);
3220 if (found && aImage)
3221 *aImage = *it;
3222 }
3223 }
3224
3225 HRESULT rc = found ? S_OK : E_INVALIDARG;
3226
3227 if (aSetError && !found)
3228 {
3229 if (aId && !aFilePathFull)
3230 setError (rc, tr ("Could not find a registered CD/DVD image "
3231 "with UUID {%s}"), aId->toString().raw());
3232 else if (aFilePathFull && !aId)
3233 setError (rc, tr ("Could not find a registered CD/DVD image "
3234 "with the file path '%ls'"), aFilePathFull);
3235 }
3236
3237 return rc;
3238}
3239
3240/**
3241 * Searches for a FloppyImage object with the given ID or file path in the
3242 * collection of registered floppy images. If both ID and file path are specified,
3243 * the first object that matches either of them (not necessarily both)
3244 * is returned.
3245 *
3246 * @param aId
3247 * ID of the floppy image (unused when NULL)
3248 * @param aFilePathFull
3249 * full path to the image file (unused when NULL)
3250 * @param aSetError
3251 * if TRUE, the appropriate error info is set in case when the image is not
3252 * found and only one search criteria (ID or file name) is specified.
3253 * @param aImage
3254 * where to store the found floppy image object (can be NULL)
3255 *
3256 * @return
3257 * S_OK when found or E_INVALIDARG when not found
3258 *
3259 * @note Locks this object for reading.
3260 */
3261HRESULT VirtualBox::findFloppyImage (const Guid *aId, const BSTR aFilePathFull,
3262 bool aSetError,
3263 ComObjPtr <FloppyImage> *aImage /* = NULL */)
3264{
3265 ComAssertRet (aId || aFilePathFull, E_INVALIDARG);
3266
3267 bool found = false;
3268
3269 {
3270 AutoReaderLock alock (this);
3271
3272 for (FloppyImageList::iterator it = mData.mFloppyImages.begin();
3273 !found && it != mData.mFloppyImages.end();
3274 ++ it)
3275 {
3276 /* FloppyImage fields are constant, so no need to lock */
3277 found = (aId && (*it)->id() == *aId) ||
3278 (aFilePathFull &&
3279 RTPathCompare (Utf8Str (aFilePathFull),
3280 Utf8Str ((*it)->filePathFull())) == 0);
3281 if (found && aImage)
3282 *aImage = *it;
3283 }
3284 }
3285
3286 HRESULT rc = found ? S_OK : E_INVALIDARG;
3287
3288 if (aSetError && !found)
3289 {
3290 if (aId && !aFilePathFull)
3291 setError (rc, tr ("Could not find a registered floppy image "
3292 "with UUID {%s}"), aId->toString().raw());
3293 else if (aFilePathFull && !aId)
3294 setError (rc, tr ("Could not find a registered floppy image "
3295 "with the file path '%ls'"), aFilePathFull);
3296 }
3297
3298 return rc;
3299}
3300
3301/**
3302 * When \a aHardDisk is not NULL, searches for an object equal to the given
3303 * hard disk in the collection of registered hard disks, or, if the given hard
3304 * disk is HVirtualDiskImage, for an object with the given file path in the
3305 * collection of all registered non-hard disk images (DVDs and floppies).
3306 * Other parameters are unused.
3307 *
3308 * When \a aHardDisk is NULL, searches for an object with the given ID or file
3309 * path in the collection of all registered images (VDIs, DVDs and floppies).
3310 * If both ID and file path are specified, matching either of them will satisfy
3311 * the search.
3312 *
3313 * If a matching object is found, this method returns E_INVALIDARG and sets the
3314 * appropriate error info. Otherwise, S_OK is returned.
3315 *
3316 * @param aHardDisk hard disk object to check against registered media
3317 * (NULL when unused)
3318 * @param aId UUID of the media to check (NULL when unused)
3319 * @param aFilePathFull full path to the image file (NULL when unused)
3320 *
3321 * @note Locks objects!
3322 */
3323HRESULT VirtualBox::checkMediaForConflicts (HardDisk *aHardDisk,
3324 const Guid *aId,
3325 const BSTR aFilePathFull)
3326{
3327 AssertReturn (aHardDisk || aId || aFilePathFull, E_FAIL);
3328
3329 HRESULT rc = S_OK;
3330
3331 AutoReaderLock alock (this);
3332
3333 if (aHardDisk)
3334 {
3335 for (HardDiskMap::const_iterator it = mData.mHardDiskMap.begin();
3336 it != mData.mHardDiskMap.end();
3337 ++ it)
3338 {
3339 const ComObjPtr <HardDisk> &hd = (*it).second;
3340 if (hd->sameAs (aHardDisk))
3341 return setError (E_INVALIDARG,
3342 tr ("A hard disk with UUID {%Vuuid} or with the same properties "
3343 "('%ls') is already registered"),
3344 aHardDisk->id().raw(), aHardDisk->toString().raw());
3345 }
3346
3347 aId = &aHardDisk->id();
3348 if (aHardDisk->storageType() == HardDiskStorageType_VirtualDiskImage)
3349#if !defined (VBOX_WITH_XPCOM)
3350#if defined(RT_OS_WINDOWS)
3351 /// @todo (dmik) stupid BSTR declaration lacks the BCSTR counterpart
3352 const_cast <BSTR> (aFilePathFull) = aHardDisk->asVDI()->filePathFull();
3353#endif
3354#else
3355 aFilePathFull = aHardDisk->asVDI()->filePathFull();
3356#endif
3357 }
3358
3359 bool found = false;
3360
3361 if (aId || aFilePathFull) do
3362 {
3363 if (!aHardDisk)
3364 {
3365 rc = findHardDisk (aId, aFilePathFull, false /* aSetError */);
3366 found = SUCCEEDED (rc);
3367 if (found)
3368 break;
3369 }
3370
3371 rc = findDVDImage (aId, aFilePathFull, false /* aSetError */);
3372 found = SUCCEEDED (rc);
3373 if (found)
3374 break;
3375
3376 rc = findFloppyImage (aId, aFilePathFull, false /* aSetError */);
3377 found = SUCCEEDED (rc);
3378 if (found)
3379 break;
3380 }
3381 while (0);
3382
3383 if (found)
3384 {
3385 if (aId && !aFilePathFull)
3386 rc = setError (E_INVALIDARG,
3387 tr ("A disk image with UUID {%Vuuid} is already registered"),
3388 aId->raw());
3389 else if (aFilePathFull && !aId)
3390 rc = setError (E_INVALIDARG,
3391 tr ("A disk image with file path '%ls' is already registered"),
3392 aFilePathFull);
3393 else
3394 rc = setError (E_INVALIDARG,
3395 tr ("A disk image with UUID {%Vuuid} or file path '%ls' "
3396 "is already registered"), aId->raw(), aFilePathFull);
3397 }
3398 else
3399 rc = S_OK;
3400
3401 return rc;
3402}
3403
3404/**
3405 * Reads in the machine definitions from the configuration loader
3406 * and creates the relevant objects.
3407 *
3408 * @note Can be called only from #init().
3409 * @note Doesn't lock anything.
3410 */
3411HRESULT VirtualBox::loadMachines (CFGNODE aGlobal)
3412{
3413 AutoCaller autoCaller (this);
3414 AssertReturn (autoCaller.state() == InInit, E_FAIL);
3415
3416 HRESULT rc = S_OK;
3417 CFGNODE machineRegistry = 0;
3418 unsigned count = 0;
3419
3420 CFGLDRGetChildNode (aGlobal, "MachineRegistry", 0, &machineRegistry);
3421 Assert (machineRegistry);
3422
3423 CFGLDRCountChildren(machineRegistry, "MachineEntry", &count);
3424 for (unsigned i = 0; i < count && SUCCEEDED (rc); i++)
3425 {
3426 CFGNODE vm = 0;
3427 CFGLDRGetChildNode(machineRegistry, "MachineEntry", i, &vm);
3428 /* get the UUID */
3429 Guid uuid;
3430 CFGLDRQueryUUID(vm, "uuid", uuid.ptr());
3431 /* get the machine configuration file name */
3432 Bstr src;
3433 CFGLDRQueryBSTR(vm, "src", src.asOutParam());
3434
3435 /* create a new object */
3436 ComObjPtr <Machine> machine;
3437 rc = machine.createObject();
3438 if (SUCCEEDED (rc))
3439 {
3440 /* initialize the machine object and register it */
3441 rc = machine->init (this, src, Machine::Init_Registered,
3442 NULL, FALSE, &uuid);
3443 if (SUCCEEDED (rc))
3444 rc = registerMachine (machine);
3445 }
3446
3447 CFGLDRReleaseNode(vm);
3448 }
3449
3450 CFGLDRReleaseNode(machineRegistry);
3451
3452 return rc;
3453}
3454
3455/**
3456 * Reads in the disk registration entries from the global settings file
3457 * and creates the relevant objects
3458 *
3459 * @param aGlobal <Global> node
3460 *
3461 * @note Can be called only from #init().
3462 * @note Doesn't lock anything.
3463 */
3464HRESULT VirtualBox::loadDisks (CFGNODE aGlobal)
3465{
3466 AutoCaller autoCaller (this);
3467 AssertReturn (autoCaller.state() == InInit, E_FAIL);
3468
3469 HRESULT rc = S_OK;
3470 CFGNODE registryNode = 0;
3471
3472 CFGLDRGetChildNode (aGlobal, "DiskRegistry", 0, &registryNode);
3473 ComAssertRet (registryNode, E_FAIL);
3474
3475 const char *ImagesNodes[] = { "HardDisks", "DVDImages", "FloppyImages" };
3476
3477 for (size_t node = 0; node < ELEMENTS (ImagesNodes) && SUCCEEDED (rc); node ++)
3478 {
3479 CFGNODE imagesNode = 0;
3480 CFGLDRGetChildNode (registryNode, ImagesNodes [node], 0, &imagesNode);
3481
3482 // all three nodes are optional
3483 if (!imagesNode)
3484 continue;
3485
3486 if (node == 0) // HardDisks node
3487 rc = loadHardDisks (imagesNode);
3488 else
3489 {
3490 unsigned count = 0;
3491 CFGLDRCountChildren (imagesNode, "Image", &count);
3492 for (unsigned i = 0; i < count && SUCCEEDED (rc); ++ i)
3493 {
3494 CFGNODE imageNode = 0;
3495 CFGLDRGetChildNode (imagesNode, "Image", i, &imageNode);
3496 ComAssertBreak (imageNode, rc = E_FAIL);
3497
3498 Guid uuid; // uuid (required)
3499 CFGLDRQueryUUID (imageNode, "uuid", uuid.ptr());
3500 Bstr src; // source (required)
3501 CFGLDRQueryBSTR (imageNode, "src", src.asOutParam());
3502
3503 switch (node)
3504 {
3505 case 1: // DVDImages
3506 {
3507 ComObjPtr <DVDImage> dvdImage;
3508 dvdImage.createObject();
3509 rc = dvdImage->init (this, src, TRUE /* isRegistered */, uuid);
3510 if (SUCCEEDED (rc))
3511 rc = registerDVDImage (dvdImage, TRUE /* aOnStartUp */);
3512
3513 break;
3514 }
3515 case 2: // FloppyImages
3516 {
3517 ComObjPtr <FloppyImage> floppyImage;
3518 floppyImage.createObject();
3519 rc = floppyImage->init (this, src, TRUE /* isRegistered */, uuid);
3520 if (SUCCEEDED (rc))
3521 rc = registerFloppyImage (floppyImage, TRUE /* aOnStartUp */);
3522
3523 break;
3524 }
3525 default:
3526 AssertFailed();
3527 }
3528
3529 CFGLDRReleaseNode (imageNode);
3530 }
3531 }
3532
3533 CFGLDRReleaseNode (imagesNode);
3534 }
3535
3536 CFGLDRReleaseNode (registryNode);
3537
3538 return rc;
3539}
3540
3541/**
3542 * Loads all hard disks from the given <HardDisks> node.
3543 * Note that all loaded hard disks register themselves within this VirtualBox.
3544 *
3545 * @param aNode <HardDisks> node
3546 *
3547 * @note Can be called only from #init().
3548 * @note Doesn't lock anything.
3549 */
3550HRESULT VirtualBox::loadHardDisks (CFGNODE aNode)
3551{
3552 AutoCaller autoCaller (this);
3553 AssertReturn (autoCaller.state() == InInit, E_FAIL);
3554
3555 AssertReturn (aNode, E_INVALIDARG);
3556
3557 HRESULT rc = S_OK;
3558
3559 unsigned count = 0;
3560 CFGLDRCountChildren (aNode, "HardDisk", &count);
3561 for (unsigned i = 0; i < count && SUCCEEDED (rc); ++ i)
3562 {
3563 CFGNODE hdNode = 0;
3564
3565 CFGLDRGetChildNode (aNode, "HardDisk", i, &hdNode);
3566 ComAssertBreak (hdNode, rc = E_FAIL);
3567
3568 {
3569 CFGNODE storageNode = 0;
3570
3571 // detect the type of the hard disk
3572 // (either one of HVirtualDiskImage, HISCSIHardDisk or HPhysicalVolume
3573 do
3574 {
3575 CFGLDRGetChildNode (hdNode, "VirtualDiskImage", 0, &storageNode);
3576 if (storageNode)
3577 {
3578 ComObjPtr <HVirtualDiskImage> vdi;
3579 vdi.createObject();
3580 rc = vdi->init (this, NULL, hdNode, storageNode);
3581 break;
3582 }
3583
3584 CFGLDRGetChildNode (hdNode, "ISCSIHardDisk", 0, &storageNode);
3585 if (storageNode)
3586 {
3587 ComObjPtr <HISCSIHardDisk> iscsi;
3588 iscsi.createObject();
3589 rc = iscsi->init (this, hdNode, storageNode);
3590 break;
3591 }
3592
3593 CFGLDRGetChildNode (hdNode, "VMDKImage", 0, &storageNode);
3594 if (storageNode)
3595 {
3596 ComObjPtr <HVMDKImage> vmdk;
3597 vmdk.createObject();
3598 rc = vmdk->init (this, NULL, hdNode, storageNode);
3599 break;
3600 }
3601
3602 /// @todo (dmik) later
3603// CFGLDRGetChildNode (hdNode, "PhysicalVolume", 0, &storageNode);
3604// if (storageNode)
3605// {
3606// break;
3607// }
3608
3609 ComAssertMsgFailedBreak (("No valid hard disk storage node!\n"),
3610 rc = E_FAIL);
3611 }
3612 while (0);
3613
3614 if (storageNode)
3615 CFGLDRReleaseNode (storageNode);
3616 }
3617
3618 CFGLDRReleaseNode (hdNode);
3619 }
3620
3621 return rc;
3622}
3623
3624/**
3625 * Helper function to write out the configuration to XML.
3626 *
3627 * @note Locks objects!
3628 */
3629HRESULT VirtualBox::saveConfig()
3630{
3631 AutoCaller autoCaller (this);
3632 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
3633
3634 ComAssertRet (!!mData.mCfgFile.mName, E_FAIL);
3635
3636 HRESULT rc = S_OK;
3637
3638 AutoLock alock (this);
3639
3640 CFGHANDLE configLoader;
3641
3642 /* load the config file */
3643 int vrc = CFGLDRLoad (&configLoader, Utf8Str (mData.mCfgFile.mName),
3644 mData.mCfgFile.mHandle,
3645 XmlSchemaNS, true, cfgLdrEntityResolver, NULL);
3646 ComAssertRCRet (vrc, E_FAIL);
3647
3648 const char * Global = "VirtualBox/Global";
3649 CFGNODE global;
3650
3651 vrc = CFGLDRGetNode(configLoader, Global, 0, &global);
3652 if (VBOX_FAILURE (vrc))
3653 CFGLDRCreateNode (configLoader, Global, &global);
3654
3655 do
3656 {
3657 ComAssertBreak (global, rc = E_FAIL);
3658
3659 /* machines */
3660 do
3661 {
3662 const char *Registry = "MachineRegistry";
3663 const char *Entry = "MachineEntry";
3664 CFGNODE registryNode = NULL;
3665
3666 /* first, delete the entire machine registry */
3667 if (VBOX_SUCCESS (CFGLDRGetChildNode (global, Registry, 0, &registryNode)))
3668 CFGLDRDeleteNode (registryNode);
3669
3670 /* then, recreate it */
3671 CFGLDRCreateChildNode (global, Registry, &registryNode);
3672
3673 /* write out the machines */
3674 for (MachineList::iterator it = mData.mMachines.begin();
3675 it != mData.mMachines.end();
3676 ++ it)
3677 {
3678 /// @todo (dmik) move this part to Machine for better incapsulation
3679
3680 ComObjPtr <Machine> m = *it;
3681 AutoReaderLock machineLock (m);
3682
3683 AssertMsg (m->data(), ("Machine data must not be NULL"));
3684 CFGNODE entryNode;
3685 CFGLDRAppendChildNode (registryNode, Entry, &entryNode);
3686 /* UUID + curly brackets */
3687 CFGLDRSetUUID (entryNode, "uuid", unconst (m->data()->mUuid).ptr());
3688 /* source */
3689 CFGLDRSetBSTR (entryNode, "src", m->data()->mConfigFile);
3690 /* done */
3691 CFGLDRReleaseNode (entryNode);
3692 }
3693
3694 CFGLDRReleaseNode (registryNode);
3695 }
3696 while (0);
3697 if (FAILED (rc))
3698 break;
3699
3700 /* disk images */
3701 do
3702 {
3703 CFGNODE registryNode = 0;
3704 CFGLDRGetChildNode (global, "DiskRegistry", 0, &registryNode);
3705 /* first, delete the entire disk image registr */
3706 if (registryNode)
3707 CFGLDRDeleteNode (registryNode);
3708 /* then, recreate it */
3709 CFGLDRCreateChildNode (global, "DiskRegistry", &registryNode);
3710 ComAssertBreak (registryNode, rc = E_FAIL);
3711
3712 /* write out the hard disks */
3713 {
3714 CFGNODE imagesNode = 0;
3715 CFGLDRCreateChildNode (registryNode, "HardDisks", &imagesNode);
3716 rc = saveHardDisks (imagesNode);
3717 CFGLDRReleaseNode (imagesNode);
3718 if (FAILED (rc))
3719 break;
3720 }
3721
3722 /* write out the CD/DVD images */
3723 {
3724 CFGNODE imagesNode = 0;
3725 CFGLDRCreateChildNode (registryNode, "DVDImages", &imagesNode);
3726
3727 for (DVDImageList::iterator it = mData.mDVDImages.begin();
3728 it != mData.mDVDImages.end();
3729 ++ it)
3730 {
3731 ComObjPtr <DVDImage> dvd = *it;
3732 /* no need to lock: fields are constant */
3733 CFGNODE imageNode = 0;
3734 CFGLDRAppendChildNode (imagesNode, "Image", &imageNode);
3735 CFGLDRSetUUID (imageNode, "uuid", dvd->id());
3736 CFGLDRSetBSTR (imageNode, "src", dvd->filePath());
3737 CFGLDRReleaseNode (imageNode);
3738 }
3739
3740 CFGLDRReleaseNode (imagesNode);
3741 }
3742
3743 /* write out the floppy images */
3744 {
3745 CFGNODE imagesNode = 0;
3746 CFGLDRCreateChildNode (registryNode, "FloppyImages", &imagesNode);
3747
3748 for (FloppyImageList::iterator it = mData.mFloppyImages.begin();
3749 it != mData.mFloppyImages.end();
3750 ++ it)
3751 {
3752 ComObjPtr <FloppyImage> fd = *it;
3753 /* no need to lock: fields are constant */
3754 CFGNODE imageNode = 0;
3755 CFGLDRAppendChildNode (imagesNode, "Image", &imageNode);
3756 CFGLDRSetUUID (imageNode, "uuid", fd->id());
3757 CFGLDRSetBSTR (imageNode, "src", fd->filePath());
3758 CFGLDRReleaseNode (imageNode);
3759 }
3760
3761 CFGLDRReleaseNode (imagesNode);
3762 }
3763
3764 CFGLDRReleaseNode (registryNode);
3765 }
3766 while (0);
3767 if (FAILED (rc))
3768 break;
3769
3770 do
3771 {
3772 /* host data (USB filters) */
3773 rc = mData.mHost->saveSettings (global);
3774 if (FAILED (rc))
3775 break;
3776
3777 rc = mData.mSystemProperties->saveSettings (global);
3778 if (FAILED (rc))
3779 break;
3780 }
3781 while (0);
3782 }
3783 while (0);
3784
3785 if (global)
3786 CFGLDRReleaseNode (global);
3787
3788 if (SUCCEEDED (rc))
3789 {
3790 char *loaderError = NULL;
3791 vrc = CFGLDRSave (configLoader, &loaderError);
3792 if (VBOX_FAILURE (vrc))
3793 {
3794 rc = setError (E_FAIL,
3795 tr ("Could not save the settings file '%ls' (%Vrc)%s%s"),
3796 mData.mCfgFile.mName.raw(), vrc,
3797 loaderError ? ".\n" : "", loaderError ? loaderError : "");
3798 if (loaderError)
3799 RTMemTmpFree (loaderError);
3800 }
3801 }
3802
3803 CFGLDRFree(configLoader);
3804
3805 return rc;
3806}
3807
3808/**
3809 * Saves all hard disks to the given <HardDisks> node.
3810 *
3811 * @param aNode <HardDisks> node
3812 *
3813 * @note Locks this object for reding.
3814 */
3815HRESULT VirtualBox::saveHardDisks (CFGNODE aNode)
3816{
3817 AssertReturn (aNode, E_INVALIDARG);
3818
3819 HRESULT rc = S_OK;
3820
3821 AutoReaderLock alock (this);
3822
3823 for (HardDiskList::const_iterator it = mData.mHardDisks.begin();
3824 it != mData.mHardDisks.end() && SUCCEEDED (rc);
3825 ++ it)
3826 {
3827 ComObjPtr <HardDisk> hd = *it;
3828 AutoReaderLock hdLock (hd);
3829
3830 CFGNODE hdNode = 0;
3831 CFGLDRAppendChildNode (aNode, "HardDisk", &hdNode);
3832 ComAssertBreak (hdNode, rc = E_FAIL);
3833
3834 CFGNODE storageNode = 0;
3835
3836 switch (hd->storageType())
3837 {
3838 case HardDiskStorageType_VirtualDiskImage:
3839 {
3840 CFGLDRAppendChildNode (hdNode, "VirtualDiskImage", &storageNode);
3841 ComAssertBreak (storageNode, rc = E_FAIL);
3842 rc = hd->saveSettings (hdNode, storageNode);
3843 break;
3844 }
3845
3846 case HardDiskStorageType_ISCSIHardDisk:
3847 {
3848 CFGLDRAppendChildNode (hdNode, "ISCSIHardDisk", &storageNode);
3849 ComAssertBreak (storageNode, rc = E_FAIL);
3850 rc = hd->saveSettings (hdNode, storageNode);
3851 break;
3852 }
3853
3854 case HardDiskStorageType_VMDKImage:
3855 {
3856 CFGLDRAppendChildNode (hdNode, "VMDKImage", &storageNode);
3857 ComAssertBreak (storageNode, rc = E_FAIL);
3858 rc = hd->saveSettings (hdNode, storageNode);
3859 break;
3860 }
3861
3862 /// @todo (dmik) later
3863// case HardDiskStorageType_PhysicalVolume:
3864// {
3865// break;
3866// }
3867 }
3868
3869 if (storageNode)
3870 CFGLDRReleaseNode (storageNode);
3871
3872 CFGLDRReleaseNode (hdNode);
3873 }
3874
3875 return rc;
3876}
3877
3878/**
3879 * Helper to register the machine.
3880 *
3881 * When called during VirtualBox startup, adds the given machine to the
3882 * collection of registered machines. Otherwise tries to mark the machine
3883 * as registered, and, if succeeded, adds it to the collection and
3884 * saves global settings.
3885 *
3886 * @param aMachine machine to register
3887 *
3888 * @note Locks objects!
3889 */
3890HRESULT VirtualBox::registerMachine (Machine *aMachine)
3891{
3892 ComAssertRet (aMachine, E_INVALIDARG);
3893
3894 AutoCaller autoCaller (this);
3895 CheckComRCReturnRC (autoCaller.rc());
3896
3897 AutoLock alock (this);
3898
3899 ComAssertRet (findMachine (aMachine->data()->mUuid,
3900 false /* aDoSetError */, NULL) == E_INVALIDARG,
3901 E_FAIL);
3902
3903 HRESULT rc = S_OK;
3904
3905 if (autoCaller.state() != InInit)
3906 {
3907 /* Machine::trySetRegistered() will commit and save machine settings */
3908 rc = aMachine->trySetRegistered (TRUE);
3909 CheckComRCReturnRC (rc);
3910 }
3911
3912 /* add to the collection of registered machines */
3913 mData.mMachines.push_back (aMachine);
3914
3915 if (autoCaller.state() != InInit)
3916 rc = saveConfig();
3917
3918 return rc;
3919}
3920
3921/**
3922 * Helper to register the hard disk.
3923 *
3924 * @param aHardDisk object to register
3925 * @param aFlags one of RHD_* values
3926 *
3927 * @note Locks objects!
3928 */
3929HRESULT VirtualBox::registerHardDisk (HardDisk *aHardDisk, RHD_Flags aFlags)
3930{
3931 ComAssertRet (aHardDisk, E_INVALIDARG);
3932
3933 AutoCaller autoCaller (this);
3934 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
3935
3936 AutoLock alock (this);
3937
3938 HRESULT rc = checkMediaForConflicts (aHardDisk, NULL, NULL);
3939 CheckComRCReturnRC (rc);
3940
3941 /* mark the hard disk as registered only when registration is external */
3942 if (aFlags == RHD_External)
3943 {
3944 rc = aHardDisk->trySetRegistered (TRUE);
3945 CheckComRCReturnRC (rc);
3946 }
3947
3948 if (!aHardDisk->parent())
3949 {
3950 /* add to the collection of top-level images */
3951 mData.mHardDisks.push_back (aHardDisk);
3952 }
3953
3954 /* insert to the map of hard disks */
3955 mData.mHardDiskMap
3956 .insert (HardDiskMap::value_type (aHardDisk->id(), aHardDisk));
3957
3958 /* save global config file if not on startup */
3959 /// @todo (dmik) optimize later to save only the <HardDisks> node
3960 if (aFlags != RHD_OnStartUp)
3961 rc = saveConfig();
3962
3963 return rc;
3964}
3965
3966/**
3967 * Helper to unregister the hard disk.
3968 *
3969 * If the hard disk is a differencing hard disk and if the unregistration
3970 * succeeds, the hard disk image is deleted and the object is uninitialized.
3971 *
3972 * @param aHardDisk hard disk to unregister
3973 *
3974 * @note Locks objects!
3975 */
3976HRESULT VirtualBox::unregisterHardDisk (HardDisk *aHardDisk)
3977{
3978 AssertReturn (aHardDisk, E_INVALIDARG);
3979
3980 AutoCaller autoCaller (this);
3981 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
3982
3983 LogFlowThisFunc (("image='%ls'\n", aHardDisk->toString().raw()));
3984
3985 AutoLock alock (this);
3986
3987 /* Lock the hard disk to ensure nobody registers it again before we delete
3988 * the differencing image (sanity check actually -- should never happen). */
3989 AutoLock hdLock (aHardDisk);
3990
3991 /* try to unregister */
3992 HRESULT rc = aHardDisk->trySetRegistered (FALSE);
3993 CheckComRCReturnRC (rc);
3994
3995 /* remove from the map of hard disks */
3996 mData.mHardDiskMap.erase (aHardDisk->id());
3997
3998 if (!aHardDisk->parent())
3999 {
4000 /* non-differencing hard disk:
4001 * remove from the collection of top-level hard disks */
4002 mData.mHardDisks.remove (aHardDisk);
4003 }
4004 else
4005 {
4006 Assert (aHardDisk->isDifferencing());
4007
4008 /* differencing hard disk: delete (only if the last access check
4009 * succeeded) and uninitialize */
4010 if (aHardDisk->asVDI()->lastAccessError().isNull())
4011 rc = aHardDisk->asVDI()->DeleteImage();
4012 aHardDisk->uninit();
4013 }
4014
4015 /* save the global config file anyway (already unregistered) */
4016 /// @todo (dmik) optimize later to save only the <HardDisks> node
4017 HRESULT rc2 = saveConfig();
4018 if (SUCCEEDED (rc))
4019 rc = rc2;
4020
4021 return rc;
4022}
4023
4024/**
4025 * Helper to unregister the differencing hard disk image.
4026 * Resets machine ID of the hard disk (to let the unregistration succeed)
4027 * and then calls #unregisterHardDisk().
4028 *
4029 * @param aHardDisk differencing hard disk image to unregister
4030 *
4031 * @note Locks objects!
4032 */
4033HRESULT VirtualBox::unregisterDiffHardDisk (HardDisk *aHardDisk)
4034{
4035 AssertReturn (aHardDisk, E_INVALIDARG);
4036
4037 AutoCaller autoCaller (this);
4038 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4039
4040 AutoLock alock (this);
4041
4042 /*
4043 * Note: it's safe to lock aHardDisk here because the same object
4044 * will be locked by #unregisterHardDisk().
4045 */
4046 AutoLock hdLock (aHardDisk);
4047
4048 AssertReturn (aHardDisk->isDifferencing(), E_INVALIDARG);
4049
4050 /*
4051 * deassociate the machine from the hard disk
4052 * (otherwise trySetRegistered() will definitely fail)
4053 */
4054 aHardDisk->setMachineId (Guid());
4055
4056 return unregisterHardDisk (aHardDisk);
4057}
4058
4059
4060/**
4061 * Helper to update the global settings file when the name of some machine
4062 * changes so that file and directory renaming occurs. This method ensures
4063 * that all affected paths in the disk registry are properly updated.
4064 *
4065 * @param aOldPath old path (full)
4066 * @param aNewPath new path (full)
4067 *
4068 * @note Locks this object + DVD, Floppy and HardDisk children for writing.
4069 */
4070HRESULT VirtualBox::updateSettings (const char *aOldPath, const char *aNewPath)
4071{
4072 LogFlowThisFunc (("aOldPath={%s} aNewPath={%s}\n", aOldPath, aNewPath));
4073
4074 AssertReturn (aOldPath, E_INVALIDARG);
4075 AssertReturn (aNewPath, E_INVALIDARG);
4076
4077 AutoCaller autoCaller (this);
4078 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4079
4080 AutoLock alock (this);
4081
4082 size_t oldPathLen = strlen (aOldPath);
4083
4084 /* check DVD paths */
4085 for (DVDImageList::iterator it = mData.mDVDImages.begin();
4086 it != mData.mDVDImages.end();
4087 ++ it)
4088 {
4089 ComObjPtr <DVDImage> image = *it;
4090
4091 /* no need to lock: fields are constant */
4092 Utf8Str path = image->filePathFull();
4093 LogFlowThisFunc (("DVD.fullPath={%s}\n", path.raw()));
4094
4095 if (RTPathStartsWith (path, aOldPath))
4096 {
4097 Utf8Str newPath = Utf8StrFmt ("%s%s", aNewPath,
4098 path.raw() + oldPathLen);
4099 path = newPath;
4100 calculateRelativePath (path, path);
4101 image->updatePath (newPath, path);
4102
4103 LogFlowThisFunc (("-> updated: full={%s} rel={%s}\n",
4104 newPath.raw(), path.raw()));
4105 }
4106 }
4107
4108 /* check Floppy paths */
4109 for (FloppyImageList::iterator it = mData.mFloppyImages.begin();
4110 it != mData.mFloppyImages.end();
4111 ++ it)
4112 {
4113 ComObjPtr <FloppyImage> image = *it;
4114
4115 /* no need to lock: fields are constant */
4116 Utf8Str path = image->filePathFull();
4117 LogFlowThisFunc (("Floppy.fullPath={%s}\n", path.raw()));
4118
4119 if (RTPathStartsWith (path, aOldPath))
4120 {
4121 Utf8Str newPath = Utf8StrFmt ("%s%s", aNewPath,
4122 path.raw() + oldPathLen);
4123 path = newPath;
4124 calculateRelativePath (path, path);
4125 image->updatePath (newPath, path);
4126
4127 LogFlowThisFunc (("-> updated: full={%s} rel={%s}\n",
4128 newPath.raw(), path.raw()));
4129 }
4130 }
4131
4132 /* check HardDisk paths */
4133 for (HardDiskList::const_iterator it = mData.mHardDisks.begin();
4134 it != mData.mHardDisks.end();
4135 ++ it)
4136 {
4137 (*it)->updatePaths (aOldPath, aNewPath);
4138 }
4139
4140 HRESULT rc = saveConfig();
4141
4142 return rc;
4143}
4144
4145/**
4146 * Helper to register the DVD image.
4147 *
4148 * @param aImage object to register
4149 * @param aOnStartUp whether this method called during VirtualBox init or not
4150 *
4151 * @return COM status code
4152 *
4153 * @note Locks objects!
4154 */
4155HRESULT VirtualBox::registerDVDImage (DVDImage *aImage, bool aOnStartUp)
4156{
4157 AssertReturn (aImage, E_INVALIDARG);
4158
4159 AutoCaller autoCaller (this);
4160 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4161
4162 AutoLock alock (this);
4163
4164 HRESULT rc = checkMediaForConflicts (NULL, &aImage->id(),
4165 aImage->filePathFull());
4166 CheckComRCReturnRC (rc);
4167
4168 /* add to the collection */
4169 mData.mDVDImages.push_back (aImage);
4170
4171 /* save global config file if we're supposed to */
4172 if (!aOnStartUp)
4173 rc = saveConfig();
4174
4175 return rc;
4176}
4177
4178/**
4179 * Helper to register the floppy image.
4180 *
4181 * @param aImage object to register
4182 * @param aOnStartUp whether this method called during VirtualBox init or not
4183 *
4184 * @return COM status code
4185 *
4186 * @note Locks objects!
4187 */
4188HRESULT VirtualBox::registerFloppyImage (FloppyImage *aImage, bool aOnStartUp)
4189{
4190 AssertReturn (aImage, E_INVALIDARG);
4191
4192 AutoCaller autoCaller (this);
4193 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4194
4195 AutoLock alock (this);
4196
4197 HRESULT rc = checkMediaForConflicts (NULL, &aImage->id(),
4198 aImage->filePathFull());
4199 CheckComRCReturnRC (rc);
4200
4201 /* add to the collection */
4202 mData.mFloppyImages.push_back (aImage);
4203
4204 /* save global config file if we're supposed to */
4205 if (!aOnStartUp)
4206 rc = saveConfig();
4207
4208 return rc;
4209}
4210
4211/**
4212 * Helper function to create the guest OS type objects and our collection
4213 *
4214 * @returns COM status code
4215 */
4216HRESULT VirtualBox::registerGuestOSTypes()
4217{
4218 AutoCaller autoCaller (this);
4219 AssertComRCReturn (autoCaller.rc(), E_FAIL);
4220 AssertReturn (autoCaller.state() == InInit, E_FAIL);
4221
4222 HRESULT rc = S_OK;
4223
4224 // this table represents our os type / string mapping
4225 static struct
4226 {
4227 const char *id; // utf-8
4228 const char *description; // utf-8
4229 const OSType osType;
4230 const uint32_t recommendedRAM;
4231 const uint32_t recommendedVRAM;
4232 const uint32_t recommendedHDD;
4233 } OSTypes[] =
4234 {
4235 /// @todo (dmik) get the list of OS types from the XML schema
4236 /* NOTE1: we assume that unknown is always the first entry!
4237 * NOTE2: please use powers of 2 when specifying the size of harddisks since
4238 * '2GB' looks better than '1.95GB' (= 2000MB) */
4239 { "unknown", tr ("Other/Unknown"), OSTypeUnknown, 64, 4, 2 * _1K },
4240 { "dos", "DOS", OSTypeDOS, 32, 4, 512 },
4241 { "win31", "Windows 3.1", OSTypeWin31, 32, 4, 1 * _1K },
4242 { "win95", "Windows 95", OSTypeWin95, 64, 4, 2 * _1K },
4243 { "win98", "Windows 98", OSTypeWin98, 64, 4, 2 * _1K },
4244 { "winme", "Windows Me", OSTypeWinMe, 64, 4, 4 * _1K },
4245 { "winnt4", "Windows NT 4", OSTypeWinNT4, 128, 4, 2 * _1K },
4246 { "win2k", "Windows 2000", OSTypeWin2k, 168, 4, 4 * _1K },
4247 { "winxp", "Windows XP", OSTypeWinXP, 192, 4, 10 * _1K },
4248 { "win2k3", "Windows Server 2003", OSTypeWin2k3, 256, 4, 20 * _1K },
4249 { "winvista", "Windows Vista", OSTypeWinVista, 512, 4, 20 * _1K },
4250 { "os2warp3", "OS/2 Warp 3", OSTypeOS2Warp3, 48, 4, 1 * _1K },
4251 { "os2warp4", "OS/2 Warp 4", OSTypeOS2Warp4, 64, 4, 2 * _1K },
4252 { "os2warp45", "OS/2 Warp 4.5", OSTypeOS2Warp45, 96, 4, 2 * _1K },
4253 { "linux22", "Linux 2.2", OSTypeLinux22, 64, 4, 2 * _1K },
4254 { "linux24", "Linux 2.4", OSTypeLinux24, 128, 4, 4 * _1K },
4255 { "linux26", "Linux 2.6", OSTypeLinux26, 256, 4, 8 * _1K },
4256 { "freebsd", "FreeBSD", OSTypeFreeBSD, 64, 4, 2 * _1K },
4257 { "openbsd", "OpenBSD", OSTypeOpenBSD, 64, 4, 2 * _1K },
4258 { "netbsd", "NetBSD", OSTypeNetBSD, 64, 4, 2 * _1K },
4259 { "netware", "Netware", OSTypeNetware, 128, 4, 4 * _1K },
4260 { "solaris", "Solaris", OSTypeSolaris, 128, 4, 8 * _1K },
4261 { "l4", "L4", OSTypeL4, 64, 4, 2 * _1K }
4262 };
4263
4264 for (uint32_t i = 0; i < ELEMENTS (OSTypes) && SUCCEEDED (rc); i++)
4265 {
4266 ComObjPtr <GuestOSType> guestOSTypeObj;
4267 rc = guestOSTypeObj.createObject();
4268 if (SUCCEEDED (rc))
4269 {
4270 rc = guestOSTypeObj->init (OSTypes[i].id,
4271 OSTypes[i].description,
4272 OSTypes[i].osType,
4273 OSTypes[i].recommendedRAM,
4274 OSTypes[i].recommendedVRAM,
4275 OSTypes[i].recommendedHDD);
4276 if (SUCCEEDED (rc))
4277 mData.mGuestOSTypes.push_back (guestOSTypeObj);
4278 }
4279 }
4280
4281 return rc;
4282}
4283
4284/**
4285 * Helper to lock the VirtualBox configuration for write access.
4286 *
4287 * @note This method is not thread safe (must be called only from #init()
4288 * or #uninit()).
4289 *
4290 * @note If the configuration file is not found, the method returns
4291 * S_OK, but subsequent #isConfigLocked() will return FALSE. This is used
4292 * in some places to determine the (valid) situation when no config file
4293 * exists yet, and therefore a new one should be created from scatch.
4294 */
4295HRESULT VirtualBox::lockConfig()
4296{
4297 AutoCaller autoCaller (this);
4298 AssertComRCReturn (autoCaller.rc(), autoCaller.rc());
4299 AssertReturn (autoCaller.state() == InInit, E_FAIL);
4300
4301 HRESULT rc = S_OK;
4302
4303 Assert (!isConfigLocked());
4304 if (!isConfigLocked())
4305 {
4306 /* open the associated config file */
4307 int vrc = RTFileOpen (&mData.mCfgFile.mHandle,
4308 Utf8Str (mData.mCfgFile.mName),
4309 RTFILE_O_READWRITE | RTFILE_O_OPEN |
4310 RTFILE_O_DENY_WRITE);
4311 if (VBOX_FAILURE (vrc))
4312 {
4313 mData.mCfgFile.mHandle = NIL_RTFILE;
4314
4315 /*
4316 * It is ok if the file is not found, it will be created by
4317 * init(). Otherwise return an error.
4318 */
4319 if (vrc != VERR_FILE_NOT_FOUND)
4320 rc = setError (E_FAIL,
4321 tr ("Could not lock the settings file '%ls' (%Vrc)"),
4322 mData.mCfgFile.mName.raw(), vrc);
4323 }
4324
4325 LogFlowThisFunc (("mCfgFile.mName='%ls', mCfgFile.mHandle=%d, rc=%08X\n",
4326 mData.mCfgFile.mName.raw(), mData.mCfgFile.mHandle, rc));
4327 }
4328
4329 return rc;
4330}
4331
4332/**
4333 * Helper to unlock the VirtualBox configuration from write access.
4334 *
4335 * @note This method is not thread safe (must be called only from #init()
4336 * or #uninit()).
4337 */
4338HRESULT VirtualBox::unlockConfig()
4339{
4340 AutoCaller autoCaller (this);
4341 AssertComRCReturn (autoCaller.rc(), E_FAIL);
4342 AssertReturn (autoCaller.state() == InUninit, E_FAIL);
4343
4344 HRESULT rc = S_OK;
4345
4346 if (isConfigLocked())
4347 {
4348 RTFileFlush (mData.mCfgFile.mHandle);
4349 RTFileClose (mData.mCfgFile.mHandle);
4350 /** @todo flush the directory too. */
4351 mData.mCfgFile.mHandle = NIL_RTFILE;
4352 LogFlowThisFunc (("\n"));
4353 }
4354
4355 return rc;
4356}
4357
4358/**
4359 * Thread function that watches the termination of all client processes
4360 * that have opened sessions using IVirtualBox::OpenSession()
4361 */
4362// static
4363DECLCALLBACK(int) VirtualBox::ClientWatcher (RTTHREAD thread, void *pvUser)
4364{
4365 LogFlowFuncEnter();
4366
4367 VirtualBox *that = (VirtualBox *) pvUser;
4368 Assert (that);
4369
4370 SessionMachineVector machines;
4371 size_t cnt = 0;
4372
4373#if defined(RT_OS_WINDOWS)
4374
4375 HRESULT hrc = CoInitializeEx (NULL,
4376 COINIT_MULTITHREADED | COINIT_DISABLE_OLE1DDE |
4377 COINIT_SPEED_OVER_MEMORY);
4378 AssertComRC (hrc);
4379
4380 /// @todo (dmik) processes reaping!
4381
4382 HANDLE *handles = new HANDLE [1];
4383 handles [0] = that->mWatcherData.mUpdateReq;
4384
4385 do
4386 {
4387 AutoCaller autoCaller (that);
4388 /* VirtualBox has been early uninitialized, terminate */
4389 if (!autoCaller.isOk())
4390 break;
4391
4392 do
4393 {
4394 /* release the caller to let uninit() ever proceed */
4395 autoCaller.release();
4396
4397 DWORD rc = ::WaitForMultipleObjects (cnt + 1, handles, FALSE, INFINITE);
4398
4399 /*
4400 * Restore the caller before using VirtualBox. If it fails, this
4401 * means VirtualBox is being uninitialized and we must terminate.
4402 */
4403 autoCaller.add();
4404 if (!autoCaller.isOk())
4405 break;
4406
4407 bool update = false;
4408 if (rc == WAIT_OBJECT_0)
4409 {
4410 /* update event is signaled */
4411 update = true;
4412 }
4413 else if (rc > WAIT_OBJECT_0 && rc <= (WAIT_OBJECT_0 + cnt))
4414 {
4415 /* machine mutex is released */
4416 (machines [rc - WAIT_OBJECT_0 - 1])->checkForDeath();
4417 update = true;
4418 }
4419 else if (rc > WAIT_ABANDONED_0 && rc <= (WAIT_ABANDONED_0 + cnt))
4420 {
4421 /* machine mutex is abandoned due to client process termination */
4422 (machines [rc - WAIT_ABANDONED_0 - 1])->checkForDeath();
4423 update = true;
4424 }
4425 if (update)
4426 {
4427 /* obtain a new set of opened machines */
4428 that->getOpenedMachines (machines);
4429 cnt = machines.size();
4430 LogFlowFunc (("UPDATE: direct session count = %d\n", cnt));
4431 AssertMsg ((cnt + 1) <= MAXIMUM_WAIT_OBJECTS,
4432 ("MAXIMUM_WAIT_OBJECTS reached"));
4433 /* renew the set of event handles */
4434 delete [] handles;
4435 handles = new HANDLE [cnt + 1];
4436 handles [0] = that->mWatcherData.mUpdateReq;
4437 for (size_t i = 0; i < cnt; ++ i)
4438 handles [i + 1] = (machines [i])->ipcSem();
4439 }
4440 }
4441 while (true);
4442 }
4443 while (0);
4444
4445 /* delete the set of event handles */
4446 delete [] handles;
4447
4448 /* delete the set of opened machines if any */
4449 machines.clear();
4450
4451 ::CoUninitialize();
4452
4453#elif defined (RT_OS_OS2)
4454
4455 /// @todo (dmik) processes reaping!
4456
4457 /* according to PMREF, 64 is the maximum for the muxwait list */
4458 SEMRECORD handles [64];
4459
4460 HMUX muxSem = NULLHANDLE;
4461
4462 do
4463 {
4464 AutoCaller autoCaller (that);
4465 /* VirtualBox has been early uninitialized, terminate */
4466 if (!autoCaller.isOk())
4467 break;
4468
4469 do
4470 {
4471 /* release the caller to let uninit() ever proceed */
4472 autoCaller.release();
4473
4474 int vrc = RTSemEventWait (that->mWatcherData.mUpdateReq, 500);
4475
4476 /*
4477 * Restore the caller before using VirtualBox. If it fails, this
4478 * means VirtualBox is being uninitialized and we must terminate.
4479 */
4480 autoCaller.add();
4481 if (!autoCaller.isOk())
4482 break;
4483
4484 bool update = false;
4485
4486 if (VBOX_SUCCESS (vrc))
4487 {
4488 /* update event is signaled */
4489 update = true;
4490 }
4491 else
4492 {
4493 AssertMsg (vrc == VERR_TIMEOUT || vrc == VERR_INTERRUPTED,
4494 ("RTSemEventWait returned %Vrc\n", vrc));
4495
4496 /* are there any mutexes? */
4497 if (cnt > 0)
4498 {
4499 /* figure out what's going on with machines */
4500
4501 unsigned long semId = 0;
4502 APIRET arc = ::DosWaitMuxWaitSem (muxSem,
4503 SEM_IMMEDIATE_RETURN, &semId);
4504
4505 if (arc == NO_ERROR)
4506 {
4507 /* machine mutex is normally released */
4508 Assert (semId >= 0 && semId < cnt);
4509 if (semId >= 0 && semId < cnt)
4510 machines [semId]->checkForDeath();
4511 update = true;
4512 }
4513 else if (arc == ERROR_SEM_OWNER_DIED)
4514 {
4515 /* machine mutex is abandoned due to client process
4516 * termination; find which mutex is in the Owner Died
4517 * state */
4518 for (size_t i = 0; i < cnt; ++ i)
4519 {
4520 PID pid; TID tid;
4521 unsigned long reqCnt;
4522 arc = DosQueryMutexSem ((HMTX) handles [i].hsemCur, &pid,
4523 &tid, &reqCnt);
4524 if (arc == ERROR_SEM_OWNER_DIED)
4525 {
4526 /* close the dead mutex as asked by PMREF */
4527 ::DosCloseMutexSem ((HMTX) handles [i].hsemCur);
4528
4529 Assert (i >= 0 && i < cnt);
4530 if (i >= 0 && i < cnt)
4531 machines [i]->checkForDeath();
4532 }
4533 }
4534 update = true;
4535 }
4536 else
4537 AssertMsg (arc == ERROR_INTERRUPT || arc == ERROR_TIMEOUT,
4538 ("DosWaitMuxWaitSem returned %d\n", arc));
4539 }
4540 }
4541
4542 if (update)
4543 {
4544 /* close the old muxsem */
4545 if (muxSem != NULLHANDLE)
4546 ::DosCloseMuxWaitSem (muxSem);
4547 /* obtain a new set of opened machines */
4548 that->getOpenedMachines (machines);
4549 cnt = machines.size();
4550 LogFlowFunc (("UPDATE: direct session count = %d\n", cnt));
4551 /// @todo use several muxwait sems if cnt is > 64
4552 AssertMsg (cnt <= 64 /* according to PMREF */,
4553 ("maximum of 64 mutex semaphores reached (%d)", cnt));
4554 if (cnt > 64)
4555 cnt = 64;
4556 if (cnt > 0)
4557 {
4558 /* renew the set of event handles */
4559 for (size_t i = 0; i < cnt; ++ i)
4560 {
4561 handles [i].hsemCur = (HSEM) machines [i]->ipcSem();
4562 handles [i].ulUser = i;
4563 }
4564 /* create a new muxsem */
4565 APIRET arc = ::DosCreateMuxWaitSem (NULL, &muxSem, cnt, handles,
4566 DCMW_WAIT_ANY);
4567 AssertMsg (arc == NO_ERROR,
4568 ("DosCreateMuxWaitSem returned %d\n", arc));
4569 }
4570 }
4571 }
4572 while (true);
4573 }
4574 while (0);
4575
4576 /* close the muxsem */
4577 if (muxSem != NULLHANDLE)
4578 ::DosCloseMuxWaitSem (muxSem);
4579
4580 /* delete the set of opened machines if any */
4581 machines.clear();
4582
4583#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
4584
4585 bool need_update = false;
4586
4587 do
4588 {
4589 AutoCaller autoCaller (that);
4590 if (!autoCaller.isOk())
4591 break;
4592
4593 do
4594 {
4595 /* release the caller to let uninit() ever proceed */
4596 autoCaller.release();
4597
4598 int rc = RTSemEventWait (that->mWatcherData.mUpdateReq, 500);
4599
4600 /*
4601 * Restore the caller before using VirtualBox. If it fails, this
4602 * means VirtualBox is being uninitialized and we must terminate.
4603 */
4604 autoCaller.add();
4605 if (!autoCaller.isOk())
4606 break;
4607
4608 if (VBOX_SUCCESS (rc) || need_update)
4609 {
4610 /* VBOX_SUCCESS (rc) means an update event is signaled */
4611
4612 /* obtain a new set of opened machines */
4613 that->getOpenedMachines (machines);
4614 cnt = machines.size();
4615 LogFlowFunc (("UPDATE: direct session count = %d\n", cnt));
4616 }
4617
4618 need_update = false;
4619 for (size_t i = 0; i < cnt; ++ i)
4620 need_update |= (machines [i])->checkForDeath();
4621
4622 /* reap child processes */
4623 {
4624 AutoLock alock (that);
4625 if (that->mWatcherData.mProcesses.size())
4626 {
4627 LogFlowFunc (("UPDATE: child process count = %d\n",
4628 that->mWatcherData.mProcesses.size()));
4629 ClientWatcherData::ProcessList::iterator it =
4630 that->mWatcherData.mProcesses.begin();
4631 while (it != that->mWatcherData.mProcesses.end())
4632 {
4633 RTPROCESS pid = *it;
4634 RTPROCSTATUS status;
4635 int vrc = ::RTProcWait (pid, RTPROCWAIT_FLAGS_NOBLOCK,
4636 &status);
4637 if (vrc == VINF_SUCCESS)
4638 {
4639 LogFlowFunc (("pid %d (%x) was reaped, "
4640 "status=%d, reason=%d\n",
4641 pid, pid, status.iStatus,
4642 status.enmReason));
4643 it = that->mWatcherData.mProcesses.erase (it);
4644 }
4645 else
4646 {
4647 LogFlowFunc (("pid %d (%x) was NOT reaped, vrc=%Vrc\n",
4648 pid, pid, vrc));
4649 if (vrc != VERR_PROCESS_RUNNING)
4650 {
4651 /* remove the process if it is not already running */
4652 it = that->mWatcherData.mProcesses.erase (it);
4653 }
4654 else
4655 ++ it;
4656 }
4657 }
4658 }
4659 }
4660 }
4661 while (true);
4662 }
4663 while (0);
4664
4665 /* delete the set of opened machines if any */
4666 machines.clear();
4667
4668#else
4669# error "Port me!"
4670#endif
4671
4672 LogFlowFuncLeave();
4673 return 0;
4674}
4675
4676/**
4677 * Thread function that handles custom events posted using #postEvent().
4678 */
4679// static
4680DECLCALLBACK(int) VirtualBox::AsyncEventHandler (RTTHREAD thread, void *pvUser)
4681{
4682 LogFlowFuncEnter();
4683
4684 AssertReturn (pvUser, VERR_INVALID_POINTER);
4685
4686 // create an event queue for the current thread
4687 EventQueue *eventQ = new EventQueue();
4688 AssertReturn (eventQ, VERR_NO_MEMORY);
4689
4690 // return the queue to the one who created this thread
4691 *(static_cast <EventQueue **> (pvUser)) = eventQ;
4692 // signal that we're ready
4693 RTThreadUserSignal (thread);
4694
4695 BOOL ok = TRUE;
4696 Event *event = NULL;
4697
4698 while ((ok = eventQ->waitForEvent (&event)) && event)
4699 eventQ->handleEvent (event);
4700
4701 AssertReturn (ok, VERR_GENERAL_FAILURE);
4702
4703 delete eventQ;
4704
4705 LogFlowFuncLeave();
4706
4707 return 0;
4708}
4709
4710////////////////////////////////////////////////////////////////////////////////
4711
4712/**
4713 * Takes the current list of registered callbacks of the managed VirtualBox
4714 * instance, and calls #handleCallback() for every callback item from the
4715 * list, passing the item as an argument.
4716 *
4717 * @note Locks the managed VirtualBox object for reading but leaves the lock
4718 * before iterating over callbacks and calling their methods.
4719 */
4720void *VirtualBox::CallbackEvent::handler()
4721{
4722 if (mVirtualBox.isNull())
4723 return NULL;
4724
4725 AutoCaller autoCaller (mVirtualBox);
4726 if (!autoCaller.isOk())
4727 {
4728 LogWarningFunc (("VirtualBox has been uninitialized (state=%d), "
4729 "the callback event is discarded!\n",
4730 autoCaller.state()));
4731 /* We don't need mVirtualBox any more, so release it */
4732 mVirtualBox.setNull();
4733 return NULL;
4734 }
4735
4736 CallbackVector callbacks;
4737 {
4738 /* Make a copy to release the lock before iterating */
4739 AutoReaderLock alock (mVirtualBox);
4740 callbacks = CallbackVector (mVirtualBox->mData.mCallbacks.begin(),
4741 mVirtualBox->mData.mCallbacks.end());
4742 /* We don't need mVirtualBox any more, so release it */
4743 mVirtualBox.setNull();
4744 }
4745
4746 for (VirtualBox::CallbackVector::const_iterator it = callbacks.begin();
4747 it != callbacks.end(); ++ it)
4748 handleCallback (*it);
4749
4750 return NULL;
4751}
4752
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