VirtualBox

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

Last change on this file since 24557 was 24555, checked in by vboxsync, 15 years ago

Main/VirtualBox: fix a bunch of error messages which weren't converted to utf8 parameters

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