/** @file * * VBox Client Session COM Class implementation */ /* * Copyright (C) 2006 InnoTek Systemberatung GmbH * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License as published by the Free Software Foundation, * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE * distribution. VirtualBox OSE is distributed in the hope that it will * be useful, but WITHOUT ANY WARRANTY of any kind. * * If you received this file as part of a commercial VirtualBox * distribution, then only the terms of your commercial VirtualBox * license agreement apply instead of the previous paragraph. */ #if defined(__WIN__) #elif defined(__LINUX__) #endif #ifdef VBOX_WITH_SYS_V_IPC_SESSION_WATCHER # include # include # include # include # include #endif #include "SessionImpl.h" #include "ConsoleImpl.h" #include "Logging.h" #include #include #if defined(__WIN__) /** VM IPC mutex holder thread */ static DECLCALLBACK(int) IPCMutexHolderThread (RTTHREAD Thread, void *pvUser); #endif /** * Local macro to check whether the session is open and return an error if not. * @note Don't forget to do |Auto[Reader]Lock alock (this);| before using this * macro. */ #define CHECK_OPEN() \ do { \ if (mState != SessionState_SessionOpen) \ return setError (E_UNEXPECTED, \ tr ("The session is not open")); \ } while (0) // constructor / destructor ///////////////////////////////////////////////////////////////////////////// HRESULT Session::FinalConstruct() { LogFlowThisFunc (("\n")); return init(); } void Session::FinalRelease() { LogFlowThisFunc (("\n")); uninit (true /* aFinalRelease */); } // public initializer/uninitializer for internal purposes only ///////////////////////////////////////////////////////////////////////////// /** * Initializes the Session object. */ HRESULT Session::init() { /* Enclose the state transition NotReady->InInit->Ready */ AutoInitSpan autoInitSpan (this); AssertReturn (autoInitSpan.isOk(), E_UNEXPECTED); LogFlowThisFuncEnter(); mState = SessionState_SessionClosed; mType = SessionType_InvalidSessionType; #if defined(__WIN__) mIPCSem = NULL; mIPCThreadSem = NULL; #elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) mIPCSem = -1; #endif /* Confirm a successful initialization when it's the case */ autoInitSpan.setSucceeded(); LogFlowThisFuncLeave(); return S_OK; } /** * Uninitializes the Session object. * * @note Locks this object for writing. */ void Session::uninit (bool aFinalRelease) { LogFlowThisFuncEnter(); LogFlowThisFunc (("aFinalRelease=%d\n", aFinalRelease)); /* Enclose the state transition Ready->InUninit->NotReady */ AutoUninitSpan autoUninitSpan (this); if (autoUninitSpan.uninitDone()) { LogFlowThisFunc (("Already uninitialized.\n")); LogFlowThisFuncLeave(); return; } AutoLock alock (this); if (mState != SessionState_SessionClosed) { Assert (mState == SessionState_SessionOpen || mState == SessionState_SessionSpawning); HRESULT rc = close (aFinalRelease, false /* aFromServer */); AssertComRC (rc); } LogFlowThisFuncLeave(); } // ISession properties ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP Session::COMGETTER(State) (SessionState_T *aState) { if (!aState) return E_POINTER; AutoCaller autoCaller (this); CheckComRCReturnRC (autoCaller.rc()); AutoReaderLock alock (this); *aState = mState; return S_OK; } STDMETHODIMP Session::COMGETTER(Type) (SessionType_T *aType) { if (!aType) return E_POINTER; AutoCaller autoCaller (this); CheckComRCReturnRC (autoCaller.rc()); AutoReaderLock alock (this); CHECK_OPEN(); *aType = mType; return S_OK; } STDMETHODIMP Session::COMGETTER(Machine) (IMachine **aMachine) { if (!aMachine) return E_POINTER; AutoCaller autoCaller (this); CheckComRCReturnRC (autoCaller.rc()); AutoReaderLock alock (this); CHECK_OPEN(); HRESULT rc = E_FAIL; if (mConsole) rc = mConsole->machine().queryInterfaceTo (aMachine); else rc = mRemoteMachine.queryInterfaceTo (aMachine); ComAssertComRC (rc); return rc; } STDMETHODIMP Session::COMGETTER(Console) (IConsole **aConsole) { if (!aConsole) return E_POINTER; AutoCaller autoCaller (this); CheckComRCReturnRC (autoCaller.rc()); AutoReaderLock alock (this); CHECK_OPEN(); HRESULT rc = E_FAIL; if (mConsole) rc = mConsole.queryInterfaceTo (aConsole); else rc = mRemoteConsole.queryInterfaceTo (aConsole); ComAssertComRC (rc); return rc; } // ISession methods ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP Session::Close() { LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType)); AutoCaller autoCaller (this); CheckComRCReturnRC (autoCaller.rc()); /* close() needs write lock */ AutoLock alock (this); CHECK_OPEN(); return close (false /* aFinalRelease */, false /* aFromServer */); } // IInternalSessionControl methods ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP Session::GetPID (ULONG *aPid) { AssertReturn (aPid, E_POINTER); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); *aPid = (ULONG) RTProcSelf(); AssertCompile (sizeof (*aPid) == sizeof (RTPROCESS)); return S_OK; } STDMETHODIMP Session::GetRemoteConsole (IConsole **aConsole) { AssertReturn (aConsole, E_POINTER); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen, E_FAIL); AssertMsgReturn (mType == SessionType_DirectSession && !!mConsole, ("This is not a direct session!\n"), E_FAIL); mConsole.queryInterfaceTo (aConsole); return S_OK; } STDMETHODIMP Session::AssignMachine (IMachine *aMachine) { LogFlowThisFuncEnter(); LogFlowThisFunc (("aMachine=%p\n", aMachine)); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoLock alock (this); AssertReturn (mState == SessionState_SessionClosed, E_FAIL); if (!aMachine) { /* * A special case: the server informs us that this session has been * passed to IVirtualBox::OpenRemoteSession() so this session will * become remote (but not existing) when AssignRemoteMachine() is * called. */ AssertReturn (mType == SessionType_InvalidSessionType, E_FAIL); mType = SessionType_RemoteSession; mState = SessionState_SessionSpawning; LogFlowThisFuncLeave(); return S_OK; } HRESULT rc = E_FAIL; /* query IInternalMachineControl interface */ mControl = aMachine; AssertReturn (!!mControl, E_FAIL); rc = mConsole.createObject(); AssertComRCReturn (rc, rc); rc = mConsole->init (aMachine, mControl); AssertComRCReturn (rc, rc); rc = grabIPCSemaphore(); /* * Reference the VirtualBox object to ensure the server is up * until the session is closed */ if (SUCCEEDED (rc)) rc = aMachine->COMGETTER(Parent) (mVirtualBox.asOutParam()); if (SUCCEEDED (rc)) { mType = SessionType_DirectSession; mState = SessionState_SessionOpen; } else { /* some cleanup */ mControl.setNull(); mConsole->uninit(); mConsole.setNull(); } LogFlowThisFunc (("rc=%08X\n", rc)); LogFlowThisFuncLeave(); return rc; } STDMETHODIMP Session::AssignRemoteMachine (IMachine *aMachine, IConsole *aConsole) { LogFlowThisFuncEnter(); LogFlowThisFunc (("aMachine=%p, aConsole=%p\n", aMachine, aConsole)); AssertReturn (aMachine && aConsole, E_INVALIDARG); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoLock alock (this); AssertReturn (mState == SessionState_SessionClosed || mState == SessionState_SessionSpawning, E_FAIL); HRESULT rc = E_FAIL; /* query IInternalMachineControl interface */ mControl = aMachine; AssertReturn (!!mControl, E_FAIL); /// @todo (dmik) // currently, the remote session returns the same machine and // console objects as the direct session, thus giving the // (remote) client full control over the direct session. For the // console, it is the desired behavior (the ability to control // VM execution is a must for the remote session). What about // the machine object, we may want to prevent the remote client // from modifying machine data. In this case, we must: // 1) assign the Machine object (instead of the SessionMachine // object that is passed to this method) to mRemoteMachine; // 2) remove GetMachine() property from the IConsole interface // because it always returns the SessionMachine object // (alternatively, we can supply a separate IConsole // implementation that will return the Machine object in // response to GetMachine()). mRemoteMachine = aMachine; mRemoteConsole = aConsole; /* * Reference the VirtualBox object to ensure the server is up * until the session is closed */ rc = aMachine->COMGETTER(Parent) (mVirtualBox.asOutParam()); if (SUCCEEDED (rc)) { /* * RemoteSession type can be already set by AssignMachine() when its * argument is NULL (a special case) */ if (mType != SessionType_RemoteSession) mType = SessionType_ExistingSession; else Assert (mState == SessionState_SessionSpawning); mState = SessionState_SessionOpen; } else { /* some cleanup */ mControl.setNull(); mRemoteMachine.setNull(); mRemoteConsole.setNull(); } LogFlowThisFunc (("rc=%08X\n", rc)); LogFlowThisFuncLeave(); return rc; } STDMETHODIMP Session::UpdateMachineState (MachineState_T aMachineState) { AutoCaller autoCaller (this); if (autoCaller.state() != Ready) { /* * We might have already entered Session::uninit() at this point, so * return silently (not interested in the state change during uninit) */ LogFlowThisFunc (("Already uninitialized.\n")); return S_OK; } AutoReaderLock alock (this); if (mState == SessionState_SessionClosing) { LogFlowThisFunc (("Already being closed.\n")); return S_OK; } AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); AssertReturn (!mControl.isNull(), E_FAIL); AssertReturn (!mConsole.isNull(), E_FAIL); return mConsole->updateMachineState (aMachineState); } STDMETHODIMP Session::Uninitialize() { LogFlowThisFuncEnter(); AutoCaller autoCaller (this); HRESULT rc = S_OK; if (autoCaller.state() == Ready) { AutoReaderLock alock (this); LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType)); if (mState == SessionState_SessionClosing) { LogFlowThisFunc (("Already being closed.\n")); return S_OK; } AssertReturn (mState == SessionState_SessionOpen, E_FAIL); /* close ourselves */ rc = close (false /* aFinalRelease */, true /* aFromServer */); } else if (autoCaller.state() == InUninit) { /* * We might have already entered Session::uninit() at this point, * return silently */ LogFlowThisFunc (("Already uninitialized.\n")); } else { LogWarningThisFunc (("UNEXPECTED uninitialization!\n")); rc = autoCaller.rc(); } LogFlowThisFunc (("rc=%08X\n", rc)); LogFlowThisFuncLeave(); return rc; } STDMETHODIMP Session::OnDVDDriveChange() { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onDVDDriveChange(); } STDMETHODIMP Session::OnFloppyDriveChange() { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onFloppyDriveChange(); } STDMETHODIMP Session::OnNetworkAdapterChange(INetworkAdapter *networkAdapter) { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onNetworkAdapterChange(networkAdapter); } STDMETHODIMP Session::OnVRDPServerChange() { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onVRDPServerChange(); } STDMETHODIMP Session::OnUSBControllerChange() { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onUSBControllerChange(); } STDMETHODIMP Session::OnUSBDeviceAttach (IUSBDevice *aDevice) { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onUSBDeviceAttach (aDevice); } STDMETHODIMP Session::OnUSBDeviceDetach (INPTR GUIDPARAM aId) { LogFlowThisFunc (("\n")); AutoCaller autoCaller (this); AssertComRCReturn (autoCaller.rc(), autoCaller.rc()); AutoReaderLock alock (this); AssertReturn (mState == SessionState_SessionOpen && mType == SessionType_DirectSession, E_FAIL); return mConsole->onUSBDeviceDetach (aId); } // private methods /////////////////////////////////////////////////////////////////////////////// /** * Closes the current session. * * @param aFinalRelease called as a result of FinalRelease() * @param aFromServer called as a result of Uninitialize() * * @note To be called only from #uninit(), #Close() or #Uninitialize(). * @note Locks this object for writing. */ HRESULT Session::close (bool aFinalRelease, bool aFromServer) { LogFlowThisFuncEnter(); LogFlowThisFunc (("aFinalRelease=%d, isFromServer=%d\n", aFinalRelease, aFromServer)); AutoCaller autoCaller (this); AssertComRCReturnRC (autoCaller.rc()); AutoLock alock (this); LogFlowThisFunc (("mState=%d, mType=%d\n", mState, mType)); if (mState != SessionState_SessionOpen) { Assert (mState == SessionState_SessionSpawning); /* The session object is going to be uninitialized by the client before * it has been assigned a direct console of the machine the client * requested to open a remote session to using IVirtualBox:: * openRemoteSession(). Theoretically it should not happen because * openRemoteSession() doesn't return control to the client until the * procedure is fully complete, so assert here. */ AssertFailed(); mState = SessionState_SessionClosed; mType = SessionType_InvalidSessionType; #if defined(__WIN__) Assert (!mIPCSem && !mIPCThreadSem); #elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) Assert (mIPCSem == -1); #endif LogFlowThisFuncLeave(); return S_OK; } /* go to the closing state */ mState = SessionState_SessionClosing; if (mType == SessionType_DirectSession) { mConsole->uninit(); mConsole.setNull(); } else { mRemoteMachine.setNull(); mRemoteConsole.setNull(); } ComPtr progress; if (!aFinalRelease && !aFromServer) { /* * We trigger OnSessionEnd() only when the session closes itself using * Close(). Note that if isFinalRelease = TRUE here, this means that * the client process has already initialized the termination procedure * without issuing Close() and the IPC channel is no more operational -- * so we cannot call the server's method (it will definitely fail). The * server will instead simply detect the abnormal client death (since * OnSessionEnd() is not called) and reset the machine state to Aborted. */ /* * while waiting for OnSessionEnd() to complete one of our methods * can be called by the server (for example, Uninitialize(), if the * direct session has initiated a closure just a bit before us) so * we need to release the lock to avoid deadlocks. The state is already * SessionState_SessionClosing here, so it's safe. */ alock.leave(); LogFlowThisFunc (("Calling mControl->OnSessionEnd()...\n")); HRESULT rc = mControl->OnSessionEnd (this, progress.asOutParam()); LogFlowThisFunc (("mControl->OnSessionEnd()=%08X\n", rc)); alock.enter(); /* * If we get E_UNEXPECTED this means that the direct session has already * been closed, we're just too late with our notification and nothing more */ if (mType != SessionType_DirectSession && rc == E_UNEXPECTED) rc = S_OK; AssertComRC (rc); } mControl.setNull(); if (mType == SessionType_DirectSession) { releaseIPCSemaphore(); if (!aFinalRelease && !aFromServer) { /* * Wait for the server to grab the semaphore and destroy the session * machine (allowing us to open a new session with the same machine * once this method returns) */ Assert (!!progress); if (progress) progress->WaitForCompletion (-1); } } mState = SessionState_SessionClosed; mType = SessionType_InvalidSessionType; /* release the VirtualBox instance as the very last step */ mVirtualBox.setNull(); LogFlowThisFuncLeave(); return S_OK; } /** @note To be called only from #AssignMachine() */ HRESULT Session::grabIPCSemaphore() { HRESULT rc = E_FAIL; /* open the IPC semaphore based on the sessionId and try to grab it */ Bstr ipcId; rc = mControl->GetIPCId (ipcId.asOutParam()); AssertComRCReturn (rc, rc); LogFlowThisFunc (("ipcId: {%ls}\n", ipcId.raw())); #if defined(__WIN__) /* * Since Session is an MTA object, this method can be executed on * any thread, and this thread will not necessarily match the thread on * which close() will be called later. Therefore, we need a separate * thread to hold the IPC mutex and then release it in close(). */ mIPCThreadSem = ::CreateEvent (NULL, FALSE, FALSE, NULL); AssertMsgReturn (mIPCThreadSem, ("Cannot create an event sem, err=%d", ::GetLastError()), E_FAIL); void *data [3]; data [0] = (void *) (BSTR) ipcId; data [1] = (void *) mIPCThreadSem; data [2] = 0; /* will get an output from the thread */ /* create the thread to hold the IPC mutex until signalled to release it */ RTTHREAD tid; int vrc = RTThreadCreate (&tid, IPCMutexHolderThread, (void *) data, 0, RTTHREADTYPE_MAIN_WORKER, 0, "IPCHolder"); AssertRCReturn (vrc, E_FAIL); /* wait until thread init is completed */ DWORD wrc = ::WaitForSingleObject (mIPCThreadSem, INFINITE); AssertMsg (wrc == WAIT_OBJECT_0, ("Wait failed, err=%d", ::GetLastError())); Assert (data [2]); if (wrc == WAIT_OBJECT_0 && data [2]) { /* memorize the event sem we should signal in close() */ mIPCSem = (HANDLE) data [2]; rc = S_OK; } else { ::CloseHandle (mIPCThreadSem); mIPCThreadSem = NULL; rc = E_FAIL; } #elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) Utf8Str semName = ipcId; char *pszSemName = NULL; RTStrUtf8ToCurrentCP (&pszSemName, semName); key_t key = ::ftok (pszSemName, 0); RTStrFree (pszSemName); mIPCSem = ::semget (key, 0, 0); AssertMsgReturn (mIPCSem >= 0, ("Cannot open IPC semaphore, errno=%d", errno), E_FAIL); /* grab the semaphore */ ::sembuf sop = { 0, -1, SEM_UNDO }; int rv = ::semop (mIPCSem, &sop, 1); AssertMsgReturn (rv == 0, ("Cannot grab IPC semaphore, errno=%d", errno), E_FAIL); #endif return rc; } /** @note To be called only from #close() */ void Session::releaseIPCSemaphore() { /* release the IPC semaphore */ #if defined(__WIN__) if (mIPCSem && mIPCThreadSem) { /* * say the thread holding the IPC mutex to release it; * it will close mIPCSem handle */ ::SetEvent (mIPCSem); /* wait for the thread to finish */ ::WaitForSingleObject (mIPCThreadSem, INFINITE); ::CloseHandle (mIPCThreadSem); } #elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER) if (mIPCSem >= 0) { ::sembuf sop = { 0, 1, SEM_UNDO }; ::semop (mIPCSem, &sop, 1); } #endif } #if defined(__WIN__) /** VM IPC mutex holder thread */ DECLCALLBACK(int) IPCMutexHolderThread (RTTHREAD Thread, void *pvUser) { LogFlowFuncEnter(); Assert (pvUser); void **data = (void **) pvUser; BSTR sessionId = (BSTR) data [0]; HANDLE initDoneSem = (HANDLE) data [1]; HANDLE ipcMutex = ::OpenMutex (MUTEX_ALL_ACCESS, FALSE, sessionId); AssertMsg (ipcMutex, ("cannot open IPC mutex, err=%08X\n", ::GetLastError())); if (ipcMutex) { /* grab the mutex */ DWORD wrc = ::WaitForSingleObject (ipcMutex, 0); AssertMsg (wrc == WAIT_OBJECT_0, ("cannot grab IPC mutex, err=%d\n", wrc)); if (wrc == WAIT_OBJECT_0) { HANDLE finishSem = ::CreateEvent (NULL, FALSE, FALSE, NULL); AssertMsg (finishSem, ("cannot create event sem, err=%d\n", ::GetLastError())); if (finishSem) { data [2] = (void *) finishSem; /* signal we're done with init */ ::SetEvent (initDoneSem); /* wait until we're signaled to release the IPC mutex */ ::WaitForSingleObject (finishSem, INFINITE); /* release the IPC mutex */ LogFlow (("IPCMutexHolderThread(): releasing IPC mutex...\n")); BOOL success = ::ReleaseMutex (ipcMutex); AssertMsg (success, ("cannot release mutex, err=%d\n", ::GetLastError())); ::CloseHandle (ipcMutex); ::CloseHandle (finishSem); } } } /* signal we're done */ ::SetEvent (initDoneSem); LogFlowFuncLeave(); return 0; } #endif