VirtualBox

source: vbox/trunk/src/VBox/Main/ConsoleImpl.cpp@ 27143

Last change on this file since 27143 was 26782, checked in by vboxsync, 15 years ago

Main, Frontends: added support for virtual pointing devices with no relative reporting and cleaned up the VMMDev/mouse device absolute reporting interaction

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 267.3 KB
Line 
1/* $Id: ConsoleImpl.cpp 26782 2010-02-25 11:17:30Z vboxsync $ */
2/** @file
3 * VBox Console COM Class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/** @todo Move the TAP mess back into the driver! */
23#if defined(RT_OS_WINDOWS)
24#elif defined(RT_OS_LINUX)
25# include <errno.h>
26# include <sys/ioctl.h>
27# include <sys/poll.h>
28# include <sys/fcntl.h>
29# include <sys/types.h>
30# include <sys/wait.h>
31# include <net/if.h>
32# include <linux/if_tun.h>
33# include <stdio.h>
34# include <stdlib.h>
35# include <string.h>
36#elif defined(RT_OS_FREEBSD)
37# include <errno.h>
38# include <sys/ioctl.h>
39# include <sys/poll.h>
40# include <sys/fcntl.h>
41# include <sys/types.h>
42# include <sys/wait.h>
43# include <stdio.h>
44# include <stdlib.h>
45# include <string.h>
46#endif
47
48#include "ConsoleImpl.h"
49
50#include "Global.h"
51#include "VirtualBoxErrorInfoImpl.h"
52#include "GuestImpl.h"
53#include "KeyboardImpl.h"
54#include "MouseImpl.h"
55#include "DisplayImpl.h"
56#include "MachineDebuggerImpl.h"
57#include "USBDeviceImpl.h"
58#include "RemoteUSBDeviceImpl.h"
59#include "SharedFolderImpl.h"
60#include "AudioSnifferInterface.h"
61#include "ProgressCombinedImpl.h"
62#include "ConsoleVRDPServer.h"
63#include "VMMDev.h"
64#include "package-generated.h"
65
66// generated header
67#include "SchemaDefs.h"
68
69#include "AutoCaller.h"
70#include "Logging.h"
71
72#include <VBox/com/array.h>
73
74#include <iprt/asm.h>
75#include <iprt/buildconfig.h>
76#include <iprt/cpp/utils.h>
77#include <iprt/dir.h>
78#include <iprt/file.h>
79#include <iprt/ldr.h>
80#include <iprt/path.h>
81#include <iprt/process.h>
82#include <iprt/string.h>
83#include <iprt/system.h>
84
85#include <VBox/vmapi.h>
86#include <VBox/err.h>
87#include <VBox/param.h>
88#include <VBox/pdmnetifs.h>
89#include <VBox/vusb.h>
90#include <VBox/mm.h>
91#include <VBox/ssm.h>
92#include <VBox/version.h>
93#ifdef VBOX_WITH_USB
94# include <VBox/pdmusb.h>
95#endif
96
97#include <VBox/VMMDev.h>
98
99#include <VBox/HostServices/VBoxClipboardSvc.h>
100#ifdef VBOX_WITH_GUEST_PROPS
101# include <VBox/HostServices/GuestPropertySvc.h>
102# include <VBox/com/array.h>
103#endif
104
105#include <set>
106#include <algorithm>
107#include <memory> // for auto_ptr
108#include <vector>
109#include <typeinfo>
110
111
112// VMTask and friends
113////////////////////////////////////////////////////////////////////////////////
114
115/**
116 * Task structure for asynchronous VM operations.
117 *
118 * Once created, the task structure adds itself as a Console caller. This means:
119 *
120 * 1. The user must check for #rc() before using the created structure
121 * (e.g. passing it as a thread function argument). If #rc() returns a
122 * failure, the Console object may not be used by the task (see
123 * Console::addCaller() for more details).
124 * 2. On successful initialization, the structure keeps the Console caller
125 * until destruction (to ensure Console remains in the Ready state and won't
126 * be accidentally uninitialized). Forgetting to delete the created task
127 * will lead to Console::uninit() stuck waiting for releasing all added
128 * callers.
129 *
130 * If \a aUsesVMPtr parameter is true, the task structure will also add itself
131 * as a Console::mpVM caller with the same meaning as above. See
132 * Console::addVMCaller() for more info.
133 */
134struct VMTask
135{
136 VMTask(Console *aConsole, bool aUsesVMPtr)
137 : mConsole(aConsole),
138 mCallerAdded(false),
139 mVMCallerAdded(false)
140 {
141 AssertReturnVoid(aConsole);
142 mRC = aConsole->addCaller();
143 if (SUCCEEDED(mRC))
144 {
145 mCallerAdded = true;
146 if (aUsesVMPtr)
147 {
148 mRC = aConsole->addVMCaller();
149 if (SUCCEEDED(mRC))
150 mVMCallerAdded = true;
151 }
152 }
153 }
154
155 ~VMTask()
156 {
157 if (mVMCallerAdded)
158 mConsole->releaseVMCaller();
159 if (mCallerAdded)
160 mConsole->releaseCaller();
161 }
162
163 HRESULT rc() const { return mRC; }
164 bool isOk() const { return SUCCEEDED(rc()); }
165
166 /** Releases the Console caller before destruction. Not normally necessary. */
167 void releaseCaller()
168 {
169 AssertReturnVoid(mCallerAdded);
170 mConsole->releaseCaller();
171 mCallerAdded = false;
172 }
173
174 /** Releases the VM caller before destruction. Not normally necessary. */
175 void releaseVMCaller()
176 {
177 AssertReturnVoid(mVMCallerAdded);
178 mConsole->releaseVMCaller();
179 mVMCallerAdded = false;
180 }
181
182 const ComObjPtr<Console> mConsole;
183
184private:
185
186 HRESULT mRC;
187 bool mCallerAdded : 1;
188 bool mVMCallerAdded : 1;
189};
190
191struct VMProgressTask : public VMTask
192{
193 VMProgressTask(Console *aConsole,
194 Progress *aProgress,
195 bool aUsesVMPtr)
196 : VMTask(aConsole, aUsesVMPtr),
197 mProgress(aProgress)
198 {}
199
200 const ComObjPtr<Progress> mProgress;
201
202 Utf8Str mErrorMsg;
203};
204
205struct VMTakeSnapshotTask : public VMProgressTask
206{
207 VMTakeSnapshotTask(Console *aConsole,
208 Progress *aProgress,
209 IN_BSTR aName,
210 IN_BSTR aDescription)
211 : VMProgressTask(aConsole, aProgress, false /* aUsesVMPtr */),
212 bstrName(aName),
213 bstrDescription(aDescription),
214 lastMachineState(MachineState_Null)
215 {}
216
217 Bstr bstrName,
218 bstrDescription;
219 Bstr bstrSavedStateFile; // received from BeginTakeSnapshot()
220 MachineState_T lastMachineState;
221 bool fTakingSnapshotOnline;
222 ULONG ulMemSize;
223};
224
225struct VMPowerUpTask : public VMProgressTask
226{
227 VMPowerUpTask(Console *aConsole,
228 Progress *aProgress)
229 : VMProgressTask(aConsole, aProgress, false /* aUsesVMPtr */),
230 mSetVMErrorCallback(NULL),
231 mConfigConstructor(NULL),
232 mStartPaused(false),
233 mTeleporterEnabled(FALSE)
234 {}
235
236 PFNVMATERROR mSetVMErrorCallback;
237 PFNCFGMCONSTRUCTOR mConfigConstructor;
238 Utf8Str mSavedStateFile;
239 Console::SharedFolderDataMap mSharedFolders;
240 bool mStartPaused;
241 BOOL mTeleporterEnabled;
242
243 typedef std::list< ComPtr<IMedium> > HardDiskList;
244 HardDiskList hardDisks;
245
246 /* array of progress objects for hard disk reset operations */
247 typedef std::list< ComPtr<IProgress> > ProgressList;
248 ProgressList hardDiskProgresses;
249};
250
251struct VMSaveTask : public VMProgressTask
252{
253 VMSaveTask(Console *aConsole, Progress *aProgress)
254 : VMProgressTask(aConsole, aProgress, true /* aUsesVMPtr */),
255 mLastMachineState(MachineState_Null)
256 {}
257
258 Utf8Str mSavedStateFile;
259 MachineState_T mLastMachineState;
260 ComPtr<IProgress> mServerProgress;
261};
262
263// constructor / destructor
264/////////////////////////////////////////////////////////////////////////////
265
266Console::Console()
267 : mSavedStateDataLoaded(false)
268 , mConsoleVRDPServer(NULL)
269 , mpVM(NULL)
270 , mVMCallers(0)
271 , mVMZeroCallersSem(NIL_RTSEMEVENT)
272 , mVMDestroying(false)
273 , mVMPoweredOff(false)
274 , mVMIsAlreadyPoweringOff(false)
275 , mVMMDev(NULL)
276 , mAudioSniffer(NULL)
277 , mVMStateChangeCallbackDisabled(false)
278 , mMachineState(MachineState_PoweredOff)
279{
280 for (ULONG slot = 0; slot < SchemaDefs::NetworkAdapterCount; ++slot)
281 meAttachmentType[slot] = NetworkAttachmentType_Null;
282}
283
284Console::~Console()
285{}
286
287HRESULT Console::FinalConstruct()
288{
289 LogFlowThisFunc(("\n"));
290
291 memset(mapStorageLeds, 0, sizeof(mapStorageLeds));
292 memset(mapNetworkLeds, 0, sizeof(mapNetworkLeds));
293 memset(&mapUSBLed, 0, sizeof(mapUSBLed));
294 memset(&mapSharedFolderLed, 0, sizeof(mapSharedFolderLed));
295
296 for (unsigned i = 0; i < RT_ELEMENTS(maStorageDevType); ++ i)
297 maStorageDevType[i] = DeviceType_Null;
298
299 return S_OK;
300}
301
302void Console::FinalRelease()
303{
304 LogFlowThisFunc(("\n"));
305
306 uninit();
307}
308
309// public initializer/uninitializer for internal purposes only
310/////////////////////////////////////////////////////////////////////////////
311
312HRESULT Console::init(IMachine *aMachine, IInternalMachineControl *aControl)
313{
314 AssertReturn(aMachine && aControl, E_INVALIDARG);
315
316 /* Enclose the state transition NotReady->InInit->Ready */
317 AutoInitSpan autoInitSpan(this);
318 AssertReturn(autoInitSpan.isOk(), E_FAIL);
319
320 LogFlowThisFuncEnter();
321 LogFlowThisFunc(("aMachine=%p, aControl=%p\n", aMachine, aControl));
322
323 HRESULT rc = E_FAIL;
324
325 unconst(mMachine) = aMachine;
326 unconst(mControl) = aControl;
327
328 memset(&mCallbackData, 0, sizeof(mCallbackData));
329
330 /* Cache essential properties and objects */
331
332 rc = mMachine->COMGETTER(State)(&mMachineState);
333 AssertComRCReturnRC(rc);
334
335#ifdef VBOX_WITH_VRDP
336 rc = mMachine->COMGETTER(VRDPServer)(unconst(mVRDPServer).asOutParam());
337 AssertComRCReturnRC(rc);
338#endif
339
340 /* Create associated child COM objects */
341
342 unconst(mGuest).createObject();
343 rc = mGuest->init(this);
344 AssertComRCReturnRC(rc);
345
346 unconst(mKeyboard).createObject();
347 rc = mKeyboard->init(this);
348 AssertComRCReturnRC(rc);
349
350 unconst(mMouse).createObject();
351 rc = mMouse->init(this);
352 AssertComRCReturnRC(rc);
353
354 unconst(mDisplay).createObject();
355 rc = mDisplay->init(this);
356 AssertComRCReturnRC(rc);
357
358 unconst(mRemoteDisplayInfo).createObject();
359 rc = mRemoteDisplayInfo->init(this);
360 AssertComRCReturnRC(rc);
361
362 /* Grab global and machine shared folder lists */
363
364 rc = fetchSharedFolders(true /* aGlobal */);
365 AssertComRCReturnRC(rc);
366 rc = fetchSharedFolders(false /* aGlobal */);
367 AssertComRCReturnRC(rc);
368
369 /* Create other child objects */
370
371 unconst(mConsoleVRDPServer) = new ConsoleVRDPServer(this);
372 AssertReturn(mConsoleVRDPServer, E_FAIL);
373
374 mcAudioRefs = 0;
375 mcVRDPClients = 0;
376 mu32SingleRDPClientId = 0;
377
378 unconst(mVMMDev) = new VMMDev(this);
379 AssertReturn(mVMMDev, E_FAIL);
380
381 unconst(mAudioSniffer) = new AudioSniffer(this);
382 AssertReturn(mAudioSniffer, E_FAIL);
383
384 /* Confirm a successful initialization when it's the case */
385 autoInitSpan.setSucceeded();
386
387 LogFlowThisFuncLeave();
388
389 return S_OK;
390}
391
392/**
393 * Uninitializes the Console object.
394 */
395void Console::uninit()
396{
397 LogFlowThisFuncEnter();
398
399 /* Enclose the state transition Ready->InUninit->NotReady */
400 AutoUninitSpan autoUninitSpan(this);
401 if (autoUninitSpan.uninitDone())
402 {
403 LogFlowThisFunc(("Already uninitialized.\n"));
404 LogFlowThisFuncLeave();
405 return;
406 }
407
408 LogFlowThisFunc(("initFailed()=%d\n", autoUninitSpan.initFailed()));
409
410 /*
411 * Uninit all children that use addDependentChild()/removeDependentChild()
412 * in their init()/uninit() methods.
413 */
414 uninitDependentChildren();
415
416 /* power down the VM if necessary */
417 if (mpVM)
418 {
419 powerDown();
420 Assert(mpVM == NULL);
421 }
422
423 if (mVMZeroCallersSem != NIL_RTSEMEVENT)
424 {
425 RTSemEventDestroy(mVMZeroCallersSem);
426 mVMZeroCallersSem = NIL_RTSEMEVENT;
427 }
428
429 if (mAudioSniffer)
430 {
431 delete mAudioSniffer;
432 unconst(mAudioSniffer) = NULL;
433 }
434
435 if (mVMMDev)
436 {
437 delete mVMMDev;
438 unconst(mVMMDev) = NULL;
439 }
440
441 mGlobalSharedFolders.clear();
442 mMachineSharedFolders.clear();
443
444 mSharedFolders.clear();
445 mRemoteUSBDevices.clear();
446 mUSBDevices.clear();
447
448 if (mRemoteDisplayInfo)
449 {
450 mRemoteDisplayInfo->uninit();
451 unconst(mRemoteDisplayInfo).setNull();;
452 }
453
454 if (mDebugger)
455 {
456 mDebugger->uninit();
457 unconst(mDebugger).setNull();
458 }
459
460 if (mDisplay)
461 {
462 mDisplay->uninit();
463 unconst(mDisplay).setNull();
464 }
465
466 if (mMouse)
467 {
468 mMouse->uninit();
469 unconst(mMouse).setNull();
470 }
471
472 if (mKeyboard)
473 {
474 mKeyboard->uninit();
475 unconst(mKeyboard).setNull();;
476 }
477
478 if (mGuest)
479 {
480 mGuest->uninit();
481 unconst(mGuest).setNull();;
482 }
483
484 if (mConsoleVRDPServer)
485 {
486 delete mConsoleVRDPServer;
487 unconst(mConsoleVRDPServer) = NULL;
488 }
489
490#ifdef VBOX_WITH_VRDP
491 unconst(mVRDPServer).setNull();
492#endif
493
494 unconst(mControl).setNull();
495 unconst(mMachine).setNull();
496
497 /* Release all callbacks. Do this after uninitializing the components,
498 * as some of them are well-behaved and unregister their callbacks.
499 * These would trigger error messages complaining about trying to
500 * unregister a non-registered callback. */
501 mCallbacks.clear();
502
503 /* dynamically allocated members of mCallbackData are uninitialized
504 * at the end of powerDown() */
505 Assert(!mCallbackData.mpsc.valid && mCallbackData.mpsc.shape == NULL);
506 Assert(!mCallbackData.mcc.valid);
507 Assert(!mCallbackData.klc.valid);
508
509 LogFlowThisFuncLeave();
510}
511
512#ifdef VBOX_WITH_GUEST_PROPS
513
514bool Console::enabledGuestPropertiesVRDP(void)
515{
516 Bstr value;
517 HRESULT hrc = mMachine->GetExtraData(Bstr("VBoxInternal2/EnableGuestPropertiesVRDP"), value.asOutParam());
518 if (hrc == S_OK)
519 {
520 if (value == "1")
521 {
522 return true;
523 }
524 }
525 return false;
526}
527
528void Console::updateGuestPropertiesVRDPLogon(uint32_t u32ClientId, const char *pszUser, const char *pszDomain)
529{
530 if (!enabledGuestPropertiesVRDP())
531 {
532 return;
533 }
534
535 int rc;
536 char *pszPropertyName;
537
538 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/Name", u32ClientId);
539 if (RT_SUCCESS(rc))
540 {
541 Bstr clientName;
542 mRemoteDisplayInfo->COMGETTER(ClientName)(clientName.asOutParam());
543
544 mMachine->SetGuestProperty(Bstr(pszPropertyName), clientName, Bstr("RDONLYGUEST"));
545 RTStrFree(pszPropertyName);
546 }
547
548 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/User", u32ClientId);
549 if (RT_SUCCESS(rc))
550 {
551 mMachine->SetGuestProperty(Bstr(pszPropertyName), Bstr(pszUser), Bstr("RDONLYGUEST"));
552 RTStrFree(pszPropertyName);
553 }
554
555 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/Domain", u32ClientId);
556 if (RT_SUCCESS(rc))
557 {
558 mMachine->SetGuestProperty(Bstr(pszPropertyName), Bstr(pszDomain), Bstr("RDONLYGUEST"));
559 RTStrFree(pszPropertyName);
560 }
561
562 char *pszClientId;
563 rc = RTStrAPrintf(&pszClientId, "%d", u32ClientId);
564 if (RT_SUCCESS(rc))
565 {
566 mMachine->SetGuestProperty(Bstr("/VirtualBox/HostInfo/VRDP/LastConnectedClient"), Bstr(pszClientId), Bstr("RDONLYGUEST"));
567 RTStrFree(pszClientId);
568 }
569
570 return;
571}
572
573void Console::updateGuestPropertiesVRDPDisconnect(uint32_t u32ClientId)
574{
575 if (!enabledGuestPropertiesVRDP())
576 return;
577
578 Bstr bstrReadOnlyGuest(L"RDONLYGUEST");
579
580 int rc;
581 char *pszPropertyName;
582
583 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/Name", u32ClientId);
584 if (RT_SUCCESS(rc))
585 {
586 mMachine->SetGuestProperty(Bstr(pszPropertyName), NULL, bstrReadOnlyGuest);
587 RTStrFree(pszPropertyName);
588 }
589
590 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/User", u32ClientId);
591 if (RT_SUCCESS(rc))
592 {
593 mMachine->SetGuestProperty(Bstr(pszPropertyName), NULL, bstrReadOnlyGuest);
594 RTStrFree(pszPropertyName);
595 }
596
597 rc = RTStrAPrintf(&pszPropertyName, "/VirtualBox/HostInfo/VRDP/Client/%u/Domain", u32ClientId);
598 if (RT_SUCCESS(rc))
599 {
600 mMachine->SetGuestProperty(Bstr(pszPropertyName), NULL, bstrReadOnlyGuest);
601 RTStrFree(pszPropertyName);
602 }
603
604 char *pszClientId;
605 rc = RTStrAPrintf(&pszClientId, "%d", u32ClientId);
606 if (RT_SUCCESS(rc))
607 {
608 mMachine->SetGuestProperty(Bstr("/VirtualBox/HostInfo/VRDP/LastDisconnectedClient"), Bstr(pszClientId), bstrReadOnlyGuest);
609 RTStrFree(pszClientId);
610 }
611
612 return;
613}
614
615#endif /* VBOX_WITH_GUEST_PROPS */
616
617
618int Console::VRDPClientLogon(uint32_t u32ClientId, const char *pszUser, const char *pszPassword, const char *pszDomain)
619{
620 LogFlowFuncEnter();
621 LogFlowFunc(("%d, %s, %s, %s\n", u32ClientId, pszUser, pszPassword, pszDomain));
622
623 AutoCaller autoCaller(this);
624 if (!autoCaller.isOk())
625 {
626 /* Console has been already uninitialized, deny request */
627 LogRel(("VRDPAUTH: Access denied (Console uninitialized).\n"));
628 LogFlowFuncLeave();
629 return VERR_ACCESS_DENIED;
630 }
631
632 Bstr id;
633 HRESULT hrc = mMachine->COMGETTER(Id)(id.asOutParam());
634 Guid uuid = Guid(id);
635
636 AssertComRCReturn(hrc, VERR_ACCESS_DENIED);
637
638 VRDPAuthType_T authType = VRDPAuthType_Null;
639 hrc = mVRDPServer->COMGETTER(AuthType)(&authType);
640 AssertComRCReturn(hrc, VERR_ACCESS_DENIED);
641
642 ULONG authTimeout = 0;
643 hrc = mVRDPServer->COMGETTER(AuthTimeout)(&authTimeout);
644 AssertComRCReturn(hrc, VERR_ACCESS_DENIED);
645
646 VRDPAuthResult result = VRDPAuthAccessDenied;
647 VRDPAuthGuestJudgement guestJudgement = VRDPAuthGuestNotAsked;
648
649 LogFlowFunc(("Auth type %d\n", authType));
650
651 LogRel(("VRDPAUTH: User: [%s]. Domain: [%s]. Authentication type: [%s]\n",
652 pszUser, pszDomain,
653 authType == VRDPAuthType_Null?
654 "Null":
655 (authType == VRDPAuthType_External?
656 "External":
657 (authType == VRDPAuthType_Guest?
658 "Guest":
659 "INVALID"
660 )
661 )
662 ));
663
664 switch (authType)
665 {
666 case VRDPAuthType_Null:
667 {
668 result = VRDPAuthAccessGranted;
669 break;
670 }
671
672 case VRDPAuthType_External:
673 {
674 /* Call the external library. */
675 result = mConsoleVRDPServer->Authenticate(uuid, guestJudgement, pszUser, pszPassword, pszDomain, u32ClientId);
676
677 if (result != VRDPAuthDelegateToGuest)
678 {
679 break;
680 }
681
682 LogRel(("VRDPAUTH: Delegated to guest.\n"));
683
684 LogFlowFunc(("External auth asked for guest judgement\n"));
685 } /* pass through */
686
687 case VRDPAuthType_Guest:
688 {
689 guestJudgement = VRDPAuthGuestNotReacted;
690
691 if (mVMMDev)
692 {
693 /* Issue the request to guest. Assume that the call does not require EMT. It should not. */
694
695 /* Ask the guest to judge these credentials. */
696 uint32_t u32GuestFlags = VMMDEV_SETCREDENTIALS_JUDGE;
697
698 int rc = mVMMDev->getVMMDevPort()->pfnSetCredentials(mVMMDev->getVMMDevPort(),
699 pszUser, pszPassword, pszDomain, u32GuestFlags);
700
701 if (RT_SUCCESS(rc))
702 {
703 /* Wait for guest. */
704 rc = mVMMDev->WaitCredentialsJudgement(authTimeout, &u32GuestFlags);
705
706 if (RT_SUCCESS(rc))
707 {
708 switch (u32GuestFlags & (VMMDEV_CREDENTIALS_JUDGE_OK | VMMDEV_CREDENTIALS_JUDGE_DENY | VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT))
709 {
710 case VMMDEV_CREDENTIALS_JUDGE_DENY: guestJudgement = VRDPAuthGuestAccessDenied; break;
711 case VMMDEV_CREDENTIALS_JUDGE_NOJUDGEMENT: guestJudgement = VRDPAuthGuestNoJudgement; break;
712 case VMMDEV_CREDENTIALS_JUDGE_OK: guestJudgement = VRDPAuthGuestAccessGranted; break;
713 default:
714 LogFlowFunc(("Invalid guest flags %08X!!!\n", u32GuestFlags)); break;
715 }
716 }
717 else
718 {
719 LogFlowFunc(("Wait for credentials judgement rc = %Rrc!!!\n", rc));
720 }
721
722 LogFlowFunc(("Guest judgement %d\n", guestJudgement));
723 }
724 else
725 {
726 LogFlowFunc(("Could not set credentials rc = %Rrc!!!\n", rc));
727 }
728 }
729
730 if (authType == VRDPAuthType_External)
731 {
732 LogRel(("VRDPAUTH: Guest judgement %d.\n", guestJudgement));
733 LogFlowFunc(("External auth called again with guest judgement = %d\n", guestJudgement));
734 result = mConsoleVRDPServer->Authenticate(uuid, guestJudgement, pszUser, pszPassword, pszDomain, u32ClientId);
735 }
736 else
737 {
738 switch (guestJudgement)
739 {
740 case VRDPAuthGuestAccessGranted:
741 result = VRDPAuthAccessGranted;
742 break;
743 default:
744 result = VRDPAuthAccessDenied;
745 break;
746 }
747 }
748 } break;
749
750 default:
751 AssertFailed();
752 }
753
754 LogFlowFunc(("Result = %d\n", result));
755 LogFlowFuncLeave();
756
757 if (result != VRDPAuthAccessGranted)
758 {
759 /* Reject. */
760 LogRel(("VRDPAUTH: Access denied.\n"));
761 return VERR_ACCESS_DENIED;
762 }
763
764 LogRel(("VRDPAUTH: Access granted.\n"));
765
766 /* Multiconnection check must be made after authentication, so bad clients would not interfere with a good one. */
767 BOOL allowMultiConnection = FALSE;
768 hrc = mVRDPServer->COMGETTER(AllowMultiConnection)(&allowMultiConnection);
769 AssertComRCReturn(hrc, VERR_ACCESS_DENIED);
770
771 BOOL reuseSingleConnection = FALSE;
772 hrc = mVRDPServer->COMGETTER(ReuseSingleConnection)(&reuseSingleConnection);
773 AssertComRCReturn(hrc, VERR_ACCESS_DENIED);
774
775 LogFlowFunc(("allowMultiConnection %d, reuseSingleConnection = %d, mcVRDPClients = %d, mu32SingleRDPClientId = %d\n", allowMultiConnection, reuseSingleConnection, mcVRDPClients, mu32SingleRDPClientId));
776
777 if (allowMultiConnection == FALSE)
778 {
779 /* Note: the 'mcVRDPClients' variable is incremented in ClientConnect callback, which is called when the client
780 * is successfully connected, that is after the ClientLogon callback. Therefore the mcVRDPClients
781 * value is 0 for first client.
782 */
783 if (mcVRDPClients != 0)
784 {
785 Assert(mcVRDPClients == 1);
786 /* There is a client already.
787 * If required drop the existing client connection and let the connecting one in.
788 */
789 if (reuseSingleConnection)
790 {
791 LogRel(("VRDPAUTH: Multiple connections are not enabled. Disconnecting existing client.\n"));
792 mConsoleVRDPServer->DisconnectClient(mu32SingleRDPClientId, false);
793 }
794 else
795 {
796 /* Reject. */
797 LogRel(("VRDPAUTH: Multiple connections are not enabled. Access denied.\n"));
798 return VERR_ACCESS_DENIED;
799 }
800 }
801
802 /* Save the connected client id. From now on it will be necessary to disconnect this one. */
803 mu32SingleRDPClientId = u32ClientId;
804 }
805
806#ifdef VBOX_WITH_GUEST_PROPS
807 updateGuestPropertiesVRDPLogon(u32ClientId, pszUser, pszDomain);
808#endif /* VBOX_WITH_GUEST_PROPS */
809
810 /* Check if the successfully verified credentials are to be sent to the guest. */
811 BOOL fProvideGuestCredentials = FALSE;
812
813 Bstr value;
814 hrc = mMachine->GetExtraData(Bstr("VRDP/ProvideGuestCredentials"), value.asOutParam());
815 if (SUCCEEDED(hrc) && value == "1")
816 {
817 fProvideGuestCredentials = TRUE;
818 }
819
820 if ( fProvideGuestCredentials
821 && mVMMDev)
822 {
823 uint32_t u32GuestFlags = VMMDEV_SETCREDENTIALS_GUESTLOGON;
824
825 int rc = mVMMDev->getVMMDevPort()->pfnSetCredentials(mVMMDev->getVMMDevPort(),
826 pszUser, pszPassword, pszDomain, u32GuestFlags);
827 AssertRC(rc);
828 }
829
830 return VINF_SUCCESS;
831}
832
833void Console::VRDPClientConnect(uint32_t u32ClientId)
834{
835 LogFlowFuncEnter();
836
837 AutoCaller autoCaller(this);
838 AssertComRCReturnVoid(autoCaller.rc());
839
840#ifdef VBOX_WITH_VRDP
841 uint32_t u32Clients = ASMAtomicIncU32(&mcVRDPClients);
842
843 if (u32Clients == 1)
844 {
845 getVMMDev()->getVMMDevPort()->
846 pfnVRDPChange(getVMMDev()->getVMMDevPort(),
847 true, VRDP_EXPERIENCE_LEVEL_FULL); // @todo configurable
848 }
849
850 NOREF(u32ClientId);
851 mDisplay->VideoAccelVRDP(true);
852#endif /* VBOX_WITH_VRDP */
853
854 LogFlowFuncLeave();
855 return;
856}
857
858void Console::VRDPClientDisconnect(uint32_t u32ClientId,
859 uint32_t fu32Intercepted)
860{
861 LogFlowFuncEnter();
862
863 AutoCaller autoCaller(this);
864 AssertComRCReturnVoid(autoCaller.rc());
865
866 AssertReturnVoid(mConsoleVRDPServer);
867
868#ifdef VBOX_WITH_VRDP
869 uint32_t u32Clients = ASMAtomicDecU32(&mcVRDPClients);
870
871 if (u32Clients == 0)
872 {
873 getVMMDev()->getVMMDevPort()->
874 pfnVRDPChange(getVMMDev()->getVMMDevPort(),
875 false, 0);
876 }
877
878 mDisplay->VideoAccelVRDP(false);
879#endif /* VBOX_WITH_VRDP */
880
881 if (fu32Intercepted & VRDP_CLIENT_INTERCEPT_USB)
882 {
883 mConsoleVRDPServer->USBBackendDelete(u32ClientId);
884 }
885
886#ifdef VBOX_WITH_VRDP
887 if (fu32Intercepted & VRDP_CLIENT_INTERCEPT_CLIPBOARD)
888 {
889 mConsoleVRDPServer->ClipboardDelete(u32ClientId);
890 }
891
892 if (fu32Intercepted & VRDP_CLIENT_INTERCEPT_AUDIO)
893 {
894 mcAudioRefs--;
895
896 if (mcAudioRefs <= 0)
897 {
898 if (mAudioSniffer)
899 {
900 PPDMIAUDIOSNIFFERPORT port = mAudioSniffer->getAudioSnifferPort();
901 if (port)
902 {
903 port->pfnSetup(port, false, false);
904 }
905 }
906 }
907 }
908#endif /* VBOX_WITH_VRDP */
909
910 Bstr uuid;
911 HRESULT hrc = mMachine->COMGETTER(Id)(uuid.asOutParam());
912 AssertComRC(hrc);
913
914 VRDPAuthType_T authType = VRDPAuthType_Null;
915 hrc = mVRDPServer->COMGETTER(AuthType)(&authType);
916 AssertComRC(hrc);
917
918 if (authType == VRDPAuthType_External)
919 mConsoleVRDPServer->AuthDisconnect(uuid, u32ClientId);
920
921#ifdef VBOX_WITH_GUEST_PROPS
922 updateGuestPropertiesVRDPDisconnect(u32ClientId);
923#endif /* VBOX_WITH_GUEST_PROPS */
924
925 LogFlowFuncLeave();
926 return;
927}
928
929void Console::VRDPInterceptAudio(uint32_t u32ClientId)
930{
931 LogFlowFuncEnter();
932
933 AutoCaller autoCaller(this);
934 AssertComRCReturnVoid(autoCaller.rc());
935
936 LogFlowFunc(("mAudioSniffer %p, u32ClientId %d.\n",
937 mAudioSniffer, u32ClientId));
938 NOREF(u32ClientId);
939
940#ifdef VBOX_WITH_VRDP
941 ++mcAudioRefs;
942
943 if (mcAudioRefs == 1)
944 {
945 if (mAudioSniffer)
946 {
947 PPDMIAUDIOSNIFFERPORT port = mAudioSniffer->getAudioSnifferPort();
948 if (port)
949 {
950 port->pfnSetup(port, true, true);
951 }
952 }
953 }
954#endif
955
956 LogFlowFuncLeave();
957 return;
958}
959
960void Console::VRDPInterceptUSB(uint32_t u32ClientId, void **ppvIntercept)
961{
962 LogFlowFuncEnter();
963
964 AutoCaller autoCaller(this);
965 AssertComRCReturnVoid(autoCaller.rc());
966
967 AssertReturnVoid(mConsoleVRDPServer);
968
969 mConsoleVRDPServer->USBBackendCreate(u32ClientId, ppvIntercept);
970
971 LogFlowFuncLeave();
972 return;
973}
974
975void Console::VRDPInterceptClipboard(uint32_t u32ClientId)
976{
977 LogFlowFuncEnter();
978
979 AutoCaller autoCaller(this);
980 AssertComRCReturnVoid(autoCaller.rc());
981
982 AssertReturnVoid(mConsoleVRDPServer);
983
984#ifdef VBOX_WITH_VRDP
985 mConsoleVRDPServer->ClipboardCreate(u32ClientId);
986#endif /* VBOX_WITH_VRDP */
987
988 LogFlowFuncLeave();
989 return;
990}
991
992
993//static
994const char *Console::sSSMConsoleUnit = "ConsoleData";
995//static
996uint32_t Console::sSSMConsoleVer = 0x00010001;
997
998/**
999 * Loads various console data stored in the saved state file.
1000 * This method does validation of the state file and returns an error info
1001 * when appropriate.
1002 *
1003 * The method does nothing if the machine is not in the Saved file or if
1004 * console data from it has already been loaded.
1005 *
1006 * @note The caller must lock this object for writing.
1007 */
1008HRESULT Console::loadDataFromSavedState()
1009{
1010 if (mMachineState != MachineState_Saved || mSavedStateDataLoaded)
1011 return S_OK;
1012
1013 Bstr savedStateFile;
1014 HRESULT rc = mMachine->COMGETTER(StateFilePath)(savedStateFile.asOutParam());
1015 if (FAILED(rc))
1016 return rc;
1017
1018 PSSMHANDLE ssm;
1019 int vrc = SSMR3Open(Utf8Str(savedStateFile).c_str(), 0, &ssm);
1020 if (RT_SUCCESS(vrc))
1021 {
1022 uint32_t version = 0;
1023 vrc = SSMR3Seek(ssm, sSSMConsoleUnit, 0 /* iInstance */, &version);
1024 if (SSM_VERSION_MAJOR(version) == SSM_VERSION_MAJOR(sSSMConsoleVer))
1025 {
1026 if (RT_SUCCESS(vrc))
1027 vrc = loadStateFileExecInternal(ssm, version);
1028 else if (vrc == VERR_SSM_UNIT_NOT_FOUND)
1029 vrc = VINF_SUCCESS;
1030 }
1031 else
1032 vrc = VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1033
1034 SSMR3Close(ssm);
1035 }
1036
1037 if (RT_FAILURE(vrc))
1038 rc = setError(VBOX_E_FILE_ERROR,
1039 tr("The saved state file '%ls' is invalid (%Rrc). Discard the saved state and try again"),
1040 savedStateFile.raw(), vrc);
1041
1042 mSavedStateDataLoaded = true;
1043
1044 return rc;
1045}
1046
1047/**
1048 * Callback handler to save various console data to the state file,
1049 * called when the user saves the VM state.
1050 *
1051 * @param pvUser pointer to Console
1052 *
1053 * @note Locks the Console object for reading.
1054 */
1055//static
1056DECLCALLBACK(void)
1057Console::saveStateFileExec(PSSMHANDLE pSSM, void *pvUser)
1058{
1059 LogFlowFunc(("\n"));
1060
1061 Console *that = static_cast<Console *>(pvUser);
1062 AssertReturnVoid(that);
1063
1064 AutoCaller autoCaller(that);
1065 AssertComRCReturnVoid(autoCaller.rc());
1066
1067 AutoReadLock alock(that COMMA_LOCKVAL_SRC_POS);
1068
1069 int vrc = SSMR3PutU32(pSSM, (uint32_t)that->mSharedFolders.size());
1070 AssertRC(vrc);
1071
1072 for (SharedFolderMap::const_iterator it = that->mSharedFolders.begin();
1073 it != that->mSharedFolders.end();
1074 ++ it)
1075 {
1076 ComObjPtr<SharedFolder> folder = (*it).second;
1077 // don't lock the folder because methods we access are const
1078
1079 Utf8Str name = folder->getName();
1080 vrc = SSMR3PutU32(pSSM, (uint32_t)name.length() + 1 /* term. 0 */);
1081 AssertRC(vrc);
1082 vrc = SSMR3PutStrZ(pSSM, name.c_str());
1083 AssertRC(vrc);
1084
1085 Utf8Str hostPath = folder->getHostPath();
1086 vrc = SSMR3PutU32(pSSM, (uint32_t)hostPath.length() + 1 /* term. 0 */);
1087 AssertRC(vrc);
1088 vrc = SSMR3PutStrZ(pSSM, hostPath.c_str());
1089 AssertRC(vrc);
1090
1091 vrc = SSMR3PutBool(pSSM, !!folder->isWritable());
1092 AssertRC(vrc);
1093 }
1094
1095 return;
1096}
1097
1098/**
1099 * Callback handler to load various console data from the state file.
1100 * Called when the VM is being restored from the saved state.
1101 *
1102 * @param pvUser pointer to Console
1103 * @param uVersion Console unit version.
1104 * Should match sSSMConsoleVer.
1105 * @param uPass The data pass.
1106 *
1107 * @note Should locks the Console object for writing, if necessary.
1108 */
1109//static
1110DECLCALLBACK(int)
1111Console::loadStateFileExec(PSSMHANDLE pSSM, void *pvUser, uint32_t uVersion, uint32_t uPass)
1112{
1113 LogFlowFunc(("\n"));
1114
1115 if (SSM_VERSION_MAJOR_CHANGED(uVersion, sSSMConsoleVer))
1116 return VERR_VERSION_MISMATCH;
1117 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
1118
1119 Console *that = static_cast<Console *>(pvUser);
1120 AssertReturn(that, VERR_INVALID_PARAMETER);
1121
1122 /* Currently, nothing to do when we've been called from VMR3Load*. */
1123 return SSMR3SkipToEndOfUnit(pSSM);
1124}
1125
1126/**
1127 * Method to load various console data from the state file.
1128 * Called from #loadDataFromSavedState.
1129 *
1130 * @param pvUser pointer to Console
1131 * @param u32Version Console unit version.
1132 * Should match sSSMConsoleVer.
1133 *
1134 * @note Locks the Console object for writing.
1135 */
1136int
1137Console::loadStateFileExecInternal(PSSMHANDLE pSSM, uint32_t u32Version)
1138{
1139 AutoCaller autoCaller(this);
1140 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
1141
1142 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1143
1144 AssertReturn(mSharedFolders.size() == 0, VERR_INTERNAL_ERROR);
1145
1146 uint32_t size = 0;
1147 int vrc = SSMR3GetU32(pSSM, &size);
1148 AssertRCReturn(vrc, vrc);
1149
1150 for (uint32_t i = 0; i < size; ++ i)
1151 {
1152 Bstr name;
1153 Bstr hostPath;
1154 bool writable = true;
1155
1156 uint32_t szBuf = 0;
1157 char *buf = NULL;
1158
1159 vrc = SSMR3GetU32(pSSM, &szBuf);
1160 AssertRCReturn(vrc, vrc);
1161 buf = new char[szBuf];
1162 vrc = SSMR3GetStrZ(pSSM, buf, szBuf);
1163 AssertRC(vrc);
1164 name = buf;
1165 delete[] buf;
1166
1167 vrc = SSMR3GetU32(pSSM, &szBuf);
1168 AssertRCReturn(vrc, vrc);
1169 buf = new char[szBuf];
1170 vrc = SSMR3GetStrZ(pSSM, buf, szBuf);
1171 AssertRC(vrc);
1172 hostPath = buf;
1173 delete[] buf;
1174
1175 if (u32Version > 0x00010000)
1176 SSMR3GetBool(pSSM, &writable);
1177
1178 ComObjPtr<SharedFolder> sharedFolder;
1179 sharedFolder.createObject();
1180 HRESULT rc = sharedFolder->init(this, name, hostPath, writable);
1181 AssertComRCReturn(rc, VERR_INTERNAL_ERROR);
1182
1183 mSharedFolders.insert(std::make_pair(name, sharedFolder));
1184 }
1185
1186 return VINF_SUCCESS;
1187}
1188
1189#ifdef VBOX_WITH_GUEST_PROPS
1190
1191// static
1192DECLCALLBACK(int) Console::doGuestPropNotification(void *pvExtension,
1193 uint32_t u32Function,
1194 void *pvParms,
1195 uint32_t cbParms)
1196{
1197 using namespace guestProp;
1198
1199 Assert(u32Function == 0); NOREF(u32Function);
1200
1201 /*
1202 * No locking, as this is purely a notification which does not make any
1203 * changes to the object state.
1204 */
1205 PHOSTCALLBACKDATA pCBData = reinterpret_cast<PHOSTCALLBACKDATA>(pvParms);
1206 AssertReturn(sizeof(HOSTCALLBACKDATA) == cbParms, VERR_INVALID_PARAMETER);
1207 AssertReturn(HOSTCALLBACKMAGIC == pCBData->u32Magic, VERR_INVALID_PARAMETER);
1208 Log5(("Console::doGuestPropNotification: pCBData={.pcszName=%s, .pcszValue=%s, .pcszFlags=%s}\n",
1209 pCBData->pcszName, pCBData->pcszValue, pCBData->pcszFlags));
1210
1211 int rc;
1212 Bstr name(pCBData->pcszName);
1213 Bstr value(pCBData->pcszValue);
1214 Bstr flags(pCBData->pcszFlags);
1215 ComObjPtr<Console> ptrConsole = reinterpret_cast<Console *>(pvExtension);
1216 HRESULT hrc = ptrConsole->mControl->PushGuestProperty(name,
1217 value,
1218 pCBData->u64Timestamp,
1219 flags);
1220 if (SUCCEEDED(hrc))
1221 rc = VINF_SUCCESS;
1222 else
1223 {
1224 LogFunc(("Console::doGuestPropNotification: hrc=%Rhrc pCBData={.pcszName=%s, .pcszValue=%s, .pcszFlags=%s}\n",
1225 pCBData->pcszName, pCBData->pcszValue, pCBData->pcszFlags));
1226 rc = Global::vboxStatusCodeFromCOM(hrc);
1227 }
1228 return rc;
1229}
1230
1231HRESULT Console::doEnumerateGuestProperties(CBSTR aPatterns,
1232 ComSafeArrayOut(BSTR, aNames),
1233 ComSafeArrayOut(BSTR, aValues),
1234 ComSafeArrayOut(ULONG64, aTimestamps),
1235 ComSafeArrayOut(BSTR, aFlags))
1236{
1237 using namespace guestProp;
1238
1239 VBOXHGCMSVCPARM parm[3];
1240
1241 Utf8Str utf8Patterns(aPatterns);
1242 parm[0].type = VBOX_HGCM_SVC_PARM_PTR;
1243 // mutableRaw() returns NULL for an empty string
1244// if ((parm[0].u.pointer.addr = utf8Patterns.mutableRaw()))
1245// parm[0].u.pointer.size = (uint32_t)utf8Patterns.length() + 1;
1246// else
1247// {
1248// parm[0].u.pointer.addr = (void*)"";
1249// parm[0].u.pointer.size = 1;
1250// }
1251 parm[0].u.pointer.addr = utf8Patterns.mutableRaw();
1252 parm[0].u.pointer.size = (uint32_t)utf8Patterns.length() + 1;
1253
1254 /*
1255 * Now things get slightly complicated. Due to a race with the guest adding
1256 * properties, there is no good way to know how much to enlarge a buffer for
1257 * the service to enumerate into. We choose a decent starting size and loop a
1258 * few times, each time retrying with the size suggested by the service plus
1259 * one Kb.
1260 */
1261 size_t cchBuf = 4096;
1262 Utf8Str Utf8Buf;
1263 int vrc = VERR_BUFFER_OVERFLOW;
1264 for (unsigned i = 0; i < 10 && (VERR_BUFFER_OVERFLOW == vrc); ++i)
1265 {
1266 try
1267 {
1268 Utf8Buf.reserve(cchBuf + 1024);
1269 }
1270 catch(...)
1271 {
1272 return E_OUTOFMEMORY;
1273 }
1274 parm[1].type = VBOX_HGCM_SVC_PARM_PTR;
1275 parm[1].u.pointer.addr = Utf8Buf.mutableRaw();
1276 parm[1].u.pointer.size = (uint32_t)cchBuf + 1024;
1277 vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", ENUM_PROPS_HOST, 3,
1278 &parm[0]);
1279 Utf8Buf.jolt();
1280 if (parm[2].type != VBOX_HGCM_SVC_PARM_32BIT)
1281 return setError(E_FAIL, tr("Internal application error"));
1282 cchBuf = parm[2].u.uint32;
1283 }
1284 if (VERR_BUFFER_OVERFLOW == vrc)
1285 return setError(E_UNEXPECTED,
1286 tr("Temporary failure due to guest activity, please retry"));
1287
1288 /*
1289 * Finally we have to unpack the data returned by the service into the safe
1290 * arrays supplied by the caller. We start by counting the number of entries.
1291 */
1292 const char *pszBuf
1293 = reinterpret_cast<const char *>(parm[1].u.pointer.addr);
1294 unsigned cEntries = 0;
1295 /* The list is terminated by a zero-length string at the end of a set
1296 * of four strings. */
1297 for (size_t i = 0; strlen(pszBuf + i) != 0; )
1298 {
1299 /* We are counting sets of four strings. */
1300 for (unsigned j = 0; j < 4; ++j)
1301 i += strlen(pszBuf + i) + 1;
1302 ++cEntries;
1303 }
1304
1305 /*
1306 * And now we create the COM safe arrays and fill them in.
1307 */
1308 com::SafeArray<BSTR> names(cEntries);
1309 com::SafeArray<BSTR> values(cEntries);
1310 com::SafeArray<ULONG64> timestamps(cEntries);
1311 com::SafeArray<BSTR> flags(cEntries);
1312 size_t iBuf = 0;
1313 /* Rely on the service to have formated the data correctly. */
1314 for (unsigned i = 0; i < cEntries; ++i)
1315 {
1316 size_t cchName = strlen(pszBuf + iBuf);
1317 Bstr(pszBuf + iBuf).detachTo(&names[i]);
1318 iBuf += cchName + 1;
1319 size_t cchValue = strlen(pszBuf + iBuf);
1320 Bstr(pszBuf + iBuf).detachTo(&values[i]);
1321 iBuf += cchValue + 1;
1322 size_t cchTimestamp = strlen(pszBuf + iBuf);
1323 timestamps[i] = RTStrToUInt64(pszBuf + iBuf);
1324 iBuf += cchTimestamp + 1;
1325 size_t cchFlags = strlen(pszBuf + iBuf);
1326 Bstr(pszBuf + iBuf).detachTo(&flags[i]);
1327 iBuf += cchFlags + 1;
1328 }
1329 names.detachTo(ComSafeArrayOutArg(aNames));
1330 values.detachTo(ComSafeArrayOutArg(aValues));
1331 timestamps.detachTo(ComSafeArrayOutArg(aTimestamps));
1332 flags.detachTo(ComSafeArrayOutArg(aFlags));
1333 return S_OK;
1334}
1335
1336/**
1337 * Helper that is used by powerDown to move the guest properties to VBoxSVC.
1338 *
1339 * @param fSaving Whether we're saving a machine state and should
1340 * therefore save transient properties as well.
1341 *
1342 * @returns COM status code.
1343 *
1344 * @remarks This is called without holding the console lock.
1345 */
1346HRESULT Console::doMoveGuestPropertiesOnPowerOff(bool fSaving)
1347{
1348 /*
1349 * First, flush any pending notifications.
1350 */
1351 VBOXHGCMSVCPARM parm[1];
1352 parm[0].setUInt32(20*1000/*ms*/);
1353 int vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", guestProp::FLUSH_NOTIFICATIONS_HOST, 1, &parm[0]);
1354 if (RT_FAILURE(vrc))
1355 LogRelFunc(("Flushing notifications failed with rc=%Rrc\n", vrc));
1356
1357 /*
1358 * Enumerate the properties and
1359 */
1360 HRESULT hrc;
1361 com::SafeArray<BSTR> namesOut;
1362 com::SafeArray<BSTR> valuesOut;
1363 com::SafeArray<ULONG64> timestampsOut;
1364 com::SafeArray<BSTR> flagsOut;
1365 try
1366 {
1367 Bstr pattern;
1368 hrc = doEnumerateGuestProperties(pattern,
1369 ComSafeArrayAsOutParam(namesOut),
1370 ComSafeArrayAsOutParam(valuesOut),
1371 ComSafeArrayAsOutParam(timestampsOut),
1372 ComSafeArrayAsOutParam(flagsOut));
1373 if (SUCCEEDED(hrc))
1374 {
1375 std::vector<BSTR> names;
1376 std::vector<BSTR> values;
1377 std::vector<ULONG64> timestamps;
1378 std::vector<BSTR> flags;
1379 for (size_t i = 0; i < namesOut.size(); ++i)
1380 {
1381 uint32_t fFlags = guestProp::NILFLAG;
1382 vrc = guestProp::validateFlags(Utf8Str(flagsOut[i]).raw(), &fFlags); AssertRC(vrc);
1383 if ( fSaving
1384 || !(fFlags & guestProp::TRANSIENT))
1385 {
1386 names.push_back(namesOut[i]);
1387 values.push_back(valuesOut[i]);
1388 timestamps.push_back(timestampsOut[i]);
1389 flags.push_back(flagsOut[i]);
1390 }
1391 }
1392 com::SafeArray<BSTR> namesIn(names);
1393 com::SafeArray<BSTR> valuesIn(values);
1394 com::SafeArray<ULONG64> timestampsIn(timestamps);
1395 com::SafeArray<BSTR> flagsIn(flags);
1396 /* PushGuestProperties() calls DiscardSettings(), which calls us back */
1397 mControl->PushGuestProperties(ComSafeArrayAsInParam(namesIn),
1398 ComSafeArrayAsInParam(valuesIn),
1399 ComSafeArrayAsInParam(timestampsIn),
1400 ComSafeArrayAsInParam(flagsIn));
1401 }
1402 }
1403 catch (...)
1404 {
1405 hrc = Console::handleUnexpectedExceptions(RT_SRC_POS);
1406 }
1407 if (FAILED(hrc))
1408 LogRelFunc(("Failed with hrc=%Rhrc\n", hrc));
1409 return hrc;
1410}
1411
1412
1413
1414#endif /* VBOX_WITH_GUEST_PROPS */
1415
1416
1417// IConsole properties
1418/////////////////////////////////////////////////////////////////////////////
1419
1420STDMETHODIMP Console::COMGETTER(Machine)(IMachine **aMachine)
1421{
1422 CheckComArgOutPointerValid(aMachine);
1423
1424 AutoCaller autoCaller(this);
1425 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1426
1427 /* mMachine is constant during life time, no need to lock */
1428 mMachine.queryInterfaceTo(aMachine);
1429
1430 /* callers expect to get a valid reference, better fail than crash them */
1431 if (mMachine.isNull())
1432 return E_FAIL;
1433
1434 return S_OK;
1435}
1436
1437STDMETHODIMP Console::COMGETTER(State)(MachineState_T *aMachineState)
1438{
1439 CheckComArgOutPointerValid(aMachineState);
1440
1441 AutoCaller autoCaller(this);
1442 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1443
1444 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1445
1446 /* we return our local state (since it's always the same as on the server) */
1447 *aMachineState = mMachineState;
1448
1449 return S_OK;
1450}
1451
1452STDMETHODIMP Console::COMGETTER(Guest)(IGuest **aGuest)
1453{
1454 CheckComArgOutPointerValid(aGuest);
1455
1456 AutoCaller autoCaller(this);
1457 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1458
1459 /* mGuest is constant during life time, no need to lock */
1460 mGuest.queryInterfaceTo(aGuest);
1461
1462 return S_OK;
1463}
1464
1465STDMETHODIMP Console::COMGETTER(Keyboard)(IKeyboard **aKeyboard)
1466{
1467 CheckComArgOutPointerValid(aKeyboard);
1468
1469 AutoCaller autoCaller(this);
1470 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1471
1472 /* mKeyboard is constant during life time, no need to lock */
1473 mKeyboard.queryInterfaceTo(aKeyboard);
1474
1475 return S_OK;
1476}
1477
1478STDMETHODIMP Console::COMGETTER(Mouse)(IMouse **aMouse)
1479{
1480 CheckComArgOutPointerValid(aMouse);
1481
1482 AutoCaller autoCaller(this);
1483 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1484
1485 /* mMouse is constant during life time, no need to lock */
1486 mMouse.queryInterfaceTo(aMouse);
1487
1488 return S_OK;
1489}
1490
1491STDMETHODIMP Console::COMGETTER(Display)(IDisplay **aDisplay)
1492{
1493 CheckComArgOutPointerValid(aDisplay);
1494
1495 AutoCaller autoCaller(this);
1496 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1497
1498 /* mDisplay is constant during life time, no need to lock */
1499 mDisplay.queryInterfaceTo(aDisplay);
1500
1501 return S_OK;
1502}
1503
1504STDMETHODIMP Console::COMGETTER(Debugger)(IMachineDebugger **aDebugger)
1505{
1506 CheckComArgOutPointerValid(aDebugger);
1507
1508 AutoCaller autoCaller(this);
1509 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1510
1511 /* we need a write lock because of the lazy mDebugger initialization*/
1512 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1513
1514 /* check if we have to create the debugger object */
1515 if (!mDebugger)
1516 {
1517 unconst(mDebugger).createObject();
1518 mDebugger->init(this);
1519 }
1520
1521 mDebugger.queryInterfaceTo(aDebugger);
1522
1523 return S_OK;
1524}
1525
1526STDMETHODIMP Console::COMGETTER(USBDevices)(ComSafeArrayOut(IUSBDevice *, aUSBDevices))
1527{
1528 CheckComArgOutSafeArrayPointerValid(aUSBDevices);
1529
1530 AutoCaller autoCaller(this);
1531 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1532
1533 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1534
1535 SafeIfaceArray<IUSBDevice> collection(mUSBDevices);
1536 collection.detachTo(ComSafeArrayOutArg(aUSBDevices));
1537
1538 return S_OK;
1539}
1540
1541STDMETHODIMP Console::COMGETTER(RemoteUSBDevices)(ComSafeArrayOut(IHostUSBDevice *, aRemoteUSBDevices))
1542{
1543 CheckComArgOutSafeArrayPointerValid(aRemoteUSBDevices);
1544
1545 AutoCaller autoCaller(this);
1546 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1547
1548 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1549
1550 SafeIfaceArray<IHostUSBDevice> collection(mRemoteUSBDevices);
1551 collection.detachTo(ComSafeArrayOutArg(aRemoteUSBDevices));
1552
1553 return S_OK;
1554}
1555
1556STDMETHODIMP Console::COMGETTER(RemoteDisplayInfo)(IRemoteDisplayInfo **aRemoteDisplayInfo)
1557{
1558 CheckComArgOutPointerValid(aRemoteDisplayInfo);
1559
1560 AutoCaller autoCaller(this);
1561 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1562
1563 /* mDisplay is constant during life time, no need to lock */
1564 mRemoteDisplayInfo.queryInterfaceTo(aRemoteDisplayInfo);
1565
1566 return S_OK;
1567}
1568
1569STDMETHODIMP
1570Console::COMGETTER(SharedFolders)(ComSafeArrayOut(ISharedFolder *, aSharedFolders))
1571{
1572 CheckComArgOutSafeArrayPointerValid(aSharedFolders);
1573
1574 AutoCaller autoCaller(this);
1575 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1576
1577 /* loadDataFromSavedState() needs a write lock */
1578 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1579
1580 /* Read console data stored in the saved state file (if not yet done) */
1581 HRESULT rc = loadDataFromSavedState();
1582 if (FAILED(rc)) return rc;
1583
1584 SafeIfaceArray<ISharedFolder> sf(mSharedFolders);
1585 sf.detachTo(ComSafeArrayOutArg(aSharedFolders));
1586
1587 return S_OK;
1588}
1589
1590
1591// IConsole methods
1592/////////////////////////////////////////////////////////////////////////////
1593
1594
1595STDMETHODIMP Console::PowerUp(IProgress **aProgress)
1596{
1597 return powerUp(aProgress, false /* aPaused */);
1598}
1599
1600STDMETHODIMP Console::PowerUpPaused(IProgress **aProgress)
1601{
1602 return powerUp(aProgress, true /* aPaused */);
1603}
1604
1605STDMETHODIMP Console::PowerDown(IProgress **aProgress)
1606{
1607 if (aProgress == NULL)
1608 return E_POINTER;
1609
1610 LogFlowThisFuncEnter();
1611 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
1612
1613 AutoCaller autoCaller(this);
1614 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1615
1616 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1617
1618 switch (mMachineState)
1619 {
1620 case MachineState_Running:
1621 case MachineState_Paused:
1622 case MachineState_Stuck:
1623 break;
1624
1625 /* Try cancel the teleportation. */
1626 case MachineState_Teleporting:
1627 case MachineState_TeleportingPausedVM:
1628 if (!mptrCancelableProgress.isNull())
1629 {
1630 HRESULT hrc = mptrCancelableProgress->Cancel();
1631 if (SUCCEEDED(hrc))
1632 break;
1633 }
1634 return setError(VBOX_E_INVALID_VM_STATE, tr("Cannot power down at this point in a teleportation"));
1635
1636 /* Try cancel the live snapshot. */
1637 case MachineState_LiveSnapshotting:
1638 if (!mptrCancelableProgress.isNull())
1639 {
1640 HRESULT hrc = mptrCancelableProgress->Cancel();
1641 if (SUCCEEDED(hrc))
1642 break;
1643 }
1644 return setError(VBOX_E_INVALID_VM_STATE, tr("Cannot power down at this point in a live snapshot"));
1645
1646 /* extra nice error message for a common case */
1647 case MachineState_Saved:
1648 return setError(VBOX_E_INVALID_VM_STATE, tr("Cannot power down a saved virtual machine"));
1649 case MachineState_Stopping:
1650 return setError(VBOX_E_INVALID_VM_STATE, tr("Virtual machine is being powered down"));
1651 default:
1652 return setError(VBOX_E_INVALID_VM_STATE,
1653 tr("Invalid machine state: %s (must be Running, Paused or Stuck)"),
1654 Global::stringifyMachineState(mMachineState));
1655 }
1656
1657 LogFlowThisFunc(("Initiating SHUTDOWN request...\n"));
1658
1659 /* create an IProgress object to track progress of this operation */
1660 ComObjPtr<Progress> progress;
1661 progress.createObject();
1662 progress->init(static_cast<IConsole *>(this),
1663 Bstr(tr("Stopping virtual machine")),
1664 FALSE /* aCancelable */);
1665
1666 /* setup task object and thread to carry out the operation asynchronously */
1667 std::auto_ptr<VMProgressTask> task(new VMProgressTask(this, progress, true /* aUsesVMPtr */));
1668 AssertReturn(task->isOk(), E_FAIL);
1669
1670 int vrc = RTThreadCreate(NULL, Console::powerDownThread,
1671 (void *) task.get(), 0,
1672 RTTHREADTYPE_MAIN_WORKER, 0,
1673 "VMPowerDown");
1674 ComAssertMsgRCRet(vrc, ("Could not create VMPowerDown thread (%Rrc)", vrc), E_FAIL);
1675
1676 /* task is now owned by powerDownThread(), so release it */
1677 task.release();
1678
1679 /* go to Stopping state to forbid state-dependant operations */
1680 setMachineState(MachineState_Stopping);
1681
1682 /* pass the progress to the caller */
1683 progress.queryInterfaceTo(aProgress);
1684
1685 LogFlowThisFuncLeave();
1686
1687 return S_OK;
1688}
1689
1690STDMETHODIMP Console::Reset()
1691{
1692 LogFlowThisFuncEnter();
1693 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
1694
1695 AutoCaller autoCaller(this);
1696 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1697
1698 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1699
1700 if ( mMachineState != MachineState_Running
1701 && mMachineState != MachineState_Teleporting
1702 && mMachineState != MachineState_LiveSnapshotting
1703 /** @todo r=bird: This should be allowed on paused VMs as well. Later. */
1704 )
1705 return setError(VBOX_E_INVALID_VM_STATE,
1706 tr("Invalid machine state: %s"),
1707 Global::stringifyMachineState(mMachineState));
1708
1709 /* protect mpVM */
1710 AutoVMCaller autoVMCaller(this);
1711 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
1712
1713 /* leave the lock before a VMR3* call (EMT will call us back)! */
1714 alock.leave();
1715
1716 int vrc = VMR3Reset(mpVM);
1717
1718 HRESULT rc = RT_SUCCESS(vrc) ? S_OK :
1719 setError(VBOX_E_VM_ERROR,
1720 tr("Could not reset the machine (%Rrc)"),
1721 vrc);
1722
1723 LogFlowThisFunc(("mMachineState=%d, rc=%08X\n", mMachineState, rc));
1724 LogFlowThisFuncLeave();
1725 return rc;
1726}
1727
1728DECLCALLBACK(int) Console::unplugCpu(Console *pThis, unsigned uCpu)
1729{
1730 LogFlowFunc(("pThis=%p uCpu=%u\n", pThis, uCpu));
1731
1732 AssertReturn(pThis, VERR_INVALID_PARAMETER);
1733
1734 int vrc = PDMR3DeviceDetach(pThis->mpVM, "acpi", 0, uCpu, 0);
1735 Log(("UnplugCpu: rc=%Rrc\n", vrc));
1736
1737 return vrc;
1738}
1739
1740HRESULT Console::doCPURemove(ULONG aCpu)
1741{
1742 HRESULT rc = S_OK;
1743
1744 LogFlowThisFuncEnter();
1745 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
1746
1747 AutoCaller autoCaller(this);
1748 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1749
1750 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1751
1752 if ( mMachineState != MachineState_Running
1753 && mMachineState != MachineState_Teleporting
1754 && mMachineState != MachineState_LiveSnapshotting
1755 )
1756 return setError(VBOX_E_INVALID_VM_STATE,
1757 tr("Invalid machine state: %s"),
1758 Global::stringifyMachineState(mMachineState));
1759
1760 /* protect mpVM */
1761 AutoVMCaller autoVMCaller(this);
1762 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
1763
1764 /* Check if the CPU is present */
1765 BOOL fCpuAttached;
1766 rc = mMachine->GetCPUStatus(aCpu, &fCpuAttached);
1767 if (FAILED(rc)) return rc;
1768
1769 if (!fCpuAttached)
1770 return setError(E_FAIL,
1771 tr("CPU %d is not attached"), aCpu);
1772
1773 /* Check if the CPU is unlocked */
1774 PPDMIBASE pBase;
1775 int vrc = PDMR3QueryDeviceLun(mpVM, "acpi", 0, aCpu, &pBase);
1776 bool fLocked = true;
1777 if (RT_SUCCESS(vrc))
1778 {
1779 uint32_t idCpuCore, idCpuPackage;
1780
1781 /* Notify the guest if possible. */
1782 vrc = VMR3GetCpuCoreAndPackageIdFromCpuId(mpVM, aCpu, &idCpuCore, &idCpuPackage);
1783 AssertRC(vrc);
1784
1785 Assert(pBase);
1786
1787 PPDMIACPIPORT pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIACPIPORT);
1788
1789 vrc = getVMMDev()->getVMMDevPort()->pfnCpuHotUnplug(getVMMDev()->getVMMDevPort(), idCpuCore, idCpuPackage);
1790 if (RT_SUCCESS(vrc))
1791 {
1792 unsigned cTries = 100;
1793
1794 do
1795 {
1796 /* It will take some time until the event is processed in the guest. Wait */
1797 vrc = pPort ? pPort->pfnGetCpuStatus(pPort, aCpu, &fLocked) : VERR_INVALID_POINTER;
1798
1799 if (RT_SUCCESS(vrc) && !fLocked)
1800 break;
1801
1802 /* Sleep a bit */
1803 RTThreadSleep(100);
1804 } while (cTries-- > 0);
1805 }
1806 else if (vrc == VERR_CPU_HOTPLUG_NOT_MONITORED_BY_GUEST)
1807 {
1808 /* Query one time. It is possible that the user ejected the CPU. */
1809 vrc = pPort ? pPort->pfnGetCpuStatus(pPort, aCpu, &fLocked) : VERR_INVALID_POINTER;
1810 }
1811 }
1812
1813 /* If the CPU was unlocked we can detach it now. */
1814 if (RT_SUCCESS(vrc) && !fLocked)
1815 {
1816 /*
1817 * Call worker in EMT, that's faster and safer than doing everything
1818 * using VMR3ReqCall. Note that we separate VMR3ReqCall from VMR3ReqWait
1819 * here to make requests from under the lock in order to serialize them.
1820 */
1821 PVMREQ pReq;
1822 vrc = VMR3ReqCall(mpVM, 0, &pReq, 0 /* no wait! */, VMREQFLAGS_VBOX_STATUS,
1823 (PFNRT)Console::unplugCpu, 2,
1824 this, aCpu);
1825
1826 /* leave the lock before a VMR3* call (EMT will call us back)! */
1827 alock.leave();
1828
1829 if (vrc == VERR_TIMEOUT || RT_SUCCESS(vrc))
1830 {
1831 vrc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
1832 AssertRC(vrc);
1833 if (RT_SUCCESS(vrc))
1834 vrc = pReq->iStatus;
1835 }
1836 VMR3ReqFree(pReq);
1837
1838 if (RT_SUCCESS(vrc))
1839 {
1840 /* Detach it from the VM */
1841 vrc = VMR3HotUnplugCpu(mpVM, aCpu);
1842 AssertRC(vrc);
1843 }
1844 else
1845 rc = setError(VBOX_E_VM_ERROR,
1846 tr("Hot-Remove failed (rc=%Rrc)"), vrc);
1847 }
1848 else
1849 rc = setError(VBOX_E_VM_ERROR,
1850 tr("Hot-Remove was aborted because the CPU may still be used by the guest"), VERR_RESOURCE_BUSY);
1851
1852 LogFlowThisFunc(("mMachineState=%d, rc=%08X\n", mMachineState, rc));
1853 LogFlowThisFuncLeave();
1854 return rc;
1855}
1856
1857DECLCALLBACK(int) Console::plugCpu(Console *pThis, unsigned uCpu)
1858{
1859 LogFlowFunc(("pThis=%p uCpu=%u\n", pThis, uCpu));
1860
1861 AssertReturn(pThis, VERR_INVALID_PARAMETER);
1862
1863 int rc = VMR3HotPlugCpu(pThis->mpVM, uCpu);
1864 AssertRC(rc);
1865
1866 PCFGMNODE pInst = CFGMR3GetChild(CFGMR3GetRoot(pThis->mpVM), "Devices/acpi/0/");
1867 AssertRelease(pInst);
1868 /* nuke anything which might have been left behind. */
1869 CFGMR3RemoveNode(CFGMR3GetChildF(pInst, "LUN#%d", uCpu));
1870
1871#define RC_CHECK() do { if (RT_FAILURE(rc)) { AssertReleaseRC(rc); break; } } while (0)
1872
1873 PCFGMNODE pLunL0;
1874 PCFGMNODE pCfg;
1875 rc = CFGMR3InsertNodeF(pInst, &pLunL0, "LUN#%d", uCpu); RC_CHECK();
1876 rc = CFGMR3InsertString(pLunL0, "Driver", "ACPICpu"); RC_CHECK();
1877 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1878
1879 /*
1880 * Attach the driver.
1881 */
1882 PPDMIBASE pBase;
1883 rc = PDMR3DeviceAttach(pThis->mpVM, "acpi", 0, uCpu, 0, &pBase); RC_CHECK();
1884
1885 Log(("PlugCpu: rc=%Rrc\n", rc));
1886
1887 CFGMR3Dump(pInst);
1888
1889 return VINF_SUCCESS;
1890}
1891
1892HRESULT Console::doCPUAdd(ULONG aCpu)
1893{
1894 HRESULT rc = S_OK;
1895
1896 LogFlowThisFuncEnter();
1897 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
1898
1899 AutoCaller autoCaller(this);
1900 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1901
1902 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1903
1904 if ( mMachineState != MachineState_Running
1905 && mMachineState != MachineState_Teleporting
1906 && mMachineState != MachineState_LiveSnapshotting
1907 /** @todo r=bird: This should be allowed on paused VMs as well. Later. */
1908 )
1909 return setError(VBOX_E_INVALID_VM_STATE,
1910 tr("Invalid machine state: %s"),
1911 Global::stringifyMachineState(mMachineState));
1912
1913 /* protect mpVM */
1914 AutoVMCaller autoVMCaller(this);
1915 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
1916
1917 /* Check if the CPU is present */
1918 BOOL fCpuAttached;
1919 rc = mMachine->GetCPUStatus(aCpu, &fCpuAttached);
1920 if (FAILED(rc)) return rc;
1921
1922 if (fCpuAttached)
1923 return setError(E_FAIL,
1924 tr("CPU %d is already attached"), aCpu);
1925
1926 /*
1927 * Call worker in EMT, that's faster and safer than doing everything
1928 * using VMR3ReqCall. Note that we separate VMR3ReqCall from VMR3ReqWait
1929 * here to make requests from under the lock in order to serialize them.
1930 */
1931 PVMREQ pReq;
1932 int vrc = VMR3ReqCall(mpVM, 0, &pReq, 0 /* no wait! */, VMREQFLAGS_VBOX_STATUS,
1933 (PFNRT)Console::plugCpu, 2,
1934 this, aCpu);
1935
1936 /* leave the lock before a VMR3* call (EMT will call us back)! */
1937 alock.leave();
1938
1939 if (vrc == VERR_TIMEOUT || RT_SUCCESS(vrc))
1940 {
1941 vrc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
1942 AssertRC(vrc);
1943 if (RT_SUCCESS(vrc))
1944 vrc = pReq->iStatus;
1945 }
1946 VMR3ReqFree(pReq);
1947
1948 rc = RT_SUCCESS(vrc) ? S_OK :
1949 setError(VBOX_E_VM_ERROR,
1950 tr("Could not add CPU to the machine (%Rrc)"),
1951 vrc);
1952
1953 if (RT_SUCCESS(vrc))
1954 {
1955 uint32_t idCpuCore, idCpuPackage;
1956
1957 /* Notify the guest if possible. */
1958 vrc = VMR3GetCpuCoreAndPackageIdFromCpuId(mpVM, aCpu, &idCpuCore, &idCpuPackage);
1959 AssertRC(vrc);
1960
1961 vrc = getVMMDev()->getVMMDevPort()->pfnCpuHotPlug(getVMMDev()->getVMMDevPort(), idCpuCore, idCpuPackage);
1962 /** @todo warning if the guest doesn't support it */
1963 }
1964
1965 LogFlowThisFunc(("mMachineState=%d, rc=%08X\n", mMachineState, rc));
1966 LogFlowThisFuncLeave();
1967 return rc;
1968}
1969
1970STDMETHODIMP Console::Pause()
1971{
1972 LogFlowThisFuncEnter();
1973
1974 AutoCaller autoCaller(this);
1975 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1976
1977 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1978
1979 switch (mMachineState)
1980 {
1981 case MachineState_Running:
1982 case MachineState_Teleporting:
1983 case MachineState_LiveSnapshotting:
1984 break;
1985
1986 case MachineState_Paused:
1987 case MachineState_TeleportingPausedVM:
1988 case MachineState_Saving:
1989 return setError(VBOX_E_INVALID_VM_STATE, tr("Already paused"));
1990
1991 default:
1992 return setError(VBOX_E_INVALID_VM_STATE,
1993 tr("Invalid machine state: %s"),
1994 Global::stringifyMachineState(mMachineState));
1995 }
1996
1997 /* protect mpVM */
1998 AutoVMCaller autoVMCaller(this);
1999 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2000
2001 LogFlowThisFunc(("Sending PAUSE request...\n"));
2002
2003 /* leave the lock before a VMR3* call (EMT will call us back)! */
2004 alock.leave();
2005
2006 int vrc = VMR3Suspend(mpVM);
2007
2008 HRESULT hrc = S_OK;
2009 if (RT_FAILURE(vrc))
2010 hrc = setError(VBOX_E_VM_ERROR, tr("Could not suspend the machine execution (%Rrc)"), vrc);
2011
2012 LogFlowThisFunc(("hrc=%Rhrc\n", hrc));
2013 LogFlowThisFuncLeave();
2014 return hrc;
2015}
2016
2017STDMETHODIMP Console::Resume()
2018{
2019 LogFlowThisFuncEnter();
2020
2021 AutoCaller autoCaller(this);
2022 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2023
2024 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2025
2026 if (mMachineState != MachineState_Paused)
2027 return setError(VBOX_E_INVALID_VM_STATE,
2028 tr("Cannot resume the machine as it is not paused (machine state: %s)"),
2029 Global::stringifyMachineState(mMachineState));
2030
2031 /* protect mpVM */
2032 AutoVMCaller autoVMCaller(this);
2033 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2034
2035 LogFlowThisFunc(("Sending RESUME request...\n"));
2036
2037 /* leave the lock before a VMR3* call (EMT will call us back)! */
2038 alock.leave();
2039
2040 int vrc;
2041 if (VMR3GetState(mpVM) == VMSTATE_CREATED)
2042 vrc = VMR3PowerOn(mpVM); /* (PowerUpPaused) */
2043 else
2044 vrc = VMR3Resume(mpVM);
2045
2046 HRESULT rc = RT_SUCCESS(vrc) ? S_OK :
2047 setError(VBOX_E_VM_ERROR,
2048 tr("Could not resume the machine execution (%Rrc)"),
2049 vrc);
2050
2051 LogFlowThisFunc(("rc=%08X\n", rc));
2052 LogFlowThisFuncLeave();
2053 return rc;
2054}
2055
2056STDMETHODIMP Console::PowerButton()
2057{
2058 LogFlowThisFuncEnter();
2059
2060 AutoCaller autoCaller(this);
2061 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2062
2063 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2064
2065 if ( mMachineState != MachineState_Running
2066 && mMachineState != MachineState_Teleporting
2067 && mMachineState != MachineState_LiveSnapshotting
2068 )
2069 return setError(VBOX_E_INVALID_VM_STATE,
2070 tr("Invalid machine state: %s"),
2071 Global::stringifyMachineState(mMachineState));
2072
2073 /* protect mpVM */
2074 AutoVMCaller autoVMCaller(this);
2075 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2076
2077 PPDMIBASE pBase;
2078 int vrc = PDMR3QueryDeviceLun(mpVM, "acpi", 0, 0, &pBase);
2079 if (RT_SUCCESS(vrc))
2080 {
2081 Assert(pBase);
2082 PPDMIACPIPORT pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIACPIPORT);
2083 vrc = pPort ? pPort->pfnPowerButtonPress(pPort) : VERR_INVALID_POINTER;
2084 }
2085
2086 HRESULT rc = RT_SUCCESS(vrc) ? S_OK :
2087 setError(VBOX_E_PDM_ERROR,
2088 tr("Controlled power off failed (%Rrc)"),
2089 vrc);
2090
2091 LogFlowThisFunc(("rc=%08X\n", rc));
2092 LogFlowThisFuncLeave();
2093 return rc;
2094}
2095
2096STDMETHODIMP Console::GetPowerButtonHandled(BOOL *aHandled)
2097{
2098 LogFlowThisFuncEnter();
2099
2100 CheckComArgOutPointerValid(aHandled);
2101
2102 *aHandled = FALSE;
2103
2104 AutoCaller autoCaller(this);
2105
2106 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2107
2108 if ( mMachineState != MachineState_Running
2109 && mMachineState != MachineState_Teleporting
2110 && mMachineState != MachineState_LiveSnapshotting
2111 )
2112 return setError(VBOX_E_INVALID_VM_STATE,
2113 tr("Invalid machine state: %s"),
2114 Global::stringifyMachineState(mMachineState));
2115
2116 /* protect mpVM */
2117 AutoVMCaller autoVMCaller(this);
2118 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2119
2120 PPDMIBASE pBase;
2121 int vrc = PDMR3QueryDeviceLun(mpVM, "acpi", 0, 0, &pBase);
2122 bool handled = false;
2123 if (RT_SUCCESS(vrc))
2124 {
2125 Assert(pBase);
2126 PPDMIACPIPORT pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIACPIPORT);
2127 vrc = pPort ? pPort->pfnGetPowerButtonHandled(pPort, &handled) : VERR_INVALID_POINTER;
2128 }
2129
2130 HRESULT rc = RT_SUCCESS(vrc) ? S_OK :
2131 setError(VBOX_E_PDM_ERROR,
2132 tr("Checking if the ACPI Power Button event was handled by the guest OS failed (%Rrc)"),
2133 vrc);
2134
2135 *aHandled = handled;
2136
2137 LogFlowThisFunc(("rc=%08X\n", rc));
2138 LogFlowThisFuncLeave();
2139 return rc;
2140}
2141
2142STDMETHODIMP Console::GetGuestEnteredACPIMode(BOOL *aEntered)
2143{
2144 LogFlowThisFuncEnter();
2145
2146 CheckComArgOutPointerValid(aEntered);
2147
2148 *aEntered = FALSE;
2149
2150 AutoCaller autoCaller(this);
2151
2152 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2153
2154 if ( mMachineState != MachineState_Running
2155 && mMachineState != MachineState_Teleporting
2156 && mMachineState != MachineState_LiveSnapshotting
2157 )
2158 return setError(VBOX_E_INVALID_VM_STATE,
2159 tr("Invalid machine state %s when checking if the guest entered the ACPI mode)"),
2160 Global::stringifyMachineState(mMachineState));
2161
2162 /* protect mpVM */
2163 AutoVMCaller autoVMCaller(this);
2164 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2165
2166 PPDMIBASE pBase;
2167 int vrc = PDMR3QueryDeviceLun(mpVM, "acpi", 0, 0, &pBase);
2168 bool entered = false;
2169 if (RT_SUCCESS(vrc))
2170 {
2171 Assert(pBase);
2172 PPDMIACPIPORT pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIACPIPORT);
2173 vrc = pPort ? pPort->pfnGetGuestEnteredACPIMode(pPort, &entered) : VERR_INVALID_POINTER;
2174 }
2175
2176 *aEntered = RT_SUCCESS(vrc) ? entered : false;
2177
2178 LogFlowThisFuncLeave();
2179 return S_OK;
2180}
2181
2182STDMETHODIMP Console::SleepButton()
2183{
2184 LogFlowThisFuncEnter();
2185
2186 AutoCaller autoCaller(this);
2187 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2188
2189 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2190
2191 if (mMachineState != MachineState_Running) /** @todo Live Migration: ??? */
2192 return setError(VBOX_E_INVALID_VM_STATE,
2193 tr("Invalid machine state: %s)"),
2194 Global::stringifyMachineState(mMachineState));
2195
2196 /* protect mpVM */
2197 AutoVMCaller autoVMCaller(this);
2198 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2199
2200 PPDMIBASE pBase;
2201 int vrc = PDMR3QueryDeviceLun(mpVM, "acpi", 0, 0, &pBase);
2202 if (RT_SUCCESS(vrc))
2203 {
2204 Assert(pBase);
2205 PPDMIACPIPORT pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIACPIPORT);
2206 vrc = pPort ? pPort->pfnSleepButtonPress(pPort) : VERR_INVALID_POINTER;
2207 }
2208
2209 HRESULT rc = RT_SUCCESS(vrc) ? S_OK :
2210 setError(VBOX_E_PDM_ERROR,
2211 tr("Sending sleep button event failed (%Rrc)"),
2212 vrc);
2213
2214 LogFlowThisFunc(("rc=%08X\n", rc));
2215 LogFlowThisFuncLeave();
2216 return rc;
2217}
2218
2219STDMETHODIMP Console::SaveState(IProgress **aProgress)
2220{
2221 LogFlowThisFuncEnter();
2222 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
2223
2224 CheckComArgOutPointerValid(aProgress);
2225
2226 AutoCaller autoCaller(this);
2227 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2228
2229 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2230
2231 if ( mMachineState != MachineState_Running
2232 && mMachineState != MachineState_Paused)
2233 {
2234 return setError(VBOX_E_INVALID_VM_STATE,
2235 tr("Cannot save the execution state as the machine is not running or paused (machine state: %s)"),
2236 Global::stringifyMachineState(mMachineState));
2237 }
2238
2239 /* memorize the current machine state */
2240 MachineState_T lastMachineState = mMachineState;
2241
2242 if (mMachineState == MachineState_Running)
2243 {
2244 HRESULT rc = Pause();
2245 if (FAILED(rc)) return rc;
2246 }
2247
2248 HRESULT rc = S_OK;
2249
2250 /* create a progress object to track operation completion */
2251 ComObjPtr<Progress> progress;
2252 progress.createObject();
2253 progress->init(static_cast<IConsole *>(this),
2254 Bstr(tr("Saving the execution state of the virtual machine")),
2255 FALSE /* aCancelable */);
2256
2257 bool fBeganSavingState = false;
2258 bool fTaskCreationFailed = false;
2259
2260 do
2261 {
2262 /* create a task object early to ensure mpVM protection is successful */
2263 std::auto_ptr <VMSaveTask> task(new VMSaveTask(this, progress));
2264 rc = task->rc();
2265 /*
2266 * If we fail here it means a PowerDown() call happened on another
2267 * thread while we were doing Pause() (which leaves the Console lock).
2268 * We assign PowerDown() a higher precedence than SaveState(),
2269 * therefore just return the error to the caller.
2270 */
2271 if (FAILED(rc))
2272 {
2273 fTaskCreationFailed = true;
2274 break;
2275 }
2276
2277 Bstr stateFilePath;
2278
2279 /*
2280 * request a saved state file path from the server
2281 * (this will set the machine state to Saving on the server to block
2282 * others from accessing this machine)
2283 */
2284 rc = mControl->BeginSavingState(progress, stateFilePath.asOutParam());
2285 if (FAILED(rc)) break;
2286
2287 fBeganSavingState = true;
2288
2289 /* sync the state with the server */
2290 setMachineStateLocally(MachineState_Saving);
2291
2292 /* ensure the directory for the saved state file exists */
2293 {
2294 Utf8Str dir = stateFilePath;
2295 dir.stripFilename();
2296 if (!RTDirExists(dir.c_str()))
2297 {
2298 int vrc = RTDirCreateFullPath(dir.c_str(), 0777);
2299 if (RT_FAILURE(vrc))
2300 {
2301 rc = setError(VBOX_E_FILE_ERROR,
2302 tr("Could not create a directory '%s' to save the state to (%Rrc)"),
2303 dir.raw(), vrc);
2304 break;
2305 }
2306 }
2307 }
2308
2309 /* setup task object and thread to carry out the operation asynchronously */
2310 task->mSavedStateFile = stateFilePath;
2311 /* set the state the operation thread will restore when it is finished */
2312 task->mLastMachineState = lastMachineState;
2313
2314 /* create a thread to wait until the VM state is saved */
2315 int vrc = RTThreadCreate(NULL, Console::saveStateThread, (void *) task.get(),
2316 0, RTTHREADTYPE_MAIN_WORKER, 0, "VMSave");
2317
2318 ComAssertMsgRCBreak(vrc, ("Could not create VMSave thread (%Rrc)", vrc),
2319 rc = E_FAIL);
2320
2321 /* task is now owned by saveStateThread(), so release it */
2322 task.release();
2323
2324 /* return the progress to the caller */
2325 progress.queryInterfaceTo(aProgress);
2326 }
2327 while (0);
2328
2329 if (FAILED(rc) && !fTaskCreationFailed)
2330 {
2331 /* preserve existing error info */
2332 ErrorInfoKeeper eik;
2333
2334 if (fBeganSavingState)
2335 {
2336 /*
2337 * cancel the requested save state procedure.
2338 * This will reset the machine state to the state it had right
2339 * before calling mControl->BeginSavingState().
2340 */
2341 mControl->EndSavingState(FALSE);
2342 }
2343
2344 if (lastMachineState == MachineState_Running)
2345 {
2346 /* restore the paused state if appropriate */
2347 setMachineStateLocally(MachineState_Paused);
2348 /* restore the running state if appropriate */
2349 Resume();
2350 }
2351 else
2352 setMachineStateLocally(lastMachineState);
2353 }
2354
2355 LogFlowThisFunc(("rc=%08X\n", rc));
2356 LogFlowThisFuncLeave();
2357 return rc;
2358}
2359
2360STDMETHODIMP Console::AdoptSavedState(IN_BSTR aSavedStateFile)
2361{
2362 CheckComArgStrNotEmptyOrNull(aSavedStateFile);
2363
2364 AutoCaller autoCaller(this);
2365 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2366
2367 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2368
2369 if ( mMachineState != MachineState_PoweredOff
2370 && mMachineState != MachineState_Teleported
2371 && mMachineState != MachineState_Aborted
2372 )
2373 return setError(VBOX_E_INVALID_VM_STATE,
2374 tr("Cannot adopt the saved machine state as the machine is not in Powered Off, Teleported or Aborted state (machine state: %s)"),
2375 Global::stringifyMachineState(mMachineState));
2376
2377 return mControl->AdoptSavedState(aSavedStateFile);
2378}
2379
2380STDMETHODIMP Console::ForgetSavedState(BOOL aRemove)
2381{
2382 AutoCaller autoCaller(this);
2383 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2384
2385 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2386
2387 if (mMachineState != MachineState_Saved)
2388 return setError(VBOX_E_INVALID_VM_STATE,
2389 tr("Cannot discard the machine state as the machine is not in the saved state (machine state: %s)"),
2390 Global::stringifyMachineState(mMachineState));
2391
2392 HRESULT rc = S_OK;
2393
2394 rc = mControl->SetRemoveSavedState(aRemove);
2395 if (FAILED(rc)) return rc;
2396
2397 /*
2398 * Saved -> PoweredOff transition will be detected in the SessionMachine
2399 * and properly handled.
2400 */
2401 rc = setMachineState(MachineState_PoweredOff);
2402
2403 return rc;
2404}
2405
2406/** read the value of a LEd. */
2407inline uint32_t readAndClearLed(PPDMLED pLed)
2408{
2409 if (!pLed)
2410 return 0;
2411 uint32_t u32 = pLed->Actual.u32 | pLed->Asserted.u32;
2412 pLed->Asserted.u32 = 0;
2413 return u32;
2414}
2415
2416STDMETHODIMP Console::GetDeviceActivity(DeviceType_T aDeviceType,
2417 DeviceActivity_T *aDeviceActivity)
2418{
2419 CheckComArgNotNull(aDeviceActivity);
2420
2421 AutoCaller autoCaller(this);
2422 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2423
2424 /*
2425 * Note: we don't lock the console object here because
2426 * readAndClearLed() should be thread safe.
2427 */
2428
2429 /* Get LED array to read */
2430 PDMLEDCORE SumLed = {0};
2431 switch (aDeviceType)
2432 {
2433 case DeviceType_Floppy:
2434 case DeviceType_DVD:
2435 case DeviceType_HardDisk:
2436 {
2437 for (unsigned i = 0; i < RT_ELEMENTS(mapStorageLeds); ++i)
2438 if (maStorageDevType[i] == aDeviceType)
2439 SumLed.u32 |= readAndClearLed(mapStorageLeds[i]);
2440 break;
2441 }
2442
2443 case DeviceType_Network:
2444 {
2445 for (unsigned i = 0; i < RT_ELEMENTS(mapNetworkLeds); ++i)
2446 SumLed.u32 |= readAndClearLed(mapNetworkLeds[i]);
2447 break;
2448 }
2449
2450 case DeviceType_USB:
2451 {
2452 for (unsigned i = 0; i < RT_ELEMENTS(mapUSBLed); ++i)
2453 SumLed.u32 |= readAndClearLed(mapUSBLed[i]);
2454 break;
2455 }
2456
2457 case DeviceType_SharedFolder:
2458 {
2459 SumLed.u32 |= readAndClearLed(mapSharedFolderLed);
2460 break;
2461 }
2462
2463 default:
2464 return setError(E_INVALIDARG,
2465 tr("Invalid device type: %d"),
2466 aDeviceType);
2467 }
2468
2469 /* Compose the result */
2470 switch (SumLed.u32 & (PDMLED_READING | PDMLED_WRITING))
2471 {
2472 case 0:
2473 *aDeviceActivity = DeviceActivity_Idle;
2474 break;
2475 case PDMLED_READING:
2476 *aDeviceActivity = DeviceActivity_Reading;
2477 break;
2478 case PDMLED_WRITING:
2479 case PDMLED_READING | PDMLED_WRITING:
2480 *aDeviceActivity = DeviceActivity_Writing;
2481 break;
2482 }
2483
2484 return S_OK;
2485}
2486
2487STDMETHODIMP Console::AttachUSBDevice(IN_BSTR aId)
2488{
2489#ifdef VBOX_WITH_USB
2490 AutoCaller autoCaller(this);
2491 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2492
2493 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2494
2495 if ( mMachineState != MachineState_Running
2496 && mMachineState != MachineState_Paused)
2497 return setError(VBOX_E_INVALID_VM_STATE,
2498 tr("Cannot attach a USB device to the machine which is not running or paused (machine state: %s)"),
2499 Global::stringifyMachineState(mMachineState));
2500
2501 /* protect mpVM */
2502 AutoVMCaller autoVMCaller(this);
2503 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
2504
2505 /* Don't proceed unless we've found the usb controller. */
2506 PPDMIBASE pBase = NULL;
2507 int vrc = PDMR3QueryLun(mpVM, "usb-ohci", 0, 0, &pBase);
2508 if (RT_FAILURE(vrc))
2509 return setError(VBOX_E_PDM_ERROR,
2510 tr("The virtual machine does not have a USB controller"));
2511
2512 /* leave the lock because the USB Proxy service may call us back
2513 * (via onUSBDeviceAttach()) */
2514 alock.leave();
2515
2516 /* Request the device capture */
2517 HRESULT rc = mControl->CaptureUSBDevice(aId);
2518 if (FAILED(rc)) return rc;
2519
2520 return rc;
2521
2522#else /* !VBOX_WITH_USB */
2523 return setError(VBOX_E_PDM_ERROR,
2524 tr("The virtual machine does not have a USB controller"));
2525#endif /* !VBOX_WITH_USB */
2526}
2527
2528STDMETHODIMP Console::DetachUSBDevice(IN_BSTR aId, IUSBDevice **aDevice)
2529{
2530#ifdef VBOX_WITH_USB
2531 CheckComArgOutPointerValid(aDevice);
2532
2533 AutoCaller autoCaller(this);
2534 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2535
2536 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2537
2538 /* Find it. */
2539 ComObjPtr<OUSBDevice> device;
2540 USBDeviceList::iterator it = mUSBDevices.begin();
2541 Guid uuid(aId);
2542 while (it != mUSBDevices.end())
2543 {
2544 if ((*it)->id() == uuid)
2545 {
2546 device = *it;
2547 break;
2548 }
2549 ++ it;
2550 }
2551
2552 if (!device)
2553 return setError(E_INVALIDARG,
2554 tr("USB device with UUID {%RTuuid} is not attached to this machine"),
2555 Guid(aId).raw());
2556
2557 /*
2558 * Inform the USB device and USB proxy about what's cooking.
2559 */
2560 alock.leave();
2561 HRESULT rc2 = mControl->DetachUSBDevice(aId, false /* aDone */);
2562 if (FAILED(rc2))
2563 return rc2;
2564 alock.enter();
2565
2566 /* Request the PDM to detach the USB device. */
2567 HRESULT rc = detachUSBDevice(it);
2568
2569 if (SUCCEEDED(rc))
2570 {
2571 /* leave the lock since we don't need it any more (note though that
2572 * the USB Proxy service must not call us back here) */
2573 alock.leave();
2574
2575 /* Request the device release. Even if it fails, the device will
2576 * remain as held by proxy, which is OK for us (the VM process). */
2577 rc = mControl->DetachUSBDevice(aId, true /* aDone */);
2578 }
2579
2580 return rc;
2581
2582
2583#else /* !VBOX_WITH_USB */
2584 return setError(VBOX_E_PDM_ERROR,
2585 tr("The virtual machine does not have a USB controller"));
2586#endif /* !VBOX_WITH_USB */
2587}
2588
2589STDMETHODIMP Console::FindUSBDeviceByAddress(IN_BSTR aAddress, IUSBDevice **aDevice)
2590{
2591#ifdef VBOX_WITH_USB
2592 CheckComArgStrNotEmptyOrNull(aAddress);
2593 CheckComArgOutPointerValid(aDevice);
2594
2595 *aDevice = NULL;
2596
2597 SafeIfaceArray<IUSBDevice> devsvec;
2598 HRESULT rc = COMGETTER(USBDevices)(ComSafeArrayAsOutParam(devsvec));
2599 if (FAILED(rc)) return rc;
2600
2601 for (size_t i = 0; i < devsvec.size(); ++i)
2602 {
2603 Bstr address;
2604 rc = devsvec[i]->COMGETTER(Address)(address.asOutParam());
2605 if (FAILED(rc)) return rc;
2606 if (address == aAddress)
2607 {
2608 ComObjPtr<OUSBDevice> found;
2609 found.createObject();
2610 found->init(devsvec[i]);
2611 return found.queryInterfaceTo(aDevice);
2612 }
2613 }
2614
2615 return setErrorNoLog(VBOX_E_OBJECT_NOT_FOUND,
2616 tr("Could not find a USB device with address '%ls'"),
2617 aAddress);
2618
2619#else /* !VBOX_WITH_USB */
2620 return E_NOTIMPL;
2621#endif /* !VBOX_WITH_USB */
2622}
2623
2624STDMETHODIMP Console::FindUSBDeviceById(IN_BSTR aId, IUSBDevice **aDevice)
2625{
2626#ifdef VBOX_WITH_USB
2627 CheckComArgExpr(aId, Guid(aId).isEmpty() == false);
2628 CheckComArgOutPointerValid(aDevice);
2629
2630 *aDevice = NULL;
2631
2632 SafeIfaceArray<IUSBDevice> devsvec;
2633 HRESULT rc = COMGETTER(USBDevices)(ComSafeArrayAsOutParam(devsvec));
2634 if (FAILED(rc)) return rc;
2635
2636 for (size_t i = 0; i < devsvec.size(); ++i)
2637 {
2638 Bstr id;
2639 rc = devsvec[i]->COMGETTER(Id)(id.asOutParam());
2640 if (FAILED(rc)) return rc;
2641 if (id == aId)
2642 {
2643 ComObjPtr<OUSBDevice> found;
2644 found.createObject();
2645 found->init(devsvec[i]);
2646 return found.queryInterfaceTo(aDevice);
2647 }
2648 }
2649
2650 return setErrorNoLog(VBOX_E_OBJECT_NOT_FOUND,
2651 tr("Could not find a USB device with uuid {%RTuuid}"),
2652 Guid(aId).raw());
2653
2654#else /* !VBOX_WITH_USB */
2655 return E_NOTIMPL;
2656#endif /* !VBOX_WITH_USB */
2657}
2658
2659STDMETHODIMP
2660Console::CreateSharedFolder(IN_BSTR aName, IN_BSTR aHostPath, BOOL aWritable)
2661{
2662 CheckComArgStrNotEmptyOrNull(aName);
2663 CheckComArgStrNotEmptyOrNull(aHostPath);
2664
2665 AutoCaller autoCaller(this);
2666 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2667
2668 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2669
2670 /// @todo see @todo in AttachUSBDevice() about the Paused state
2671 if (mMachineState == MachineState_Saved)
2672 return setError(VBOX_E_INVALID_VM_STATE,
2673 tr("Cannot create a transient shared folder on the machine in the saved state"));
2674 if ( mMachineState != MachineState_PoweredOff
2675 && mMachineState != MachineState_Teleported
2676 && mMachineState != MachineState_Aborted
2677 && mMachineState != MachineState_Running
2678 && mMachineState != MachineState_Paused
2679 )
2680 return setError(VBOX_E_INVALID_VM_STATE,
2681 tr("Cannot create a transient shared folder on the machine while it is changing the state (machine state: %s)"),
2682 Global::stringifyMachineState(mMachineState));
2683
2684 ComObjPtr<SharedFolder> sharedFolder;
2685 HRESULT rc = findSharedFolder(aName, sharedFolder, false /* aSetError */);
2686 if (SUCCEEDED(rc))
2687 return setError(VBOX_E_FILE_ERROR,
2688 tr("Shared folder named '%ls' already exists"),
2689 aName);
2690
2691 sharedFolder.createObject();
2692 rc = sharedFolder->init(this, aName, aHostPath, aWritable);
2693 if (FAILED(rc)) return rc;
2694
2695 /* protect mpVM (if not NULL) */
2696 AutoVMCallerQuietWeak autoVMCaller(this);
2697
2698 if (mpVM && autoVMCaller.isOk() && mVMMDev->isShFlActive())
2699 {
2700 /* If the VM is online and supports shared folders, share this folder
2701 * under the specified name. */
2702
2703 /* first, remove the machine or the global folder if there is any */
2704 SharedFolderDataMap::const_iterator it;
2705 if (findOtherSharedFolder(aName, it))
2706 {
2707 rc = removeSharedFolder(aName);
2708 if (FAILED(rc)) return rc;
2709 }
2710
2711 /* second, create the given folder */
2712 rc = createSharedFolder(aName, SharedFolderData(aHostPath, aWritable));
2713 if (FAILED(rc)) return rc;
2714 }
2715
2716 mSharedFolders.insert(std::make_pair(aName, sharedFolder));
2717
2718 /* notify console callbacks after the folder is added to the list */
2719 {
2720 CallbackList::iterator it = mCallbacks.begin();
2721 while (it != mCallbacks.end())
2722 (*it++)->OnSharedFolderChange(Scope_Session);
2723 }
2724
2725 return rc;
2726}
2727
2728STDMETHODIMP Console::RemoveSharedFolder(IN_BSTR aName)
2729{
2730 CheckComArgStrNotEmptyOrNull(aName);
2731
2732 AutoCaller autoCaller(this);
2733 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2734
2735 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2736
2737 /// @todo see @todo in AttachUSBDevice() about the Paused state
2738 if (mMachineState == MachineState_Saved)
2739 return setError(VBOX_E_INVALID_VM_STATE,
2740 tr("Cannot remove a transient shared folder from the machine in the saved state"));
2741 if ( mMachineState != MachineState_PoweredOff
2742 && mMachineState != MachineState_Teleported
2743 && mMachineState != MachineState_Aborted
2744 && mMachineState != MachineState_Running
2745 && mMachineState != MachineState_Paused
2746 )
2747 return setError(VBOX_E_INVALID_VM_STATE,
2748 tr("Cannot remove a transient shared folder from the machine while it is changing the state (machine state: %s)"),
2749 Global::stringifyMachineState(mMachineState));
2750
2751 ComObjPtr<SharedFolder> sharedFolder;
2752 HRESULT rc = findSharedFolder(aName, sharedFolder, true /* aSetError */);
2753 if (FAILED(rc)) return rc;
2754
2755 /* protect mpVM (if not NULL) */
2756 AutoVMCallerQuietWeak autoVMCaller(this);
2757
2758 if (mpVM && autoVMCaller.isOk() && mVMMDev->isShFlActive())
2759 {
2760 /* if the VM is online and supports shared folders, UNshare this
2761 * folder. */
2762
2763 /* first, remove the given folder */
2764 rc = removeSharedFolder(aName);
2765 if (FAILED(rc)) return rc;
2766
2767 /* first, remove the machine or the global folder if there is any */
2768 SharedFolderDataMap::const_iterator it;
2769 if (findOtherSharedFolder(aName, it))
2770 {
2771 rc = createSharedFolder(aName, it->second);
2772 /* don't check rc here because we need to remove the console
2773 * folder from the collection even on failure */
2774 }
2775 }
2776
2777 mSharedFolders.erase(aName);
2778
2779 /* notify console callbacks after the folder is removed to the list */
2780 {
2781 CallbackList::iterator it = mCallbacks.begin();
2782 while (it != mCallbacks.end())
2783 (*it++)->OnSharedFolderChange(Scope_Session);
2784 }
2785
2786 return rc;
2787}
2788
2789STDMETHODIMP Console::TakeSnapshot(IN_BSTR aName,
2790 IN_BSTR aDescription,
2791 IProgress **aProgress)
2792{
2793 LogFlowThisFuncEnter();
2794 LogFlowThisFunc(("aName='%ls' mMachineState=%08X\n", aName, mMachineState));
2795
2796 CheckComArgStrNotEmptyOrNull(aName);
2797 CheckComArgOutPointerValid(aProgress);
2798
2799 AutoCaller autoCaller(this);
2800 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2801
2802 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2803
2804 if (Global::IsTransient(mMachineState))
2805 return setError(VBOX_E_INVALID_VM_STATE,
2806 tr("Cannot take a snapshot of the machine while it is changing the state (machine state: %s)"),
2807 Global::stringifyMachineState(mMachineState));
2808
2809 HRESULT rc = S_OK;
2810
2811 /* prepare the progress object:
2812 a) count the no. of hard disk attachments to get a matching no. of progress sub-operations */
2813 ULONG cOperations = 2; // always at least setting up + finishing up
2814 ULONG ulTotalOperationsWeight = 2; // one each for setting up + finishing up
2815 SafeIfaceArray<IMediumAttachment> aMediumAttachments;
2816 rc = mMachine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(aMediumAttachments));
2817 if (FAILED(rc))
2818 return setError(rc, tr("Cannot get medium attachments of the machine"));
2819
2820 ULONG ulMemSize;
2821 rc = mMachine->COMGETTER(MemorySize)(&ulMemSize);
2822 if (FAILED(rc))
2823 return rc;
2824
2825 for (size_t i = 0;
2826 i < aMediumAttachments.size();
2827 ++i)
2828 {
2829 DeviceType_T type;
2830 rc = aMediumAttachments[i]->COMGETTER(Type)(&type);
2831 if (FAILED(rc))
2832 return rc;
2833
2834 if (type == DeviceType_HardDisk)
2835 {
2836 ++cOperations;
2837
2838 // assume that creating a diff image takes as long as saving a 1 MB state
2839 // (note, the same value must be used in SessionMachine::BeginTakingSnapshot() on the server!)
2840 ulTotalOperationsWeight += 1;
2841 }
2842 }
2843
2844 // b) one extra sub-operations for online snapshots OR offline snapshots that have a saved state (needs to be copied)
2845 bool fTakingSnapshotOnline = ((mMachineState == MachineState_Running) || (mMachineState == MachineState_Paused));
2846
2847 LogFlowFunc(("fTakingSnapshotOnline = %d, mMachineState = %d\n", fTakingSnapshotOnline, mMachineState));
2848
2849 if ( fTakingSnapshotOnline
2850 || mMachineState == MachineState_Saved
2851 )
2852 {
2853 ++cOperations;
2854
2855 ulTotalOperationsWeight += ulMemSize;
2856 }
2857
2858 // finally, create the progress object
2859 ComObjPtr<Progress> pProgress;
2860 pProgress.createObject();
2861 rc = pProgress->init(static_cast<IConsole*>(this),
2862 Bstr(tr("Taking a snapshot of the virtual machine")),
2863 mMachineState == MachineState_Running /* aCancelable */,
2864 cOperations,
2865 ulTotalOperationsWeight,
2866 Bstr(tr("Setting up snapshot operation")), // first sub-op description
2867 1); // ulFirstOperationWeight
2868
2869 if (FAILED(rc))
2870 return rc;
2871
2872 VMTakeSnapshotTask *pTask;
2873 if (!(pTask = new VMTakeSnapshotTask(this, pProgress, aName, aDescription)))
2874 return E_OUTOFMEMORY;
2875
2876 Assert(pTask->mProgress);
2877
2878 try
2879 {
2880 mptrCancelableProgress = pProgress;
2881
2882 /*
2883 * If we fail here it means a PowerDown() call happened on another
2884 * thread while we were doing Pause() (which leaves the Console lock).
2885 * We assign PowerDown() a higher precedence than TakeSnapshot(),
2886 * therefore just return the error to the caller.
2887 */
2888 rc = pTask->rc();
2889 if (FAILED(rc)) throw rc;
2890
2891 pTask->ulMemSize = ulMemSize;
2892
2893 /* memorize the current machine state */
2894 pTask->lastMachineState = mMachineState;
2895 pTask->fTakingSnapshotOnline = fTakingSnapshotOnline;
2896
2897 int vrc = RTThreadCreate(NULL,
2898 Console::fntTakeSnapshotWorker,
2899 (void*)pTask,
2900 0,
2901 RTTHREADTYPE_MAIN_WORKER,
2902 0,
2903 "ConsoleTakeSnap");
2904 if (FAILED(vrc))
2905 throw setError(E_FAIL,
2906 tr("Could not create VMTakeSnap thread (%Rrc)"),
2907 vrc);
2908
2909 pTask->mProgress.queryInterfaceTo(aProgress);
2910 }
2911 catch (HRESULT erc)
2912 {
2913 delete pTask;
2914 NOREF(erc);
2915 mptrCancelableProgress.setNull();
2916 }
2917
2918 LogFlowThisFunc(("rc=%Rhrc\n", rc));
2919 LogFlowThisFuncLeave();
2920 return rc;
2921}
2922
2923STDMETHODIMP Console::DeleteSnapshot(IN_BSTR aId, IProgress **aProgress)
2924{
2925 CheckComArgExpr(aId, Guid(aId).isEmpty() == false);
2926 CheckComArgOutPointerValid(aProgress);
2927
2928 AutoCaller autoCaller(this);
2929 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2930
2931 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2932
2933 if (Global::IsOnlineOrTransient(mMachineState))
2934 return setError(VBOX_E_INVALID_VM_STATE,
2935 tr("Cannot discard a snapshot of the running machine (machine state: %s)"),
2936 Global::stringifyMachineState(mMachineState));
2937
2938 MachineState_T machineState = MachineState_Null;
2939 HRESULT rc = mControl->DeleteSnapshot(this, aId, &machineState, aProgress);
2940 if (FAILED(rc)) return rc;
2941
2942 setMachineStateLocally(machineState);
2943 return S_OK;
2944}
2945
2946STDMETHODIMP Console::RestoreSnapshot(ISnapshot *aSnapshot, IProgress **aProgress)
2947{
2948 AutoCaller autoCaller(this);
2949 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2950
2951 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2952
2953 if (Global::IsOnlineOrTransient(mMachineState))
2954 return setError(VBOX_E_INVALID_VM_STATE,
2955 tr("Cannot discard the current state of the running machine (machine state: %s)"),
2956 Global::stringifyMachineState(mMachineState));
2957
2958 MachineState_T machineState = MachineState_Null;
2959 HRESULT rc = mControl->RestoreSnapshot(this, aSnapshot, &machineState, aProgress);
2960 if (FAILED(rc)) return rc;
2961
2962 setMachineStateLocally(machineState);
2963 return S_OK;
2964}
2965
2966STDMETHODIMP Console::RegisterCallback(IConsoleCallback *aCallback)
2967{
2968 CheckComArgNotNull(aCallback);
2969
2970 AutoCaller autoCaller(this);
2971 if (FAILED(autoCaller.rc())) return autoCaller.rc();
2972
2973#if 0 /** @todo r=bird,r=pritesh: must check that the interface id match correct or we might screw up with old code! */
2974 void *dummy;
2975 HRESULT hrc = aCallback->QueryInterface(NS_GET_IID(IConsoleCallback), &dummy);
2976 if (FAILED(hrc))
2977 return hrc;
2978 aCallback->Release();
2979#endif
2980
2981 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2982
2983 mCallbacks.push_back(CallbackList::value_type(aCallback));
2984
2985 /* Inform the callback about the current status (for example, the new
2986 * callback must know the current mouse capabilities and the pointer
2987 * shape in order to properly integrate the mouse pointer). */
2988
2989 if (mCallbackData.mpsc.valid)
2990 aCallback->OnMousePointerShapeChange(mCallbackData.mpsc.visible,
2991 mCallbackData.mpsc.alpha,
2992 mCallbackData.mpsc.xHot,
2993 mCallbackData.mpsc.yHot,
2994 mCallbackData.mpsc.width,
2995 mCallbackData.mpsc.height,
2996 mCallbackData.mpsc.shape);
2997 if (mCallbackData.mcc.valid)
2998 aCallback->OnMouseCapabilityChange(mCallbackData.mcc.supportsAbsolute,
2999 mCallbackData.mcc.supportsRelative,
3000 mCallbackData.mcc.needsHostCursor);
3001
3002 aCallback->OnAdditionsStateChange();
3003
3004 if (mCallbackData.klc.valid)
3005 aCallback->OnKeyboardLedsChange(mCallbackData.klc.numLock,
3006 mCallbackData.klc.capsLock,
3007 mCallbackData.klc.scrollLock);
3008
3009 /* Note: we don't call OnStateChange for new callbacks because the
3010 * machine state is a) not actually changed on callback registration
3011 * and b) can be always queried from Console. */
3012
3013 return S_OK;
3014}
3015
3016STDMETHODIMP Console::UnregisterCallback(IConsoleCallback *aCallback)
3017{
3018 CheckComArgNotNull(aCallback);
3019
3020 AutoCaller autoCaller(this);
3021 if (FAILED(autoCaller.rc())) return autoCaller.rc();
3022
3023 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3024
3025 CallbackList::iterator it;
3026 it = std::find(mCallbacks.begin(),
3027 mCallbacks.end(),
3028 CallbackList::value_type(aCallback));
3029 if (it == mCallbacks.end())
3030 return setError(E_INVALIDARG,
3031 tr("The given callback handler is not registered"));
3032
3033 mCallbacks.erase(it);
3034 return S_OK;
3035}
3036
3037// Non-interface public methods
3038/////////////////////////////////////////////////////////////////////////////
3039
3040/**
3041 * @copydoc VirtualBox::handleUnexpectedExceptions
3042 */
3043/* static */
3044HRESULT Console::handleUnexpectedExceptions(RT_SRC_POS_DECL)
3045{
3046 try
3047 {
3048 /* re-throw the current exception */
3049 throw;
3050 }
3051 catch (const std::exception &err)
3052 {
3053 return setError(E_FAIL, tr("Unexpected exception: %s [%s]\n%s[%d] (%s)"),
3054 err.what(), typeid(err).name(),
3055 pszFile, iLine, pszFunction);
3056 }
3057 catch (...)
3058 {
3059 return setError(E_FAIL, tr("Unknown exception\n%s[%d] (%s)"),
3060 pszFile, iLine, pszFunction);
3061 }
3062
3063 /* should not get here */
3064 AssertFailed();
3065 return E_FAIL;
3066}
3067
3068/* static */
3069const char *Console::convertControllerTypeToDev(StorageControllerType_T enmCtrlType)
3070{
3071 switch (enmCtrlType)
3072 {
3073 case StorageControllerType_LsiLogic:
3074 case StorageControllerType_LsiLogicSas:
3075 return "lsilogicscsi";
3076 case StorageControllerType_BusLogic:
3077 return "buslogic";
3078 case StorageControllerType_IntelAhci:
3079 return "ahci";
3080 case StorageControllerType_PIIX3:
3081 case StorageControllerType_PIIX4:
3082 case StorageControllerType_ICH6:
3083 return "piix3ide";
3084 case StorageControllerType_I82078:
3085 return "i82078";
3086 default:
3087 return NULL;
3088 }
3089}
3090
3091HRESULT Console::convertBusPortDeviceToLun(StorageBus_T enmBus, LONG port, LONG device, unsigned &uLun)
3092{
3093 switch (enmBus)
3094 {
3095 case StorageBus_IDE:
3096 case StorageBus_Floppy:
3097 {
3098 AssertMsgReturn(port < 2 && port >= 0, ("%d\n", port), E_INVALIDARG);
3099 AssertMsgReturn(device < 2 && device >= 0, ("%d\n", device), E_INVALIDARG);
3100 uLun = 2 * port + device;
3101 return S_OK;
3102 }
3103 case StorageBus_SATA:
3104 case StorageBus_SCSI:
3105 case StorageBus_SAS:
3106 {
3107 uLun = port;
3108 return S_OK;
3109 }
3110 default:
3111 uLun = 0;
3112 AssertMsgFailedReturn(("%d\n", enmBus), E_INVALIDARG);
3113 }
3114}
3115
3116// private methods
3117/////////////////////////////////////////////////////////////////////////////
3118
3119/**
3120 * Process a medium change.
3121 *
3122 * @param aMediumAttachment The medium attachment with the new medium state.
3123 * @param fForce Force medium chance, if it is locked or not.
3124 *
3125 * @note Locks this object for writing.
3126 */
3127HRESULT Console::doMediumChange(IMediumAttachment *aMediumAttachment, bool fForce)
3128{
3129 AutoCaller autoCaller(this);
3130 AssertComRCReturnRC(autoCaller.rc());
3131
3132 /* We will need to release the write lock before calling EMT */
3133 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3134
3135 HRESULT rc = S_OK;
3136 const char *pszDevice = NULL;
3137 unsigned uInstance = 0;
3138 unsigned uLun = 0;
3139 BOOL fHostDrive = FALSE;
3140 Utf8Str location;
3141 Utf8Str format;
3142 BOOL fPassthrough = FALSE;
3143
3144 SafeIfaceArray<IStorageController> ctrls;
3145 rc = mMachine->COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(ctrls));
3146 AssertComRC(rc);
3147 Bstr attCtrlName;
3148 rc = aMediumAttachment->COMGETTER(Controller)(attCtrlName.asOutParam());
3149 AssertComRC(rc);
3150 ComPtr<IStorageController> ctrl;
3151 for (size_t i = 0; i < ctrls.size(); ++i)
3152 {
3153 Bstr ctrlName;
3154 rc = ctrls[i]->COMGETTER(Name)(ctrlName.asOutParam());
3155 AssertComRC(rc);
3156 if (attCtrlName == ctrlName)
3157 {
3158 ctrl = ctrls[i];
3159 break;
3160 }
3161 }
3162 if (ctrl.isNull())
3163 {
3164 return setError(E_FAIL,
3165 tr("Could not find storage controller '%ls'"), attCtrlName.raw());
3166 }
3167 StorageControllerType_T enmCtrlType;
3168 rc = ctrl->COMGETTER(ControllerType)(&enmCtrlType);
3169 AssertComRC(rc);
3170 pszDevice = convertControllerTypeToDev(enmCtrlType);
3171
3172 /** @todo support multiple instances of a controller */
3173 uInstance = 0;
3174
3175 LONG device;
3176 rc = aMediumAttachment->COMGETTER(Device)(&device);
3177 AssertComRC(rc);
3178 LONG port;
3179 rc = aMediumAttachment->COMGETTER(Port)(&port);
3180 AssertComRC(rc);
3181 StorageBus_T enmBus;
3182 rc = ctrl->COMGETTER(Bus)(&enmBus);
3183 AssertComRC(rc);
3184 rc = convertBusPortDeviceToLun(enmBus, port, device, uLun);
3185 AssertComRCReturnRC(rc);
3186
3187 ComPtr<IMedium> medium;
3188 rc = aMediumAttachment->COMGETTER(Medium)(medium.asOutParam());
3189 if (SUCCEEDED(rc) && !medium.isNull())
3190 {
3191 Bstr loc;
3192 rc = medium->COMGETTER(Location)(loc.asOutParam());
3193 AssertComRC(rc);
3194 location = loc;
3195 Bstr fmt;
3196 rc = medium->COMGETTER(Format)(fmt.asOutParam());
3197 AssertComRC(rc);
3198 format = fmt;
3199 rc = medium->COMGETTER(HostDrive)(&fHostDrive);
3200 AssertComRC(rc);
3201 }
3202 rc = aMediumAttachment->COMGETTER(Passthrough)(&fPassthrough);
3203 AssertComRC(rc);
3204
3205 /* protect mpVM */
3206 AutoVMCaller autoVMCaller(this);
3207 AssertComRCReturnRC(autoVMCaller.rc());
3208
3209 /*
3210 * Call worker in EMT, that's faster and safer than doing everything
3211 * using VMR3ReqCall. Note that we separate VMR3ReqCall from VMR3ReqWait
3212 * here to make requests from under the lock in order to serialize them.
3213 */
3214 PVMREQ pReq;
3215 int vrc = VMR3ReqCall(mpVM, VMCPUID_ANY, &pReq, 0 /* no wait! */, VMREQFLAGS_VBOX_STATUS,
3216 (PFNRT)Console::changeDrive, 9,
3217 this, pszDevice, uInstance, uLun, !!fHostDrive, location.raw(), format.raw(), !!fPassthrough, fForce);
3218
3219 /* leave the lock before waiting for a result (EMT will call us back!) */
3220 alock.leave();
3221
3222 if (vrc == VERR_TIMEOUT || RT_SUCCESS(vrc))
3223 {
3224 vrc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
3225 AssertRC(vrc);
3226 if (RT_SUCCESS(vrc))
3227 vrc = pReq->iStatus;
3228 }
3229 VMR3ReqFree(pReq);
3230
3231 if (RT_SUCCESS(vrc))
3232 {
3233 LogFlowThisFunc(("Returns S_OK\n"));
3234 return S_OK;
3235 }
3236
3237 if (!location.isEmpty())
3238 return setError(E_FAIL,
3239 tr("Could not mount the media/drive '%s' (%Rrc)"),
3240 location.raw(), vrc);
3241
3242 return setError(E_FAIL,
3243 tr("Could not unmount the currently mounted media/drive (%Rrc)"),
3244 vrc);
3245}
3246
3247/**
3248 * Performs the medium change in EMT.
3249 *
3250 * @returns VBox status code.
3251 *
3252 * @param pThis Pointer to the Console object.
3253 * @param pszDevice The PDM device name.
3254 * @param uInstance The PDM device instance.
3255 * @param uLun The PDM LUN number of the drive.
3256 * @param fHostDrive True if this is a host drive attachment.
3257 * @param pszPath The path to the media / drive which is now being mounted / captured.
3258 * If NULL no media or drive is attached and the LUN will be configured with
3259 * the default block driver with no media. This will also be the state if
3260 * mounting / capturing the specified media / drive fails.
3261 * @param pszFormat Medium format string, usually "RAW".
3262 * @param fPassthrough Enables using passthrough mode of the host DVD drive if applicable.
3263 *
3264 * @thread EMT
3265 * @note Locks the Console object for writing.
3266 * @todo the error handling in this method needs to be improved seriously - what if mounting fails...
3267 */
3268DECLCALLBACK(int) Console::changeDrive(Console *pThis, const char *pszDevice, unsigned uInstance, unsigned uLun,
3269 bool fHostDrive, const char *pszPath, const char *pszFormat, bool fPassthrough, bool fForce)
3270{
3271/// @todo change this to use the same code as in ConsoleImpl2.cpp
3272 LogFlowFunc(("pThis=%p pszDevice=%p:{%s} uInstance=%u uLun=%u fHostDrive=%d pszPath=%p:{%s} pszFormat=%p:{%s} fPassthrough=%d fForce=%d\n",
3273 pThis, pszDevice, pszDevice, uInstance, uLun, fHostDrive, pszPath, pszPath, pszFormat, pszFormat, fPassthrough, fForce));
3274
3275 AssertReturn(pThis, VERR_INVALID_PARAMETER);
3276
3277 AutoCaller autoCaller(pThis);
3278 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
3279
3280 /* protect mpVM */
3281 AutoVMCaller autoVMCaller(pThis);
3282 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3283
3284 PVM pVM = pThis->mpVM;
3285
3286 /*
3287 * Suspend the VM first.
3288 *
3289 * The VM must not be running since it might have pending I/O to
3290 * the drive which is being changed.
3291 */
3292 bool fResume;
3293 VMSTATE enmVMState = VMR3GetState(pVM);
3294 switch (enmVMState)
3295 {
3296 case VMSTATE_RESETTING:
3297 case VMSTATE_RUNNING:
3298 {
3299 LogFlowFunc(("Suspending the VM...\n"));
3300 /* disable the callback to prevent Console-level state change */
3301 pThis->mVMStateChangeCallbackDisabled = true;
3302 int rc = VMR3Suspend(pVM);
3303 pThis->mVMStateChangeCallbackDisabled = false;
3304 AssertRCReturn(rc, rc);
3305 fResume = true;
3306 break;
3307 }
3308
3309 case VMSTATE_SUSPENDED:
3310 case VMSTATE_CREATED:
3311 case VMSTATE_OFF:
3312 fResume = false;
3313 break;
3314
3315 case VMSTATE_RUNNING_LS:
3316 return setError(VBOX_E_INVALID_VM_STATE, tr("Cannot change drive during live migration"));
3317
3318 default:
3319 AssertMsgFailedReturn(("enmVMState=%d\n", enmVMState), VERR_ACCESS_DENIED);
3320 }
3321
3322 int rc = VINF_SUCCESS;
3323 int rcRet = VINF_SUCCESS;
3324
3325 /*
3326 In general locking the object before doing VMR3* calls is quite safe
3327 here, since we're on EMT. Anyway we lock for write after eventually
3328 suspending the vm. The reason is that in the vmstateChangeCallback the
3329 var mVMStateChangeCallbackDisabled is checked under a lock also, which
3330 can lead to an dead lock. The write lock is necessary because we
3331 indirectly modify the meDVDState/meFloppyState members (pointed to by
3332 peState).
3333 */
3334 AutoWriteLock alock(pThis COMMA_LOCKVAL_SRC_POS);
3335
3336 do
3337 {
3338 /*
3339 * Unmount existing media / detach host drive.
3340 */
3341 PPDMIBASE pBase;
3342 rc = PDMR3QueryLun(pVM, pszDevice, uInstance, uLun, &pBase);
3343 if (RT_FAILURE(rc))
3344 {
3345 if (rc == VERR_PDM_LUN_NOT_FOUND || rc == VERR_PDM_NO_DRIVER_ATTACHED_TO_LUN)
3346 rc = VINF_SUCCESS;
3347 AssertRC(rc);
3348 }
3349 else
3350 {
3351 PPDMIMOUNT pIMount = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMOUNT);
3352 AssertBreakStmt(pIMount, rc = VERR_INVALID_POINTER);
3353
3354 /*
3355 * Unmount the media.
3356 */
3357 rc = pIMount->pfnUnmount(pIMount, fForce);
3358 if (rc == VERR_PDM_MEDIA_NOT_MOUNTED)
3359 rc = VINF_SUCCESS;
3360
3361 if (RT_SUCCESS(rc))
3362 {
3363 rc = PDMR3DeviceDetach(pVM, pszDevice, uInstance, uLun, PDM_TACH_FLAGS_NOT_HOT_PLUG);
3364 if (rc == VERR_PDM_NO_DRIVER_ATTACHED_TO_LUN)
3365 rc = VINF_SUCCESS;
3366 }
3367 }
3368
3369 if (RT_FAILURE(rc))
3370 {
3371 rcRet = rc;
3372 break;
3373 }
3374
3375 /** @todo this does a very thorough job. usually it's too much,
3376 * as a simple medium change (without changing between host attachment
3377 * and image) could be done with a lot less effort, by just using the
3378 * pfnUnmount and pfnMount interfaces. Later. */
3379
3380 /*
3381 * Construct a new driver configuration.
3382 */
3383 PCFGMNODE pInst = CFGMR3GetChildF(CFGMR3GetRoot(pVM), "Devices/%s/%d/", pszDevice, uInstance);
3384 AssertRelease(pInst);
3385 /* nuke anything which might have been left behind. */
3386 CFGMR3RemoveNode(CFGMR3GetChildF(pInst, "LUN#%d", uLun));
3387
3388#define RC_CHECK() do { if (RT_FAILURE(rc)) { AssertReleaseRC(rc); break; } } while (0)
3389
3390 PCFGMNODE pLunL0;
3391 PCFGMNODE pCfg;
3392 rc = CFGMR3InsertNodeF(pInst, &pLunL0, "LUN#%d", uLun); RC_CHECK();
3393
3394 if (fHostDrive)
3395 {
3396 rc = CFGMR3InsertString(pLunL0, "Driver", !strcmp(pszDevice, "i82078") ? "HostFloppy" : "HostDVD"); RC_CHECK();
3397 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
3398 Assert(pszPath && *pszPath);
3399 rc = CFGMR3InsertString(pCfg, "Path", pszPath); RC_CHECK();
3400 if (strcmp(pszDevice, "i82078"))
3401 {
3402 rc = CFGMR3InsertInteger(pCfg, "Passthrough", fPassthrough); RC_CHECK();
3403 }
3404 }
3405 else
3406 {
3407 /* create a new block driver config */
3408 rc = CFGMR3InsertString(pLunL0, "Driver", "Block"); RC_CHECK();
3409 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
3410 rc = CFGMR3InsertString(pCfg, "Type", !strcmp(pszDevice, "i82078") ? "Floppy 1.44" : "DVD"); RC_CHECK();
3411 rc = CFGMR3InsertInteger(pCfg, "Mountable", 1); RC_CHECK();
3412 }
3413
3414 /*
3415 * Attach the driver.
3416 */
3417 rc = PDMR3DeviceAttach(pVM, pszDevice, uInstance, uLun, PDM_TACH_FLAGS_NOT_HOT_PLUG, &pBase); RC_CHECK();
3418
3419 if (!fHostDrive && pszPath && *pszPath)
3420 {
3421 PCFGMNODE pLunL1;
3422 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
3423 rc = CFGMR3InsertString(pLunL1, "Driver", "VD"); RC_CHECK();
3424 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
3425 rc = CFGMR3InsertString(pCfg, "Path", pszPath); RC_CHECK();
3426 rc = CFGMR3InsertString(pCfg, "Format", pszFormat); RC_CHECK();
3427 if (strcmp(pszDevice, "i82078"))
3428 {
3429 rc = CFGMR3InsertInteger(pCfg, "ReadOnly", 1); RC_CHECK();
3430 }
3431 /** @todo later pass full VDConfig information and parent images */
3432 }
3433
3434 /* Dump the new controller configuration. */
3435 CFGMR3Dump(pInst);
3436
3437 if (!fHostDrive && pszPath && *pszPath)
3438 {
3439 PPDMIMOUNT pIMount = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMOUNT);
3440 if (!pIMount)
3441 {
3442 AssertFailed();
3443 return rc;
3444 }
3445
3446 rc = pIMount->pfnMount(pIMount, NULL , NULL);
3447 }
3448
3449#undef RC_CHECK
3450
3451 if (RT_FAILURE(rc) && RT_SUCCESS(rcRet))
3452 rcRet = rc;
3453
3454 }
3455 while (0);
3456
3457 /*
3458 Unlock before resuming because the vmstateChangeCallback problem
3459 described above.
3460 */
3461 alock.release();
3462
3463 /*
3464 * Resume the VM if necessary.
3465 */
3466 if (fResume)
3467 {
3468 LogFlowFunc(("Resuming the VM...\n"));
3469 /* disable the callback to prevent Console-level state change */
3470 pThis->mVMStateChangeCallbackDisabled = true;
3471 rc = VMR3Resume(pVM);
3472 pThis->mVMStateChangeCallbackDisabled = false;
3473 AssertRC(rc);
3474 if (RT_FAILURE(rc))
3475 {
3476 /* too bad, we failed. try to sync the console state with the VMM state */
3477 vmstateChangeCallback(pVM, VMSTATE_SUSPENDED, enmVMState, pThis);
3478 }
3479 /// @todo (r=dmik) if we failed with drive mount, then the VMR3Resume
3480 // error (if any) will be hidden from the caller. For proper reporting
3481 // of such multiple errors to the caller we need to enhance the
3482 // IVirtualBoxError interface. For now, give the first error the higher
3483 // priority.
3484 if (RT_SUCCESS(rcRet))
3485 rcRet = rc;
3486 }
3487
3488 LogFlowFunc(("Returning %Rrc\n", rcRet));
3489 return rcRet;
3490}
3491
3492
3493/**
3494 * Called by IInternalSessionControl::OnNetworkAdapterChange().
3495 *
3496 * @note Locks this object for writing.
3497 */
3498HRESULT Console::onNetworkAdapterChange(INetworkAdapter *aNetworkAdapter, BOOL changeAdapter)
3499{
3500 LogFlowThisFunc(("\n"));
3501
3502 AutoCaller autoCaller(this);
3503 AssertComRCReturnRC(autoCaller.rc());
3504
3505 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3506
3507 /* Don't do anything if the VM isn't running */
3508 if (!mpVM)
3509 return S_OK;
3510
3511 /* protect mpVM */
3512 AutoVMCaller autoVMCaller(this);
3513 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3514
3515 /* Get the properties we need from the adapter */
3516 BOOL fCableConnected, fTraceEnabled;
3517 HRESULT rc = aNetworkAdapter->COMGETTER(CableConnected)(&fCableConnected);
3518 AssertComRC(rc);
3519 if (SUCCEEDED(rc))
3520 {
3521 rc = aNetworkAdapter->COMGETTER(TraceEnabled)(&fTraceEnabled);
3522 AssertComRC(rc);
3523 }
3524 if (SUCCEEDED(rc))
3525 {
3526 ULONG ulInstance;
3527 rc = aNetworkAdapter->COMGETTER(Slot)(&ulInstance);
3528 AssertComRC(rc);
3529 if (SUCCEEDED(rc))
3530 {
3531 /*
3532 * Find the pcnet instance, get the config interface and update
3533 * the link state.
3534 */
3535 NetworkAdapterType_T adapterType;
3536 rc = aNetworkAdapter->COMGETTER(AdapterType)(&adapterType);
3537 AssertComRC(rc);
3538 const char *pszAdapterName = NULL;
3539 switch (adapterType)
3540 {
3541 case NetworkAdapterType_Am79C970A:
3542 case NetworkAdapterType_Am79C973:
3543 pszAdapterName = "pcnet";
3544 break;
3545#ifdef VBOX_WITH_E1000
3546 case NetworkAdapterType_I82540EM:
3547 case NetworkAdapterType_I82543GC:
3548 case NetworkAdapterType_I82545EM:
3549 pszAdapterName = "e1000";
3550 break;
3551#endif
3552#ifdef VBOX_WITH_VIRTIO
3553 case NetworkAdapterType_Virtio:
3554 pszAdapterName = "virtio-net";
3555 break;
3556#endif
3557 default:
3558 AssertFailed();
3559 pszAdapterName = "unknown";
3560 break;
3561 }
3562
3563 PPDMIBASE pBase;
3564 int vrc = PDMR3QueryDeviceLun(mpVM, pszAdapterName, ulInstance, 0, &pBase);
3565 ComAssertRC(vrc);
3566 if (RT_SUCCESS(vrc))
3567 {
3568 Assert(pBase);
3569 PPDMINETWORKCONFIG pINetCfg;
3570 pINetCfg = PDMIBASE_QUERY_INTERFACE(pBase, PDMINETWORKCONFIG);
3571 if (pINetCfg)
3572 {
3573 Log(("Console::onNetworkAdapterChange: setting link state to %d\n",
3574 fCableConnected));
3575 vrc = pINetCfg->pfnSetLinkState(pINetCfg,
3576 fCableConnected ? PDMNETWORKLINKSTATE_UP
3577 : PDMNETWORKLINKSTATE_DOWN);
3578 ComAssertRC(vrc);
3579 }
3580#ifdef VBOX_DYNAMIC_NET_ATTACH
3581 if (RT_SUCCESS(vrc) && changeAdapter)
3582 {
3583 VMSTATE enmVMState = VMR3GetState(mpVM);
3584 if ( enmVMState == VMSTATE_RUNNING /** @todo LiveMigration: Forbit or deal correctly with the _LS variants */
3585 || enmVMState == VMSTATE_SUSPENDED)
3586 {
3587 if (fTraceEnabled && fCableConnected && pINetCfg)
3588 {
3589 vrc = pINetCfg->pfnSetLinkState(pINetCfg, PDMNETWORKLINKSTATE_DOWN);
3590 ComAssertRC(vrc);
3591 }
3592
3593 rc = doNetworkAdapterChange(pszAdapterName, ulInstance, 0, aNetworkAdapter);
3594
3595 if (fTraceEnabled && fCableConnected && pINetCfg)
3596 {
3597 vrc = pINetCfg->pfnSetLinkState(pINetCfg, PDMNETWORKLINKSTATE_UP);
3598 ComAssertRC(vrc);
3599 }
3600 }
3601 }
3602#endif /* VBOX_DYNAMIC_NET_ATTACH */
3603 }
3604
3605 if (RT_FAILURE(vrc))
3606 rc = E_FAIL;
3607 }
3608 }
3609
3610 /* notify console callbacks on success */
3611 if (SUCCEEDED(rc))
3612 {
3613 CallbackList::iterator it = mCallbacks.begin();
3614 while (it != mCallbacks.end())
3615 (*it++)->OnNetworkAdapterChange(aNetworkAdapter);
3616 }
3617
3618 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3619 return rc;
3620}
3621
3622
3623#ifdef VBOX_DYNAMIC_NET_ATTACH
3624/**
3625 * Process a network adaptor change.
3626 *
3627 * @returns COM status code.
3628 *
3629 * @param pszDevice The PDM device name.
3630 * @param uInstance The PDM device instance.
3631 * @param uLun The PDM LUN number of the drive.
3632 * @param aNetworkAdapter The network adapter whose attachment needs to be changed
3633 *
3634 * @note Locks this object for writing.
3635 */
3636HRESULT Console::doNetworkAdapterChange(const char *pszDevice,
3637 unsigned uInstance,
3638 unsigned uLun,
3639 INetworkAdapter *aNetworkAdapter)
3640{
3641 LogFlowThisFunc(("pszDevice=%p:{%s} uInstance=%u uLun=%u aNetworkAdapter=%p\n",
3642 pszDevice, pszDevice, uInstance, uLun, aNetworkAdapter));
3643
3644 AutoCaller autoCaller(this);
3645 AssertComRCReturnRC(autoCaller.rc());
3646
3647 /* We will need to release the write lock before calling EMT */
3648 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3649
3650 /* protect mpVM */
3651 AutoVMCaller autoVMCaller(this);
3652 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3653
3654 /*
3655 * Call worker in EMT, that's faster and safer than doing everything
3656 * using VM3ReqCall. Note that we separate VMR3ReqCall from VMR3ReqWait
3657 * here to make requests from under the lock in order to serialize them.
3658 */
3659 PVMREQ pReq;
3660 int vrc = VMR3ReqCall(mpVM, 0 /*idDstCpu*/, &pReq, 0 /* no wait! */, VMREQFLAGS_VBOX_STATUS,
3661 (PFNRT) Console::changeNetworkAttachment, 5,
3662 this, pszDevice, uInstance, uLun, aNetworkAdapter);
3663
3664 /* leave the lock before waiting for a result (EMT will call us back!) */
3665 alock.leave();
3666
3667 if (vrc == VERR_TIMEOUT || RT_SUCCESS(vrc))
3668 {
3669 vrc = VMR3ReqWait(pReq, RT_INDEFINITE_WAIT);
3670 AssertRC(vrc);
3671 if (RT_SUCCESS(vrc))
3672 vrc = pReq->iStatus;
3673 }
3674 VMR3ReqFree(pReq);
3675
3676 if (RT_SUCCESS(vrc))
3677 {
3678 LogFlowThisFunc(("Returns S_OK\n"));
3679 return S_OK;
3680 }
3681
3682 return setError(E_FAIL,
3683 tr("Could not change the network adaptor attachement type (%Rrc)"),
3684 vrc);
3685}
3686
3687
3688/**
3689 * Performs the Network Adaptor change in EMT.
3690 *
3691 * @returns VBox status code.
3692 *
3693 * @param pThis Pointer to the Console object.
3694 * @param pszDevice The PDM device name.
3695 * @param uInstance The PDM device instance.
3696 * @param uLun The PDM LUN number of the drive.
3697 * @param aNetworkAdapter The network adapter whose attachment needs to be changed
3698 *
3699 * @thread EMT
3700 * @note Locks the Console object for writing.
3701 */
3702DECLCALLBACK(int) Console::changeNetworkAttachment(Console *pThis,
3703 const char *pszDevice,
3704 unsigned uInstance,
3705 unsigned uLun,
3706 INetworkAdapter *aNetworkAdapter)
3707{
3708 LogFlowFunc(("pThis=%p pszDevice=%p:{%s} uInstance=%u uLun=%u aNetworkAdapter=%p\n",
3709 pThis, pszDevice, pszDevice, uInstance, uLun, aNetworkAdapter));
3710
3711 AssertReturn(pThis, VERR_INVALID_PARAMETER);
3712
3713 AssertMsg( ( !strcmp(pszDevice, "pcnet")
3714 || !strcmp(pszDevice, "e1000")
3715 || !strcmp(pszDevice, "virtio-net"))
3716 && (uLun == 0)
3717 && (uInstance < SchemaDefs::NetworkAdapterCount),
3718 ("pszDevice=%s uLun=%d uInstance=%d\n", pszDevice, uLun, uInstance));
3719 Log(("pszDevice=%s uLun=%d uInstance=%d\n", pszDevice, uLun, uInstance));
3720
3721 AutoCaller autoCaller(pThis);
3722 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
3723
3724 /* protect mpVM */
3725 AutoVMCaller autoVMCaller(pThis);
3726 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3727
3728 PVM pVM = pThis->mpVM;
3729
3730 /*
3731 * Suspend the VM first.
3732 *
3733 * The VM must not be running since it might have pending I/O to
3734 * the drive which is being changed.
3735 */
3736 bool fResume;
3737 VMSTATE enmVMState = VMR3GetState(pVM);
3738 switch (enmVMState)
3739 {
3740 case VMSTATE_RESETTING:
3741 case VMSTATE_RUNNING:
3742 {
3743 LogFlowFunc(("Suspending the VM...\n"));
3744 /* disable the callback to prevent Console-level state change */
3745 pThis->mVMStateChangeCallbackDisabled = true;
3746 int rc = VMR3Suspend(pVM);
3747 pThis->mVMStateChangeCallbackDisabled = false;
3748 AssertRCReturn(rc, rc);
3749 fResume = true;
3750 break;
3751 }
3752
3753 case VMSTATE_SUSPENDED:
3754 case VMSTATE_CREATED:
3755 case VMSTATE_OFF:
3756 fResume = false;
3757 break;
3758
3759 default:
3760 AssertLogRelMsgFailedReturn(("enmVMState=%d\n", enmVMState), VERR_ACCESS_DENIED);
3761 }
3762
3763 int rc = VINF_SUCCESS;
3764 int rcRet = VINF_SUCCESS;
3765
3766 PCFGMNODE pCfg = NULL; /* /Devices/Dev/.../Config/ */
3767 PCFGMNODE pLunL0 = NULL; /* /Devices/Dev/0/LUN#0/ */
3768 PCFGMNODE pInst = CFGMR3GetChildF(CFGMR3GetRoot(pVM), "Devices/%s/%d/", pszDevice, uInstance);
3769 AssertRelease(pInst);
3770
3771 rcRet = configNetwork(pThis, pszDevice, uInstance, uLun, aNetworkAdapter, pCfg, pLunL0, pInst, true);
3772
3773 /*
3774 * Resume the VM if necessary.
3775 */
3776 if (fResume)
3777 {
3778 LogFlowFunc(("Resuming the VM...\n"));
3779 /* disable the callback to prevent Console-level state change */
3780 pThis->mVMStateChangeCallbackDisabled = true;
3781 rc = VMR3Resume(pVM);
3782 pThis->mVMStateChangeCallbackDisabled = false;
3783 AssertRC(rc);
3784 if (RT_FAILURE(rc))
3785 {
3786 /* too bad, we failed. try to sync the console state with the VMM state */
3787 vmstateChangeCallback(pVM, VMSTATE_SUSPENDED, enmVMState, pThis);
3788 }
3789 /// @todo (r=dmik) if we failed with drive mount, then the VMR3Resume
3790 // error (if any) will be hidden from the caller. For proper reporting
3791 // of such multiple errors to the caller we need to enhance the
3792 // IVirtualBoxError interface. For now, give the first error the higher
3793 // priority.
3794 if (RT_SUCCESS(rcRet))
3795 rcRet = rc;
3796 }
3797
3798 LogFlowFunc(("Returning %Rrc\n", rcRet));
3799 return rcRet;
3800}
3801#endif /* VBOX_DYNAMIC_NET_ATTACH */
3802
3803
3804/**
3805 * Called by IInternalSessionControl::OnSerialPortChange().
3806 *
3807 * @note Locks this object for writing.
3808 */
3809HRESULT Console::onSerialPortChange(ISerialPort *aSerialPort)
3810{
3811 LogFlowThisFunc(("\n"));
3812
3813 AutoCaller autoCaller(this);
3814 AssertComRCReturnRC(autoCaller.rc());
3815
3816 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3817
3818 /* Don't do anything if the VM isn't running */
3819 if (!mpVM)
3820 return S_OK;
3821
3822 HRESULT rc = S_OK;
3823
3824 /* protect mpVM */
3825 AutoVMCaller autoVMCaller(this);
3826 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3827
3828 /* nothing to do so far */
3829
3830 /* notify console callbacks on success */
3831 if (SUCCEEDED(rc))
3832 {
3833 CallbackList::iterator it = mCallbacks.begin();
3834 while (it != mCallbacks.end())
3835 (*it++)->OnSerialPortChange(aSerialPort);
3836 }
3837
3838 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3839 return rc;
3840}
3841
3842/**
3843 * Called by IInternalSessionControl::OnParallelPortChange().
3844 *
3845 * @note Locks this object for writing.
3846 */
3847HRESULT Console::onParallelPortChange(IParallelPort *aParallelPort)
3848{
3849 LogFlowThisFunc(("\n"));
3850
3851 AutoCaller autoCaller(this);
3852 AssertComRCReturnRC(autoCaller.rc());
3853
3854 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3855
3856 /* Don't do anything if the VM isn't running */
3857 if (!mpVM)
3858 return S_OK;
3859
3860 HRESULT rc = S_OK;
3861
3862 /* protect mpVM */
3863 AutoVMCaller autoVMCaller(this);
3864 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3865
3866 /* nothing to do so far */
3867
3868 /* notify console callbacks on success */
3869 if (SUCCEEDED(rc))
3870 {
3871 CallbackList::iterator it = mCallbacks.begin();
3872 while (it != mCallbacks.end())
3873 (*it++)->OnParallelPortChange(aParallelPort);
3874 }
3875
3876 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3877 return rc;
3878}
3879
3880/**
3881 * Called by IInternalSessionControl::OnStorageControllerChange().
3882 *
3883 * @note Locks this object for writing.
3884 */
3885HRESULT Console::onStorageControllerChange()
3886{
3887 LogFlowThisFunc(("\n"));
3888
3889 AutoCaller autoCaller(this);
3890 AssertComRCReturnRC(autoCaller.rc());
3891
3892 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3893
3894 /* Don't do anything if the VM isn't running */
3895 if (!mpVM)
3896 return S_OK;
3897
3898 HRESULT rc = S_OK;
3899
3900 /* protect mpVM */
3901 AutoVMCaller autoVMCaller(this);
3902 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3903
3904 /* nothing to do so far */
3905
3906 /* notify console callbacks on success */
3907 if (SUCCEEDED(rc))
3908 {
3909 CallbackList::iterator it = mCallbacks.begin();
3910 while (it != mCallbacks.end())
3911 (*it++)->OnStorageControllerChange();
3912 }
3913
3914 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3915 return rc;
3916}
3917
3918/**
3919 * Called by IInternalSessionControl::OnMediumChange().
3920 *
3921 * @note Locks this object for writing.
3922 */
3923HRESULT Console::onMediumChange(IMediumAttachment *aMediumAttachment, BOOL aForce)
3924{
3925 LogFlowThisFunc(("\n"));
3926
3927 AutoCaller autoCaller(this);
3928 AssertComRCReturnRC(autoCaller.rc());
3929
3930 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3931
3932 /* Don't do anything if the VM isn't running */
3933 if (!mpVM)
3934 return S_OK;
3935
3936 HRESULT rc = S_OK;
3937
3938 /* protect mpVM */
3939 AutoVMCaller autoVMCaller(this);
3940 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3941
3942 rc = doMediumChange(aMediumAttachment, !!aForce);
3943
3944 /* notify console callbacks on success */
3945 if (SUCCEEDED(rc))
3946 {
3947 CallbackList::iterator it = mCallbacks.begin();
3948 while (it != mCallbacks.end())
3949 (*it++)->OnMediumChange(aMediumAttachment);
3950 }
3951
3952 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3953 return rc;
3954}
3955
3956/**
3957 * Called by IInternalSessionControl::OnCPUChange().
3958 *
3959 * @note Locks this object for writing.
3960 */
3961HRESULT Console::onCPUChange(ULONG aCPU, BOOL aRemove)
3962{
3963 LogFlowThisFunc(("\n"));
3964
3965 AutoCaller autoCaller(this);
3966 AssertComRCReturnRC(autoCaller.rc());
3967
3968 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
3969
3970 /* Don't do anything if the VM isn't running */
3971 if (!mpVM)
3972 return S_OK;
3973
3974 HRESULT rc = S_OK;
3975
3976 /* protect mpVM */
3977 AutoVMCaller autoVMCaller(this);
3978 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
3979
3980 if (aRemove)
3981 rc = doCPURemove(aCPU);
3982 else
3983 rc = doCPUAdd(aCPU);
3984
3985 /* notify console callbacks on success */
3986 if (SUCCEEDED(rc))
3987 {
3988 CallbackList::iterator it = mCallbacks.begin();
3989 while (it != mCallbacks.end())
3990 (*it++)->OnCPUChange(aCPU, aRemove);
3991 }
3992
3993 LogFlowThisFunc(("Leaving rc=%#x\n", rc));
3994 return rc;
3995}
3996
3997/**
3998 * Called by IInternalSessionControl::OnVRDPServerChange().
3999 *
4000 * @note Locks this object for writing.
4001 */
4002HRESULT Console::onVRDPServerChange()
4003{
4004 AutoCaller autoCaller(this);
4005 AssertComRCReturnRC(autoCaller.rc());
4006
4007 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4008
4009 HRESULT rc = S_OK;
4010
4011 if ( mVRDPServer
4012 && ( mMachineState == MachineState_Running
4013 || mMachineState == MachineState_Teleporting
4014 || mMachineState == MachineState_LiveSnapshotting
4015 )
4016 )
4017 {
4018 BOOL vrdpEnabled = FALSE;
4019
4020 rc = mVRDPServer->COMGETTER(Enabled)(&vrdpEnabled);
4021 ComAssertComRCRetRC(rc);
4022
4023 /* VRDP server may call this Console object back from other threads (VRDP INPUT or OUTPUT). */
4024 alock.leave();
4025
4026 if (vrdpEnabled)
4027 {
4028 // If there was no VRDP server started the 'stop' will do nothing.
4029 // However if a server was started and this notification was called,
4030 // we have to restart the server.
4031 mConsoleVRDPServer->Stop();
4032
4033 if (RT_FAILURE(mConsoleVRDPServer->Launch()))
4034 {
4035 rc = E_FAIL;
4036 }
4037 else
4038 {
4039 mConsoleVRDPServer->EnableConnections();
4040 }
4041 }
4042 else
4043 {
4044 mConsoleVRDPServer->Stop();
4045 }
4046
4047 alock.enter();
4048 }
4049
4050 /* notify console callbacks on success */
4051 if (SUCCEEDED(rc))
4052 {
4053 CallbackList::iterator it = mCallbacks.begin();
4054 while (it != mCallbacks.end())
4055 (*it++)->OnVRDPServerChange();
4056 }
4057
4058 return rc;
4059}
4060
4061/**
4062 * @note Locks this object for reading.
4063 */
4064void Console::onRemoteDisplayInfoChange()
4065{
4066 AutoCaller autoCaller(this);
4067 AssertComRCReturnVoid(autoCaller.rc());
4068
4069 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4070
4071 CallbackList::iterator it = mCallbacks.begin();
4072 while (it != mCallbacks.end())
4073 (*it++)->OnRemoteDisplayInfoChange();
4074}
4075
4076
4077
4078/**
4079 * Called by IInternalSessionControl::OnUSBControllerChange().
4080 *
4081 * @note Locks this object for writing.
4082 */
4083HRESULT Console::onUSBControllerChange()
4084{
4085 LogFlowThisFunc(("\n"));
4086
4087 AutoCaller autoCaller(this);
4088 AssertComRCReturnRC(autoCaller.rc());
4089
4090 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4091
4092 /* Ignore if no VM is running yet. */
4093 if (!mpVM)
4094 return S_OK;
4095
4096 HRESULT rc = S_OK;
4097
4098/// @todo (dmik)
4099// check for the Enabled state and disable virtual USB controller??
4100// Anyway, if we want to query the machine's USB Controller we need to cache
4101// it to mUSBController in #init() (as it is done with mDVDDrive).
4102//
4103// bird: While the VM supports hot-plugging, I doubt any guest can handle it at this time... :-)
4104//
4105// /* protect mpVM */
4106// AutoVMCaller autoVMCaller(this);
4107// if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
4108
4109 /* notify console callbacks on success */
4110 if (SUCCEEDED(rc))
4111 {
4112 CallbackList::iterator it = mCallbacks.begin();
4113 while (it != mCallbacks.end())
4114 (*it++)->OnUSBControllerChange();
4115 }
4116
4117 return rc;
4118}
4119
4120/**
4121 * Called by IInternalSessionControl::OnSharedFolderChange().
4122 *
4123 * @note Locks this object for writing.
4124 */
4125HRESULT Console::onSharedFolderChange(BOOL aGlobal)
4126{
4127 LogFlowThisFunc(("aGlobal=%RTbool\n", aGlobal));
4128
4129 AutoCaller autoCaller(this);
4130 AssertComRCReturnRC(autoCaller.rc());
4131
4132 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4133
4134 HRESULT rc = fetchSharedFolders(aGlobal);
4135
4136 /* notify console callbacks on success */
4137 if (SUCCEEDED(rc))
4138 {
4139 CallbackList::iterator it = mCallbacks.begin();
4140 while (it != mCallbacks.end())
4141 (*it++)->OnSharedFolderChange(aGlobal ? (Scope_T)Scope_Global
4142 : (Scope_T)Scope_Machine);
4143 }
4144
4145 return rc;
4146}
4147
4148/**
4149 * Called by IInternalSessionControl::OnUSBDeviceAttach() or locally by
4150 * processRemoteUSBDevices() after IInternalMachineControl::RunUSBDeviceFilters()
4151 * returns TRUE for a given remote USB device.
4152 *
4153 * @return S_OK if the device was attached to the VM.
4154 * @return failure if not attached.
4155 *
4156 * @param aDevice
4157 * The device in question.
4158 * @param aMaskedIfs
4159 * The interfaces to hide from the guest.
4160 *
4161 * @note Locks this object for writing.
4162 */
4163HRESULT Console::onUSBDeviceAttach(IUSBDevice *aDevice, IVirtualBoxErrorInfo *aError, ULONG aMaskedIfs)
4164{
4165#ifdef VBOX_WITH_USB
4166 LogFlowThisFunc(("aDevice=%p aError=%p\n", aDevice, aError));
4167
4168 AutoCaller autoCaller(this);
4169 ComAssertComRCRetRC(autoCaller.rc());
4170
4171 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4172
4173 /* protect mpVM (we don't need error info, since it's a callback) */
4174 AutoVMCallerQuiet autoVMCaller(this);
4175 if (FAILED(autoVMCaller.rc()))
4176 {
4177 /* The VM may be no more operational when this message arrives
4178 * (e.g. it may be Saving or Stopping or just PoweredOff) --
4179 * autoVMCaller.rc() will return a failure in this case. */
4180 LogFlowThisFunc(("Attach request ignored (mMachineState=%d).\n",
4181 mMachineState));
4182 return autoVMCaller.rc();
4183 }
4184
4185 if (aError != NULL)
4186 {
4187 /* notify callbacks about the error */
4188 onUSBDeviceStateChange(aDevice, true /* aAttached */, aError);
4189 return S_OK;
4190 }
4191
4192 /* Don't proceed unless there's at least one USB hub. */
4193 if (!PDMR3USBHasHub(mpVM))
4194 {
4195 LogFlowThisFunc(("Attach request ignored (no USB controller).\n"));
4196 return E_FAIL;
4197 }
4198
4199 HRESULT rc = attachUSBDevice(aDevice, aMaskedIfs);
4200 if (FAILED(rc))
4201 {
4202 /* take the current error info */
4203 com::ErrorInfoKeeper eik;
4204 /* the error must be a VirtualBoxErrorInfo instance */
4205 ComPtr<IVirtualBoxErrorInfo> error = eik.takeError();
4206 Assert(!error.isNull());
4207 if (!error.isNull())
4208 {
4209 /* notify callbacks about the error */
4210 onUSBDeviceStateChange(aDevice, true /* aAttached */, error);
4211 }
4212 }
4213
4214 return rc;
4215
4216#else /* !VBOX_WITH_USB */
4217 return E_FAIL;
4218#endif /* !VBOX_WITH_USB */
4219}
4220
4221/**
4222 * Called by IInternalSessionControl::OnUSBDeviceDetach() and locally by
4223 * processRemoteUSBDevices().
4224 *
4225 * @note Locks this object for writing.
4226 */
4227HRESULT Console::onUSBDeviceDetach(IN_BSTR aId,
4228 IVirtualBoxErrorInfo *aError)
4229{
4230#ifdef VBOX_WITH_USB
4231 Guid Uuid(aId);
4232 LogFlowThisFunc(("aId={%RTuuid} aError=%p\n", Uuid.raw(), aError));
4233
4234 AutoCaller autoCaller(this);
4235 AssertComRCReturnRC(autoCaller.rc());
4236
4237 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4238
4239 /* Find the device. */
4240 ComObjPtr<OUSBDevice> device;
4241 USBDeviceList::iterator it = mUSBDevices.begin();
4242 while (it != mUSBDevices.end())
4243 {
4244 LogFlowThisFunc(("it={%RTuuid}\n", (*it)->id().raw()));
4245 if ((*it)->id() == Uuid)
4246 {
4247 device = *it;
4248 break;
4249 }
4250 ++ it;
4251 }
4252
4253
4254 if (device.isNull())
4255 {
4256 LogFlowThisFunc(("USB device not found.\n"));
4257
4258 /* The VM may be no more operational when this message arrives
4259 * (e.g. it may be Saving or Stopping or just PoweredOff). Use
4260 * AutoVMCaller to detect it -- AutoVMCaller::rc() will return a
4261 * failure in this case. */
4262
4263 AutoVMCallerQuiet autoVMCaller(this);
4264 if (FAILED(autoVMCaller.rc()))
4265 {
4266 LogFlowThisFunc(("Detach request ignored (mMachineState=%d).\n",
4267 mMachineState));
4268 return autoVMCaller.rc();
4269 }
4270
4271 /* the device must be in the list otherwise */
4272 AssertFailedReturn(E_FAIL);
4273 }
4274
4275 if (aError != NULL)
4276 {
4277 /* notify callback about an error */
4278 onUSBDeviceStateChange(device, false /* aAttached */, aError);
4279 return S_OK;
4280 }
4281
4282 HRESULT rc = detachUSBDevice(it);
4283
4284 if (FAILED(rc))
4285 {
4286 /* take the current error info */
4287 com::ErrorInfoKeeper eik;
4288 /* the error must be a VirtualBoxErrorInfo instance */
4289 ComPtr<IVirtualBoxErrorInfo> error = eik.takeError();
4290 Assert(!error.isNull());
4291 if (!error.isNull())
4292 {
4293 /* notify callbacks about the error */
4294 onUSBDeviceStateChange(device, false /* aAttached */, error);
4295 }
4296 }
4297
4298 return rc;
4299
4300#else /* !VBOX_WITH_USB */
4301 return E_FAIL;
4302#endif /* !VBOX_WITH_USB */
4303}
4304
4305/**
4306 * @note Temporarily locks this object for writing.
4307 */
4308HRESULT Console::getGuestProperty(IN_BSTR aName, BSTR *aValue,
4309 ULONG64 *aTimestamp, BSTR *aFlags)
4310{
4311#ifndef VBOX_WITH_GUEST_PROPS
4312 ReturnComNotImplemented();
4313#else /* VBOX_WITH_GUEST_PROPS */
4314 if (!VALID_PTR(aName))
4315 return E_INVALIDARG;
4316 if (!VALID_PTR(aValue))
4317 return E_POINTER;
4318 if ((aTimestamp != NULL) && !VALID_PTR(aTimestamp))
4319 return E_POINTER;
4320 if ((aFlags != NULL) && !VALID_PTR(aFlags))
4321 return E_POINTER;
4322
4323 AutoCaller autoCaller(this);
4324 AssertComRCReturnRC(autoCaller.rc());
4325
4326 /* protect mpVM (if not NULL) */
4327 AutoVMCallerWeak autoVMCaller(this);
4328 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
4329
4330 /* Note: validity of mVMMDev which is bound to uninit() is guaranteed by
4331 * autoVMCaller, so there is no need to hold a lock of this */
4332
4333 HRESULT rc = E_UNEXPECTED;
4334 using namespace guestProp;
4335
4336 try
4337 {
4338 VBOXHGCMSVCPARM parm[4];
4339 Utf8Str Utf8Name = aName;
4340 char pszBuffer[MAX_VALUE_LEN + MAX_FLAGS_LEN];
4341
4342 parm[0].type = VBOX_HGCM_SVC_PARM_PTR;
4343 parm[0].u.pointer.addr = (void*)Utf8Name.c_str();
4344 /* The + 1 is the null terminator */
4345 parm[0].u.pointer.size = (uint32_t)Utf8Name.length() + 1;
4346 parm[1].type = VBOX_HGCM_SVC_PARM_PTR;
4347 parm[1].u.pointer.addr = pszBuffer;
4348 parm[1].u.pointer.size = sizeof(pszBuffer);
4349 int vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", GET_PROP_HOST,
4350 4, &parm[0]);
4351 /* The returned string should never be able to be greater than our buffer */
4352 AssertLogRel(vrc != VERR_BUFFER_OVERFLOW);
4353 AssertLogRel(RT_FAILURE(vrc) || VBOX_HGCM_SVC_PARM_64BIT == parm[2].type);
4354 if (RT_SUCCESS(vrc) || (VERR_NOT_FOUND == vrc))
4355 {
4356 rc = S_OK;
4357 if (vrc != VERR_NOT_FOUND)
4358 {
4359 Utf8Str strBuffer(pszBuffer);
4360 strBuffer.cloneTo(aValue);
4361
4362 *aTimestamp = parm[2].u.uint64;
4363
4364 size_t iFlags = strBuffer.length() + 1;
4365 Utf8Str(pszBuffer + iFlags).cloneTo(aFlags);
4366 }
4367 else
4368 aValue = NULL;
4369 }
4370 else
4371 rc = setError(E_UNEXPECTED,
4372 tr("The service call failed with the error %Rrc"),
4373 vrc);
4374 }
4375 catch(std::bad_alloc & /*e*/)
4376 {
4377 rc = E_OUTOFMEMORY;
4378 }
4379 return rc;
4380#endif /* VBOX_WITH_GUEST_PROPS */
4381}
4382
4383/**
4384 * @note Temporarily locks this object for writing.
4385 */
4386HRESULT Console::setGuestProperty(IN_BSTR aName, IN_BSTR aValue, IN_BSTR aFlags)
4387{
4388#ifndef VBOX_WITH_GUEST_PROPS
4389 ReturnComNotImplemented();
4390#else /* VBOX_WITH_GUEST_PROPS */
4391 if (!VALID_PTR(aName))
4392 return E_INVALIDARG;
4393 if ((aValue != NULL) && !VALID_PTR(aValue))
4394 return E_INVALIDARG;
4395 if ((aFlags != NULL) && !VALID_PTR(aFlags))
4396 return E_INVALIDARG;
4397
4398 AutoCaller autoCaller(this);
4399 AssertComRCReturnRC(autoCaller.rc());
4400
4401 /* protect mpVM (if not NULL) */
4402 AutoVMCallerWeak autoVMCaller(this);
4403 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
4404
4405 /* Note: validity of mVMMDev which is bound to uninit() is guaranteed by
4406 * autoVMCaller, so there is no need to hold a lock of this */
4407
4408 HRESULT rc = E_UNEXPECTED;
4409 using namespace guestProp;
4410
4411 VBOXHGCMSVCPARM parm[3];
4412 Utf8Str Utf8Name = aName;
4413 int vrc = VINF_SUCCESS;
4414
4415 parm[0].type = VBOX_HGCM_SVC_PARM_PTR;
4416 parm[0].u.pointer.addr = (void*)Utf8Name.c_str();
4417 /* The + 1 is the null terminator */
4418 parm[0].u.pointer.size = (uint32_t)Utf8Name.length() + 1;
4419 Utf8Str Utf8Value = aValue;
4420 if (aValue != NULL)
4421 {
4422 parm[1].type = VBOX_HGCM_SVC_PARM_PTR;
4423 parm[1].u.pointer.addr = (void*)Utf8Value.c_str();
4424 /* The + 1 is the null terminator */
4425 parm[1].u.pointer.size = (uint32_t)Utf8Value.length() + 1;
4426 }
4427 Utf8Str Utf8Flags = aFlags;
4428 if (aFlags != NULL)
4429 {
4430 parm[2].type = VBOX_HGCM_SVC_PARM_PTR;
4431 parm[2].u.pointer.addr = (void*)Utf8Flags.c_str();
4432 /* The + 1 is the null terminator */
4433 parm[2].u.pointer.size = (uint32_t)Utf8Flags.length() + 1;
4434 }
4435 if ((aValue != NULL) && (aFlags != NULL))
4436 vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", SET_PROP_HOST,
4437 3, &parm[0]);
4438 else if (aValue != NULL)
4439 vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", SET_PROP_VALUE_HOST,
4440 2, &parm[0]);
4441 else
4442 vrc = mVMMDev->hgcmHostCall("VBoxGuestPropSvc", DEL_PROP_HOST,
4443 1, &parm[0]);
4444 if (RT_SUCCESS(vrc))
4445 rc = S_OK;
4446 else
4447 rc = setError(E_UNEXPECTED,
4448 tr("The service call failed with the error %Rrc"),
4449 vrc);
4450 return rc;
4451#endif /* VBOX_WITH_GUEST_PROPS */
4452}
4453
4454
4455/**
4456 * @note Temporarily locks this object for writing.
4457 */
4458HRESULT Console::enumerateGuestProperties(IN_BSTR aPatterns,
4459 ComSafeArrayOut(BSTR, aNames),
4460 ComSafeArrayOut(BSTR, aValues),
4461 ComSafeArrayOut(ULONG64, aTimestamps),
4462 ComSafeArrayOut(BSTR, aFlags))
4463{
4464#ifndef VBOX_WITH_GUEST_PROPS
4465 ReturnComNotImplemented();
4466#else /* VBOX_WITH_GUEST_PROPS */
4467 if (!VALID_PTR(aPatterns) && (aPatterns != NULL))
4468 return E_POINTER;
4469 if (ComSafeArrayOutIsNull(aNames))
4470 return E_POINTER;
4471 if (ComSafeArrayOutIsNull(aValues))
4472 return E_POINTER;
4473 if (ComSafeArrayOutIsNull(aTimestamps))
4474 return E_POINTER;
4475 if (ComSafeArrayOutIsNull(aFlags))
4476 return E_POINTER;
4477
4478 AutoCaller autoCaller(this);
4479 AssertComRCReturnRC(autoCaller.rc());
4480
4481 /* protect mpVM (if not NULL) */
4482 AutoVMCallerWeak autoVMCaller(this);
4483 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
4484
4485 /* Note: validity of mVMMDev which is bound to uninit() is guaranteed by
4486 * autoVMCaller, so there is no need to hold a lock of this */
4487
4488 return doEnumerateGuestProperties(aPatterns, ComSafeArrayOutArg(aNames),
4489 ComSafeArrayOutArg(aValues),
4490 ComSafeArrayOutArg(aTimestamps),
4491 ComSafeArrayOutArg(aFlags));
4492#endif /* VBOX_WITH_GUEST_PROPS */
4493}
4494
4495/**
4496 * Gets called by Session::UpdateMachineState()
4497 * (IInternalSessionControl::updateMachineState()).
4498 *
4499 * Must be called only in certain cases (see the implementation).
4500 *
4501 * @note Locks this object for writing.
4502 */
4503HRESULT Console::updateMachineState(MachineState_T aMachineState)
4504{
4505 AutoCaller autoCaller(this);
4506 AssertComRCReturnRC(autoCaller.rc());
4507
4508 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4509
4510 AssertReturn( mMachineState == MachineState_Saving
4511 || mMachineState == MachineState_LiveSnapshotting
4512 || mMachineState == MachineState_RestoringSnapshot
4513 || mMachineState == MachineState_DeletingSnapshot
4514 , E_FAIL);
4515
4516 return setMachineStateLocally(aMachineState);
4517}
4518
4519/**
4520 * @note Locks this object for writing.
4521 */
4522void Console::onMousePointerShapeChange(bool fVisible, bool fAlpha,
4523 uint32_t xHot, uint32_t yHot,
4524 uint32_t width, uint32_t height,
4525 void *pShape)
4526{
4527#if 0
4528 LogFlowThisFuncEnter();
4529 LogFlowThisFunc(("fVisible=%d, fAlpha=%d, xHot = %d, yHot = %d, width=%d, height=%d, shape=%p\n",
4530 fVisible, fAlpha, xHot, yHot, width, height, pShape));
4531#endif
4532
4533 AutoCaller autoCaller(this);
4534 AssertComRCReturnVoid(autoCaller.rc());
4535
4536 /* We need a write lock because we alter the cached callback data */
4537 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4538
4539 /* Save the callback arguments */
4540 mCallbackData.mpsc.visible = fVisible;
4541 mCallbackData.mpsc.alpha = fAlpha;
4542 mCallbackData.mpsc.xHot = xHot;
4543 mCallbackData.mpsc.yHot = yHot;
4544 mCallbackData.mpsc.width = width;
4545 mCallbackData.mpsc.height = height;
4546
4547 /* start with not valid */
4548 bool wasValid = mCallbackData.mpsc.valid;
4549 mCallbackData.mpsc.valid = false;
4550
4551 if (pShape != NULL)
4552 {
4553 size_t cb = (width + 7) / 8 * height; /* size of the AND mask */
4554 cb = ((cb + 3) & ~3) + width * 4 * height; /* + gap + size of the XOR mask */
4555 /* try to reuse the old shape buffer if the size is the same */
4556 if (!wasValid)
4557 mCallbackData.mpsc.shape = NULL;
4558 else
4559 if (mCallbackData.mpsc.shape != NULL && mCallbackData.mpsc.shapeSize != cb)
4560 {
4561 RTMemFree(mCallbackData.mpsc.shape);
4562 mCallbackData.mpsc.shape = NULL;
4563 }
4564 if (mCallbackData.mpsc.shape == NULL)
4565 {
4566 mCallbackData.mpsc.shape = (BYTE *) RTMemAllocZ(cb);
4567 AssertReturnVoid(mCallbackData.mpsc.shape);
4568 }
4569 mCallbackData.mpsc.shapeSize = cb;
4570 memcpy(mCallbackData.mpsc.shape, pShape, cb);
4571 }
4572 else
4573 {
4574 if (wasValid && mCallbackData.mpsc.shape != NULL)
4575 RTMemFree(mCallbackData.mpsc.shape);
4576 mCallbackData.mpsc.shape = NULL;
4577 mCallbackData.mpsc.shapeSize = 0;
4578 }
4579
4580 mCallbackData.mpsc.valid = true;
4581
4582 CallbackList::iterator it = mCallbacks.begin();
4583 while (it != mCallbacks.end())
4584 (*it++)->OnMousePointerShapeChange(fVisible, fAlpha, xHot, yHot,
4585 width, height, (BYTE *) pShape);
4586
4587#if 0
4588 LogFlowThisFuncLeave();
4589#endif
4590}
4591
4592/**
4593 * @note Locks this object for writing.
4594 */
4595void Console::onMouseCapabilityChange(BOOL supportsAbsolute, BOOL supportsRelative, BOOL needsHostCursor)
4596{
4597 LogFlowThisFunc(("supportsAbsolute=%d supportsRelative=%d needsHostCursor=%d\n",
4598 supportsAbsolute, supportsRelative, needsHostCursor));
4599
4600 AutoCaller autoCaller(this);
4601 AssertComRCReturnVoid(autoCaller.rc());
4602
4603 /* We need a write lock because we alter the cached callback data */
4604 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4605
4606 /* save the callback arguments */
4607 mCallbackData.mcc.supportsAbsolute = supportsAbsolute;
4608 mCallbackData.mcc.supportsRelative = supportsRelative;
4609 mCallbackData.mcc.needsHostCursor = needsHostCursor;
4610 mCallbackData.mcc.valid = true;
4611
4612 CallbackList::iterator it = mCallbacks.begin();
4613 while (it != mCallbacks.end())
4614 {
4615 Log2(("Console::onMouseCapabilityChange: calling %p\n", (void*)*it));
4616 (*it++)->OnMouseCapabilityChange(supportsAbsolute, supportsRelative, needsHostCursor);
4617 }
4618}
4619
4620/**
4621 * @note Locks this object for reading.
4622 */
4623void Console::onStateChange(MachineState_T machineState)
4624{
4625 AutoCaller autoCaller(this);
4626 AssertComRCReturnVoid(autoCaller.rc());
4627
4628 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4629
4630 CallbackList::iterator it = mCallbacks.begin();
4631 while (it != mCallbacks.end())
4632 (*it++)->OnStateChange(machineState);
4633}
4634
4635/**
4636 * @note Locks this object for reading.
4637 */
4638void Console::onAdditionsStateChange()
4639{
4640 AutoCaller autoCaller(this);
4641 AssertComRCReturnVoid(autoCaller.rc());
4642
4643 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4644
4645 CallbackList::iterator it = mCallbacks.begin();
4646 while (it != mCallbacks.end())
4647 (*it++)->OnAdditionsStateChange();
4648}
4649
4650/**
4651 * @note Locks this object for reading.
4652 */
4653void Console::onAdditionsOutdated()
4654{
4655 AutoCaller autoCaller(this);
4656 AssertComRCReturnVoid(autoCaller.rc());
4657
4658 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4659
4660 /** @todo Use the On-Screen Display feature to report the fact.
4661 * The user should be told to install additions that are
4662 * provided with the current VBox build:
4663 * VBOX_VERSION_MAJOR.VBOX_VERSION_MINOR.VBOX_VERSION_BUILD
4664 */
4665}
4666
4667/**
4668 * @note Locks this object for writing.
4669 */
4670void Console::onKeyboardLedsChange(bool fNumLock, bool fCapsLock, bool fScrollLock)
4671{
4672 AutoCaller autoCaller(this);
4673 AssertComRCReturnVoid(autoCaller.rc());
4674
4675 /* We need a write lock because we alter the cached callback data */
4676 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4677
4678 /* save the callback arguments */
4679 mCallbackData.klc.numLock = fNumLock;
4680 mCallbackData.klc.capsLock = fCapsLock;
4681 mCallbackData.klc.scrollLock = fScrollLock;
4682 mCallbackData.klc.valid = true;
4683
4684 CallbackList::iterator it = mCallbacks.begin();
4685 while (it != mCallbacks.end())
4686 (*it++)->OnKeyboardLedsChange(fNumLock, fCapsLock, fScrollLock);
4687}
4688
4689/**
4690 * @note Locks this object for reading.
4691 */
4692void Console::onUSBDeviceStateChange(IUSBDevice *aDevice, bool aAttached,
4693 IVirtualBoxErrorInfo *aError)
4694{
4695 AutoCaller autoCaller(this);
4696 AssertComRCReturnVoid(autoCaller.rc());
4697
4698 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4699
4700 CallbackList::iterator it = mCallbacks.begin();
4701 while (it != mCallbacks.end())
4702 (*it++)->OnUSBDeviceStateChange(aDevice, aAttached, aError);
4703}
4704
4705/**
4706 * @note Locks this object for reading.
4707 */
4708void Console::onRuntimeError(BOOL aFatal, IN_BSTR aErrorID, IN_BSTR aMessage)
4709{
4710 AutoCaller autoCaller(this);
4711 AssertComRCReturnVoid(autoCaller.rc());
4712
4713 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4714
4715 CallbackList::iterator it = mCallbacks.begin();
4716 while (it != mCallbacks.end())
4717 (*it++)->OnRuntimeError(aFatal, aErrorID, aMessage);
4718}
4719
4720/**
4721 * @note Locks this object for reading.
4722 */
4723HRESULT Console::onShowWindow(BOOL aCheck, BOOL *aCanShow, ULONG64 *aWinId)
4724{
4725 AssertReturn(aCanShow, E_POINTER);
4726 AssertReturn(aWinId, E_POINTER);
4727
4728 *aCanShow = FALSE;
4729 *aWinId = 0;
4730
4731 AutoCaller autoCaller(this);
4732 AssertComRCReturnRC(autoCaller.rc());
4733
4734 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
4735
4736 HRESULT rc = S_OK;
4737 CallbackList::iterator it = mCallbacks.begin();
4738
4739 if (aCheck)
4740 {
4741 while (it != mCallbacks.end())
4742 {
4743 BOOL canShow = FALSE;
4744 rc = (*it++)->OnCanShowWindow(&canShow);
4745 AssertComRC(rc);
4746 if (FAILED(rc) || !canShow)
4747 return rc;
4748 }
4749 *aCanShow = TRUE;
4750 }
4751 else
4752 {
4753 while (it != mCallbacks.end())
4754 {
4755 ULONG64 winId = 0;
4756 rc = (*it++)->OnShowWindow(&winId);
4757 AssertComRC(rc);
4758 if (FAILED(rc))
4759 return rc;
4760 /* only one callback may return non-null winId */
4761 Assert(*aWinId == 0 || winId == 0);
4762 if (*aWinId == 0)
4763 *aWinId = winId;
4764 }
4765 }
4766
4767 return S_OK;
4768}
4769
4770// private methods
4771////////////////////////////////////////////////////////////////////////////////
4772
4773/**
4774 * Increases the usage counter of the mpVM pointer. Guarantees that
4775 * VMR3Destroy() will not be called on it at least until releaseVMCaller()
4776 * is called.
4777 *
4778 * If this method returns a failure, the caller is not allowed to use mpVM
4779 * and may return the failed result code to the upper level. This method sets
4780 * the extended error info on failure if \a aQuiet is false.
4781 *
4782 * Setting \a aQuiet to true is useful for methods that don't want to return
4783 * the failed result code to the caller when this method fails (e.g. need to
4784 * silently check for the mpVM availability).
4785 *
4786 * When mpVM is NULL but \a aAllowNullVM is true, a corresponding error will be
4787 * returned instead of asserting. Having it false is intended as a sanity check
4788 * for methods that have checked mMachineState and expect mpVM *NOT* to be NULL.
4789 *
4790 * @param aQuiet true to suppress setting error info
4791 * @param aAllowNullVM true to accept mpVM being NULL and return a failure
4792 * (otherwise this method will assert if mpVM is NULL)
4793 *
4794 * @note Locks this object for writing.
4795 */
4796HRESULT Console::addVMCaller(bool aQuiet /* = false */,
4797 bool aAllowNullVM /* = false */)
4798{
4799 AutoCaller autoCaller(this);
4800 AssertComRCReturnRC(autoCaller.rc());
4801
4802 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4803
4804 if (mVMDestroying)
4805 {
4806 /* powerDown() is waiting for all callers to finish */
4807 return aQuiet ? E_ACCESSDENIED : setError(E_ACCESSDENIED,
4808 tr("Virtual machine is being powered down"));
4809 }
4810
4811 if (mpVM == NULL)
4812 {
4813 Assert(aAllowNullVM == true);
4814
4815 /* The machine is not powered up */
4816 return aQuiet ? E_ACCESSDENIED : setError(E_ACCESSDENIED,
4817 tr("Virtual machine is not powered up"));
4818 }
4819
4820 ++ mVMCallers;
4821
4822 return S_OK;
4823}
4824
4825/**
4826 * Decreases the usage counter of the mpVM pointer. Must always complete
4827 * the addVMCaller() call after the mpVM pointer is no more necessary.
4828 *
4829 * @note Locks this object for writing.
4830 */
4831void Console::releaseVMCaller()
4832{
4833 AutoCaller autoCaller(this);
4834 AssertComRCReturnVoid(autoCaller.rc());
4835
4836 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4837
4838 AssertReturnVoid(mpVM != NULL);
4839
4840 Assert(mVMCallers > 0);
4841 --mVMCallers;
4842
4843 if (mVMCallers == 0 && mVMDestroying)
4844 {
4845 /* inform powerDown() there are no more callers */
4846 RTSemEventSignal(mVMZeroCallersSem);
4847 }
4848}
4849
4850/**
4851 * Initialize the release logging facility. In case something
4852 * goes wrong, there will be no release logging. Maybe in the future
4853 * we can add some logic to use different file names in this case.
4854 * Note that the logic must be in sync with Machine::DeleteSettings().
4855 */
4856HRESULT Console::consoleInitReleaseLog(const ComPtr<IMachine> aMachine)
4857{
4858 HRESULT hrc = S_OK;
4859
4860 Bstr logFolder;
4861 hrc = aMachine->COMGETTER(LogFolder)(logFolder.asOutParam());
4862 if (FAILED(hrc)) return hrc;
4863
4864 Utf8Str logDir = logFolder;
4865
4866 /* make sure the Logs folder exists */
4867 Assert(logDir.length());
4868 if (!RTDirExists(logDir.c_str()))
4869 RTDirCreateFullPath(logDir.c_str(), 0777);
4870
4871 Utf8Str logFile = Utf8StrFmt("%s%cVBox.log",
4872 logDir.raw(), RTPATH_DELIMITER);
4873 Utf8Str pngFile = Utf8StrFmt("%s%cVBox.png",
4874 logDir.raw(), RTPATH_DELIMITER);
4875
4876 /*
4877 * Age the old log files
4878 * Rename .(n-1) to .(n), .(n-2) to .(n-1), ..., and the last log file to .1
4879 * Overwrite target files in case they exist.
4880 */
4881 ComPtr<IVirtualBox> virtualBox;
4882 aMachine->COMGETTER(Parent)(virtualBox.asOutParam());
4883 ComPtr<ISystemProperties> systemProperties;
4884 virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam());
4885 ULONG uLogHistoryCount = 3;
4886 systemProperties->COMGETTER(LogHistoryCount)(&uLogHistoryCount);
4887 ComPtr<IHost> host;
4888 virtualBox->COMGETTER(Host)(host.asOutParam());
4889 ULONG uHostRamMb = 0, uHostRamAvailMb = 0;
4890 host->COMGETTER(MemorySize)(&uHostRamMb);
4891 host->COMGETTER(MemoryAvailable)(&uHostRamAvailMb);
4892 if (uLogHistoryCount)
4893 {
4894 for (int i = uLogHistoryCount-1; i >= 0; i--)
4895 {
4896 Utf8Str *files[] = { &logFile, &pngFile };
4897 Utf8Str oldName, newName;
4898
4899 for (unsigned int j = 0; j < RT_ELEMENTS(files); ++ j)
4900 {
4901 if (i > 0)
4902 oldName = Utf8StrFmt("%s.%d", files[j]->raw(), i);
4903 else
4904 oldName = *files[j];
4905 newName = Utf8StrFmt("%s.%d", files[j]->raw(), i + 1);
4906 /* If the old file doesn't exist, delete the new file (if it
4907 * exists) to provide correct rotation even if the sequence is
4908 * broken */
4909 if ( RTFileRename(oldName.c_str(), newName.c_str(), RTFILEMOVE_FLAGS_REPLACE)
4910 == VERR_FILE_NOT_FOUND)
4911 RTFileDelete(newName.c_str());
4912 }
4913 }
4914 }
4915
4916 PRTLOGGER loggerRelease;
4917 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
4918 RTUINT fFlags = RTLOGFLAGS_PREFIX_TIME_PROG;
4919#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
4920 fFlags |= RTLOGFLAGS_USECRLF;
4921#endif
4922 char szError[RTPATH_MAX + 128] = "";
4923 int vrc = RTLogCreateEx(&loggerRelease, fFlags, "all",
4924 "VBOX_RELEASE_LOG", RT_ELEMENTS(s_apszGroups), s_apszGroups,
4925 RTLOGDEST_FILE, szError, sizeof(szError), logFile.raw());
4926 if (RT_SUCCESS(vrc))
4927 {
4928 /* some introductory information */
4929 RTTIMESPEC timeSpec;
4930 char szTmp[256];
4931 RTTimeSpecToString(RTTimeNow(&timeSpec), szTmp, sizeof(szTmp));
4932 RTLogRelLogger(loggerRelease, 0, ~0U,
4933 "VirtualBox %s r%u %s (%s %s) release log\n"
4934#ifdef VBOX_BLEEDING_EDGE
4935 "EXPERIMENTAL build " VBOX_BLEEDING_EDGE "\n"
4936#endif
4937 "Log opened %s\n",
4938 VBOX_VERSION_STRING, RTBldCfgRevision(), VBOX_BUILD_TARGET,
4939 __DATE__, __TIME__, szTmp);
4940
4941 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_PRODUCT, szTmp, sizeof(szTmp));
4942 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
4943 RTLogRelLogger(loggerRelease, 0, ~0U, "OS Product: %s\n", szTmp);
4944 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_RELEASE, szTmp, sizeof(szTmp));
4945 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
4946 RTLogRelLogger(loggerRelease, 0, ~0U, "OS Release: %s\n", szTmp);
4947 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_VERSION, szTmp, sizeof(szTmp));
4948 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
4949 RTLogRelLogger(loggerRelease, 0, ~0U, "OS Version: %s\n", szTmp);
4950 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_SERVICE_PACK, szTmp, sizeof(szTmp));
4951 if (RT_SUCCESS(vrc) || vrc == VERR_BUFFER_OVERFLOW)
4952 RTLogRelLogger(loggerRelease, 0, ~0U, "OS Service Pack: %s\n", szTmp);
4953 RTLogRelLogger(loggerRelease, 0, ~0U, "Host RAM: %uMB RAM, available: %uMB\n",
4954 uHostRamMb, uHostRamAvailMb);
4955 /* the package type is interesting for Linux distributions */
4956 char szExecName[RTPATH_MAX];
4957 char *pszExecName = RTProcGetExecutableName(szExecName, sizeof(szExecName));
4958 RTLogRelLogger(loggerRelease, 0, ~0U,
4959 "Executable: %s\n"
4960 "Process ID: %u\n"
4961 "Package type: %s"
4962#ifdef VBOX_OSE
4963 " (OSE)"
4964#endif
4965 "\n",
4966 pszExecName ? pszExecName : "unknown",
4967 RTProcSelf(),
4968 VBOX_PACKAGE_STRING);
4969
4970 /* register this logger as the release logger */
4971 RTLogRelSetDefaultInstance(loggerRelease);
4972 hrc = S_OK;
4973 }
4974 else
4975 hrc = setError(E_FAIL,
4976 tr("Failed to open release log (%s, %Rrc)"),
4977 szError, vrc);
4978
4979 return hrc;
4980}
4981
4982/**
4983 * Common worker for PowerUp and PowerUpPaused.
4984 *
4985 * @returns COM status code.
4986 *
4987 * @param aProgress Where to return the progress object.
4988 * @param aPaused true if PowerUpPaused called.
4989 *
4990 * @todo move down to powerDown();
4991 */
4992HRESULT Console::powerUp(IProgress **aProgress, bool aPaused)
4993{
4994 if (aProgress == NULL)
4995 return E_POINTER;
4996
4997 LogFlowThisFuncEnter();
4998 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
4999
5000 AutoCaller autoCaller(this);
5001 if (FAILED(autoCaller.rc())) return autoCaller.rc();
5002
5003 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5004
5005 if (Global::IsOnlineOrTransient(mMachineState))
5006 return setError(VBOX_E_INVALID_VM_STATE,
5007 tr("Virtual machine is already running or busy (machine state: %s)"),
5008 Global::stringifyMachineState(mMachineState));
5009
5010 HRESULT rc = S_OK;
5011
5012 /* the network cards will undergo a quick consistency check */
5013 for (ULONG slot = 0;
5014 slot < SchemaDefs::NetworkAdapterCount;
5015 ++slot)
5016 {
5017 ComPtr<INetworkAdapter> adapter;
5018 mMachine->GetNetworkAdapter(slot, adapter.asOutParam());
5019 BOOL enabled = FALSE;
5020 adapter->COMGETTER(Enabled)(&enabled);
5021 if (!enabled)
5022 continue;
5023
5024 NetworkAttachmentType_T netattach;
5025 adapter->COMGETTER(AttachmentType)(&netattach);
5026 switch (netattach)
5027 {
5028 case NetworkAttachmentType_Bridged:
5029 {
5030#ifdef RT_OS_WINDOWS
5031 /* a valid host interface must have been set */
5032 Bstr hostif;
5033 adapter->COMGETTER(HostInterface)(hostif.asOutParam());
5034 if (!hostif)
5035 {
5036 return setError(VBOX_E_HOST_ERROR,
5037 tr("VM cannot start because host interface networking requires a host interface name to be set"));
5038 }
5039 ComPtr<IVirtualBox> virtualBox;
5040 mMachine->COMGETTER(Parent)(virtualBox.asOutParam());
5041 ComPtr<IHost> host;
5042 virtualBox->COMGETTER(Host)(host.asOutParam());
5043 ComPtr<IHostNetworkInterface> hostInterface;
5044 if (!SUCCEEDED(host->FindHostNetworkInterfaceByName(hostif, hostInterface.asOutParam())))
5045 {
5046 return setError(VBOX_E_HOST_ERROR,
5047 tr("VM cannot start because the host interface '%ls' does not exist"),
5048 hostif.raw());
5049 }
5050#endif /* RT_OS_WINDOWS */
5051 break;
5052 }
5053 default:
5054 break;
5055 }
5056 }
5057
5058 /* Read console data stored in the saved state file (if not yet done) */
5059 rc = loadDataFromSavedState();
5060 if (FAILED(rc)) return rc;
5061
5062 /* Check all types of shared folders and compose a single list */
5063 SharedFolderDataMap sharedFolders;
5064 {
5065 /* first, insert global folders */
5066 for (SharedFolderDataMap::const_iterator it = mGlobalSharedFolders.begin();
5067 it != mGlobalSharedFolders.end(); ++ it)
5068 sharedFolders[it->first] = it->second;
5069
5070 /* second, insert machine folders */
5071 for (SharedFolderDataMap::const_iterator it = mMachineSharedFolders.begin();
5072 it != mMachineSharedFolders.end(); ++ it)
5073 sharedFolders[it->first] = it->second;
5074
5075 /* third, insert console folders */
5076 for (SharedFolderMap::const_iterator it = mSharedFolders.begin();
5077 it != mSharedFolders.end(); ++ it)
5078 sharedFolders[it->first] = SharedFolderData(it->second->getHostPath(), it->second->isWritable());
5079 }
5080
5081 Bstr savedStateFile;
5082
5083 /*
5084 * Saved VMs will have to prove that their saved states seem kosher.
5085 */
5086 if (mMachineState == MachineState_Saved)
5087 {
5088 rc = mMachine->COMGETTER(StateFilePath)(savedStateFile.asOutParam());
5089 if (FAILED(rc)) return rc;
5090 ComAssertRet(!!savedStateFile, E_FAIL);
5091 int vrc = SSMR3ValidateFile(Utf8Str(savedStateFile).c_str(), false /* fChecksumIt */);
5092 if (RT_FAILURE(vrc))
5093 return setError(VBOX_E_FILE_ERROR,
5094 tr("VM cannot start because the saved state file '%ls' is invalid (%Rrc). Discard the saved state prior to starting the VM"),
5095 savedStateFile.raw(), vrc);
5096 }
5097
5098 /* test and clear the TeleporterEnabled property */
5099 BOOL fTeleporterEnabled;
5100 rc = mMachine->COMGETTER(TeleporterEnabled)(&fTeleporterEnabled);
5101 if (FAILED(rc)) return rc;
5102#if 0 /** @todo we should save it afterwards, but that isn't necessarily a good idea. Find a better place for this (VBoxSVC). */
5103 if (fTeleporterEnabled)
5104 {
5105 rc = mMachine->COMSETTER(TeleporterEnabled)(FALSE);
5106 if (FAILED(rc)) return rc;
5107 }
5108#endif
5109
5110 /* create a progress object to track progress of this operation */
5111 ComObjPtr<Progress> powerupProgress;
5112 powerupProgress.createObject();
5113 Bstr progressDesc;
5114 if (mMachineState == MachineState_Saved)
5115 progressDesc = tr("Restoring virtual machine");
5116 else if (fTeleporterEnabled)
5117 progressDesc = tr("Teleporting virtual machine");
5118 else
5119 progressDesc = tr("Starting virtual machine");
5120 rc = powerupProgress->init(static_cast<IConsole *>(this),
5121 progressDesc,
5122 fTeleporterEnabled /* aCancelable */);
5123 if (FAILED(rc)) return rc;
5124
5125 /* setup task object and thread to carry out the operation
5126 * asynchronously */
5127
5128 std::auto_ptr<VMPowerUpTask> task(new VMPowerUpTask(this, powerupProgress));
5129 ComAssertComRCRetRC(task->rc());
5130
5131 task->mSetVMErrorCallback = setVMErrorCallback;
5132 task->mConfigConstructor = configConstructor;
5133 task->mSharedFolders = sharedFolders;
5134 task->mStartPaused = aPaused;
5135 if (mMachineState == MachineState_Saved)
5136 task->mSavedStateFile = savedStateFile;
5137 task->mTeleporterEnabled = fTeleporterEnabled;
5138
5139 /* Reset differencing hard disks for which autoReset is true,
5140 * but only if the machine has no snapshots OR the current snapshot
5141 * is an OFFLINE snapshot; otherwise we would reset the current differencing
5142 * image of an ONLINE snapshot which contains the disk state of the machine
5143 * while it was previously running, but without the corresponding machine
5144 * state, which is equivalent to powering off a running machine and not
5145 * good idea
5146 */
5147 ComPtr<ISnapshot> pCurrentSnapshot;
5148 rc = mMachine->COMGETTER(CurrentSnapshot)(pCurrentSnapshot.asOutParam());
5149 if (FAILED(rc)) return rc;
5150
5151 BOOL fCurrentSnapshotIsOnline = false;
5152 if (pCurrentSnapshot)
5153 {
5154 rc = pCurrentSnapshot->COMGETTER(Online)(&fCurrentSnapshotIsOnline);
5155 if (FAILED(rc)) return rc;
5156 }
5157
5158 if (!fCurrentSnapshotIsOnline)
5159 {
5160 LogFlowThisFunc(("Looking for immutable images to reset\n"));
5161
5162 com::SafeIfaceArray<IMediumAttachment> atts;
5163 rc = mMachine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(atts));
5164 if (FAILED(rc)) return rc;
5165
5166 for (size_t i = 0;
5167 i < atts.size();
5168 ++i)
5169 {
5170 DeviceType_T devType;
5171 rc = atts[i]->COMGETTER(Type)(&devType);
5172 /** @todo later applies to floppies as well */
5173 if (devType == DeviceType_HardDisk)
5174 {
5175 ComPtr<IMedium> medium;
5176 rc = atts[i]->COMGETTER(Medium)(medium.asOutParam());
5177 if (FAILED(rc)) return rc;
5178
5179 /* save for later use on the powerup thread */
5180 task->hardDisks.push_back(medium);
5181
5182 /* needs autoreset? */
5183 BOOL autoReset = FALSE;
5184 rc = medium->COMGETTER(AutoReset)(&autoReset);
5185 if (FAILED(rc)) return rc;
5186
5187 if (autoReset)
5188 {
5189 ComPtr<IProgress> resetProgress;
5190 rc = medium->Reset(resetProgress.asOutParam());
5191 if (FAILED(rc)) return rc;
5192
5193 /* save for later use on the powerup thread */
5194 task->hardDiskProgresses.push_back(resetProgress);
5195 }
5196 }
5197 }
5198 }
5199 else
5200 LogFlowThisFunc(("Machine has a current snapshot which is online, skipping immutable images reset\n"));
5201
5202 rc = consoleInitReleaseLog(mMachine);
5203 if (FAILED(rc)) return rc;
5204
5205 /* pass the progress object to the caller if requested */
5206 if (aProgress)
5207 {
5208 if (task->hardDiskProgresses.size() == 0)
5209 {
5210 /* there are no other operations to track, return the powerup
5211 * progress only */
5212 powerupProgress.queryInterfaceTo(aProgress);
5213 }
5214 else
5215 {
5216 /* create a combined progress object */
5217 ComObjPtr<CombinedProgress> progress;
5218 progress.createObject();
5219 VMPowerUpTask::ProgressList progresses(task->hardDiskProgresses);
5220 progresses.push_back(ComPtr<IProgress> (powerupProgress));
5221 rc = progress->init(static_cast<IConsole *>(this),
5222 progressDesc, progresses.begin(),
5223 progresses.end());
5224 AssertComRCReturnRC(rc);
5225 progress.queryInterfaceTo(aProgress);
5226 }
5227 }
5228
5229 int vrc = RTThreadCreate(NULL, Console::powerUpThread, (void *) task.get(),
5230 0, RTTHREADTYPE_MAIN_WORKER, 0, "VMPowerUp");
5231
5232 ComAssertMsgRCRet(vrc, ("Could not create VMPowerUp thread (%Rrc)", vrc),
5233 E_FAIL);
5234
5235 /* task is now owned by powerUpThread(), so release it */
5236 task.release();
5237
5238 /* finally, set the state: no right to fail in this method afterwards
5239 * since we've already started the thread and it is now responsible for
5240 * any error reporting and appropriate state change! */
5241
5242 if (mMachineState == MachineState_Saved)
5243 setMachineState(MachineState_Restoring);
5244 else if (fTeleporterEnabled)
5245 setMachineState(MachineState_TeleportingIn);
5246 else
5247 setMachineState(MachineState_Starting);
5248
5249 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
5250 LogFlowThisFuncLeave();
5251 return S_OK;
5252}
5253
5254/**
5255 * Internal power off worker routine.
5256 *
5257 * This method may be called only at certain places with the following meaning
5258 * as shown below:
5259 *
5260 * - if the machine state is either Running or Paused, a normal
5261 * Console-initiated powerdown takes place (e.g. PowerDown());
5262 * - if the machine state is Saving, saveStateThread() has successfully done its
5263 * job;
5264 * - if the machine state is Starting or Restoring, powerUpThread() has failed
5265 * to start/load the VM;
5266 * - if the machine state is Stopping, the VM has powered itself off (i.e. not
5267 * as a result of the powerDown() call).
5268 *
5269 * Calling it in situations other than the above will cause unexpected behavior.
5270 *
5271 * Note that this method should be the only one that destroys mpVM and sets it
5272 * to NULL.
5273 *
5274 * @param aProgress Progress object to run (may be NULL).
5275 *
5276 * @note Locks this object for writing.
5277 *
5278 * @note Never call this method from a thread that called addVMCaller() or
5279 * instantiated an AutoVMCaller object; first call releaseVMCaller() or
5280 * release(). Otherwise it will deadlock.
5281 */
5282HRESULT Console::powerDown(Progress *aProgress /*= NULL*/)
5283{
5284 LogFlowThisFuncEnter();
5285
5286 AutoCaller autoCaller(this);
5287 AssertComRCReturnRC(autoCaller.rc());
5288
5289 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5290
5291 /* Total # of steps for the progress object. Must correspond to the
5292 * number of "advance percent count" comments in this method! */
5293 enum { StepCount = 7 };
5294 /* current step */
5295 ULONG step = 0;
5296
5297 HRESULT rc = S_OK;
5298 int vrc = VINF_SUCCESS;
5299
5300 /* sanity */
5301 Assert(mVMDestroying == false);
5302
5303 Assert(mpVM != NULL);
5304
5305 AssertMsg( mMachineState == MachineState_Running
5306 || mMachineState == MachineState_Paused
5307 || mMachineState == MachineState_Stuck
5308 || mMachineState == MachineState_Starting
5309 || mMachineState == MachineState_Stopping
5310 || mMachineState == MachineState_Saving
5311 || mMachineState == MachineState_Restoring
5312 || mMachineState == MachineState_TeleportingPausedVM
5313 || mMachineState == MachineState_TeleportingIn
5314 , ("Invalid machine state: %s\n", Global::stringifyMachineState(mMachineState)));
5315
5316 LogRel(("Console::powerDown(): A request to power off the VM has been issued (mMachineState=%s, InUninit=%d)\n",
5317 Global::stringifyMachineState(mMachineState), autoCaller.state() == InUninit));
5318
5319 /* Check if we need to power off the VM. In case of mVMPoweredOff=true, the
5320 * VM has already powered itself off in vmstateChangeCallback() and is just
5321 * notifying Console about that. In case of Starting or Restoring,
5322 * powerUpThread() is calling us on failure, so the VM is already off at
5323 * that point. */
5324 if ( !mVMPoweredOff
5325 && ( mMachineState == MachineState_Starting
5326 || mMachineState == MachineState_Restoring
5327 || mMachineState == MachineState_TeleportingIn)
5328 )
5329 mVMPoweredOff = true;
5330
5331 /*
5332 * Go to Stopping state if not already there.
5333 *
5334 * Note that we don't go from Saving/Restoring to Stopping because
5335 * vmstateChangeCallback() needs it to set the state to Saved on
5336 * VMSTATE_TERMINATED. In terms of protecting from inappropriate operations
5337 * while leaving the lock below, Saving or Restoring should be fine too.
5338 * Ditto for TeleportingPausedVM -> Teleported.
5339 */
5340 if ( mMachineState != MachineState_Saving
5341 && mMachineState != MachineState_Restoring
5342 && mMachineState != MachineState_Stopping
5343 && mMachineState != MachineState_TeleportingIn
5344 && mMachineState != MachineState_TeleportingPausedVM
5345 )
5346 setMachineState(MachineState_Stopping);
5347
5348 /* ----------------------------------------------------------------------
5349 * DONE with necessary state changes, perform the power down actions (it's
5350 * safe to leave the object lock now if needed)
5351 * ---------------------------------------------------------------------- */
5352
5353 /* Stop the VRDP server to prevent new clients connection while VM is being
5354 * powered off. */
5355 if (mConsoleVRDPServer)
5356 {
5357 LogFlowThisFunc(("Stopping VRDP server...\n"));
5358
5359 /* Leave the lock since EMT will call us back as addVMCaller()
5360 * in updateDisplayData(). */
5361 alock.leave();
5362
5363 mConsoleVRDPServer->Stop();
5364
5365 alock.enter();
5366 }
5367
5368 /* advance percent count */
5369 if (aProgress)
5370 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5371
5372
5373 /* ----------------------------------------------------------------------
5374 * Now, wait for all mpVM callers to finish their work if there are still
5375 * some on other threads. NO methods that need mpVM (or initiate other calls
5376 * that need it) may be called after this point
5377 * ---------------------------------------------------------------------- */
5378
5379 if (mVMCallers > 0)
5380 {
5381 /* go to the destroying state to prevent from adding new callers */
5382 mVMDestroying = true;
5383
5384 /* lazy creation */
5385 if (mVMZeroCallersSem == NIL_RTSEMEVENT)
5386 RTSemEventCreate(&mVMZeroCallersSem);
5387
5388 LogFlowThisFunc(("Waiting for mpVM callers (%d) to drop to zero...\n",
5389 mVMCallers));
5390
5391 alock.leave();
5392
5393 RTSemEventWait(mVMZeroCallersSem, RT_INDEFINITE_WAIT);
5394
5395 alock.enter();
5396 }
5397
5398 /* advance percent count */
5399 if (aProgress)
5400 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5401
5402 vrc = VINF_SUCCESS;
5403
5404 /*
5405 * Power off the VM if not already done that.
5406 * Leave the lock since EMT will call vmstateChangeCallback.
5407 *
5408 * Note that VMR3PowerOff() may fail here (invalid VMSTATE) if the
5409 * VM-(guest-)initiated power off happened in parallel a ms before this
5410 * call. So far, we let this error pop up on the user's side.
5411 */
5412 if (!mVMPoweredOff)
5413 {
5414 LogFlowThisFunc(("Powering off the VM...\n"));
5415 alock.leave();
5416 vrc = VMR3PowerOff(mpVM);
5417 alock.enter();
5418 }
5419 else
5420 {
5421 /** @todo r=bird: Doesn't make sense. Please remove after 3.1 has been branched
5422 * off. */
5423 /* reset the flag for future re-use */
5424 mVMPoweredOff = false;
5425 }
5426
5427 /* advance percent count */
5428 if (aProgress)
5429 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5430
5431#ifdef VBOX_WITH_HGCM
5432# ifdef VBOX_WITH_GUEST_PROPS
5433 /*
5434 * Save all guest property store entries to the machine XML file
5435 * and hand controll over to VBoxSVC. Ignoring failure for now.
5436 */
5437 LogFlowThisFunc(("Moving Guest Properties to XML/VBoxSVC...\n"));
5438 bool fIsSaving = mMachineState == MachineState_Saving
5439 || mMachineState == MachineState_LiveSnapshotting;
5440 alock.leave();
5441 doMoveGuestPropertiesOnPowerOff(fIsSaving);
5442 alock.enter();
5443
5444 /* advance percent count */
5445 if (aProgress)
5446 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5447
5448# endif /* VBOX_WITH_GUEST_PROPS defined */
5449
5450 /* Shutdown HGCM services before destroying the VM. */
5451 if (mVMMDev)
5452 {
5453 LogFlowThisFunc(("Shutdown HGCM...\n"));
5454
5455 /* Leave the lock since EMT will call us back as addVMCaller() */
5456 alock.leave();
5457
5458 mVMMDev->hgcmShutdown();
5459
5460 alock.enter();
5461 }
5462
5463 /* advance percent count */
5464 if (aProgress)
5465 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5466
5467#endif /* VBOX_WITH_HGCM */
5468
5469 LogFlowThisFunc(("Ready for VM destruction.\n"));
5470
5471 /* If we are called from Console::uninit(), then try to destroy the VM even
5472 * on failure (this will most likely fail too, but what to do?..) */
5473 if (RT_SUCCESS(vrc) || autoCaller.state() == InUninit)
5474 {
5475 /* If the machine has an USB controller, release all USB devices
5476 * (symmetric to the code in captureUSBDevices()) */
5477 bool fHasUSBController = false;
5478 {
5479 PPDMIBASE pBase;
5480 vrc = PDMR3QueryLun(mpVM, "usb-ohci", 0, 0, &pBase);
5481 if (RT_SUCCESS(vrc))
5482 {
5483 fHasUSBController = true;
5484 detachAllUSBDevices(false /* aDone */);
5485 }
5486 }
5487
5488 /* Now we've got to destroy the VM as well. (mpVM is not valid beyond
5489 * this point). We leave the lock before calling VMR3Destroy() because
5490 * it will result into calling destructors of drivers associated with
5491 * Console children which may in turn try to lock Console (e.g. by
5492 * instantiating SafeVMPtr to access mpVM). It's safe here because
5493 * mVMDestroying is set which should prevent any activity. */
5494
5495 /* Set mpVM to NULL early just in case if some old code is not using
5496 * addVMCaller()/releaseVMCaller(). */
5497 PVM pVM = mpVM;
5498 mpVM = NULL;
5499
5500 LogFlowThisFunc(("Destroying the VM...\n"));
5501
5502 alock.leave();
5503
5504 vrc = VMR3Destroy(pVM);
5505
5506 /* take the lock again */
5507 alock.enter();
5508
5509 /* advance percent count */
5510 if (aProgress)
5511 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5512
5513 if (RT_SUCCESS(vrc))
5514 {
5515 LogFlowThisFunc(("Machine has been destroyed (mMachineState=%d)\n",
5516 mMachineState));
5517 /* Note: the Console-level machine state change happens on the
5518 * VMSTATE_TERMINATE state change in vmstateChangeCallback(). If
5519 * powerDown() is called from EMT (i.e. from vmstateChangeCallback()
5520 * on receiving VM-initiated VMSTATE_OFF), VMSTATE_TERMINATE hasn't
5521 * occurred yet. This is okay, because mMachineState is already
5522 * Stopping in this case, so any other attempt to call PowerDown()
5523 * will be rejected. */
5524 }
5525 else
5526 {
5527 /* bad bad bad, but what to do? */
5528 mpVM = pVM;
5529 rc = setError(VBOX_E_VM_ERROR,
5530 tr("Could not destroy the machine. (Error: %Rrc)"),
5531 vrc);
5532 }
5533
5534 /* Complete the detaching of the USB devices. */
5535 if (fHasUSBController)
5536 detachAllUSBDevices(true /* aDone */);
5537
5538 /* advance percent count */
5539 if (aProgress)
5540 aProgress->SetCurrentOperationProgress(99 * (++ step) / StepCount );
5541 }
5542 else
5543 {
5544 rc = setError(VBOX_E_VM_ERROR,
5545 tr("Could not power off the machine. (Error: %Rrc)"),
5546 vrc);
5547 }
5548
5549 /* Finished with destruction. Note that if something impossible happened and
5550 * we've failed to destroy the VM, mVMDestroying will remain true and
5551 * mMachineState will be something like Stopping, so most Console methods
5552 * will return an error to the caller. */
5553 if (mpVM == NULL)
5554 mVMDestroying = false;
5555
5556 if (SUCCEEDED(rc))
5557 {
5558 /* uninit dynamically allocated members of mCallbackData */
5559 if (mCallbackData.mpsc.valid)
5560 {
5561 if (mCallbackData.mpsc.shape != NULL)
5562 RTMemFree(mCallbackData.mpsc.shape);
5563 }
5564 memset(&mCallbackData, 0, sizeof(mCallbackData));
5565 }
5566
5567 /* complete the progress */
5568 if (aProgress)
5569 aProgress->notifyComplete(rc);
5570
5571 LogFlowThisFuncLeave();
5572 return rc;
5573}
5574
5575/**
5576 * @note Locks this object for writing.
5577 */
5578HRESULT Console::setMachineState(MachineState_T aMachineState,
5579 bool aUpdateServer /* = true */)
5580{
5581 AutoCaller autoCaller(this);
5582 AssertComRCReturnRC(autoCaller.rc());
5583
5584 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
5585
5586 HRESULT rc = S_OK;
5587
5588 if (mMachineState != aMachineState)
5589 {
5590 LogThisFunc(("machineState=%s -> %s aUpdateServer=%RTbool\n",
5591 Global::stringifyMachineState(mMachineState), Global::stringifyMachineState(aMachineState), aUpdateServer));
5592 mMachineState = aMachineState;
5593
5594 /// @todo (dmik)
5595 // possibly, we need to redo onStateChange() using the dedicated
5596 // Event thread, like it is done in VirtualBox. This will make it
5597 // much safer (no deadlocks possible if someone tries to use the
5598 // console from the callback), however, listeners will lose the
5599 // ability to synchronously react to state changes (is it really
5600 // necessary??)
5601 LogFlowThisFunc(("Doing onStateChange()...\n"));
5602 onStateChange(aMachineState);
5603 LogFlowThisFunc(("Done onStateChange()\n"));
5604
5605 if (aUpdateServer)
5606 {
5607 /* Server notification MUST be done from under the lock; otherwise
5608 * the machine state here and on the server might go out of sync
5609 * which can lead to various unexpected results (like the machine
5610 * state being >= MachineState_Running on the server, while the
5611 * session state is already SessionState_Closed at the same time
5612 * there).
5613 *
5614 * Cross-lock conditions should be carefully watched out: calling
5615 * UpdateState we will require Machine and SessionMachine locks
5616 * (remember that here we're holding the Console lock here, and also
5617 * all locks that have been entered by the thread before calling
5618 * this method).
5619 */
5620 LogFlowThisFunc(("Doing mControl->UpdateState()...\n"));
5621 rc = mControl->UpdateState(aMachineState);
5622 LogFlowThisFunc(("mControl->UpdateState()=%08X\n", rc));
5623 }
5624 }
5625
5626 return rc;
5627}
5628
5629/**
5630 * Searches for a shared folder with the given logical name
5631 * in the collection of shared folders.
5632 *
5633 * @param aName logical name of the shared folder
5634 * @param aSharedFolder where to return the found object
5635 * @param aSetError whether to set the error info if the folder is
5636 * not found
5637 * @return
5638 * S_OK when found or E_INVALIDARG when not found
5639 *
5640 * @note The caller must lock this object for writing.
5641 */
5642HRESULT Console::findSharedFolder(CBSTR aName,
5643 ComObjPtr<SharedFolder> &aSharedFolder,
5644 bool aSetError /* = false */)
5645{
5646 /* sanity check */
5647 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
5648
5649 SharedFolderMap::const_iterator it = mSharedFolders.find(aName);
5650 if (it != mSharedFolders.end())
5651 {
5652 aSharedFolder = it->second;
5653 return S_OK;
5654 }
5655
5656 if (aSetError)
5657 setError(VBOX_E_FILE_ERROR,
5658 tr("Could not find a shared folder named '%ls'."),
5659 aName);
5660
5661 return VBOX_E_FILE_ERROR;
5662}
5663
5664/**
5665 * Fetches the list of global or machine shared folders from the server.
5666 *
5667 * @param aGlobal true to fetch global folders.
5668 *
5669 * @note The caller must lock this object for writing.
5670 */
5671HRESULT Console::fetchSharedFolders(BOOL aGlobal)
5672{
5673 /* sanity check */
5674 AssertReturn(AutoCaller(this).state() == InInit ||
5675 isWriteLockOnCurrentThread(), E_FAIL);
5676
5677 /* protect mpVM (if not NULL) */
5678 AutoVMCallerQuietWeak autoVMCaller(this);
5679
5680 HRESULT rc = S_OK;
5681
5682 bool online = mpVM && autoVMCaller.isOk() && mVMMDev->isShFlActive();
5683
5684 if (aGlobal)
5685 {
5686 /// @todo grab & process global folders when they are done
5687 }
5688 else
5689 {
5690 SharedFolderDataMap oldFolders;
5691 if (online)
5692 oldFolders = mMachineSharedFolders;
5693
5694 mMachineSharedFolders.clear();
5695
5696 SafeIfaceArray<ISharedFolder> folders;
5697 rc = mMachine->COMGETTER(SharedFolders)(ComSafeArrayAsOutParam(folders));
5698 AssertComRCReturnRC(rc);
5699
5700 for (size_t i = 0; i < folders.size(); ++i)
5701 {
5702 ComPtr<ISharedFolder> folder = folders[i];
5703
5704 Bstr name;
5705 Bstr hostPath;
5706 BOOL writable;
5707
5708 rc = folder->COMGETTER(Name)(name.asOutParam());
5709 if (FAILED(rc)) break;
5710 rc = folder->COMGETTER(HostPath)(hostPath.asOutParam());
5711 if (FAILED(rc)) break;
5712 rc = folder->COMGETTER(Writable)(&writable);
5713
5714 mMachineSharedFolders.insert(std::make_pair(name, SharedFolderData(hostPath, writable)));
5715
5716 /* send changes to HGCM if the VM is running */
5717 /// @todo report errors as runtime warnings through VMSetError
5718 if (online)
5719 {
5720 SharedFolderDataMap::iterator it = oldFolders.find(name);
5721 if (it == oldFolders.end() || it->second.mHostPath != hostPath)
5722 {
5723 /* a new machine folder is added or
5724 * the existing machine folder is changed */
5725 if (mSharedFolders.find(name) != mSharedFolders.end())
5726 ; /* the console folder exists, nothing to do */
5727 else
5728 {
5729 /* remove the old machine folder (when changed)
5730 * or the global folder if any (when new) */
5731 if (it != oldFolders.end() ||
5732 mGlobalSharedFolders.find(name) !=
5733 mGlobalSharedFolders.end())
5734 rc = removeSharedFolder(name);
5735 /* create the new machine folder */
5736 rc = createSharedFolder(name, SharedFolderData(hostPath, writable));
5737 }
5738 }
5739 /* forget the processed (or identical) folder */
5740 if (it != oldFolders.end())
5741 oldFolders.erase(it);
5742
5743 rc = S_OK;
5744 }
5745 }
5746
5747 AssertComRCReturnRC(rc);
5748
5749 /* process outdated (removed) folders */
5750 /// @todo report errors as runtime warnings through VMSetError
5751 if (online)
5752 {
5753 for (SharedFolderDataMap::const_iterator it = oldFolders.begin();
5754 it != oldFolders.end(); ++ it)
5755 {
5756 if (mSharedFolders.find(it->first) != mSharedFolders.end())
5757 ; /* the console folder exists, nothing to do */
5758 else
5759 {
5760 /* remove the outdated machine folder */
5761 rc = removeSharedFolder(it->first);
5762 /* create the global folder if there is any */
5763 SharedFolderDataMap::const_iterator git =
5764 mGlobalSharedFolders.find(it->first);
5765 if (git != mGlobalSharedFolders.end())
5766 rc = createSharedFolder(git->first, git->second);
5767 }
5768 }
5769
5770 rc = S_OK;
5771 }
5772 }
5773
5774 return rc;
5775}
5776
5777/**
5778 * Searches for a shared folder with the given name in the list of machine
5779 * shared folders and then in the list of the global shared folders.
5780 *
5781 * @param aName Name of the folder to search for.
5782 * @param aIt Where to store the pointer to the found folder.
5783 * @return @c true if the folder was found and @c false otherwise.
5784 *
5785 * @note The caller must lock this object for reading.
5786 */
5787bool Console::findOtherSharedFolder(IN_BSTR aName,
5788 SharedFolderDataMap::const_iterator &aIt)
5789{
5790 /* sanity check */
5791 AssertReturn(isWriteLockOnCurrentThread(), false);
5792
5793 /* first, search machine folders */
5794 aIt = mMachineSharedFolders.find(aName);
5795 if (aIt != mMachineSharedFolders.end())
5796 return true;
5797
5798 /* second, search machine folders */
5799 aIt = mGlobalSharedFolders.find(aName);
5800 if (aIt != mGlobalSharedFolders.end())
5801 return true;
5802
5803 return false;
5804}
5805
5806/**
5807 * Calls the HGCM service to add a shared folder definition.
5808 *
5809 * @param aName Shared folder name.
5810 * @param aHostPath Shared folder path.
5811 *
5812 * @note Must be called from under AutoVMCaller and when mpVM != NULL!
5813 * @note Doesn't lock anything.
5814 */
5815HRESULT Console::createSharedFolder(CBSTR aName, SharedFolderData aData)
5816{
5817 ComAssertRet(aName && *aName, E_FAIL);
5818 ComAssertRet(aData.mHostPath, E_FAIL);
5819
5820 /* sanity checks */
5821 AssertReturn(mpVM, E_FAIL);
5822 AssertReturn(mVMMDev->isShFlActive(), E_FAIL);
5823
5824 VBOXHGCMSVCPARM parms[SHFL_CPARMS_ADD_MAPPING];
5825 SHFLSTRING *pFolderName, *pMapName;
5826 size_t cbString;
5827
5828 Log(("Adding shared folder '%ls' -> '%ls'\n", aName, aData.mHostPath.raw()));
5829
5830 cbString = (RTUtf16Len(aData.mHostPath) + 1) * sizeof(RTUTF16);
5831 if (cbString >= UINT16_MAX)
5832 return setError(E_INVALIDARG, tr("The name is too long"));
5833 pFolderName = (SHFLSTRING *) RTMemAllocZ(sizeof(SHFLSTRING) + cbString);
5834 Assert(pFolderName);
5835 memcpy(pFolderName->String.ucs2, aData.mHostPath, cbString);
5836
5837 pFolderName->u16Size = (uint16_t)cbString;
5838 pFolderName->u16Length = (uint16_t)cbString - sizeof(RTUTF16);
5839
5840 parms[0].type = VBOX_HGCM_SVC_PARM_PTR;
5841 parms[0].u.pointer.addr = pFolderName;
5842 parms[0].u.pointer.size = sizeof(SHFLSTRING) + (uint16_t)cbString;
5843
5844 cbString = (RTUtf16Len(aName) + 1) * sizeof(RTUTF16);
5845 if (cbString >= UINT16_MAX)
5846 {
5847 RTMemFree(pFolderName);
5848 return setError(E_INVALIDARG, tr("The host path is too long"));
5849 }
5850 pMapName = (SHFLSTRING *) RTMemAllocZ(sizeof(SHFLSTRING) + cbString);
5851 Assert(pMapName);
5852 memcpy(pMapName->String.ucs2, aName, cbString);
5853
5854 pMapName->u16Size = (uint16_t)cbString;
5855 pMapName->u16Length = (uint16_t)cbString - sizeof(RTUTF16);
5856
5857 parms[1].type = VBOX_HGCM_SVC_PARM_PTR;
5858 parms[1].u.pointer.addr = pMapName;
5859 parms[1].u.pointer.size = sizeof(SHFLSTRING) + (uint16_t)cbString;
5860
5861 parms[2].type = VBOX_HGCM_SVC_PARM_32BIT;
5862 parms[2].u.uint32 = aData.mWritable;
5863
5864 int vrc = mVMMDev->hgcmHostCall("VBoxSharedFolders",
5865 SHFL_FN_ADD_MAPPING,
5866 SHFL_CPARMS_ADD_MAPPING, &parms[0]);
5867 RTMemFree(pFolderName);
5868 RTMemFree(pMapName);
5869
5870 if (RT_FAILURE(vrc))
5871 return setError(E_FAIL,
5872 tr("Could not create a shared folder '%ls' mapped to '%ls' (%Rrc)"),
5873 aName, aData.mHostPath.raw(), vrc);
5874
5875 return S_OK;
5876}
5877
5878/**
5879 * Calls the HGCM service to remove the shared folder definition.
5880 *
5881 * @param aName Shared folder name.
5882 *
5883 * @note Must be called from under AutoVMCaller and when mpVM != NULL!
5884 * @note Doesn't lock anything.
5885 */
5886HRESULT Console::removeSharedFolder(CBSTR aName)
5887{
5888 ComAssertRet(aName && *aName, E_FAIL);
5889
5890 /* sanity checks */
5891 AssertReturn(mpVM, E_FAIL);
5892 AssertReturn(mVMMDev->isShFlActive(), E_FAIL);
5893
5894 VBOXHGCMSVCPARM parms;
5895 SHFLSTRING *pMapName;
5896 size_t cbString;
5897
5898 Log(("Removing shared folder '%ls'\n", aName));
5899
5900 cbString = (RTUtf16Len(aName) + 1) * sizeof(RTUTF16);
5901 if (cbString >= UINT16_MAX)
5902 return setError(E_INVALIDARG, tr("The name is too long"));
5903 pMapName = (SHFLSTRING *) RTMemAllocZ(sizeof(SHFLSTRING) + cbString);
5904 Assert(pMapName);
5905 memcpy(pMapName->String.ucs2, aName, cbString);
5906
5907 pMapName->u16Size = (uint16_t)cbString;
5908 pMapName->u16Length = (uint16_t)cbString - sizeof(RTUTF16);
5909
5910 parms.type = VBOX_HGCM_SVC_PARM_PTR;
5911 parms.u.pointer.addr = pMapName;
5912 parms.u.pointer.size = sizeof(SHFLSTRING) + (uint16_t)cbString;
5913
5914 int vrc = mVMMDev->hgcmHostCall("VBoxSharedFolders",
5915 SHFL_FN_REMOVE_MAPPING,
5916 1, &parms);
5917 RTMemFree(pMapName);
5918 if (RT_FAILURE(vrc))
5919 return setError(E_FAIL,
5920 tr("Could not remove the shared folder '%ls' (%Rrc)"),
5921 aName, vrc);
5922
5923 return S_OK;
5924}
5925
5926/**
5927 * VM state callback function. Called by the VMM
5928 * using its state machine states.
5929 *
5930 * Primarily used to handle VM initiated power off, suspend and state saving,
5931 * but also for doing termination completed work (VMSTATE_TERMINATE).
5932 *
5933 * In general this function is called in the context of the EMT.
5934 *
5935 * @param aVM The VM handle.
5936 * @param aState The new state.
5937 * @param aOldState The old state.
5938 * @param aUser The user argument (pointer to the Console object).
5939 *
5940 * @note Locks the Console object for writing.
5941 */
5942DECLCALLBACK(void) Console::vmstateChangeCallback(PVM aVM,
5943 VMSTATE aState,
5944 VMSTATE aOldState,
5945 void *aUser)
5946{
5947 LogFlowFunc(("Changing state from %s to %s (aVM=%p)\n",
5948 VMR3GetStateName(aOldState), VMR3GetStateName(aState), aVM));
5949
5950 Console *that = static_cast<Console *>(aUser);
5951 AssertReturnVoid(that);
5952
5953 AutoCaller autoCaller(that);
5954
5955 /* Note that we must let this method proceed even if Console::uninit() has
5956 * been already called. In such case this VMSTATE change is a result of:
5957 * 1) powerDown() called from uninit() itself, or
5958 * 2) VM-(guest-)initiated power off. */
5959 AssertReturnVoid( autoCaller.isOk()
5960 || autoCaller.state() == InUninit);
5961
5962 switch (aState)
5963 {
5964 /*
5965 * The VM has terminated
5966 */
5967 case VMSTATE_OFF:
5968 {
5969 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
5970
5971 if (that->mVMStateChangeCallbackDisabled)
5972 break;
5973
5974 /* Do we still think that it is running? It may happen if this is a
5975 * VM-(guest-)initiated shutdown/poweroff.
5976 */
5977 if ( that->mMachineState != MachineState_Stopping
5978 && that->mMachineState != MachineState_Saving
5979 && that->mMachineState != MachineState_Restoring
5980 && that->mMachineState != MachineState_TeleportingIn
5981 && that->mMachineState != MachineState_TeleportingPausedVM
5982 && !that->mVMIsAlreadyPoweringOff
5983 )
5984 {
5985 LogFlowFunc(("VM has powered itself off but Console still thinks it is running. Notifying.\n"));
5986
5987 /* prevent powerDown() from calling VMR3PowerOff() again */
5988 Assert(that->mVMPoweredOff == false);
5989 that->mVMPoweredOff = true;
5990
5991 /* we are stopping now */
5992 that->setMachineState(MachineState_Stopping);
5993
5994 /* Setup task object and thread to carry out the operation
5995 * asynchronously (if we call powerDown() right here but there
5996 * is one or more mpVM callers (added with addVMCaller()) we'll
5997 * deadlock).
5998 */
5999 std::auto_ptr<VMProgressTask> task(new VMProgressTask(that, NULL /* aProgress */,
6000 true /* aUsesVMPtr */));
6001
6002 /* If creating a task is falied, this can currently mean one of
6003 * two: either Console::uninit() has been called just a ms
6004 * before (so a powerDown() call is already on the way), or
6005 * powerDown() itself is being already executed. Just do
6006 * nothing.
6007 */
6008 if (!task->isOk())
6009 {
6010 LogFlowFunc(("Console is already being uninitialized.\n"));
6011 break;
6012 }
6013
6014 int vrc = RTThreadCreate(NULL, Console::powerDownThread,
6015 (void *) task.get(), 0,
6016 RTTHREADTYPE_MAIN_WORKER, 0,
6017 "VMPowerDown");
6018 AssertMsgRCBreak(vrc, ("Could not create VMPowerDown thread (%Rrc)\n", vrc));
6019
6020 /* task is now owned by powerDownThread(), so release it */
6021 task.release();
6022 }
6023 break;
6024 }
6025
6026 /* The VM has been completely destroyed.
6027 *
6028 * Note: This state change can happen at two points:
6029 * 1) At the end of VMR3Destroy() if it was not called from EMT.
6030 * 2) At the end of vmR3EmulationThread if VMR3Destroy() was
6031 * called by EMT.
6032 */
6033 case VMSTATE_TERMINATED:
6034 {
6035 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6036
6037 if (that->mVMStateChangeCallbackDisabled)
6038 break;
6039
6040 /* Terminate host interface networking. If aVM is NULL, we've been
6041 * manually called from powerUpThread() either before calling
6042 * VMR3Create() or after VMR3Create() failed, so no need to touch
6043 * networking.
6044 */
6045 if (aVM)
6046 that->powerDownHostInterfaces();
6047
6048 /* From now on the machine is officially powered down or remains in
6049 * the Saved state.
6050 */
6051 switch (that->mMachineState)
6052 {
6053 default:
6054 AssertFailed();
6055 /* fall through */
6056 case MachineState_Stopping:
6057 /* successfully powered down */
6058 that->setMachineState(MachineState_PoweredOff);
6059 break;
6060 case MachineState_Saving:
6061 /* successfully saved (note that the machine is already in
6062 * the Saved state on the server due to EndSavingState()
6063 * called from saveStateThread(), so only change the local
6064 * state) */
6065 that->setMachineStateLocally(MachineState_Saved);
6066 break;
6067 case MachineState_Starting:
6068 /* failed to start, but be patient: set back to PoweredOff
6069 * (for similarity with the below) */
6070 that->setMachineState(MachineState_PoweredOff);
6071 break;
6072 case MachineState_Restoring:
6073 /* failed to load the saved state file, but be patient: set
6074 * back to Saved (to preserve the saved state file) */
6075 that->setMachineState(MachineState_Saved);
6076 break;
6077 case MachineState_TeleportingIn:
6078 /* Teleportation failed or was cancelled. Back to powered off. */
6079 that->setMachineState(MachineState_PoweredOff);
6080 break;
6081 case MachineState_TeleportingPausedVM:
6082 /* Successfully teleported the VM. */
6083 that->setMachineState(MachineState_Teleported);
6084 break;
6085 }
6086 break;
6087 }
6088
6089 case VMSTATE_SUSPENDED:
6090 {
6091 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6092
6093 if (that->mVMStateChangeCallbackDisabled)
6094 break;
6095
6096 switch (that->mMachineState)
6097 {
6098 case MachineState_Teleporting:
6099 that->setMachineState(MachineState_TeleportingPausedVM);
6100 break;
6101
6102 case MachineState_LiveSnapshotting:
6103 that->setMachineState(MachineState_Saving);
6104 break;
6105
6106 case MachineState_TeleportingPausedVM:
6107 case MachineState_Saving:
6108 case MachineState_Restoring:
6109 case MachineState_Stopping:
6110 case MachineState_TeleportingIn:
6111 /* The worker threads handles the transition. */
6112 break;
6113
6114 default:
6115 AssertMsgFailed(("%s\n", Global::stringifyMachineState(that->mMachineState)));
6116 case MachineState_Running:
6117 that->setMachineState(MachineState_Paused);
6118 break;
6119 }
6120 break;
6121 }
6122
6123 case VMSTATE_SUSPENDED_LS:
6124 case VMSTATE_SUSPENDED_EXT_LS:
6125 {
6126 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6127 if (that->mVMStateChangeCallbackDisabled)
6128 break;
6129 switch (that->mMachineState)
6130 {
6131 case MachineState_Teleporting:
6132 that->setMachineState(MachineState_TeleportingPausedVM);
6133 break;
6134
6135 case MachineState_LiveSnapshotting:
6136 that->setMachineState(MachineState_Saving);
6137 break;
6138
6139 case MachineState_TeleportingPausedVM:
6140 case MachineState_Saving:
6141 /* ignore */
6142 break;
6143
6144 default:
6145 AssertMsgFailed(("%s/%s -> %s\n", Global::stringifyMachineState(that->mMachineState), VMR3GetStateName(aOldState), VMR3GetStateName(aState) ));
6146 that->setMachineState(MachineState_Paused);
6147 break;
6148 }
6149 break;
6150 }
6151
6152 case VMSTATE_RUNNING:
6153 {
6154 if ( aOldState == VMSTATE_POWERING_ON
6155 || aOldState == VMSTATE_RESUMING)
6156 {
6157 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6158
6159 if (that->mVMStateChangeCallbackDisabled)
6160 break;
6161
6162 Assert( ( ( that->mMachineState == MachineState_Starting
6163 || that->mMachineState == MachineState_Paused)
6164 && aOldState == VMSTATE_POWERING_ON)
6165 || ( ( that->mMachineState == MachineState_Restoring
6166 || that->mMachineState == MachineState_TeleportingIn
6167 || that->mMachineState == MachineState_Paused
6168 || that->mMachineState == MachineState_Saving
6169 )
6170 && aOldState == VMSTATE_RESUMING));
6171 that->setMachineState(MachineState_Running);
6172 }
6173
6174 break;
6175 }
6176
6177 case VMSTATE_RUNNING_LS:
6178 AssertMsg( that->mMachineState == MachineState_LiveSnapshotting
6179 || that->mMachineState == MachineState_Teleporting,
6180 ("%s/%s -> %s\n", Global::stringifyMachineState(that->mMachineState), VMR3GetStateName(aOldState), VMR3GetStateName(aState) ));
6181 break;
6182
6183 case VMSTATE_FATAL_ERROR:
6184 {
6185 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6186
6187 if (that->mVMStateChangeCallbackDisabled)
6188 break;
6189
6190 /* Fatal errors are only for running VMs. */
6191 Assert(Global::IsOnline(that->mMachineState));
6192
6193 /* Note! 'Pause' is used here in want of something better. There
6194 * are currently only two places where fatal errors might be
6195 * raised, so it is not worth adding a new externally
6196 * visible state for this yet. */
6197 that->setMachineState(MachineState_Paused);
6198 break;
6199 }
6200
6201 case VMSTATE_GURU_MEDITATION:
6202 {
6203 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6204
6205 if (that->mVMStateChangeCallbackDisabled)
6206 break;
6207
6208 /* Guru are only for running VMs */
6209 Assert(Global::IsOnline(that->mMachineState));
6210
6211 that->setMachineState(MachineState_Stuck);
6212 break;
6213 }
6214
6215 default: /* shut up gcc */
6216 break;
6217 }
6218}
6219
6220#ifdef VBOX_WITH_USB
6221
6222/**
6223 * Sends a request to VMM to attach the given host device.
6224 * After this method succeeds, the attached device will appear in the
6225 * mUSBDevices collection.
6226 *
6227 * @param aHostDevice device to attach
6228 *
6229 * @note Synchronously calls EMT.
6230 * @note Must be called from under this object's lock.
6231 */
6232HRESULT Console::attachUSBDevice(IUSBDevice *aHostDevice, ULONG aMaskedIfs)
6233{
6234 AssertReturn(aHostDevice, E_FAIL);
6235 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6236
6237 /* still want a lock object because we need to leave it */
6238 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6239
6240 HRESULT hrc;
6241
6242 /*
6243 * Get the address and the Uuid, and call the pfnCreateProxyDevice roothub
6244 * method in EMT (using usbAttachCallback()).
6245 */
6246 Bstr BstrAddress;
6247 hrc = aHostDevice->COMGETTER(Address)(BstrAddress.asOutParam());
6248 ComAssertComRCRetRC(hrc);
6249
6250 Utf8Str Address(BstrAddress);
6251
6252 Bstr id;
6253 hrc = aHostDevice->COMGETTER(Id)(id.asOutParam());
6254 ComAssertComRCRetRC(hrc);
6255 Guid uuid(id);
6256
6257 BOOL fRemote = FALSE;
6258 hrc = aHostDevice->COMGETTER(Remote)(&fRemote);
6259 ComAssertComRCRetRC(hrc);
6260
6261 /* protect mpVM */
6262 AutoVMCaller autoVMCaller(this);
6263 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
6264
6265 LogFlowThisFunc(("Proxying USB device '%s' {%RTuuid}...\n",
6266 Address.raw(), uuid.ptr()));
6267
6268 /* leave the lock before a VMR3* call (EMT will call us back)! */
6269 alock.leave();
6270
6271/** @todo just do everything here and only wrap the PDMR3Usb call. That'll offload some notification stuff from the EMT thread. */
6272 int vrc = VMR3ReqCallWait(mpVM, VMCPUID_ANY,
6273 (PFNRT) usbAttachCallback, 6, this, aHostDevice, uuid.ptr(), fRemote, Address.raw(), aMaskedIfs);
6274
6275 /* restore the lock */
6276 alock.enter();
6277
6278 /* hrc is S_OK here */
6279
6280 if (RT_FAILURE(vrc))
6281 {
6282 LogWarningThisFunc(("Failed to create proxy device for '%s' {%RTuuid} (%Rrc)\n",
6283 Address.raw(), uuid.ptr(), vrc));
6284
6285 switch (vrc)
6286 {
6287 case VERR_VUSB_NO_PORTS:
6288 hrc = setError(E_FAIL,
6289 tr("Failed to attach the USB device. (No available ports on the USB controller)."));
6290 break;
6291 case VERR_VUSB_USBFS_PERMISSION:
6292 hrc = setError(E_FAIL,
6293 tr("Not permitted to open the USB device, check usbfs options"));
6294 break;
6295 default:
6296 hrc = setError(E_FAIL,
6297 tr("Failed to create a proxy device for the USB device. (Error: %Rrc)"),
6298 vrc);
6299 break;
6300 }
6301 }
6302
6303 return hrc;
6304}
6305
6306/**
6307 * USB device attach callback used by AttachUSBDevice().
6308 * Note that AttachUSBDevice() doesn't return until this callback is executed,
6309 * so we don't use AutoCaller and don't care about reference counters of
6310 * interface pointers passed in.
6311 *
6312 * @thread EMT
6313 * @note Locks the console object for writing.
6314 */
6315//static
6316DECLCALLBACK(int)
6317Console::usbAttachCallback(Console *that, IUSBDevice *aHostDevice, PCRTUUID aUuid, bool aRemote, const char *aAddress, ULONG aMaskedIfs)
6318{
6319 LogFlowFuncEnter();
6320 LogFlowFunc(("that={%p}\n", that));
6321
6322 AssertReturn(that && aUuid, VERR_INVALID_PARAMETER);
6323
6324 void *pvRemoteBackend = NULL;
6325 if (aRemote)
6326 {
6327 RemoteUSBDevice *pRemoteUSBDevice = static_cast<RemoteUSBDevice *>(aHostDevice);
6328 Guid guid(*aUuid);
6329
6330 pvRemoteBackend = that->consoleVRDPServer()->USBBackendRequestPointer(pRemoteUSBDevice->clientId(), &guid);
6331 if (!pvRemoteBackend)
6332 return VERR_INVALID_PARAMETER; /* The clientId is invalid then. */
6333 }
6334
6335 USHORT portVersion = 1;
6336 HRESULT hrc = aHostDevice->COMGETTER(PortVersion)(&portVersion);
6337 AssertComRCReturn(hrc, VERR_GENERAL_FAILURE);
6338 Assert(portVersion == 1 || portVersion == 2);
6339
6340 int vrc = PDMR3USBCreateProxyDevice(that->mpVM, aUuid, aRemote, aAddress, pvRemoteBackend,
6341 portVersion == 1 ? VUSB_STDVER_11 : VUSB_STDVER_20, aMaskedIfs);
6342 if (RT_SUCCESS(vrc))
6343 {
6344 /* Create a OUSBDevice and add it to the device list */
6345 ComObjPtr<OUSBDevice> device;
6346 device.createObject();
6347 hrc = device->init(aHostDevice);
6348 AssertComRC(hrc);
6349
6350 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6351 that->mUSBDevices.push_back(device);
6352 LogFlowFunc(("Attached device {%RTuuid}\n", device->id().raw()));
6353
6354 /* notify callbacks */
6355 that->onUSBDeviceStateChange(device, true /* aAttached */, NULL);
6356 }
6357
6358 LogFlowFunc(("vrc=%Rrc\n", vrc));
6359 LogFlowFuncLeave();
6360 return vrc;
6361}
6362
6363/**
6364 * Sends a request to VMM to detach the given host device. After this method
6365 * succeeds, the detached device will disappear from the mUSBDevices
6366 * collection.
6367 *
6368 * @param aIt Iterator pointing to the device to detach.
6369 *
6370 * @note Synchronously calls EMT.
6371 * @note Must be called from under this object's lock.
6372 */
6373HRESULT Console::detachUSBDevice(USBDeviceList::iterator &aIt)
6374{
6375 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6376
6377 /* still want a lock object because we need to leave it */
6378 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6379
6380 /* protect mpVM */
6381 AutoVMCaller autoVMCaller(this);
6382 if (FAILED(autoVMCaller.rc())) return autoVMCaller.rc();
6383
6384 /* if the device is attached, then there must at least one USB hub. */
6385 AssertReturn(PDMR3USBHasHub(mpVM), E_FAIL);
6386
6387 LogFlowThisFunc(("Detaching USB proxy device {%RTuuid}...\n",
6388 (*aIt)->id().raw()));
6389
6390 /* leave the lock before a VMR3* call (EMT will call us back)! */
6391 alock.leave();
6392
6393/** @todo just do everything here and only wrap the PDMR3Usb call. That'll offload some notification stuff from the EMT thread. */
6394 int vrc = VMR3ReqCallWait(mpVM, VMCPUID_ANY,
6395 (PFNRT) usbDetachCallback, 4, this, &aIt, (*aIt)->id().raw());
6396 ComAssertRCRet(vrc, E_FAIL);
6397
6398 return S_OK;
6399}
6400
6401/**
6402 * USB device detach callback used by DetachUSBDevice().
6403 * Note that DetachUSBDevice() doesn't return until this callback is executed,
6404 * so we don't use AutoCaller and don't care about reference counters of
6405 * interface pointers passed in.
6406 *
6407 * @thread EMT
6408 * @note Locks the console object for writing.
6409 */
6410//static
6411DECLCALLBACK(int)
6412Console::usbDetachCallback(Console *that, USBDeviceList::iterator *aIt, PCRTUUID aUuid)
6413{
6414 LogFlowFuncEnter();
6415 LogFlowFunc(("that={%p}\n", that));
6416
6417 AssertReturn(that && aUuid, VERR_INVALID_PARAMETER);
6418 ComObjPtr<OUSBDevice> device = **aIt;
6419
6420 /*
6421 * If that was a remote device, release the backend pointer.
6422 * The pointer was requested in usbAttachCallback.
6423 */
6424 BOOL fRemote = FALSE;
6425
6426 HRESULT hrc2 = (**aIt)->COMGETTER(Remote)(&fRemote);
6427 ComAssertComRC(hrc2);
6428
6429 if (fRemote)
6430 {
6431 Guid guid(*aUuid);
6432 that->consoleVRDPServer()->USBBackendReleasePointer(&guid);
6433 }
6434
6435 int vrc = PDMR3USBDetachDevice(that->mpVM, aUuid);
6436
6437 if (RT_SUCCESS(vrc))
6438 {
6439 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
6440
6441 /* Remove the device from the collection */
6442 that->mUSBDevices.erase(*aIt);
6443 LogFlowFunc(("Detached device {%RTuuid}\n", device->id().raw()));
6444
6445 /* notify callbacks */
6446 that->onUSBDeviceStateChange(device, false /* aAttached */, NULL);
6447 }
6448
6449 LogFlowFunc(("vrc=%Rrc\n", vrc));
6450 LogFlowFuncLeave();
6451 return vrc;
6452}
6453
6454#endif /* VBOX_WITH_USB */
6455#if ((defined(RT_OS_LINUX) && !defined(VBOX_WITH_NETFLT)) || defined(RT_OS_FREEBSD))
6456
6457/**
6458 * Helper function to handle host interface device creation and attachment.
6459 *
6460 * @param networkAdapter the network adapter which attachment should be reset
6461 * @return COM status code
6462 *
6463 * @note The caller must lock this object for writing.
6464 *
6465 * @todo Move this back into the driver!
6466 */
6467HRESULT Console::attachToTapInterface(INetworkAdapter *networkAdapter)
6468{
6469 LogFlowThisFunc(("\n"));
6470 /* sanity check */
6471 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6472
6473# ifdef VBOX_STRICT
6474 /* paranoia */
6475 NetworkAttachmentType_T attachment;
6476 networkAdapter->COMGETTER(AttachmentType)(&attachment);
6477 Assert(attachment == NetworkAttachmentType_Bridged);
6478# endif /* VBOX_STRICT */
6479
6480 HRESULT rc = S_OK;
6481
6482 ULONG slot = 0;
6483 rc = networkAdapter->COMGETTER(Slot)(&slot);
6484 AssertComRC(rc);
6485
6486# ifdef RT_OS_LINUX
6487 /*
6488 * Allocate a host interface device
6489 */
6490 int rcVBox = RTFileOpen(&maTapFD[slot], "/dev/net/tun",
6491 RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_INHERIT);
6492 if (RT_SUCCESS(rcVBox))
6493 {
6494 /*
6495 * Set/obtain the tap interface.
6496 */
6497 struct ifreq IfReq;
6498 memset(&IfReq, 0, sizeof(IfReq));
6499 /* The name of the TAP interface we are using */
6500 Bstr tapDeviceName;
6501 rc = networkAdapter->COMGETTER(HostInterface)(tapDeviceName.asOutParam());
6502 if (FAILED(rc))
6503 tapDeviceName.setNull(); /* Is this necessary? */
6504 if (tapDeviceName.isEmpty())
6505 {
6506 LogRel(("No TAP device name was supplied.\n"));
6507 rc = setError(E_FAIL, tr("No TAP device name was supplied for the host networking interface"));
6508 }
6509
6510 if (SUCCEEDED(rc))
6511 {
6512 /* If we are using a static TAP device then try to open it. */
6513 Utf8Str str(tapDeviceName);
6514 if (str.length() <= sizeof(IfReq.ifr_name))
6515 strcpy(IfReq.ifr_name, str.raw());
6516 else
6517 memcpy(IfReq.ifr_name, str.raw(), sizeof(IfReq.ifr_name) - 1); /** @todo bitch about names which are too long... */
6518 IfReq.ifr_flags = IFF_TAP | IFF_NO_PI;
6519 rcVBox = ioctl(maTapFD[slot], TUNSETIFF, &IfReq);
6520 if (rcVBox != 0)
6521 {
6522 LogRel(("Failed to open the host network interface %ls\n", tapDeviceName.raw()));
6523 rc = setError(E_FAIL,
6524 tr("Failed to open the host network interface %ls"),
6525 tapDeviceName.raw());
6526 }
6527 }
6528 if (SUCCEEDED(rc))
6529 {
6530 /*
6531 * Make it pollable.
6532 */
6533 if (fcntl(maTapFD[slot], F_SETFL, O_NONBLOCK) != -1)
6534 {
6535 Log(("attachToTapInterface: %RTfile %ls\n", maTapFD[slot], tapDeviceName.raw()));
6536 /*
6537 * Here is the right place to communicate the TAP file descriptor and
6538 * the host interface name to the server if/when it becomes really
6539 * necessary.
6540 */
6541 maTAPDeviceName[slot] = tapDeviceName;
6542 rcVBox = VINF_SUCCESS;
6543 }
6544 else
6545 {
6546 int iErr = errno;
6547
6548 LogRel(("Configuration error: Failed to configure /dev/net/tun non blocking. Error: %s\n", strerror(iErr)));
6549 rcVBox = VERR_HOSTIF_BLOCKING;
6550 rc = setError(E_FAIL,
6551 tr("could not set up the host networking device for non blocking access: %s"),
6552 strerror(errno));
6553 }
6554 }
6555 }
6556 else
6557 {
6558 LogRel(("Configuration error: Failed to open /dev/net/tun rc=%Rrc\n", rcVBox));
6559 switch (rcVBox)
6560 {
6561 case VERR_ACCESS_DENIED:
6562 /* will be handled by our caller */
6563 rc = rcVBox;
6564 break;
6565 default:
6566 rc = setError(E_FAIL,
6567 tr("Could not set up the host networking device: %Rrc"),
6568 rcVBox);
6569 break;
6570 }
6571 }
6572
6573# elif defined(RT_OS_FREEBSD)
6574 /*
6575 * Set/obtain the tap interface.
6576 */
6577 /* The name of the TAP interface we are using */
6578 Bstr tapDeviceName;
6579 rc = networkAdapter->COMGETTER(HostInterface)(tapDeviceName.asOutParam());
6580 if (FAILED(rc))
6581 tapDeviceName.setNull(); /* Is this necessary? */
6582 if (tapDeviceName.isEmpty())
6583 {
6584 LogRel(("No TAP device name was supplied.\n"));
6585 rc = setError(E_FAIL, tr("No TAP device name was supplied for the host networking interface"));
6586 }
6587 char szTapdev[1024] = "/dev/";
6588 /* If we are using a static TAP device then try to open it. */
6589 Utf8Str str(tapDeviceName);
6590 if (str.length() + strlen(szTapdev) <= sizeof(szTapdev))
6591 strcat(szTapdev, str.raw());
6592 else
6593 memcpy(szTapdev + strlen(szTapdev), str.raw(), sizeof(szTapdev) - strlen(szTapdev) - 1); /** @todo bitch about names which are too long... */
6594 int rcVBox = RTFileOpen(&maTapFD[slot], szTapdev,
6595 RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_INHERIT | RTFILE_O_NON_BLOCK);
6596
6597 if (RT_SUCCESS(rcVBox))
6598 maTAPDeviceName[slot] = tapDeviceName;
6599 else
6600 {
6601 switch (rcVBox)
6602 {
6603 case VERR_ACCESS_DENIED:
6604 /* will be handled by our caller */
6605 rc = rcVBox;
6606 break;
6607 default:
6608 rc = setError(E_FAIL,
6609 tr("Failed to open the host network interface %ls"),
6610 tapDeviceName.raw());
6611 break;
6612 }
6613 }
6614# else
6615# error "huh?"
6616# endif
6617 /* in case of failure, cleanup. */
6618 if (RT_FAILURE(rcVBox) && SUCCEEDED(rc))
6619 {
6620 LogRel(("General failure attaching to host interface\n"));
6621 rc = setError(E_FAIL,
6622 tr("General failure attaching to host interface"));
6623 }
6624 LogFlowThisFunc(("rc=%d\n", rc));
6625 return rc;
6626}
6627
6628
6629/**
6630 * Helper function to handle detachment from a host interface
6631 *
6632 * @param networkAdapter the network adapter which attachment should be reset
6633 * @return COM status code
6634 *
6635 * @note The caller must lock this object for writing.
6636 *
6637 * @todo Move this back into the driver!
6638 */
6639HRESULT Console::detachFromTapInterface(INetworkAdapter *networkAdapter)
6640{
6641 /* sanity check */
6642 LogFlowThisFunc(("\n"));
6643 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6644
6645 HRESULT rc = S_OK;
6646# ifdef VBOX_STRICT
6647 /* paranoia */
6648 NetworkAttachmentType_T attachment;
6649 networkAdapter->COMGETTER(AttachmentType)(&attachment);
6650 Assert(attachment == NetworkAttachmentType_Bridged);
6651# endif /* VBOX_STRICT */
6652
6653 ULONG slot = 0;
6654 rc = networkAdapter->COMGETTER(Slot)(&slot);
6655 AssertComRC(rc);
6656
6657 /* is there an open TAP device? */
6658 if (maTapFD[slot] != NIL_RTFILE)
6659 {
6660 /*
6661 * Close the file handle.
6662 */
6663 Bstr tapDeviceName, tapTerminateApplication;
6664 bool isStatic = true;
6665 rc = networkAdapter->COMGETTER(HostInterface)(tapDeviceName.asOutParam());
6666 if (FAILED(rc) || tapDeviceName.isEmpty())
6667 {
6668 /* If the name is empty, this is a dynamic TAP device, so close it now,
6669 so that the termination script can remove the interface. Otherwise we still
6670 need the FD to pass to the termination script. */
6671 isStatic = false;
6672 int rcVBox = RTFileClose(maTapFD[slot]);
6673 AssertRC(rcVBox);
6674 maTapFD[slot] = NIL_RTFILE;
6675 }
6676 if (isStatic)
6677 {
6678 /* If we are using a static TAP device, we close it now, after having called the
6679 termination script. */
6680 int rcVBox = RTFileClose(maTapFD[slot]);
6681 AssertRC(rcVBox);
6682 }
6683 /* the TAP device name and handle are no longer valid */
6684 maTapFD[slot] = NIL_RTFILE;
6685 maTAPDeviceName[slot] = "";
6686 }
6687 LogFlowThisFunc(("returning %d\n", rc));
6688 return rc;
6689}
6690
6691#endif /* (RT_OS_LINUX || RT_OS_FREEBSD) && !VBOX_WITH_NETFLT */
6692
6693/**
6694 * Called at power down to terminate host interface networking.
6695 *
6696 * @note The caller must lock this object for writing.
6697 */
6698HRESULT Console::powerDownHostInterfaces()
6699{
6700 LogFlowThisFunc(("\n"));
6701
6702 /* sanity check */
6703 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
6704
6705 /*
6706 * host interface termination handling
6707 */
6708 HRESULT rc;
6709 for (ULONG slot = 0; slot < SchemaDefs::NetworkAdapterCount; slot ++)
6710 {
6711 ComPtr<INetworkAdapter> networkAdapter;
6712 rc = mMachine->GetNetworkAdapter(slot, networkAdapter.asOutParam());
6713 if (FAILED(rc)) break;
6714
6715 BOOL enabled = FALSE;
6716 networkAdapter->COMGETTER(Enabled)(&enabled);
6717 if (!enabled)
6718 continue;
6719
6720 NetworkAttachmentType_T attachment;
6721 networkAdapter->COMGETTER(AttachmentType)(&attachment);
6722 if (attachment == NetworkAttachmentType_Bridged)
6723 {
6724#if defined(RT_OS_LINUX) && !defined(VBOX_WITH_NETFLT)
6725 HRESULT rc2 = detachFromTapInterface(networkAdapter);
6726 if (FAILED(rc2) && SUCCEEDED(rc))
6727 rc = rc2;
6728#endif
6729 }
6730 }
6731
6732 return rc;
6733}
6734
6735
6736/**
6737 * Process callback handler for VMR3LoadFromFile, VMR3LoadFromStream, VMR3Save
6738 * and VMR3Teleport.
6739 *
6740 * @param pVM The VM handle.
6741 * @param uPercent Completetion precentage (0-100).
6742 * @param pvUser Pointer to the VMProgressTask structure.
6743 * @return VINF_SUCCESS.
6744 */
6745/*static*/
6746DECLCALLBACK(int) Console::stateProgressCallback(PVM pVM, unsigned uPercent, void *pvUser)
6747{
6748 VMProgressTask *task = static_cast<VMProgressTask *>(pvUser);
6749 AssertReturn(task, VERR_INVALID_PARAMETER);
6750
6751 /* update the progress object */
6752 if (task->mProgress)
6753 task->mProgress->SetCurrentOperationProgress(uPercent);
6754
6755 return VINF_SUCCESS;
6756}
6757
6758/**
6759 * VM error callback function. Called by the various VM components.
6760 *
6761 * @param pVM VM handle. Can be NULL if an error occurred before
6762 * successfully creating a VM.
6763 * @param pvUser Pointer to the VMProgressTask structure.
6764 * @param rc VBox status code.
6765 * @param pszFormat Printf-like error message.
6766 * @param args Various number of arguments for the error message.
6767 *
6768 * @thread EMT, VMPowerUp...
6769 *
6770 * @note The VMProgressTask structure modified by this callback is not thread
6771 * safe.
6772 */
6773/* static */ DECLCALLBACK(void)
6774Console::setVMErrorCallback(PVM pVM, void *pvUser, int rc, RT_SRC_POS_DECL,
6775 const char *pszFormat, va_list args)
6776{
6777 VMProgressTask *task = static_cast<VMProgressTask *>(pvUser);
6778 AssertReturnVoid(task);
6779
6780 /* we ignore RT_SRC_POS_DECL arguments to avoid confusion of end-users */
6781 va_list va2;
6782 va_copy(va2, args); /* Have to make a copy here or GCC will break. */
6783
6784 /* append to the existing error message if any */
6785 if (task->mErrorMsg.length())
6786 task->mErrorMsg = Utf8StrFmt("%s.\n%N (%Rrc)", task->mErrorMsg.raw(),
6787 pszFormat, &va2, rc, rc);
6788 else
6789 task->mErrorMsg = Utf8StrFmt("%N (%Rrc)",
6790 pszFormat, &va2, rc, rc);
6791
6792 va_end (va2);
6793}
6794
6795/**
6796 * VM runtime error callback function.
6797 * See VMSetRuntimeError for the detailed description of parameters.
6798 *
6799 * @param pVM The VM handle.
6800 * @param pvUser The user argument.
6801 * @param fFlags The action flags. See VMSETRTERR_FLAGS_*.
6802 * @param pszErrorId Error ID string.
6803 * @param pszFormat Error message format string.
6804 * @param va Error message arguments.
6805 * @thread EMT.
6806 */
6807/* static */ DECLCALLBACK(void)
6808Console::setVMRuntimeErrorCallback(PVM pVM, void *pvUser, uint32_t fFlags,
6809 const char *pszErrorId,
6810 const char *pszFormat, va_list va)
6811{
6812 bool const fFatal = !!(fFlags & VMSETRTERR_FLAGS_FATAL);
6813 LogFlowFuncEnter();
6814
6815 Console *that = static_cast<Console *>(pvUser);
6816 AssertReturnVoid(that);
6817
6818 Utf8Str message = Utf8StrFmtVA(pszFormat, va);
6819
6820 LogRel(("Console: VM runtime error: fatal=%RTbool, errorID=%s message=\"%s\"\n",
6821 fFatal, pszErrorId, message.raw()));
6822
6823 that->onRuntimeError(BOOL(fFatal), Bstr(pszErrorId), Bstr(message));
6824
6825 LogFlowFuncLeave();
6826}
6827
6828/**
6829 * Captures USB devices that match filters of the VM.
6830 * Called at VM startup.
6831 *
6832 * @param pVM The VM handle.
6833 *
6834 * @note The caller must lock this object for writing.
6835 */
6836HRESULT Console::captureUSBDevices(PVM pVM)
6837{
6838 LogFlowThisFunc(("\n"));
6839
6840 /* sanity check */
6841 ComAssertRet(isWriteLockOnCurrentThread(), E_FAIL);
6842
6843 /* If the machine has an USB controller, ask the USB proxy service to
6844 * capture devices */
6845 PPDMIBASE pBase;
6846 int vrc = PDMR3QueryLun(pVM, "usb-ohci", 0, 0, &pBase);
6847 if (RT_SUCCESS(vrc))
6848 {
6849 /* leave the lock before calling Host in VBoxSVC since Host may call
6850 * us back from under its lock (e.g. onUSBDeviceAttach()) which would
6851 * produce an inter-process dead-lock otherwise. */
6852 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6853 alock.leave();
6854
6855 HRESULT hrc = mControl->AutoCaptureUSBDevices();
6856 ComAssertComRCRetRC(hrc);
6857 }
6858 else if ( vrc == VERR_PDM_DEVICE_NOT_FOUND
6859 || vrc == VERR_PDM_DEVICE_INSTANCE_NOT_FOUND)
6860 vrc = VINF_SUCCESS;
6861 else
6862 AssertRC(vrc);
6863
6864 return RT_SUCCESS(vrc) ? S_OK : E_FAIL;
6865}
6866
6867
6868/**
6869 * Detach all USB device which are attached to the VM for the
6870 * purpose of clean up and such like.
6871 *
6872 * @note The caller must lock this object for writing.
6873 */
6874void Console::detachAllUSBDevices(bool aDone)
6875{
6876 LogFlowThisFunc(("aDone=%RTbool\n", aDone));
6877
6878 /* sanity check */
6879 AssertReturnVoid(isWriteLockOnCurrentThread());
6880
6881 mUSBDevices.clear();
6882
6883 /* leave the lock before calling Host in VBoxSVC since Host may call
6884 * us back from under its lock (e.g. onUSBDeviceAttach()) which would
6885 * produce an inter-process dead-lock otherwise. */
6886 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6887 alock.leave();
6888
6889 mControl->DetachAllUSBDevices(aDone);
6890}
6891
6892/**
6893 * @note Locks this object for writing.
6894 */
6895void Console::processRemoteUSBDevices(uint32_t u32ClientId, VRDPUSBDEVICEDESC *pDevList, uint32_t cbDevList)
6896{
6897 LogFlowThisFuncEnter();
6898 LogFlowThisFunc(("u32ClientId = %d, pDevList=%p, cbDevList = %d\n", u32ClientId, pDevList, cbDevList));
6899
6900 AutoCaller autoCaller(this);
6901 if (!autoCaller.isOk())
6902 {
6903 /* Console has been already uninitialized, deny request */
6904 AssertMsgFailed(("Console is already uninitialized\n"));
6905 LogFlowThisFunc(("Console is already uninitialized\n"));
6906 LogFlowThisFuncLeave();
6907 return;
6908 }
6909
6910 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
6911
6912 /*
6913 * Mark all existing remote USB devices as dirty.
6914 */
6915 for (RemoteUSBDeviceList::iterator it = mRemoteUSBDevices.begin();
6916 it != mRemoteUSBDevices.end();
6917 ++it)
6918 {
6919 (*it)->dirty(true);
6920 }
6921
6922 /*
6923 * Process the pDevList and add devices those are not already in the mRemoteUSBDevices list.
6924 */
6925 /** @todo (sunlover) REMOTE_USB Strict validation of the pDevList. */
6926 VRDPUSBDEVICEDESC *e = pDevList;
6927
6928 /* The cbDevList condition must be checked first, because the function can
6929 * receive pDevList = NULL and cbDevList = 0 on client disconnect.
6930 */
6931 while (cbDevList >= 2 && e->oNext)
6932 {
6933 LogFlowThisFunc(("vendor %04X, product %04X, name = %s\n",
6934 e->idVendor, e->idProduct,
6935 e->oProduct? (char *)e + e->oProduct: ""));
6936
6937 bool fNewDevice = true;
6938
6939 for (RemoteUSBDeviceList::iterator it = mRemoteUSBDevices.begin();
6940 it != mRemoteUSBDevices.end();
6941 ++it)
6942 {
6943 if ((*it)->devId() == e->id
6944 && (*it)->clientId() == u32ClientId)
6945 {
6946 /* The device is already in the list. */
6947 (*it)->dirty(false);
6948 fNewDevice = false;
6949 break;
6950 }
6951 }
6952
6953 if (fNewDevice)
6954 {
6955 LogRel(("Remote USB: ++++ Vendor %04X. Product %04X. Name = [%s].\n",
6956 e->idVendor, e->idProduct, e->oProduct? (char *)e + e->oProduct: ""));
6957
6958 /* Create the device object and add the new device to list. */
6959 ComObjPtr<RemoteUSBDevice> device;
6960 device.createObject();
6961 device->init(u32ClientId, e);
6962
6963 mRemoteUSBDevices.push_back(device);
6964
6965 /* Check if the device is ok for current USB filters. */
6966 BOOL fMatched = FALSE;
6967 ULONG fMaskedIfs = 0;
6968
6969 HRESULT hrc = mControl->RunUSBDeviceFilters(device, &fMatched, &fMaskedIfs);
6970
6971 AssertComRC(hrc);
6972
6973 LogFlowThisFunc(("USB filters return %d %#x\n", fMatched, fMaskedIfs));
6974
6975 if (fMatched)
6976 {
6977 hrc = onUSBDeviceAttach(device, NULL, fMaskedIfs);
6978
6979 /// @todo (r=dmik) warning reporting subsystem
6980
6981 if (hrc == S_OK)
6982 {
6983 LogFlowThisFunc(("Device attached\n"));
6984 device->captured(true);
6985 }
6986 }
6987 }
6988
6989 if (cbDevList < e->oNext)
6990 {
6991 LogWarningThisFunc(("cbDevList %d > oNext %d\n",
6992 cbDevList, e->oNext));
6993 break;
6994 }
6995
6996 cbDevList -= e->oNext;
6997
6998 e = (VRDPUSBDEVICEDESC *)((uint8_t *)e + e->oNext);
6999 }
7000
7001 /*
7002 * Remove dirty devices, that is those which are not reported by the server anymore.
7003 */
7004 for (;;)
7005 {
7006 ComObjPtr<RemoteUSBDevice> device;
7007
7008 RemoteUSBDeviceList::iterator it = mRemoteUSBDevices.begin();
7009 while (it != mRemoteUSBDevices.end())
7010 {
7011 if ((*it)->dirty())
7012 {
7013 device = *it;
7014 break;
7015 }
7016
7017 ++ it;
7018 }
7019
7020 if (!device)
7021 {
7022 break;
7023 }
7024
7025 USHORT vendorId = 0;
7026 device->COMGETTER(VendorId)(&vendorId);
7027
7028 USHORT productId = 0;
7029 device->COMGETTER(ProductId)(&productId);
7030
7031 Bstr product;
7032 device->COMGETTER(Product)(product.asOutParam());
7033
7034 LogRel(("Remote USB: ---- Vendor %04X. Product %04X. Name = [%ls].\n",
7035 vendorId, productId, product.raw()));
7036
7037 /* Detach the device from VM. */
7038 if (device->captured())
7039 {
7040 Bstr uuid;
7041 device->COMGETTER(Id)(uuid.asOutParam());
7042 onUSBDeviceDetach(uuid, NULL);
7043 }
7044
7045 /* And remove it from the list. */
7046 mRemoteUSBDevices.erase(it);
7047 }
7048
7049 LogFlowThisFuncLeave();
7050}
7051
7052/**
7053 * Thread function which starts the VM (also from saved state) and
7054 * track progress.
7055 *
7056 * @param Thread The thread id.
7057 * @param pvUser Pointer to a VMPowerUpTask structure.
7058 * @return VINF_SUCCESS (ignored).
7059 *
7060 * @note Locks the Console object for writing.
7061 */
7062/*static*/
7063DECLCALLBACK(int) Console::powerUpThread(RTTHREAD Thread, void *pvUser)
7064{
7065 LogFlowFuncEnter();
7066
7067 std::auto_ptr<VMPowerUpTask> task(static_cast<VMPowerUpTask *>(pvUser));
7068 AssertReturn(task.get(), VERR_INVALID_PARAMETER);
7069
7070 AssertReturn(!task->mConsole.isNull(), VERR_INVALID_PARAMETER);
7071 AssertReturn(!task->mProgress.isNull(), VERR_INVALID_PARAMETER);
7072
7073#if defined(RT_OS_WINDOWS)
7074 {
7075 /* initialize COM */
7076 HRESULT hrc = CoInitializeEx(NULL,
7077 COINIT_MULTITHREADED | COINIT_DISABLE_OLE1DDE |
7078 COINIT_SPEED_OVER_MEMORY);
7079 LogFlowFunc(("CoInitializeEx()=%08X\n", hrc));
7080 }
7081#endif
7082
7083 HRESULT rc = S_OK;
7084 int vrc = VINF_SUCCESS;
7085
7086 /* Set up a build identifier so that it can be seen from core dumps what
7087 * exact build was used to produce the core. */
7088 static char saBuildID[40];
7089 RTStrPrintf(saBuildID, sizeof(saBuildID), "%s%s%s%s VirtualBox %s r%u %s%s%s%s",
7090 "BU", "IL", "DI", "D", VBOX_VERSION_STRING, RTBldCfgRevision(), "BU", "IL", "DI", "D");
7091
7092 ComObjPtr<Console> console = task->mConsole;
7093
7094 /* Note: no need to use addCaller() because VMPowerUpTask does that */
7095
7096 /* The lock is also used as a signal from the task initiator (which
7097 * releases it only after RTThreadCreate()) that we can start the job */
7098 AutoWriteLock alock(console COMMA_LOCKVAL_SRC_POS);
7099
7100 /* sanity */
7101 Assert(console->mpVM == NULL);
7102
7103 try
7104 {
7105 /* wait for auto reset ops to complete so that we can successfully lock
7106 * the attached hard disks by calling LockMedia() below */
7107 for (VMPowerUpTask::ProgressList::const_iterator
7108 it = task->hardDiskProgresses.begin();
7109 it != task->hardDiskProgresses.end(); ++ it)
7110 {
7111 HRESULT rc2 = (*it)->WaitForCompletion(-1);
7112 AssertComRC(rc2);
7113 }
7114
7115 /*
7116 * Lock attached media. This method will also check their accessibility.
7117 * If we're a teleporter, we'll have to postpone this action so we can
7118 * migrate between local processes.
7119 *
7120 * Note! The media will be unlocked automatically by
7121 * SessionMachine::setMachineState() when the VM is powered down.
7122 */
7123 if (!task->mTeleporterEnabled)
7124 {
7125 rc = console->mControl->LockMedia();
7126 if (FAILED(rc)) throw rc;
7127 }
7128
7129#ifdef VBOX_WITH_VRDP
7130
7131 /* Create the VRDP server. In case of headless operation, this will
7132 * also create the framebuffer, required at VM creation.
7133 */
7134 ConsoleVRDPServer *server = console->consoleVRDPServer();
7135 Assert(server);
7136
7137 /* Does VRDP server call Console from the other thread?
7138 * Not sure (and can change), so leave the lock just in case.
7139 */
7140 alock.leave();
7141 vrc = server->Launch();
7142 alock.enter();
7143
7144 if (vrc == VERR_NET_ADDRESS_IN_USE)
7145 {
7146 Utf8Str errMsg;
7147 Bstr bstr;
7148 console->mVRDPServer->COMGETTER(Ports)(bstr.asOutParam());
7149 Utf8Str ports = bstr;
7150 errMsg = Utf8StrFmt(tr("VRDP server can't bind to a port: %s"),
7151 ports.raw());
7152 LogRel(("Warning: failed to launch VRDP server (%Rrc): '%s'\n",
7153 vrc, errMsg.raw()));
7154 }
7155 else if (RT_FAILURE(vrc))
7156 {
7157 Utf8Str errMsg;
7158 switch (vrc)
7159 {
7160 case VERR_FILE_NOT_FOUND:
7161 {
7162 errMsg = Utf8StrFmt(tr("Could not load the VRDP library"));
7163 break;
7164 }
7165 default:
7166 errMsg = Utf8StrFmt(tr("Failed to launch VRDP server (%Rrc)"),
7167 vrc);
7168 }
7169 LogRel(("Failed to launch VRDP server (%Rrc), error message: '%s'\n",
7170 vrc, errMsg.raw()));
7171 throw setError(E_FAIL, errMsg.c_str());
7172 }
7173
7174#endif /* VBOX_WITH_VRDP */
7175
7176 ComPtr<IMachine> pMachine = console->machine();
7177 ULONG cCpus = 1;
7178 pMachine->COMGETTER(CPUCount)(&cCpus);
7179
7180 /*
7181 * Create the VM
7182 */
7183 PVM pVM;
7184 /*
7185 * leave the lock since EMT will call Console. It's safe because
7186 * mMachineState is either Starting or Restoring state here.
7187 */
7188 alock.leave();
7189
7190 vrc = VMR3Create(cCpus, task->mSetVMErrorCallback, task.get(),
7191 task->mConfigConstructor, static_cast<Console *>(console),
7192 &pVM);
7193
7194 alock.enter();
7195
7196#ifdef VBOX_WITH_VRDP
7197 /* Enable client connections to the server. */
7198 console->consoleVRDPServer()->EnableConnections();
7199#endif /* VBOX_WITH_VRDP */
7200
7201 if (RT_SUCCESS(vrc))
7202 {
7203 do
7204 {
7205 /*
7206 * Register our load/save state file handlers
7207 */
7208 vrc = SSMR3RegisterExternal(pVM, sSSMConsoleUnit, 0 /*iInstance*/, sSSMConsoleVer, 0 /* cbGuess */,
7209 NULL, NULL, NULL,
7210 NULL, saveStateFileExec, NULL,
7211 NULL, loadStateFileExec, NULL,
7212 static_cast<Console *>(console));
7213 AssertRCBreak(vrc);
7214
7215 vrc = static_cast<Console *>(console)->getDisplay()->registerSSM(pVM);
7216 AssertRC(vrc);
7217 if (RT_FAILURE(vrc))
7218 break;
7219
7220 /*
7221 * Synchronize debugger settings
7222 */
7223 MachineDebugger *machineDebugger = console->getMachineDebugger();
7224 if (machineDebugger)
7225 {
7226 machineDebugger->flushQueuedSettings();
7227 }
7228
7229 /*
7230 * Shared Folders
7231 */
7232 if (console->getVMMDev()->isShFlActive())
7233 {
7234 /* Does the code below call Console from the other thread?
7235 * Not sure, so leave the lock just in case. */
7236 alock.leave();
7237
7238 for (SharedFolderDataMap::const_iterator
7239 it = task->mSharedFolders.begin();
7240 it != task->mSharedFolders.end();
7241 ++ it)
7242 {
7243 rc = console->createSharedFolder((*it).first, (*it).second);
7244 if (FAILED(rc)) break;
7245 }
7246 if (FAILED(rc)) break;
7247
7248 /* enter the lock again */
7249 alock.enter();
7250 }
7251
7252 /*
7253 * Capture USB devices.
7254 */
7255 rc = console->captureUSBDevices(pVM);
7256 if (FAILED(rc)) break;
7257
7258 /* leave the lock before a lengthy operation */
7259 alock.leave();
7260
7261 /* Load saved state? */
7262 if (task->mSavedStateFile.length())
7263 {
7264 LogFlowFunc(("Restoring saved state from '%s'...\n",
7265 task->mSavedStateFile.raw()));
7266
7267 vrc = VMR3LoadFromFile(pVM,
7268 task->mSavedStateFile.c_str(),
7269 Console::stateProgressCallback,
7270 static_cast<VMProgressTask*>(task.get()));
7271
7272 if (RT_SUCCESS(vrc))
7273 {
7274 if (task->mStartPaused)
7275 /* done */
7276 console->setMachineState(MachineState_Paused);
7277 else
7278 {
7279 /* Start/Resume the VM execution */
7280 vrc = VMR3Resume(pVM);
7281 AssertRC(vrc);
7282 }
7283 }
7284
7285 /* Power off in case we failed loading or resuming the VM */
7286 if (RT_FAILURE(vrc))
7287 {
7288 int vrc2 = VMR3PowerOff(pVM);
7289 AssertRC(vrc2);
7290 }
7291 }
7292 else if (task->mTeleporterEnabled)
7293 {
7294 /* -> ConsoleImplTeleporter.cpp */
7295 vrc = console->teleporterTrg(pVM, pMachine, task->mStartPaused, task->mProgress);
7296 if (RT_FAILURE(vrc) && !task->mErrorMsg.length())
7297 rc = E_FAIL; /* Avoid the "Missing error message..." assertion. */
7298 }
7299 else if (task->mStartPaused)
7300 /* done */
7301 console->setMachineState(MachineState_Paused);
7302 else
7303 {
7304 /* Power on the VM (i.e. start executing) */
7305 vrc = VMR3PowerOn(pVM);
7306 AssertRC(vrc);
7307 }
7308
7309 /* enter the lock again */
7310 alock.enter();
7311 }
7312 while (0);
7313
7314 /* On failure, destroy the VM */
7315 if (FAILED(rc) || RT_FAILURE(vrc))
7316 {
7317 /* preserve existing error info */
7318 ErrorInfoKeeper eik;
7319
7320 /* powerDown() will call VMR3Destroy() and do all necessary
7321 * cleanup (VRDP, USB devices) */
7322 HRESULT rc2 = console->powerDown();
7323 AssertComRC(rc2);
7324 }
7325 else
7326 {
7327 /*
7328 * Deregister the VMSetError callback. This is necessary as the
7329 * pfnVMAtError() function passed to VMR3Create() is supposed to
7330 * be sticky but our error callback isn't.
7331 */
7332 alock.leave();
7333 VMR3AtErrorDeregister(pVM, task->mSetVMErrorCallback, task.get());
7334 /** @todo register another VMSetError callback? */
7335 alock.enter();
7336 }
7337 }
7338 else
7339 {
7340 /*
7341 * If VMR3Create() failed it has released the VM memory.
7342 */
7343 console->mpVM = NULL;
7344 }
7345
7346 if (SUCCEEDED(rc) && RT_FAILURE(vrc))
7347 {
7348 /* If VMR3Create() or one of the other calls in this function fail,
7349 * an appropriate error message has been set in task->mErrorMsg.
7350 * However since that happens via a callback, the rc status code in
7351 * this function is not updated.
7352 */
7353 if (!task->mErrorMsg.length())
7354 {
7355 /* If the error message is not set but we've got a failure,
7356 * convert the VBox status code into a meaningful error message.
7357 * This becomes unused once all the sources of errors set the
7358 * appropriate error message themselves.
7359 */
7360 AssertMsgFailed(("Missing error message during powerup for status code %Rrc\n", vrc));
7361 task->mErrorMsg = Utf8StrFmt(tr("Failed to start VM execution (%Rrc)"),
7362 vrc);
7363 }
7364
7365 /* Set the error message as the COM error.
7366 * Progress::notifyComplete() will pick it up later. */
7367 throw setError(E_FAIL, task->mErrorMsg.c_str());
7368 }
7369 }
7370 catch (HRESULT aRC) { rc = aRC; }
7371
7372 if ( console->mMachineState == MachineState_Starting
7373 || console->mMachineState == MachineState_Restoring
7374 || console->mMachineState == MachineState_TeleportingIn
7375 )
7376 {
7377 /* We are still in the Starting/Restoring state. This means one of:
7378 *
7379 * 1) we failed before VMR3Create() was called;
7380 * 2) VMR3Create() failed.
7381 *
7382 * In both cases, there is no need to call powerDown(), but we still
7383 * need to go back to the PoweredOff/Saved state. Reuse
7384 * vmstateChangeCallback() for that purpose.
7385 */
7386
7387 /* preserve existing error info */
7388 ErrorInfoKeeper eik;
7389
7390 Assert(console->mpVM == NULL);
7391 vmstateChangeCallback(NULL, VMSTATE_TERMINATED, VMSTATE_CREATING,
7392 console);
7393 }
7394
7395 /*
7396 * Evaluate the final result. Note that the appropriate mMachineState value
7397 * is already set by vmstateChangeCallback() in all cases.
7398 */
7399
7400 /* leave the lock, don't need it any more */
7401 alock.leave();
7402
7403 if (SUCCEEDED(rc))
7404 {
7405 /* Notify the progress object of the success */
7406 task->mProgress->notifyComplete(S_OK);
7407 console->mControl->SetPowerUpInfo(NULL);
7408 }
7409 else
7410 {
7411 /* The progress object will fetch the current error info */
7412 task->mProgress->notifyComplete(rc);
7413 ProgressErrorInfo info(task->mProgress);
7414 ComObjPtr<VirtualBoxErrorInfo> errorInfo;
7415 rc = errorInfo.createObject();
7416 if (SUCCEEDED(rc))
7417 {
7418 errorInfo->init(info.getResultCode(),
7419 info.getInterfaceID(),
7420 info.getComponent(),
7421 info.getText());
7422 console->mControl->SetPowerUpInfo(errorInfo);
7423 }
7424 else
7425 {
7426 /* If it's not possible to create an IVirtualBoxErrorInfo object
7427 * signal success, as not signalling anything will cause a stuck
7428 * progress object in VBoxSVC. */
7429 console->mControl->SetPowerUpInfo(NULL);
7430 }
7431
7432 LogRel(("Power up failed (vrc=%Rrc, rc=%Rhrc (%#08X))\n", vrc, rc, rc));
7433 }
7434
7435#if defined(RT_OS_WINDOWS)
7436 /* uninitialize COM */
7437 CoUninitialize();
7438#endif
7439
7440 LogFlowFuncLeave();
7441
7442 return VINF_SUCCESS;
7443}
7444
7445
7446/**
7447 * Reconfigures a medium attachment (part of taking an online snapshot).
7448 *
7449 * @param pVM The VM handle.
7450 * @param lInstance The instance of the controller.
7451 * @param enmController The type of the controller.
7452 * @param enmBus The storage bus type of the controller.
7453 * @param aMediumAtt The medium attachment.
7454 * @param phrc Where to store com error - only valid if we return VERR_GENERAL_FAILURE.
7455 * @return VBox status code.
7456 */
7457static DECLCALLBACK(int) reconfigureMedium(PVM pVM, ULONG lInstance,
7458 StorageControllerType_T enmController,
7459 StorageBus_T enmBus,
7460 IMediumAttachment *aMediumAtt,
7461 HRESULT *phrc)
7462{
7463 LogFlowFunc(("pVM=%p aMediumAtt=%p phrc=%p\n", pVM, aMediumAtt, phrc));
7464
7465 int rc;
7466 HRESULT hrc;
7467 Bstr bstr;
7468 *phrc = S_OK;
7469#define RC_CHECK() do { if (RT_FAILURE(rc)) { AssertMsgFailed(("rc=%Rrc\n", rc)); return rc; } } while (0)
7470#define H() do { if (FAILED(hrc)) { AssertMsgFailed(("hrc=%Rhrc (%#x)\n", hrc, hrc)); *phrc = hrc; return VERR_GENERAL_FAILURE; } } while (0)
7471
7472 /*
7473 * Figure out medium and other attachment details.
7474 */
7475 ComPtr<IMedium> medium;
7476 hrc = aMediumAtt->COMGETTER(Medium)(medium.asOutParam()); H();
7477 LONG lDev;
7478 hrc = aMediumAtt->COMGETTER(Device)(&lDev); H();
7479 LONG lPort;
7480 hrc = aMediumAtt->COMGETTER(Port)(&lPort); H();
7481 DeviceType_T lType;
7482 hrc = aMediumAtt->COMGETTER(Type)(&lType); H();
7483
7484 unsigned iLUN;
7485 const char *pcszDevice = Console::convertControllerTypeToDev(enmController);
7486 AssertMsgReturn(pcszDevice, ("invalid disk controller type: %d\n", enmController), VERR_GENERAL_FAILURE);
7487 hrc = Console::convertBusPortDeviceToLun(enmBus, lPort, lDev, iLUN); H();
7488
7489 /* Ignore attachments other than hard disks, since at the moment they are
7490 * not subject to snapshotting in general. */
7491 if (lType != DeviceType_HardDisk || medium.isNull())
7492 return VINF_SUCCESS;
7493
7494 /** @todo this should be unified with the relevant part of
7495 * Console::configConstructor to avoid inconsistencies. */
7496
7497 /*
7498 * Is there an existing LUN? If not create it.
7499 */
7500 PCFGMNODE pCfg;
7501 PCFGMNODE pLunL1;
7502
7503 /* SCSI has an extra driver between the device and the block driver. */
7504 if (enmBus == StorageBus_SCSI)
7505 pLunL1 = CFGMR3GetChildF(CFGMR3GetRoot(pVM), "Devices/%s/%u/LUN#%u/AttachedDriver/AttachedDriver/", pcszDevice, lInstance, iLUN);
7506 else
7507 pLunL1 = CFGMR3GetChildF(CFGMR3GetRoot(pVM), "Devices/%s/%u/LUN#%u/AttachedDriver/", pcszDevice, lInstance, iLUN);
7508
7509 if (!pLunL1)
7510 {
7511 PCFGMNODE pInst = CFGMR3GetChildF(CFGMR3GetRoot(pVM), "Devices/%s/%u/", pcszDevice, lInstance);
7512 AssertReturn(pInst, VERR_INTERNAL_ERROR);
7513
7514 PCFGMNODE pLunL0;
7515 rc = CFGMR3InsertNodeF(pInst, &pLunL0, "LUN#%u", iLUN); RC_CHECK();
7516
7517 if (enmBus == StorageBus_SCSI)
7518 {
7519 rc = CFGMR3InsertString(pLunL0, "Driver", "SCSI"); RC_CHECK();
7520 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
7521
7522 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0); RC_CHECK();
7523 }
7524
7525 rc = CFGMR3InsertString(pLunL0, "Driver", "Block"); RC_CHECK();
7526 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
7527 rc = CFGMR3InsertString(pCfg, "Type", "HardDisk"); RC_CHECK();
7528 rc = CFGMR3InsertInteger(pCfg, "Mountable", 0); RC_CHECK();
7529
7530 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
7531 rc = CFGMR3InsertString(pLunL1, "Driver", "VD"); RC_CHECK();
7532 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
7533 }
7534 else
7535 {
7536#ifdef VBOX_STRICT
7537 char *pszDriver;
7538 rc = CFGMR3QueryStringAlloc(pLunL1, "Driver", &pszDriver); RC_CHECK();
7539 Assert(!strcmp(pszDriver, "VD"));
7540 MMR3HeapFree(pszDriver);
7541#endif
7542
7543 pCfg = CFGMR3GetChild(pLunL1, "Config");
7544 AssertReturn(pCfg, VERR_INTERNAL_ERROR);
7545
7546 /* Here used to be a lot of code checking if things have changed,
7547 * but that's not really worth it, as with snapshots there is always
7548 * some change, so the code was just logging useless information in
7549 * a hard to analyze form. */
7550
7551 /*
7552 * Detach the driver and replace the config node.
7553 */
7554 rc = PDMR3DeviceDetach(pVM, pcszDevice, 0, iLUN, PDM_TACH_FLAGS_NOT_HOT_PLUG); RC_CHECK();
7555 CFGMR3RemoveNode(pCfg);
7556 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
7557 }
7558
7559 /*
7560 * Create the driver configuration.
7561 */
7562 hrc = medium->COMGETTER(Location)(bstr.asOutParam()); H();
7563 LogFlowFunc(("LUN#%u: leaf location '%ls'\n", iLUN, bstr.raw()));
7564 rc = CFGMR3InsertString(pCfg, "Path", Utf8Str(bstr).c_str()); RC_CHECK();
7565 hrc = medium->COMGETTER(Format)(bstr.asOutParam()); H();
7566 LogFlowFunc(("LUN#%u: leaf format '%ls'\n", iLUN, bstr.raw()));
7567 rc = CFGMR3InsertString(pCfg, "Format", Utf8Str(bstr).c_str()); RC_CHECK();
7568
7569 /* Pass all custom parameters. */
7570 bool fHostIP = true;
7571 SafeArray<BSTR> names;
7572 SafeArray<BSTR> values;
7573 hrc = medium->GetProperties(NULL,
7574 ComSafeArrayAsOutParam(names),
7575 ComSafeArrayAsOutParam(values)); H();
7576
7577 if (names.size() != 0)
7578 {
7579 PCFGMNODE pVDC;
7580 rc = CFGMR3InsertNode(pCfg, "VDConfig", &pVDC); RC_CHECK();
7581 for (size_t i = 0; i < names.size(); ++ i)
7582 {
7583 if (values[i] && *values[i])
7584 {
7585 Utf8Str name = names[i];
7586 Utf8Str value = values[i];
7587 rc = CFGMR3InsertString(pVDC, name.c_str(), value.c_str());
7588 if ( !(name.compare("HostIPStack"))
7589 && !(value.compare("0")))
7590 fHostIP = false;
7591 }
7592 }
7593 }
7594
7595 /* Create an inversed tree of parents. */
7596 ComPtr<IMedium> parentMedium = medium;
7597 for (PCFGMNODE pParent = pCfg;;)
7598 {
7599 hrc = parentMedium->COMGETTER(Parent)(medium.asOutParam()); H();
7600 if (medium.isNull())
7601 break;
7602
7603 PCFGMNODE pCur;
7604 rc = CFGMR3InsertNode(pParent, "Parent", &pCur); RC_CHECK();
7605 hrc = medium->COMGETTER(Location)(bstr.asOutParam()); H();
7606 rc = CFGMR3InsertString(pCur, "Path", Utf8Str(bstr).c_str()); RC_CHECK();
7607
7608 hrc = medium->COMGETTER(Format)(bstr.asOutParam()); H();
7609 rc = CFGMR3InsertString(pCur, "Format", Utf8Str(bstr).c_str()); RC_CHECK();
7610
7611 /* Pass all custom parameters. */
7612 SafeArray<BSTR> aNames;
7613 SafeArray<BSTR> aValues;
7614 hrc = medium->GetProperties(NULL,
7615 ComSafeArrayAsOutParam(aNames),
7616 ComSafeArrayAsOutParam(aValues)); H();
7617
7618 if (aNames.size() != 0)
7619 {
7620 PCFGMNODE pVDC;
7621 rc = CFGMR3InsertNode(pCur, "VDConfig", &pVDC); RC_CHECK();
7622 for (size_t i = 0; i < aNames.size(); ++ i)
7623 {
7624 if (aValues[i] && *aValues[i])
7625 {
7626 Utf8Str name = aNames[i];
7627 Utf8Str value = aValues[i];
7628 rc = CFGMR3InsertString(pVDC, name.c_str(), value.c_str()); RC_CHECK();
7629 if ( !(name.compare("HostIPStack"))
7630 && !(value.compare("0")))
7631 fHostIP = false;
7632 }
7633 }
7634 }
7635
7636 /* Custom code: put marker to not use host IP stack to driver
7637 * configuration node. Simplifies life of DrvVD a bit. */
7638 if (!fHostIP)
7639 {
7640 rc = CFGMR3InsertInteger(pCfg, "HostIPStack", 0); RC_CHECK();
7641 }
7642
7643
7644 /* next */
7645 pParent = pCur;
7646 parentMedium = medium;
7647 }
7648
7649 CFGMR3Dump(CFGMR3GetRoot(pVM));
7650
7651 /*
7652 * Attach the new driver.
7653 */
7654 rc = PDMR3DeviceAttach(pVM, pcszDevice, 0, iLUN, PDM_TACH_FLAGS_NOT_HOT_PLUG, NULL /*ppBase*/); RC_CHECK();
7655
7656 LogFlowFunc(("Returns success\n"));
7657 return rc;
7658}
7659
7660/**
7661 * Progress cancelation callback employed by Console::fntTakeSnapshotWorker.
7662 */
7663static void takesnapshotProgressCancelCallback(void *pvUser)
7664{
7665 PVM pVM = (PVM)pvUser;
7666 SSMR3Cancel(pVM);
7667}
7668
7669/**
7670 * Worker thread created by Console::TakeSnapshot.
7671 * @param Thread The current thread (ignored).
7672 * @param pvUser The task.
7673 * @return VINF_SUCCESS (ignored).
7674 */
7675/*static*/
7676DECLCALLBACK(int) Console::fntTakeSnapshotWorker(RTTHREAD Thread, void *pvUser)
7677{
7678 VMTakeSnapshotTask *pTask = (VMTakeSnapshotTask*)pvUser;
7679
7680 // taking a snapshot consists of the following:
7681
7682 // 1) creating a diff image for each virtual hard disk, into which write operations go after
7683 // the snapshot has been created (done in VBoxSVC, in SessionMachine::BeginTakingSnapshot)
7684 // 2) creating a Snapshot object with the state of the machine (hardware + storage,
7685 // done in VBoxSVC, also in SessionMachine::BeginTakingSnapshot)
7686 // 3) saving the state of the virtual machine (here, in the VM process, if the machine is online)
7687
7688 Console *that = pTask->mConsole;
7689 bool fBeganTakingSnapshot = false;
7690 bool fSuspenededBySave = false;
7691
7692 AutoCaller autoCaller(that);
7693 if (FAILED(autoCaller.rc()))
7694 {
7695 that->mptrCancelableProgress.setNull();
7696 return autoCaller.rc();
7697 }
7698
7699 AutoWriteLock alock(that COMMA_LOCKVAL_SRC_POS);
7700
7701 HRESULT rc = S_OK;
7702
7703 try
7704 {
7705 /* STEP 1 + 2:
7706 * request creating the diff images on the server and create the snapshot object
7707 * (this will set the machine state to Saving on the server to block
7708 * others from accessing this machine)
7709 */
7710 rc = that->mControl->BeginTakingSnapshot(that,
7711 pTask->bstrName,
7712 pTask->bstrDescription,
7713 pTask->mProgress,
7714 pTask->fTakingSnapshotOnline,
7715 pTask->bstrSavedStateFile.asOutParam());
7716 if (FAILED(rc))
7717 throw rc;
7718
7719 fBeganTakingSnapshot = true;
7720
7721 /*
7722 * state file is non-null only when the VM is paused
7723 * (i.e. creating a snapshot online)
7724 */
7725 ComAssertThrow( (!pTask->bstrSavedStateFile.isEmpty() && pTask->fTakingSnapshotOnline)
7726 || ( pTask->bstrSavedStateFile.isEmpty() && !pTask->fTakingSnapshotOnline),
7727 rc = E_FAIL);
7728
7729 /* sync the state with the server */
7730 if (pTask->lastMachineState == MachineState_Running)
7731 that->setMachineStateLocally(MachineState_LiveSnapshotting);
7732 else
7733 that->setMachineStateLocally(MachineState_Saving);
7734
7735 // STEP 3: save the VM state (if online)
7736 if (pTask->fTakingSnapshotOnline)
7737 {
7738 Utf8Str strSavedStateFile(pTask->bstrSavedStateFile);
7739
7740 pTask->mProgress->SetNextOperation(Bstr(tr("Saving the machine state")),
7741 pTask->ulMemSize); // operation weight, same as computed when setting up progress object
7742 pTask->mProgress->setCancelCallback(takesnapshotProgressCancelCallback, that->mpVM);
7743
7744 alock.leave();
7745 LogFlowFunc(("VMR3Save...\n"));
7746 int vrc = VMR3Save(that->mpVM,
7747 strSavedStateFile.c_str(),
7748 true /*fContinueAfterwards*/,
7749 Console::stateProgressCallback,
7750 (void*)pTask,
7751 &fSuspenededBySave);
7752 alock.enter();
7753 if (RT_FAILURE(vrc))
7754 throw setError(E_FAIL,
7755 tr("Failed to save the machine state to '%s' (%Rrc)"),
7756 strSavedStateFile.c_str(), vrc);
7757
7758 pTask->mProgress->setCancelCallback(NULL, NULL);
7759 if (!pTask->mProgress->notifyPointOfNoReturn())
7760 throw setError(E_FAIL, tr("Cancelled"));
7761 that->mptrCancelableProgress.setNull();
7762
7763 // STEP 4: reattach hard disks
7764 LogFlowFunc(("Reattaching new differencing hard disks...\n"));
7765
7766 pTask->mProgress->SetNextOperation(Bstr(tr("Reconfiguring medium attachments")),
7767 1); // operation weight, same as computed when setting up progress object
7768
7769 com::SafeIfaceArray<IMediumAttachment> atts;
7770 rc = that->mMachine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(atts));
7771 if (FAILED(rc))
7772 throw rc;
7773
7774 for (size_t i = 0;
7775 i < atts.size();
7776 ++i)
7777 {
7778 ComPtr<IStorageController> controller;
7779 BSTR controllerName;
7780 ULONG lInstance;
7781 StorageControllerType_T enmController;
7782 StorageBus_T enmBus;
7783
7784 /*
7785 * We can't pass a storage controller object directly
7786 * (g++ complains about not being able to pass non POD types through '...')
7787 * so we have to query needed values here and pass them.
7788 */
7789 rc = atts[i]->COMGETTER(Controller)(&controllerName);
7790 if (FAILED(rc))
7791 throw rc;
7792
7793 rc = that->mMachine->GetStorageControllerByName(controllerName, controller.asOutParam());
7794 if (FAILED(rc))
7795 throw rc;
7796
7797 rc = controller->COMGETTER(ControllerType)(&enmController);
7798 if (FAILED(rc))
7799 throw rc;
7800 rc = controller->COMGETTER(Instance)(&lInstance);
7801 if (FAILED(rc))
7802 throw rc;
7803 rc = controller->COMGETTER(Bus)(&enmBus);
7804 if (FAILED(rc))
7805 throw rc;
7806
7807 /*
7808 * don't leave the lock since reconfigureMedium isn't going
7809 * to access Console.
7810 */
7811 vrc = VMR3ReqCallWait(that->mpVM,
7812 VMCPUID_ANY,
7813 (PFNRT)reconfigureMedium,
7814 6,
7815 that->mpVM,
7816 lInstance,
7817 enmController,
7818 enmBus,
7819 atts[i],
7820 &rc);
7821 if (RT_FAILURE(vrc))
7822 throw setError(E_FAIL, Console::tr("%Rrc"), vrc);
7823 if (FAILED(rc))
7824 throw rc;
7825 }
7826 }
7827
7828 /*
7829 * finalize the requested snapshot object.
7830 * This will reset the machine state to the state it had right
7831 * before calling mControl->BeginTakingSnapshot().
7832 */
7833 rc = that->mControl->EndTakingSnapshot(TRUE /*aSuccess*/);
7834 // do not throw rc here because we can't call EndTakingSnapshot() twice
7835 LogFlowFunc(("EndTakingSnapshot -> %Rhrc [mMachineState=%s]\n", rc, Global::stringifyMachineState(that->mMachineState)));
7836 }
7837 catch (HRESULT rcThrown)
7838 {
7839 /* preserve existing error info */
7840 ErrorInfoKeeper eik;
7841
7842 if (fBeganTakingSnapshot)
7843 that->mControl->EndTakingSnapshot(FALSE /*aSuccess*/);
7844
7845 rc = rcThrown;
7846 LogFunc(("Caught %Rhrc [mMachineState=%s]\n", rc, Global::stringifyMachineState(that->mMachineState)));
7847 }
7848 Assert(alock.isWriteLockOnCurrentThread());
7849
7850 if (FAILED(rc)) /* Must come before calling setMachineState. */
7851 pTask->mProgress->notifyComplete(rc);
7852
7853 /*
7854 * Fix up the machine state.
7855 *
7856 * For live snapshots we do all the work, for the two other variantions we
7857 * just update the local copy.
7858 */
7859 MachineState_T enmMachineState;
7860 that->mMachine->COMGETTER(State)(&enmMachineState);
7861 if ( that->mMachineState == MachineState_LiveSnapshotting
7862 || that->mMachineState == MachineState_Saving)
7863 {
7864
7865 if (!pTask->fTakingSnapshotOnline)
7866 that->setMachineStateLocally(pTask->lastMachineState);
7867 else if (SUCCEEDED(rc))
7868 {
7869 Assert( pTask->lastMachineState == MachineState_Running
7870 || pTask->lastMachineState == MachineState_Paused);
7871 Assert(that->mMachineState == MachineState_Saving);
7872 if (pTask->lastMachineState == MachineState_Running)
7873 {
7874 LogFlowFunc(("VMR3Resume...\n"));
7875 alock.leave();
7876 int vrc = VMR3Resume(that->mpVM);
7877 alock.enter();
7878 if (RT_FAILURE(vrc))
7879 {
7880 rc = setError(VBOX_E_VM_ERROR, tr("Could not resume the machine execution (%Rrc)"), vrc);
7881 pTask->mProgress->notifyComplete(rc);
7882 if (that->mMachineState == MachineState_Saving)
7883 that->setMachineStateLocally(MachineState_Paused);
7884 }
7885 }
7886 else
7887 that->setMachineStateLocally(MachineState_Paused);
7888 }
7889 else
7890 {
7891 /** @todo this could probably be made more generic and reused elsewhere. */
7892 /* paranoid cleanup on for a failed online snapshot. */
7893 VMSTATE enmVMState = VMR3GetState(that->mpVM);
7894 switch (enmVMState)
7895 {
7896 case VMSTATE_RUNNING:
7897 case VMSTATE_RUNNING_LS:
7898 case VMSTATE_DEBUGGING:
7899 case VMSTATE_DEBUGGING_LS:
7900 case VMSTATE_POWERING_OFF:
7901 case VMSTATE_POWERING_OFF_LS:
7902 case VMSTATE_RESETTING:
7903 case VMSTATE_RESETTING_LS:
7904 Assert(!fSuspenededBySave);
7905 that->setMachineState(MachineState_Running);
7906 break;
7907
7908 case VMSTATE_GURU_MEDITATION:
7909 case VMSTATE_GURU_MEDITATION_LS:
7910 that->setMachineState(MachineState_Stuck);
7911 break;
7912
7913 case VMSTATE_FATAL_ERROR:
7914 case VMSTATE_FATAL_ERROR_LS:
7915 if (pTask->lastMachineState == MachineState_Paused)
7916 that->setMachineStateLocally(pTask->lastMachineState);
7917 else
7918 that->setMachineState(MachineState_Paused);
7919 break;
7920
7921 default:
7922 AssertMsgFailed(("%s\n", VMR3GetStateName(enmVMState)));
7923 case VMSTATE_SUSPENDED:
7924 case VMSTATE_SUSPENDED_LS:
7925 case VMSTATE_SUSPENDING:
7926 case VMSTATE_SUSPENDING_LS:
7927 case VMSTATE_SUSPENDING_EXT_LS:
7928 if (fSuspenededBySave)
7929 {
7930 Assert(pTask->lastMachineState == MachineState_Running);
7931 LogFlowFunc(("VMR3Resume (on failure)...\n"));
7932 alock.leave();
7933 int vrc = VMR3Resume(that->mpVM);
7934 alock.enter();
7935 AssertLogRelRC(vrc);
7936 if (RT_FAILURE(vrc))
7937 that->setMachineState(MachineState_Paused);
7938 }
7939 else if (pTask->lastMachineState == MachineState_Paused)
7940 that->setMachineStateLocally(pTask->lastMachineState);
7941 else
7942 that->setMachineState(MachineState_Paused);
7943 break;
7944 }
7945
7946 }
7947 }
7948 /*else: somebody else has change the state... Leave it. */
7949
7950 /* check the remote state to see that we got it right. */
7951 that->mMachine->COMGETTER(State)(&enmMachineState);
7952 AssertLogRelMsg(that->mMachineState == enmMachineState,
7953 ("mMachineState=%s enmMachineState=%s\n", Global::stringifyMachineState(that->mMachineState),
7954 Global::stringifyMachineState(enmMachineState) ));
7955
7956
7957 if (SUCCEEDED(rc)) /* The failure cases are handled above. */
7958 pTask->mProgress->notifyComplete(rc);
7959
7960 delete pTask;
7961
7962 LogFlowFuncLeave();
7963 return VINF_SUCCESS;
7964}
7965
7966/**
7967 * Thread for executing the saved state operation.
7968 *
7969 * @param Thread The thread handle.
7970 * @param pvUser Pointer to a VMSaveTask structure.
7971 * @return VINF_SUCCESS (ignored).
7972 *
7973 * @note Locks the Console object for writing.
7974 */
7975/*static*/
7976DECLCALLBACK(int) Console::saveStateThread(RTTHREAD Thread, void *pvUser)
7977{
7978 LogFlowFuncEnter();
7979
7980 std::auto_ptr<VMSaveTask> task(static_cast<VMSaveTask*>(pvUser));
7981 AssertReturn(task.get(), VERR_INVALID_PARAMETER);
7982
7983 Assert(task->mSavedStateFile.length());
7984 Assert(!task->mProgress.isNull());
7985
7986 const ComObjPtr<Console> &that = task->mConsole;
7987 Utf8Str errMsg;
7988 HRESULT rc = S_OK;
7989
7990 LogFlowFunc(("Saving the state to '%s'...\n", task->mSavedStateFile.raw()));
7991
7992 bool fSuspenededBySave;
7993 int vrc = VMR3Save(that->mpVM,
7994 task->mSavedStateFile.c_str(),
7995 false, /*fContinueAfterwards*/
7996 Console::stateProgressCallback,
7997 static_cast<VMProgressTask*>(task.get()),
7998 &fSuspenededBySave);
7999 if (RT_FAILURE(vrc))
8000 {
8001 errMsg = Utf8StrFmt(Console::tr("Failed to save the machine state to '%s' (%Rrc)"),
8002 task->mSavedStateFile.raw(), vrc);
8003 rc = E_FAIL;
8004 }
8005 Assert(!fSuspenededBySave);
8006
8007 /* lock the console once we're going to access it */
8008 AutoWriteLock thatLock(that COMMA_LOCKVAL_SRC_POS);
8009
8010 /*
8011 * finalize the requested save state procedure.
8012 * In case of success, the server will set the machine state to Saved;
8013 * in case of failure it will reset the it to the state it had right
8014 * before calling mControl->BeginSavingState().
8015 */
8016 that->mControl->EndSavingState(SUCCEEDED(rc));
8017
8018 /* synchronize the state with the server */
8019 if (!FAILED(rc))
8020 {
8021 /*
8022 * The machine has been successfully saved, so power it down
8023 * (vmstateChangeCallback() will set state to Saved on success).
8024 * Note: we release the task's VM caller, otherwise it will
8025 * deadlock.
8026 */
8027 task->releaseVMCaller();
8028
8029 rc = that->powerDown();
8030 }
8031
8032 /* notify the progress object about operation completion */
8033 if (SUCCEEDED(rc))
8034 task->mProgress->notifyComplete(S_OK);
8035 else
8036 {
8037 if (errMsg.length())
8038 task->mProgress->notifyComplete(rc,
8039 COM_IIDOF(IConsole),
8040 (CBSTR)Console::getComponentName(),
8041 errMsg.c_str());
8042 else
8043 task->mProgress->notifyComplete(rc);
8044 }
8045
8046 LogFlowFuncLeave();
8047 return VINF_SUCCESS;
8048}
8049
8050/**
8051 * Thread for powering down the Console.
8052 *
8053 * @param Thread The thread handle.
8054 * @param pvUser Pointer to the VMTask structure.
8055 * @return VINF_SUCCESS (ignored).
8056 *
8057 * @note Locks the Console object for writing.
8058 */
8059/*static*/
8060DECLCALLBACK(int) Console::powerDownThread(RTTHREAD Thread, void *pvUser)
8061{
8062 LogFlowFuncEnter();
8063
8064 std::auto_ptr<VMProgressTask> task(static_cast<VMProgressTask *>(pvUser));
8065 AssertReturn(task.get(), VERR_INVALID_PARAMETER);
8066
8067 AssertReturn(task->isOk(), VERR_GENERAL_FAILURE);
8068
8069 const ComObjPtr<Console> &that = task->mConsole;
8070
8071 /* Note: no need to use addCaller() to protect Console because VMTask does
8072 * that */
8073
8074 /* wait until the method tat started us returns */
8075 AutoWriteLock thatLock(that COMMA_LOCKVAL_SRC_POS);
8076
8077 /* release VM caller to avoid the powerDown() deadlock */
8078 task->releaseVMCaller();
8079
8080 that->powerDown(task->mProgress);
8081
8082 LogFlowFuncLeave();
8083 return VINF_SUCCESS;
8084}
8085
8086/**
8087 * The Main status driver instance data.
8088 */
8089typedef struct DRVMAINSTATUS
8090{
8091 /** The LED connectors. */
8092 PDMILEDCONNECTORS ILedConnectors;
8093 /** Pointer to the LED ports interface above us. */
8094 PPDMILEDPORTS pLedPorts;
8095 /** Pointer to the array of LED pointers. */
8096 PPDMLED *papLeds;
8097 /** The unit number corresponding to the first entry in the LED array. */
8098 RTUINT iFirstLUN;
8099 /** The unit number corresponding to the last entry in the LED array.
8100 * (The size of the LED array is iLastLUN - iFirstLUN + 1.) */
8101 RTUINT iLastLUN;
8102} DRVMAINSTATUS, *PDRVMAINSTATUS;
8103
8104
8105/**
8106 * Notification about a unit which have been changed.
8107 *
8108 * The driver must discard any pointers to data owned by
8109 * the unit and requery it.
8110 *
8111 * @param pInterface Pointer to the interface structure containing the called function pointer.
8112 * @param iLUN The unit number.
8113 */
8114DECLCALLBACK(void) Console::drvStatus_UnitChanged(PPDMILEDCONNECTORS pInterface, unsigned iLUN)
8115{
8116 PDRVMAINSTATUS pData = (PDRVMAINSTATUS)(void *)pInterface;
8117 if (iLUN >= pData->iFirstLUN && iLUN <= pData->iLastLUN)
8118 {
8119 PPDMLED pLed;
8120 int rc = pData->pLedPorts->pfnQueryStatusLed(pData->pLedPorts, iLUN, &pLed);
8121 if (RT_FAILURE(rc))
8122 pLed = NULL;
8123 ASMAtomicXchgPtr((void * volatile *)&pData->papLeds[iLUN - pData->iFirstLUN], pLed);
8124 Log(("drvStatus_UnitChanged: iLUN=%d pLed=%p\n", iLUN, pLed));
8125 }
8126}
8127
8128
8129/**
8130 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
8131 */
8132DECLCALLBACK(void *) Console::drvStatus_QueryInterface(PPDMIBASE pInterface, const char *pszIID)
8133{
8134 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
8135 PDRVMAINSTATUS pThis = PDMINS_2_DATA(pDrvIns, PDRVMAINSTATUS);
8136 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
8137 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDCONNECTORS, &pThis->ILedConnectors);
8138 return NULL;
8139}
8140
8141
8142/**
8143 * Destruct a status driver instance.
8144 *
8145 * @returns VBox status.
8146 * @param pDrvIns The driver instance data.
8147 */
8148DECLCALLBACK(void) Console::drvStatus_Destruct(PPDMDRVINS pDrvIns)
8149{
8150 PDRVMAINSTATUS pData = PDMINS_2_DATA(pDrvIns, PDRVMAINSTATUS);
8151 LogFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
8152 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
8153
8154 if (pData->papLeds)
8155 {
8156 unsigned iLed = pData->iLastLUN - pData->iFirstLUN + 1;
8157 while (iLed-- > 0)
8158 ASMAtomicXchgPtr((void * volatile *)&pData->papLeds[iLed], NULL);
8159 }
8160}
8161
8162
8163/**
8164 * Construct a status driver instance.
8165 *
8166 * @copydoc FNPDMDRVCONSTRUCT
8167 */
8168DECLCALLBACK(int) Console::drvStatus_Construct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
8169{
8170 PDRVMAINSTATUS pData = PDMINS_2_DATA(pDrvIns, PDRVMAINSTATUS);
8171 LogFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
8172 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
8173
8174 /*
8175 * Validate configuration.
8176 */
8177 if (!CFGMR3AreValuesValid(pCfg, "papLeds\0First\0Last\0"))
8178 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
8179 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
8180 ("Configuration error: Not possible to attach anything to this driver!\n"),
8181 VERR_PDM_DRVINS_NO_ATTACH);
8182
8183 /*
8184 * Data.
8185 */
8186 pDrvIns->IBase.pfnQueryInterface = Console::drvStatus_QueryInterface;
8187 pData->ILedConnectors.pfnUnitChanged = Console::drvStatus_UnitChanged;
8188
8189 /*
8190 * Read config.
8191 */
8192 int rc = CFGMR3QueryPtr(pCfg, "papLeds", (void **)&pData->papLeds);
8193 if (RT_FAILURE(rc))
8194 {
8195 AssertMsgFailed(("Configuration error: Failed to query the \"papLeds\" value! rc=%Rrc\n", rc));
8196 return rc;
8197 }
8198
8199 rc = CFGMR3QueryU32(pCfg, "First", &pData->iFirstLUN);
8200 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
8201 pData->iFirstLUN = 0;
8202 else if (RT_FAILURE(rc))
8203 {
8204 AssertMsgFailed(("Configuration error: Failed to query the \"First\" value! rc=%Rrc\n", rc));
8205 return rc;
8206 }
8207
8208 rc = CFGMR3QueryU32(pCfg, "Last", &pData->iLastLUN);
8209 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
8210 pData->iLastLUN = 0;
8211 else if (RT_FAILURE(rc))
8212 {
8213 AssertMsgFailed(("Configuration error: Failed to query the \"Last\" value! rc=%Rrc\n", rc));
8214 return rc;
8215 }
8216 if (pData->iFirstLUN > pData->iLastLUN)
8217 {
8218 AssertMsgFailed(("Configuration error: Invalid unit range %u-%u\n", pData->iFirstLUN, pData->iLastLUN));
8219 return VERR_GENERAL_FAILURE;
8220 }
8221
8222 /*
8223 * Get the ILedPorts interface of the above driver/device and
8224 * query the LEDs we want.
8225 */
8226 pData->pLedPorts = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMILEDPORTS);
8227 AssertMsgReturn(pData->pLedPorts, ("Configuration error: No led ports interface above!\n"),
8228 VERR_PDM_MISSING_INTERFACE_ABOVE);
8229
8230 for (unsigned i = pData->iFirstLUN; i <= pData->iLastLUN; ++i)
8231 Console::drvStatus_UnitChanged(&pData->ILedConnectors, i);
8232
8233 return VINF_SUCCESS;
8234}
8235
8236
8237/**
8238 * Keyboard driver registration record.
8239 */
8240const PDMDRVREG Console::DrvStatusReg =
8241{
8242 /* u32Version */
8243 PDM_DRVREG_VERSION,
8244 /* szName */
8245 "MainStatus",
8246 /* szRCMod */
8247 "",
8248 /* szR0Mod */
8249 "",
8250 /* pszDescription */
8251 "Main status driver (Main as in the API).",
8252 /* fFlags */
8253 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
8254 /* fClass. */
8255 PDM_DRVREG_CLASS_STATUS,
8256 /* cMaxInstances */
8257 ~0,
8258 /* cbInstance */
8259 sizeof(DRVMAINSTATUS),
8260 /* pfnConstruct */
8261 Console::drvStatus_Construct,
8262 /* pfnDestruct */
8263 Console::drvStatus_Destruct,
8264 /* pfnRelocate */
8265 NULL,
8266 /* pfnIOCtl */
8267 NULL,
8268 /* pfnPowerOn */
8269 NULL,
8270 /* pfnReset */
8271 NULL,
8272 /* pfnSuspend */
8273 NULL,
8274 /* pfnResume */
8275 NULL,
8276 /* pfnAttach */
8277 NULL,
8278 /* pfnDetach */
8279 NULL,
8280 /* pfnPowerOff */
8281 NULL,
8282 /* pfnSoftReset */
8283 NULL,
8284 /* u32EndVersion */
8285 PDM_DRVREG_VERSION
8286};
8287
8288/* 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