VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/ConsoleImpl.cpp@ 39647

Last change on this file since 39647 was 39647, checked in by vboxsync, 13 years ago

VBoxInternal2

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