/* $Id: VBoxDragAndDropSvc.cpp 85407 2020-07-22 08:45:18Z vboxsync $ */ /** @file * Drag and Drop Service. */ /* * Copyright (C) 2011-2020 Oracle Corporation * * 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 (GPL) 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. */ /** @page pg_svc_dnd Drag and drop HGCM Service * * TODO */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_GUEST_DND #include #include #include #include #include #include #include #include #include #include "dndmanager.h" using namespace DragAndDropSvc; /********************************************************************************************************************************* * Service class declaration * *********************************************************************************************************************************/ class DragAndDropClient : public HGCM::Client { public: DragAndDropClient(uint32_t idClient) : HGCM::Client(idClient) { RT_ZERO(m_SvcCtx); } virtual ~DragAndDropClient(void) { disconnect(); } public: void disconnect(void) RT_NOEXCEPT; }; /** Map holding pointers to drag and drop clients. Key is the (unique) HGCM client ID. */ typedef std::map DnDClientMap; /** Simple queue (list) which holds deferred (waiting) clients. */ typedef std::list DnDClientQueue; /** * Specialized drag & drop service class. */ class DragAndDropService : public HGCM::AbstractService { public: explicit DragAndDropService(PVBOXHGCMSVCHELPERS pHelpers) : HGCM::AbstractService(pHelpers) , m_pManager(NULL) , m_u32Mode(VBOX_DRAG_AND_DROP_MODE_OFF) {} protected: int init(VBOXHGCMSVCFNTABLE *pTable) RT_NOEXCEPT RT_OVERRIDE; int uninit(void) RT_NOEXCEPT RT_OVERRIDE; int clientConnect(uint32_t idClient, void *pvClient) RT_NOEXCEPT RT_OVERRIDE; int clientDisconnect(uint32_t idClient, void *pvClient) RT_NOEXCEPT RT_OVERRIDE; void guestCall(VBOXHGCMCALLHANDLE callHandle, uint32_t idClient, void *pvClient, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) RT_NOEXCEPT RT_OVERRIDE; int hostCall(uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) RT_NOEXCEPT RT_OVERRIDE; private: int modeSet(uint32_t u32Mode) RT_NOEXCEPT; inline uint32_t modeGet(void) const RT_NOEXCEPT { return m_u32Mode; }; static DECLCALLBACK(int) progressCallback(uint32_t uStatus, uint32_t uPercentage, int rc, void *pvUser); private: /** Pointer to our DnD manager instance. */ DnDManager *m_pManager; /** Map of all connected clients. * The primary key is the (unique) client ID, the secondary value * an allocated pointer to the DragAndDropClient class, managed * by this service class. */ DnDClientMap m_clientMap; /** List of all clients which are queued up (deferred return) and ready * to process new commands. The key is the (unique) client ID. */ DnDClientQueue m_clientQueue; /** Current drag and drop mode, VBOX_DRAG_AND_DROP_MODE_XXX. */ uint32_t m_u32Mode; }; /********************************************************************************************************************************* * Client implementation * *********************************************************************************************************************************/ /** * Called when the HGCM client disconnected on the guest side. * * This function takes care of the client's data cleanup and also lets the host * know that the client has been disconnected. */ void DragAndDropClient::disconnect(void) RT_NOEXCEPT { LogFlowThisFunc(("uClient=%RU32\n", m_idClient)); if (IsDeferred()) CompleteDeferred(VERR_INTERRUPTED); /* * Let the host know. */ VBOXDNDCBDISCONNECTMSGDATA data; RT_ZERO(data); /** @todo Magic needed? */ /** @todo Add context ID. */ if (m_SvcCtx.pfnHostCallback) { int rc2 = m_SvcCtx.pfnHostCallback(m_SvcCtx.pvHostData, GUEST_DND_DISCONNECT, &data, sizeof(data)); if (RT_FAILURE(rc2)) LogFlowFunc(("Warning: Unable to notify host about client %RU32 disconnect, rc=%Rrc\n", m_idClient, rc2)); /* Not fatal. */ } } /********************************************************************************************************************************* * Service class implementation * *********************************************************************************************************************************/ int DragAndDropService::init(VBOXHGCMSVCFNTABLE *pTable) RT_NOEXCEPT { /* Register functions. */ pTable->pfnHostCall = svcHostCall; pTable->pfnSaveState = NULL; /* The service is stateless, so the normal */ pTable->pfnLoadState = NULL; /* construction done before restoring suffices */ pTable->pfnRegisterExtension = svcRegisterExtension; pTable->pfnNotify = NULL; /* Drag'n drop mode is disabled by default. */ modeSet(VBOX_DRAG_AND_DROP_MODE_OFF); int rc = VINF_SUCCESS; try { m_pManager = new DnDManager(&DragAndDropService::progressCallback, this); } catch (std::bad_alloc &) { rc = VERR_NO_MEMORY; } LogFlowFuncLeaveRC(rc); return rc; } int DragAndDropService::uninit(void) RT_NOEXCEPT { LogFlowFuncEnter(); if (m_pManager) { delete m_pManager; m_pManager = NULL; } DnDClientMap::iterator itClient = m_clientMap.begin(); while (itClient != m_clientMap.end()) { delete itClient->second; m_clientMap.erase(itClient); itClient = m_clientMap.begin(); } LogFlowFuncLeave(); return VINF_SUCCESS; } int DragAndDropService::clientConnect(uint32_t idClient, void *pvClient) RT_NOEXCEPT { RT_NOREF1(pvClient); if (m_clientMap.size() >= UINT8_MAX) /* Don't allow too much clients at the same time. */ { AssertMsgFailed(("Maximum number of clients reached\n")); return VERR_MAX_PROCS_REACHED; } /* * Add client to our client map. */ if (m_clientMap.find(idClient) != m_clientMap.end()) { LogFunc(("Client %RU32 is already connected!\n", idClient)); return VERR_ALREADY_EXISTS; } try { DragAndDropClient *pClient = new DragAndDropClient(idClient); pClient->SetSvcContext(m_SvcCtx); m_clientMap[idClient] = pClient; } catch (std::bad_alloc &) { LogFunc(("Client %RU32 - VERR_NO_MEMORY!\n", idClient)); return VERR_NO_MEMORY; } /* * Reset the message queue as soon as a new clients connect * to ensure that every client has the same state. */ if (m_pManager) m_pManager->Reset(); LogFlowFunc(("Client %RU32 connected (VINF_SUCCESS)\n", idClient)); return VINF_SUCCESS; } int DragAndDropService::clientDisconnect(uint32_t idClient, void *pvClient) RT_NOEXCEPT { RT_NOREF1(pvClient); /* Client not found? Bail out early. */ DnDClientMap::iterator itClient = m_clientMap.find(idClient); if (itClient == m_clientMap.end()) { LogFunc(("Client %RU32 not found!\n", idClient)); return VERR_NOT_FOUND; } /* * Remove from waiters queue. */ m_clientQueue.remove(idClient); /* * Remove from client map and deallocate. */ AssertPtr(itClient->second); delete itClient->second; m_clientMap.erase(itClient); LogFlowFunc(("Client %RU32 disconnected\n", idClient)); return VINF_SUCCESS; } int DragAndDropService::modeSet(uint32_t u32Mode) RT_NOEXCEPT { #ifndef VBOX_WITH_DRAG_AND_DROP_GH if ( u32Mode == VBOX_DRAG_AND_DROP_MODE_GUEST_TO_HOST || u32Mode == VBOX_DRAG_AND_DROP_MODE_BIDIRECTIONAL) { m_u32Mode = VBOX_DRAG_AND_DROP_MODE_OFF; return VERR_NOT_SUPPORTED; } #endif switch (u32Mode) { case VBOX_DRAG_AND_DROP_MODE_OFF: case VBOX_DRAG_AND_DROP_MODE_HOST_TO_GUEST: case VBOX_DRAG_AND_DROP_MODE_GUEST_TO_HOST: case VBOX_DRAG_AND_DROP_MODE_BIDIRECTIONAL: m_u32Mode = u32Mode; break; default: m_u32Mode = VBOX_DRAG_AND_DROP_MODE_OFF; break; } return VINF_SUCCESS; } void DragAndDropService::guestCall(VBOXHGCMCALLHANDLE callHandle, uint32_t idClient, void *pvClient, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) RT_NOEXCEPT { RT_NOREF1(pvClient); LogFlowFunc(("idClient=%RU32, u32Function=%RU32, cParms=%RU32\n", idClient, u32Function, cParms)); /* Check if we've the right mode set. */ int rc = VERR_ACCESS_DENIED; /* Play safe. */ switch (u32Function) { case GUEST_DND_GET_NEXT_HOST_MSG: { if (modeGet() != VBOX_DRAG_AND_DROP_MODE_OFF) rc = VINF_SUCCESS; else { LogFlowFunc(("DnD disabled, deferring request\n")); rc = VINF_HGCM_ASYNC_EXECUTE; } break; } /* New since protocol v2. */ case GUEST_DND_CONNECT: { /* * Never block the initial connect call, as the clients do this when * initializing and might get stuck if drag and drop is set to "disabled" at * that time. */ rc = VINF_SUCCESS; break; } case GUEST_DND_HG_ACK_OP: case GUEST_DND_HG_REQ_DATA: case GUEST_DND_HG_EVT_PROGRESS: { if ( modeGet() == VBOX_DRAG_AND_DROP_MODE_BIDIRECTIONAL || modeGet() == VBOX_DRAG_AND_DROP_MODE_HOST_TO_GUEST) rc = VINF_SUCCESS; else LogFlowFunc(("Host -> Guest DnD mode disabled, failing request\n")); break; } case GUEST_DND_GH_ACK_PENDING: case GUEST_DND_GH_SND_DATA_HDR: case GUEST_DND_GH_SND_DATA: case GUEST_DND_GH_SND_DIR: case GUEST_DND_GH_SND_FILE_HDR: case GUEST_DND_GH_SND_FILE_DATA: case GUEST_DND_GH_EVT_ERROR: { #ifdef VBOX_WITH_DRAG_AND_DROP_GH if ( modeGet() == VBOX_DRAG_AND_DROP_MODE_BIDIRECTIONAL || modeGet() == VBOX_DRAG_AND_DROP_MODE_GUEST_TO_HOST) rc = VINF_SUCCESS; else #endif LogFlowFunc(("Guest -> Host DnD mode disabled, failing request\n")); break; } default: /* Reach through to DnD manager. */ rc = VINF_SUCCESS; break; } #ifdef DEBUG_andy LogFlowFunc(("Mode (%RU32) check rc=%Rrc\n", modeGet(), rc)); #endif #define DO_HOST_CALLBACK(); \ if ( RT_SUCCESS(rc) \ && m_SvcCtx.pfnHostCallback) \ { \ rc = m_SvcCtx.pfnHostCallback(m_SvcCtx.pvHostData, u32Function, &data, sizeof(data)); \ } /* * Lookup client. */ DragAndDropClient *pClient = NULL; DnDClientMap::iterator itClient = m_clientMap.find(idClient); if (itClient != m_clientMap.end()) { pClient = itClient->second; AssertPtr(pClient); } else { LogFunc(("Client %RU32 was not found\n", idClient)); rc = VERR_NOT_FOUND; } /* Verifies that an uint32 parameter has the expected buffer size set. * Will set rc to VERR_INVALID_PARAMETER otherwise. See #9777. */ #define VERIFY_BUFFER_SIZE_UINT32(a_ParmUInt32, a_SizeExpected) \ do { \ uint32_t cbTemp = 0; \ rc = HGCMSvcGetU32(&a_ParmUInt32, &cbTemp); \ ASSERT_GUEST_BREAK(RT_SUCCESS(rc) && cbTemp == a_SizeExpected); \ } while (0) /* Gets the context ID from the first parameter and store it into the data header. * Then increments idxParm by one if more than one parameter is available. */ #define GET_CONTEXT_ID_PARM0() \ if (fHasCtxID) \ { \ ASSERT_GUEST_BREAK(cParms >= 1); \ rc = HGCMSvcGetU32(&paParms[0], &data.hdr.uContextID); \ ASSERT_GUEST_BREAK(RT_SUCCESS(rc)); \ if (cParms > 1) \ idxParm++; \ } if (rc == VINF_SUCCESS) /* Note: rc might be VINF_HGCM_ASYNC_EXECUTE! */ { LogFlowFunc(("Client %RU32: Protocol v%RU32\n", pClient->GetClientID(), pClient->GetProtocolVer())); rc = VERR_INVALID_PARAMETER; /* Play safe by default. */ /* Whether the client's advertised protocol sends context IDs with commands. */ const bool fHasCtxID = pClient->GetProtocolVer() >= 3; /* Current parameter index to process. */ unsigned idxParm = 0; switch (u32Function) { /* * Note: Older VBox versions with enabled DnD guest->host support (< 5.0) * used the same message ID (300) for GUEST_DND_GET_NEXT_HOST_MSG and * HOST_DND_GH_REQ_PENDING, which led this service returning * VERR_INVALID_PARAMETER when the guest wanted to actually * handle HOST_DND_GH_REQ_PENDING. */ case GUEST_DND_GET_NEXT_HOST_MSG: { LogFlowFunc(("GUEST_DND_GET_NEXT_HOST_MSG\n")); if (cParms == 3) { rc = m_pManager->GetNextMsgInfo(&paParms[0].u.uint32 /* uMsg */, &paParms[1].u.uint32 /* cParms */); if (RT_FAILURE(rc)) /* No queued messages available? */ { if (m_SvcCtx.pfnHostCallback) /* Try asking the host. */ { VBOXDNDCBHGGETNEXTHOSTMSG data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_HG_GET_NEXT_HOST_MSG; rc = m_SvcCtx.pfnHostCallback(m_SvcCtx.pvHostData, u32Function, &data, sizeof(data)); if (RT_SUCCESS(rc)) { paParms[0].u.uint32 = data.uMsg; /* uMsg */ paParms[1].u.uint32 = data.cParms; /* cParms */ /* Note: paParms[2] was set by the guest as blocking flag. */ } } else /* No host callback in place, so drag and drop is not supported by the host. */ rc = VERR_NOT_SUPPORTED; if (RT_FAILURE(rc)) rc = m_pManager->GetNextMsg(u32Function, cParms, paParms); /* Some error occurred or no (new) messages available? */ if (RT_FAILURE(rc)) { uint32_t fFlags = 0; int rc2 = HGCMSvcGetU32(&paParms[2], &fFlags); if ( RT_SUCCESS(rc2) && fFlags) /* Blocking flag set? */ { /* Defer client returning. */ rc = VINF_HGCM_ASYNC_EXECUTE; } else rc = VERR_INVALID_PARAMETER; LogFlowFunc(("Message queue is empty, returning %Rrc to guest\n", rc)); } } } break; } case GUEST_DND_CONNECT: { LogFlowFunc(("GUEST_DND_CONNECT\n")); ASSERT_GUEST_BREAK(cParms >= 2); VBOXDNDCBCONNECTMSGDATA data; RT_ZERO(data); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.hdr.uContextID); \ ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uProtocol); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm], &data.uFlags); ASSERT_GUEST_RC_BREAK(rc); unsigned uProtocolVer = 3; /* The protocol version we're going to use. */ /* Make sure we're only setting a protocl version we're supporting on the host. */ if (data.uProtocol > uProtocolVer) data.uProtocol = uProtocolVer; pClient->SetProtocolVer(data.uProtocol); /* Return the highest protocol version we're supporting. */ AssertBreak(idxParm); ASSERT_GUEST_BREAK(idxParm); paParms[idxParm - 1].u.uint32 = data.uProtocol; LogFlowFunc(("Client %RU32 is now using protocol v%RU32\n", pClient->GetClientID(), pClient->GetProtocolVer())); DO_HOST_CALLBACK(); break; } case GUEST_DND_HG_ACK_OP: { LogFlowFunc(("GUEST_DND_HG_ACK_OP\n")); ASSERT_GUEST_BREAK(cParms >= 2); VBOXDNDCBHGACKOPDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_HG_ACK_OP; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetU32(&paParms[idxParm], &data.uAction); /* Get drop action. */ ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } case GUEST_DND_HG_REQ_DATA: { LogFlowFunc(("GUEST_DND_HG_REQ_DATA\n")); VBOXDNDCBHGREQDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_HG_REQ_DATA; switch (pClient->GetProtocolVer()) { case 3: { ASSERT_GUEST_BREAK(cParms == 3); GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void **)&data.pszFormat, &data.cbFormat); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.cbFormat); break; } case 2: RT_FALL_THROUGH(); default: { ASSERT_GUEST_BREAK(cParms == 1); rc = HGCMSvcGetPv(&paParms[idxParm], (void**)&data.pszFormat, &data.cbFormat); ASSERT_GUEST_RC_BREAK(rc); break; } } DO_HOST_CALLBACK(); break; } case GUEST_DND_HG_EVT_PROGRESS: { LogFlowFunc(("GUEST_DND_HG_EVT_PROGRESS\n")); ASSERT_GUEST_BREAK(cParms >= 3); VBOXDNDCBHGEVTPROGRESSDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_HG_EVT_PROGRESS; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uStatus); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uPercentage); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm], &data.rc); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } #ifdef VBOX_WITH_DRAG_AND_DROP_GH case GUEST_DND_GH_ACK_PENDING: { LogFlowFunc(("GUEST_DND_GH_ACK_PENDING\n")); VBOXDNDCBGHACKPENDINGDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_ACK_PENDING; switch (pClient->GetProtocolVer()) { case 3: { ASSERT_GUEST_BREAK(cParms == 5); GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uDefAction); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uAllActions); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pszFormat, &data.cbFormat); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.cbFormat); break; } case 2: default: { ASSERT_GUEST_BREAK(cParms == 3); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uDefAction); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.uAllActions); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetPv(&paParms[idxParm], (void**)&data.pszFormat, &data.cbFormat); ASSERT_GUEST_RC_BREAK(rc); break; } } DO_HOST_CALLBACK(); break; } /* New since protocol v3. */ case GUEST_DND_GH_SND_DATA_HDR: { LogFlowFunc(("GUEST_DND_GH_SND_DATA_HDR\n")); ASSERT_GUEST_BREAK(cParms == 12); VBOXDNDCBSNDDATAHDRDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_DATA_HDR; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.data.uFlags); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.data.uScreenId); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU64(&paParms[idxParm++], &data.data.cbTotal); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.data.cbMeta); ASSERT_GUEST_RC_BREAK(rc); ASSERT_GUEST_BREAK(data.data.cbMeta <= data.data.cbTotal); rc = HGCMSvcGetPv(&paParms[idxParm++], &data.data.pvMetaFmt, &data.data.cbMetaFmt); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.data.cbMetaFmt); rc = HGCMSvcGetU64(&paParms[idxParm++], &data.data.cObjects); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.data.enmCompression); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], (uint32_t *)&data.data.enmChecksumType); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetPv(&paParms[idxParm++], &data.data.pvChecksum, &data.data.cbChecksum); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.data.cbChecksum); DO_HOST_CALLBACK(); break; } case GUEST_DND_GH_SND_DATA: { LogFlowFunc(("GUEST_DND_GH_SND_DATA\n")); switch (pClient->GetProtocolVer()) { case 3: { ASSERT_GUEST_BREAK(cParms == 5); VBOXDNDCBSNDDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_DATA; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.data.u.v3.pvData, &data.data.u.v3.cbData); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.data.u.v3.cbData); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.data.u.v3.pvChecksum, &data.data.u.v3.cbChecksum); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.data.u.v3.cbChecksum); DO_HOST_CALLBACK(); break; } case 2: default: { ASSERT_GUEST_BREAK(cParms == 2); VBOXDNDCBSNDDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_DATA; rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.data.u.v1.pvData, &data.data.u.v1.cbData); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm], &data.data.u.v1.cbTotalSize); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } } break; } case GUEST_DND_GH_SND_DIR: { LogFlowFunc(("GUEST_DND_GH_SND_DIR\n")); ASSERT_GUEST_BREAK(cParms >= 3); VBOXDNDCBSNDDIRDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_DIR; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pszPath, &data.cbPath); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.cbPath); rc = HGCMSvcGetU32(&paParms[idxParm], &data.fMode); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } /* New since protocol v2 (>= VBox 5.0). */ case GUEST_DND_GH_SND_FILE_HDR: { LogFlowFunc(("GUEST_DND_GH_SND_FILE_HDR\n")); ASSERT_GUEST_BREAK(cParms == 6); VBOXDNDCBSNDFILEHDRDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_FILE_HDR; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pszFilePath, &data.cbFilePath); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.cbFilePath); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.fFlags); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU32(&paParms[idxParm++], &data.fMode); ASSERT_GUEST_RC_BREAK(rc); rc = HGCMSvcGetU64(&paParms[idxParm], &data.cbSize); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } case GUEST_DND_GH_SND_FILE_DATA: { LogFlowFunc(("GUEST_DND_GH_SND_FILE_DATA\n")); switch (pClient->GetProtocolVer()) { /* Protocol v3 adds (optional) checksums. */ case 3: { ASSERT_GUEST_BREAK(cParms == 5); VBOXDNDCBSNDFILEDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_FILE_DATA; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pvData, &data.cbData); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.cbData); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.u.v3.pvChecksum, &data.u.v3.cbChecksum); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.u.v3.cbChecksum); DO_HOST_CALLBACK(); break; } /* Protocol v2 only sends the next data chunks to reduce traffic. */ case 2: { ASSERT_GUEST_BREAK(cParms == 3); VBOXDNDCBSNDFILEDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_FILE_DATA; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pvData, &data.cbData); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm], data.cbData); DO_HOST_CALLBACK(); break; } /* Protocol v1 sends the file path and attributes for every file chunk (!). */ default: { ASSERT_GUEST_BREAK(cParms == 5); VBOXDNDCBSNDFILEDATADATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_SND_FILE_DATA; rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.u.v1.pszFilePath, &data.u.v1.cbFilePath); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.u.v1.cbFilePath); rc = HGCMSvcGetPv(&paParms[idxParm++], (void**)&data.pvData, &data.cbData); ASSERT_GUEST_RC_BREAK(rc); VERIFY_BUFFER_SIZE_UINT32(paParms[idxParm++], data.cbData); rc = HGCMSvcGetU32(&paParms[idxParm], &data.u.v1.fMode); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } } break; } case GUEST_DND_GH_EVT_ERROR: { LogFlowFunc(("GUEST_DND_GH_EVT_ERROR\n")); ASSERT_GUEST_BREAK(cParms >= 1); VBOXDNDCBEVTERRORDATA data; RT_ZERO(data); data.hdr.uMagic = CB_MAGIC_DND_GH_EVT_ERROR; GET_CONTEXT_ID_PARM0(); rc = HGCMSvcGetU32(&paParms[idxParm], (uint32_t *)&data.rc); ASSERT_GUEST_RC_BREAK(rc); DO_HOST_CALLBACK(); break; } #endif /* VBOX_WITH_DRAG_AND_DROP_GH */ default: { /* All other messages are handled by the DnD manager. */ rc = m_pManager->GetNextMsg(u32Function, cParms, paParms); if (rc == VERR_NO_DATA) /* Manager has no new messsages? Try asking the host. */ { if (m_SvcCtx.pfnHostCallback) { VBOXDNDCBHGGETNEXTHOSTMSGDATA data; RT_ZERO(data); data.hdr.uMagic = VBOX_DND_CB_MAGIC_MAKE(0 /* uFn */, 0 /* uVer */); data.uMsg = u32Function; data.cParms = cParms; data.paParms = paParms; rc = m_SvcCtx.pfnHostCallback(m_SvcCtx.pvHostData, u32Function, &data, sizeof(data)); if (RT_SUCCESS(rc)) { cParms = data.cParms; paParms = data.paParms; } else { /* * In case the guest is too fast asking for the next message * and the host did not supply it yet, just defer the client's * return until a response from the host available. */ LogFlowFunc(("No new messages from the host (yet), deferring request: %Rrc\n", rc)); rc = VINF_HGCM_ASYNC_EXECUTE; } } else /* No host callback in place, so drag and drop is not supported by the host. */ rc = VERR_NOT_SUPPORTED; } break; } } } #undef VERIFY_BUFFER_SIZE_UINT32 /* * If async execution is requested, we didn't notify the guest yet about * completion. The client is queued into the waiters list and will be * notified as soon as a new event is available. */ if (rc == VINF_HGCM_ASYNC_EXECUTE) { try { AssertPtr(pClient); pClient->SetDeferred(callHandle, u32Function, cParms, paParms); m_clientQueue.push_back(idClient); } catch (std::bad_alloc &) { rc = VERR_NO_MEMORY; /* Don't report to guest. */ } } else if (pClient) pClient->Complete(callHandle, rc); else { AssertMsgFailed(("Guest call failed with %Rrc\n", rc)); rc = VERR_NOT_IMPLEMENTED; } LogFlowFunc(("Returning rc=%Rrc\n", rc)); } int DragAndDropService::hostCall(uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) RT_NOEXCEPT { LogFlowFunc(("u32Function=%RU32, cParms=%RU32, cClients=%zu, cQueue=%zu\n", u32Function, cParms, m_clientMap.size(), m_clientQueue.size())); int rc; bool fSendToGuest = false; /* Whether to send the message down to the guest side or not. */ switch (u32Function) { case HOST_DND_SET_MODE: { if (cParms != 1) rc = VERR_INVALID_PARAMETER; else if (paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT) rc = VERR_INVALID_PARAMETER; else rc = modeSet(paParms[0].u.uint32); break; } case HOST_DND_CANCEL: { LogFlowFunc(("Cancelling all waiting clients ...\n")); /* Reset the message queue as the host cancelled the whole operation. */ m_pManager->Reset(); rc = m_pManager->AddMsg(u32Function, cParms, paParms, true /* fAppend */); if (RT_FAILURE(rc)) { AssertMsgFailed(("Adding new message of type=%RU32 failed with rc=%Rrc\n", u32Function, rc)); break; } /* * Wake up all deferred clients and tell them to process * the cancelling message next. */ DnDClientQueue::iterator itQueue = m_clientQueue.begin(); while (itQueue != m_clientQueue.end()) { DnDClientMap::iterator itClient = m_clientMap.find(*itQueue); Assert(itClient != m_clientMap.end()); DragAndDropClient *pClient = itClient->second; AssertPtr(pClient); int rc2 = pClient->SetDeferredMsgInfo(HOST_DND_CANCEL, /* Protocol v3+ also contains the context ID. */ pClient->GetProtocolVer() >= 3 ? 1 : 0); pClient->CompleteDeferred(rc2); m_clientQueue.erase(itQueue); itQueue = m_clientQueue.begin(); } Assert(m_clientQueue.empty()); /* Tell the host that everything went well. */ rc = VINF_SUCCESS; break; } case HOST_DND_HG_EVT_ENTER: { /* Reset the message queue as a new DnD operation just began. */ m_pManager->Reset(); fSendToGuest = true; rc = VINF_SUCCESS; break; } default: { fSendToGuest = true; rc = VINF_SUCCESS; break; } } do /* goto avoidance break-loop. */ { if (fSendToGuest) { if (modeGet() == VBOX_DRAG_AND_DROP_MODE_OFF) { /* Tell the host that a wrong drag'n drop mode is set. */ rc = VERR_ACCESS_DENIED; break; } if (m_clientMap.empty()) /* At least one client on the guest connected? */ { /* * Tell the host that the guest does not support drag'n drop. * This might happen due to not installed Guest Additions or * not running VBoxTray/VBoxClient. */ rc = VERR_NOT_SUPPORTED; break; } rc = m_pManager->AddMsg(u32Function, cParms, paParms, true /* fAppend */); if (RT_FAILURE(rc)) { AssertMsgFailed(("Adding new message of type=%RU32 failed with rc=%Rrc\n", u32Function, rc)); break; } /* Any clients in our queue ready for processing the next command? */ if (m_clientQueue.empty()) { LogFlowFunc(("All clients (%zu) busy -- delaying execution\n", m_clientMap.size())); break; } uint32_t uClientNext = m_clientQueue.front(); DnDClientMap::iterator itClientNext = m_clientMap.find(uClientNext); Assert(itClientNext != m_clientMap.end()); DragAndDropClient *pClient = itClientNext->second; AssertPtr(pClient); /* * Check if this was a request for getting the next host * message. If so, return the message ID and the parameter * count. The message itself has to be queued. */ uint32_t uMsgClient = pClient->GetMsgType(); uint32_t uMsgNext = 0; uint32_t cParmsNext = 0; int rcNext = m_pManager->GetNextMsgInfo(&uMsgNext, &cParmsNext); LogFlowFunc(("uMsgClient=%RU32, uMsgNext=%RU32, cParmsNext=%RU32, rcNext=%Rrc\n", uMsgClient, uMsgNext, cParmsNext, rcNext)); if (RT_SUCCESS(rcNext)) { if (uMsgClient == GUEST_DND_GET_NEXT_HOST_MSG) { rc = pClient->SetDeferredMsgInfo(uMsgNext, cParmsNext); /* Note: Report the current rc back to the guest. */ pClient->CompleteDeferred(rc); } /* * Does the message the client is waiting for match the message * next in the queue? Process it right away then. */ else if (uMsgClient == uMsgNext) { rc = m_pManager->GetNextMsg(u32Function, cParms, paParms); /* Note: Report the current rc back to the guest. */ pClient->CompleteDeferred(rc); } else /* Should not happen; cancel the operation on the guest. */ { LogFunc(("Client ID=%RU32 in wrong state with uMsg=%RU32 (next message in queue: %RU32), cancelling\n", pClient->GetClientID(), uMsgClient, uMsgNext)); pClient->CompleteDeferred(VERR_CANCELLED); } m_clientQueue.pop_front(); } } /* fSendToGuest */ } while (0); /* To use breaks. */ LogFlowFuncLeaveRC(rc); return rc; } DECLCALLBACK(int) DragAndDropService::progressCallback(uint32_t uStatus, uint32_t uPercentage, int rc, void *pvUser) { AssertPtrReturn(pvUser, VERR_INVALID_POINTER); DragAndDropService *pSelf = static_cast(pvUser); AssertPtr(pSelf); if (pSelf->m_SvcCtx.pfnHostCallback) { LogFlowFunc(("GUEST_DND_HG_EVT_PROGRESS: uStatus=%RU32, uPercentage=%RU32, rc=%Rrc\n", uStatus, uPercentage, rc)); VBOXDNDCBHGEVTPROGRESSDATA data; data.hdr.uMagic = CB_MAGIC_DND_HG_EVT_PROGRESS; data.uPercentage = RT_MIN(uPercentage, 100); data.uStatus = uStatus; data.rc = rc; /** @todo uin32_t vs. int. */ return pSelf->m_SvcCtx.pfnHostCallback(pSelf->m_SvcCtx.pvHostData, GUEST_DND_HG_EVT_PROGRESS, &data, sizeof(data)); } return VINF_SUCCESS; } /** * @copydoc FNVBOXHGCMSVCLOAD */ extern "C" DECLCALLBACK(DECLEXPORT(int)) VBoxHGCMSvcLoad(VBOXHGCMSVCFNTABLE *pTable) { return DragAndDropService::svcLoad(pTable); }