/* $Id: GuestDnDImpl.cpp 50265 2014-01-29 11:12:44Z vboxsync $ */ /** @file * VirtualBox COM class implementation: Guest Drag and Drop parts */ /* * Copyright (C) 2011-2014 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. */ #include "GuestImpl.h" #include "AutoCaller.h" #ifdef VBOX_WITH_DRAG_AND_DROP # include "ConsoleImpl.h" # include "ProgressImpl.h" # include "GuestDnDImpl.h" # include # include # include # ifdef LOG_GROUP # undef LOG_GROUP # endif # define LOG_GROUP LOG_GROUP_GUEST_DND # include # include # include # include /* How does this work: * * Drag and Drop is handled over the internal HGCM service for the host <-> * guest communication. Beside that we need to map the Drag and Drop protocols * of the various OS's we support to our internal channels, this is also highly * communicative in both directions. Unfortunately HGCM isn't really designed * for that. Next we have to foul some of the components. This includes to * trick X11 on the guest side, but also Qt needs to be tricked on the host * side a little bit. * * The following components are involved: * * 1. GUI: Uses the Qt classes for Drag and Drop and mainly forward the content * of it to the Main IGuest interface (see UIDnDHandler.cpp). * 2. Main: Public interface for doing Drag and Drop. Also manage the IProgress * interfaces for blocking the caller by showing a progress dialog (see * this file). * 3. HGCM service: Handle all messages from the host to the guest at once and * encapsulate the internal communication details (see dndmanager.cpp and * friends). * 4. Guest additions: Split into the platform neutral part (see * VBoxGuestR3LibDragAndDrop.cpp) and the guest OS specific parts. * Receive/send message from/to the HGCM service and does all guest specific * operations. Currently only X11 is supported (see draganddrop.cpp within * VBoxClient). * * Host -> Guest: * 1. There are DnD Enter, Move, Leave events which are send exactly like this * to the guest. The info includes the pos, mimetypes and allowed actions. * The guest has to respond with an action it would accept, so the GUI could * change the cursor. * 2. On drop, first a drop event is send. If this is accepted a drop data * event follows. This blocks the GUI and shows some progress indicator. * * Guest -> Host: * 1. The GUI is asking the guest if a DnD event is pending when the user moves * the cursor out of the view window. If so, this returns the mimetypes and * allowed actions. * (2. On every mouse move this is asked again, to make sure the DnD event is * still valid.) * 3. On drop the host request the data from the guest. This blocks the GUI and * shows some progress indicator. * * Some hints: * m_sstrAllowedMimeTypes here in this file defines the allowed mime-types. * This is necessary because we need special handling for some of the * mime-types. E.g. for URI lists we need to transfer the actual dirs and * files. Text EOL may to be changed. Also unknown mime-types may need special * handling as well, which may lead to undefined behavior in the host/guest, if * not done. * * Dropping of a directory, means recursively transferring _all_ the content. * * Directories and files are placed into the user's temporary directory on the * guest (e.g. /tmp/VirtualBox Dropped Files). We can't delete them after the * DnD operation, because we didn't know what the DnD target does with it. E.g. * it could just be opened in place. This could lead ofc to filling up the disk * within the guest. To inform the user about this, a small app could be * developed which scans this directory regularly and inform the user with a * tray icon hint (and maybe the possibility to clean this up instantly). The * same has to be done in the G->H direction when it is implemented. * * Of course only regularly files are supported. Symlinks are resolved and * transfered as regularly files. First we don't know if the other side support * symlinks at all and second they could point to somewhere in a directory tree * which not exists on the other side. * * The code tries to preserve the file modes of the transfered dirs/files. This * is useful (and maybe necessary) for two things: * 1. If a file is executable, it should be also after the transfer, so the * user can just execute it, without manually tweaking the modes first. * 2. If a dir/file is not accessible by group/others in the host, it shouldn't * be in the guest. * In any case, the user mode is always set to rwx (so that we can access it * ourself, in e.g. for a cleanup case after cancel). * * Cancel is supported in both directions and cleans up all previous steps * (thats is: deleting already transfered dirs/files). * * In general I propose the following changes in the VBox HGCM infrastructure * for the future: * - Currently it isn't really possible to send messages to the guest from the * host. The host informs the guest just that there is something, the guest * than has to ask which message and depending on that send the appropriate * message to the host, which is filled with the right data. * - There is no generic interface for sending bigger memory blocks to/from the * guest. This is now done here, but I guess was also necessary for e.g. * guest execution. So something generic which brake this up into smaller * blocks and send it would be nice (with all the error handling and such * ofc). * - I developed a "protocol" for the DnD communication here. So the host and * the guest have always to match in the revision. This is ofc bad, because * the additions could be outdated easily. So some generic protocol number * support in HGCM for asking the host and the guest of the support version, * would be nice. Ofc at least the host should be able to talk to the guest, * even when the version is below the host one. * All this stuff would be useful for the current services, but also for future * onces. * * Todo: * - Dragging out of the guest (partly done) * - ESC doesn't really work (on Windows guests it's already implemented) * - transfer of URIs (that is the files and patching of the data) * - testing in a multi monitor setup * ... in any case it seems a little bit difficult to handle from the Qt * side. Maybe also a host specific implementation becomes necessary ... * this would be really worst ofc. * - Win guest support (maybe there have to be done a mapping between the * official mime-types and Win Clipboard formats (see QWindowsMime, for an * idea), for VBox internally only mime-types should be used) * - EOL handling on text (should be shared with the clipboard code) * - add configuration (GH, HG, Bidirectional, None), like for the clipboard * - HG->GH and GH->HG-switch: Handle the case the user drags something out of * the guest and than return to the source view (or another window in the * multiple guest screen scenario). * - add support for more mime-types (especially images, csv) * - test unusual behavior: * - DnD service crash in the guest during a DnD op (e.g. crash of VBoxClient or X11) * - not expected order of the events between HGCM and the guest * - Security considerations: We transfer a lot of memory between the guest and * the host and even allow the creation of dirs/files. Maybe there should be * limits introduced to preventing DOS attacks or filling up all the memory * (both in the host and the guest). * - test, test, test ... */ class DnDGuestResponse { public: DnDGuestResponse(const ComObjPtr& pGuest); virtual ~DnDGuestResponse(void); public: int notifyAboutGuestResponse(void); int waitForGuestResponse(RTMSINTERVAL msTimeout = 500); void setDefAction(uint32_t a) { m_defAction = a; } uint32_t defAction(void) const { return m_defAction; } void setAllActions(uint32_t a) { m_allActions = a; } uint32_t allActions() const { return m_allActions; } void setFormat(const Utf8Str &strFormat) { m_strFormat = strFormat; } Utf8Str format(void) const { return m_strFormat; } int dataAdd(void *pvData, uint32_t cbData, uint32_t *pcbCurSize); void reset(void); const void *data(void) { return m_pvData; } size_t size(void) const { return m_cbData; } int setProgress(unsigned uPercentage, uint32_t uState, int rcOp = VINF_SUCCESS); HRESULT resetProgress(const ComObjPtr& pParent); HRESULT queryProgressTo(IProgress **ppProgress); private: RTSEMEVENT m_EventSem; uint32_t m_defAction; uint32_t m_allActions; Utf8Str m_strFormat; void *m_pvData; uint32_t m_cbData; ComObjPtr m_parent; ComObjPtr m_progress; }; class GuestDnDPrivate { private: /* todo: currently we only support one response. Maybe this needs to be extended at some time. */ GuestDnDPrivate(GuestDnD *q, const ComObjPtr& pGuest) : q_ptr(q) , p(pGuest) , m_pDnDResponse(new DnDGuestResponse(pGuest)) {} virtual ~GuestDnDPrivate(void) { delete m_pDnDResponse; } DnDGuestResponse *response(void) const { return m_pDnDResponse; } void adjustCoords(ULONG uScreenId, ULONG *puX, ULONG *puY) const; void hostCall(uint32_t u32Function, uint32_t cParms, PVBOXHGCMSVCPARM paParms) const; /* Static helper */ static RTCString toFormatString(ComSafeArrayIn(IN_BSTR, formats)); static void toFormatSafeArray(const RTCString &strFormats, ComSafeArrayOut(BSTR, formats)); static DragAndDropAction_T toMainAction(uint32_t uAction); static void toMainActions(uint32_t uActions, ComSafeArrayOut(DragAndDropAction_T, actions)); static uint32_t toHGCMAction(DragAndDropAction_T action); static void toHGCMActions(DragAndDropAction_T inDefAction, uint32_t *pOutDefAction, ComSafeArrayIn(DragAndDropAction_T, inAllowedActions), uint32_t *pOutAllowedActions); /* Private q and parent pointer */ GuestDnD *q_ptr; ComObjPtr p; /* Private helper members */ static const RTCList m_sstrAllowedMimeTypes; DnDGuestResponse *m_pDnDResponse; friend class GuestDnD; }; /* What mime-types are supported by VirtualBox. * Note: If you add something here, make sure you test it with all guest OS's! ** @todo Make this MIME list configurable / extendable (by extra data?). Currently * this is done hardcoded on every guest platform (POSIX/Windows). */ /* static */ const RTCList GuestDnDPrivate::m_sstrAllowedMimeTypes = RTCList() /* Uri's */ << "text/uri-list" /* Text */ << "text/plain;charset=utf-8" << "UTF8_STRING" << "text/plain" << "COMPOUND_TEXT" << "TEXT" << "STRING" /* OpenOffice formates */ << "application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\"" << "application/x-openoffice-drawing;windows_formatname=\"Drawing Format\""; DnDGuestResponse::DnDGuestResponse(const ComObjPtr& pGuest) : m_EventSem(NIL_RTSEMEVENT) , m_defAction(0) , m_allActions(0) , m_pvData(0) , m_cbData(0) , m_parent(pGuest) { int rc = RTSemEventCreate(&m_EventSem); AssertRC(rc); } DnDGuestResponse::~DnDGuestResponse() { reset(); int rc = RTSemEventDestroy(m_EventSem); AssertRC(rc); } int DnDGuestResponse::notifyAboutGuestResponse() { return RTSemEventSignal(m_EventSem); } int DnDGuestResponse::waitForGuestResponse(RTMSINTERVAL msTimeout /*= 500 */) { int vrc = RTSemEventWait(m_EventSem, msTimeout); #ifdef DEBUG_andy LogFlowFunc(("msTimeout=%RU32, rc=%Rrc\n", msTimeout, vrc)); #endif return vrc; } int DnDGuestResponse::dataAdd(void *pvData, uint32_t cbData, uint32_t *pcbCurSize) { int rc = VINF_SUCCESS; /** @todo Make reallocation scheme a bit smarter here. */ m_pvData = RTMemRealloc(m_pvData, m_cbData + cbData); if (m_pvData) { memcpy(&static_cast(m_pvData)[m_cbData], pvData, cbData); m_cbData += cbData; if (pcbCurSize) *pcbCurSize = m_cbData; } else rc = VERR_NO_MEMORY; return rc; } void DnDGuestResponse::reset(void) { if (m_pvData) { RTMemFree(m_pvData); m_pvData = NULL; } m_cbData = 0; } HRESULT DnDGuestResponse::resetProgress(const ComObjPtr& pParent) { m_progress.setNull(); HRESULT rc = m_progress.createObject(); if (SUCCEEDED(rc)) { rc = m_progress->init(static_cast(pParent), Bstr(pParent->tr("Dropping data")).raw(), TRUE); } return rc; } int DnDGuestResponse::setProgress(unsigned uPercentage, uint32_t uState, int rcOp /* = VINF_SUCCESS */) { LogFlowFunc(("uPercentage=%RU32, uState=%ld, rcOp=%Rrc\n", uPercentage, uState, rcOp)); int vrc = VINF_SUCCESS; if (!m_progress.isNull()) { BOOL fCompleted; HRESULT rc = m_progress->COMGETTER(Completed)(&fCompleted); if (!fCompleted) { if (uState == DragAndDropSvc::DND_PROGRESS_ERROR) { rc = m_progress->notifyComplete(E_FAIL, COM_IIDOF(IGuest), m_parent->getComponentName(), m_parent->tr("Guest error (%Rrc)"), rcOp); } else if (uState == DragAndDropSvc::DND_PROGRESS_CANCELLED) { rc = m_progress->Cancel(); vrc = VERR_CANCELLED; } else /* uState == DragAndDropSvc::DND_PROGRESS_RUNNING */ { rc = m_progress->SetCurrentOperationProgress(uPercentage); #ifndef DEBUG_andy Assert(SUCCEEDED(rc)); #endif if ( uState == DragAndDropSvc::DND_PROGRESS_COMPLETE || uPercentage >= 100) rc = m_progress->notifyComplete(S_OK); } #ifndef DEBUG_andy Assert(SUCCEEDED(rc)); #endif } } return vrc; } HRESULT DnDGuestResponse::queryProgressTo(IProgress **ppProgress) { return m_progress.queryInterfaceTo(ppProgress); } void GuestDnDPrivate::adjustCoords(ULONG uScreenId, ULONG *puX, ULONG *puY) const { /* For multi-monitor support we need to add shift values to the coordinates * (depending on the screen number). */ ComPtr pDisplay; HRESULT rc = p->mParent->COMGETTER(Display)(pDisplay.asOutParam()); if (FAILED(rc)) throw rc; ComPtr pFramebuffer; LONG xShift, yShift; rc = pDisplay->GetFramebuffer(uScreenId, pFramebuffer.asOutParam(), &xShift, &yShift); if (FAILED(rc)) throw rc; *puX += xShift; *puY += yShift; } void GuestDnDPrivate::hostCall(uint32_t u32Function, uint32_t cParms, PVBOXHGCMSVCPARM paParms) const { VMMDev *vmmDev = NULL; { /* Make sure mParent is valid, so set the read lock while using. * Do not keep this lock while doing the actual call, because in the meanwhile * another thread could request a write lock which would be a bad idea ... */ AutoReadLock alock(p COMMA_LOCKVAL_SRC_POS); /* Forward the information to the VMM device. */ AssertPtr(p->mParent); vmmDev = p->mParent->getVMMDev(); } if (!vmmDev) throw p->setError(VBOX_E_VM_ERROR, p->tr("VMM device is not available (is the VM running?)")); LogFlowFunc(("hgcmHostCall msg=%RU32, numParms=%RU32\n", u32Function, cParms)); int vrc = vmmDev->hgcmHostCall("VBoxDragAndDropSvc", u32Function, cParms, paParms); if (RT_FAILURE(vrc)) { LogFlowFunc(("hgcmHostCall error: %Rrc\n", vrc)); throw p->setError(VBOX_E_VM_ERROR, p->tr("hgcmHostCall failed (%Rrc)"), vrc); } } /* static */ RTCString GuestDnDPrivate::toFormatString(ComSafeArrayIn(IN_BSTR, formats)) { const RTCList formatList(ComSafeArrayInArg(formats)); RTCString strFormat; for (size_t i = 0; i < formatList.size(); ++i) { const RTCString &f = formatList.at(i); /* Only keep allowed format types. */ if (m_sstrAllowedMimeTypes.contains(f)) strFormat += f + "\r\n"; } return strFormat; } /* static */ void GuestDnDPrivate::toFormatSafeArray(const RTCString &strFormats, ComSafeArrayOut(BSTR, formats)) { RTCList list = strFormats.split("\r\n"); size_t i = 0; while (i < list.size()) { /* Only keep allowed format types. */ if (!m_sstrAllowedMimeTypes.contains(list.at(i))) list.removeAt(i); else ++i; } /* Create a safe array out of the cleaned list. */ com::SafeArray sfaFormats(list.size()); for (i = 0; i < list.size(); ++i) { const RTCString &f = list.at(i); if (m_sstrAllowedMimeTypes.contains(f)) { Bstr bstr(f); bstr.detachTo(&sfaFormats[i]); } } sfaFormats.detachTo(ComSafeArrayOutArg(formats)); } /* static */ uint32_t GuestDnDPrivate::toHGCMAction(DragAndDropAction_T action) { uint32_t a = DND_IGNORE_ACTION; switch (action) { case DragAndDropAction_Copy: a = DND_COPY_ACTION; break; case DragAndDropAction_Move: a = DND_MOVE_ACTION; break; case DragAndDropAction_Link: /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */ case DragAndDropAction_Ignore: /* Ignored */ break; default: AssertMsgFailed(("Action %u not recognized!\n", action)); break; } return a; } /* static */ void GuestDnDPrivate::toHGCMActions(DragAndDropAction_T inDefAction, uint32_t *pOutDefAction, ComSafeArrayIn(DragAndDropAction_T, inAllowedActions), uint32_t *pOutAllowedActions) { const com::SafeArray sfaInActions(ComSafeArrayInArg(inAllowedActions)); /* Defaults */ *pOutDefAction = toHGCMAction(inDefAction);; *pOutAllowedActions = DND_IGNORE_ACTION; /* First convert the allowed actions to a bit array. */ for (size_t i = 0; i < sfaInActions.size(); ++i) *pOutAllowedActions |= toHGCMAction(sfaInActions[i]); /* Second check if the default action is a valid action and if not so try * to find an allowed action. */ if (isDnDIgnoreAction(*pOutDefAction)) { if (hasDnDCopyAction(*pOutAllowedActions)) *pOutDefAction = DND_COPY_ACTION; else if (hasDnDMoveAction(*pOutAllowedActions)) *pOutDefAction = DND_MOVE_ACTION; } } /* static */ DragAndDropAction_T GuestDnDPrivate::toMainAction(uint32_t uAction) { /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */ return (isDnDCopyAction(uAction) ? (DragAndDropAction_T)DragAndDropAction_Copy : isDnDMoveAction(uAction) ? (DragAndDropAction_T)DragAndDropAction_Move : (DragAndDropAction_T)DragAndDropAction_Ignore); } /* static */ void GuestDnDPrivate::toMainActions(uint32_t uActions, ComSafeArrayOut(DragAndDropAction_T, actions)) { /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */ RTCList list; if (hasDnDCopyAction(uActions)) list.append(DragAndDropAction_Copy); if (hasDnDMoveAction(uActions)) list.append(DragAndDropAction_Move); com::SafeArray sfaActions(list.size()); for (size_t i = 0; i < list.size(); ++i) sfaActions[i] = list.at(i); sfaActions.detachTo(ComSafeArrayOutArg(actions)); } GuestDnD::GuestDnD(const ComObjPtr& pGuest) : d_ptr(new GuestDnDPrivate(this, pGuest)) { } GuestDnD::~GuestDnD() { delete d_ptr; } HRESULT GuestDnD::dragHGEnter(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), DragAndDropAction_T *pResultAction) { DPTR(GuestDnD); const ComObjPtr &p = d->p; /* Default is ignoring */ *pResultAction = DragAndDropAction_Ignore; /* Check & convert the drag & drop actions */ uint32_t uDefAction = 0; uint32_t uAllowedActions = 0; d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions); /* If there is no usable action, ignore this request. */ if (isDnDIgnoreAction(uDefAction)) return S_OK; /* Make a flat data string out of the mime-type list. */ RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats)); /* If there is no valid mime-type, ignore this request. */ if (strFormats.isEmpty()) return S_OK; HRESULT rc = S_OK; try { /* Adjust the coordinates in a multi-monitor setup. */ d->adjustCoords(uScreenId, &uX, &uY); VBOXHGCMSVCPARM paParms[7]; int i = 0; paParms[i++].setUInt32(uScreenId); paParms[i++].setUInt32(uX); paParms[i++].setUInt32(uY); paParms[i++].setUInt32(uDefAction); paParms[i++].setUInt32(uAllowedActions); paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1); paParms[i++].setUInt32(strFormats.length() + 1); d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_ENTER, i, paParms); DnDGuestResponse *pDnD = d->response(); /* This blocks until the request is answered (or timeout). */ if (pDnD->waitForGuestResponse() == VERR_TIMEOUT) return S_OK; /* Copy the response info */ *pResultAction = d->toMainAction(pDnD->defAction()); LogFlowFunc(("*pResultAction=%ld\n", *pResultAction)); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragHGMove(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), DragAndDropAction_T *pResultAction) { DPTR(GuestDnD); const ComObjPtr &p = d->p; /* Default is ignoring */ *pResultAction = DragAndDropAction_Ignore; /* Check & convert the drag & drop actions */ uint32_t uDefAction = 0; uint32_t uAllowedActions = 0; d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions); /* If there is no usable action, ignore this request. */ if (isDnDIgnoreAction(uDefAction)) return S_OK; /* Make a flat data string out of the mime-type list. */ RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats)); /* If there is no valid mime-type, ignore this request. */ if (strFormats.isEmpty()) return S_OK; HRESULT rc = S_OK; try { /* Adjust the coordinates in a multi-monitor setup. */ d->adjustCoords(uScreenId, &uX, &uY); VBOXHGCMSVCPARM paParms[7]; int i = 0; paParms[i++].setUInt32(uScreenId); paParms[i++].setUInt32(uX); paParms[i++].setUInt32(uY); paParms[i++].setUInt32(uDefAction); paParms[i++].setUInt32(uAllowedActions); paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1); paParms[i++].setUInt32(strFormats.length() + 1); d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_MOVE, i, paParms); DnDGuestResponse *pDnD = d->response(); /* This blocks until the request is answered (or timeout). */ if (pDnD->waitForGuestResponse() == VERR_TIMEOUT) return S_OK; /* Copy the response info */ *pResultAction = d->toMainAction(pDnD->defAction()); LogFlowFunc(("*pResultAction=%ld\n", *pResultAction)); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragHGLeave(ULONG uScreenId) { DPTR(GuestDnD); const ComObjPtr &p = d->p; HRESULT rc = S_OK; try { d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_LEAVE, 0, NULL); DnDGuestResponse *pDnD = d->response(); /* This blocks until the request is answered (or timeout). */ pDnD->waitForGuestResponse(); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragHGDrop(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), BSTR *pstrFormat, DragAndDropAction_T *pResultAction) { DPTR(GuestDnD); const ComObjPtr &p = d->p; /* Default is ignoring */ *pResultAction = DragAndDropAction_Ignore; /* Check & convert the drag & drop actions */ uint32_t uDefAction = 0; uint32_t uAllowedActions = 0; d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions); /* If there is no usable action, ignore this request. */ if (isDnDIgnoreAction(uDefAction)) return S_OK; /* Make a flat data string out of the mime-type list. */ RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats)); /* If there is no valid mime-type, ignore this request. */ if (strFormats.isEmpty()) return S_OK; HRESULT rc = S_OK; try { /* Adjust the coordinates in a multi-monitor setup. */ d->adjustCoords(uScreenId, &uX, &uY); VBOXHGCMSVCPARM paParms[7]; int i = 0; paParms[i++].setUInt32(uScreenId); paParms[i++].setUInt32(uX); paParms[i++].setUInt32(uY); paParms[i++].setUInt32(uDefAction); paParms[i++].setUInt32(uAllowedActions); paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1); paParms[i++].setUInt32(strFormats.length() + 1); d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_DROPPED, i, paParms); DnDGuestResponse *pDnD = d->response(); /* This blocks until the request is answered (or timeout). */ if (pDnD->waitForGuestResponse() == VERR_TIMEOUT) return S_OK; /* Copy the response info */ *pResultAction = d->toMainAction(pDnD->defAction()); Bstr(pDnD->format()).cloneTo(pstrFormat); LogFlowFunc(("*pResultAction=%ld\n", *pResultAction)); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragHGPutData(ULONG uScreenId, IN_BSTR bstrFormat, ComSafeArrayIn(BYTE, data), IProgress **ppProgress) { DPTR(GuestDnD); const ComObjPtr &p = d->p; HRESULT rc = S_OK; try { Utf8Str strFormat(bstrFormat); com::SafeArray sfaData(ComSafeArrayInArg(data)); VBOXHGCMSVCPARM paParms[5]; int i = 0; paParms[i++].setUInt32(uScreenId); paParms[i++].setPointer((void*)strFormat.c_str(), (uint32_t)strFormat.length() + 1); paParms[i++].setUInt32((uint32_t)strFormat.length() + 1); paParms[i++].setPointer((void*)sfaData.raw(), (uint32_t)sfaData.size()); paParms[i++].setUInt32((uint32_t)sfaData.size()); DnDGuestResponse *pDnD = d->response(); /* Reset any old progress status. */ pDnD->resetProgress(p); d->hostCall(DragAndDropSvc::HOST_DND_HG_SND_DATA, i, paParms); /* Query the progress object to the caller. */ pDnD->queryProgressTo(ppProgress); } catch (HRESULT rc2) { rc = rc2; } return rc; } #ifdef VBOX_WITH_DRAG_AND_DROP_GH HRESULT GuestDnD::dragGHPending(ULONG uScreenId, ComSafeArrayOut(BSTR, formats), ComSafeArrayOut(DragAndDropAction_T, allowedActions), DragAndDropAction_T *pDefaultAction) { DPTR(GuestDnD); const ComObjPtr &p = d->p; /* Default is ignoring */ *pDefaultAction = DragAndDropAction_Ignore; HRESULT rc = S_OK; try { VBOXHGCMSVCPARM paParms[1]; int i = 0; paParms[i++].setUInt32(uScreenId); d->hostCall(DragAndDropSvc::HOST_DND_GH_REQ_PENDING, i, paParms); DnDGuestResponse *pDnD = d->response(); /* This blocks until the request is answered (or timeout). */ if (pDnD->waitForGuestResponse() == VERR_TIMEOUT) return S_OK; if (isDnDIgnoreAction(pDnD->defAction())) return S_OK; /* Fetch the default action to use. */ *pDefaultAction = d->toMainAction(pDnD->defAction()); /* Convert the formats strings to a vector of strings. */ d->toFormatSafeArray(pDnD->format(), ComSafeArrayOutArg(formats)); /* Convert the action bit field to a vector of actions. */ d->toMainActions(pDnD->allActions(), ComSafeArrayOutArg(allowedActions)); LogFlowFunc(("*pDefaultAction=0x%x\n", *pDefaultAction)); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragGHDropped(IN_BSTR bstrFormat, DragAndDropAction_T action, IProgress **ppProgress) { DPTR(GuestDnD); const ComObjPtr &p = d->p; Utf8Str strFormat(bstrFormat); HRESULT rc = S_OK; uint32_t uAction = d->toHGCMAction(action); /* If there is no usable action, ignore this request. */ if (isDnDIgnoreAction(uAction)) return S_OK; try { LogFlowFunc(("strFormat=%s, uAction=0x%x\n", strFormat.c_str(), uAction)); VBOXHGCMSVCPARM paParms[3]; int i = 0; paParms[i++].setPointer((void*)strFormat.c_str(), (uint32_t)strFormat.length() + 1); paParms[i++].setUInt32((uint32_t)strFormat.length() + 1); paParms[i++].setUInt32(uAction); DnDGuestResponse *pDnD = d->response(); /* Reset any old data and the progress status. */ pDnD->reset(); pDnD->resetProgress(p); d->hostCall(DragAndDropSvc::HOST_DND_GH_EVT_DROPPED, i, paParms); /* Query the progress object to the caller. */ pDnD->queryProgressTo(ppProgress); } catch (HRESULT rc2) { rc = rc2; } return rc; } HRESULT GuestDnD::dragGHGetData(ComSafeArrayOut(BYTE, data)) { DPTR(GuestDnD); const ComObjPtr &p = d->p; HRESULT rc = S_OK; DnDGuestResponse *pDnD = d->response(); if (pDnD) { com::SafeArray sfaData; uint32_t cbData = pDnD->size(); if (cbData) { /* Copy the data into an safe array of bytes. */ const void *pvData = pDnD->data(); if (sfaData.resize(cbData)) memcpy(sfaData.raw(), pvData, cbData); else rc = E_OUTOFMEMORY; } #ifdef DEBUG_andy LogFlowFunc(("Received %RU32 bytes\n", cbData)); #endif /* Detach in any case, regardless of data size. */ sfaData.detachTo(ComSafeArrayOutArg(data)); /* Delete the data. */ pDnD->reset(); } else rc = VBOX_E_INVALID_OBJECT_STATE; return rc; } #endif /* VBOX_WITH_DRAG_AND_DROP_GH */ DECLCALLBACK(int) GuestDnD::notifyGuestDragAndDropEvent(void *pvExtension, uint32_t u32Function, void *pvParms, uint32_t cbParms) { LogFlowFunc(("pvExtension=%p, u32Function=%RU32, pvParms=%p, cbParms=%RU32\n", pvExtension, u32Function, pvParms, cbParms)); ComObjPtr pGuest = reinterpret_cast(pvExtension); if (!pGuest->m_pGuestDnD) return VINF_SUCCESS; GuestDnDPrivate *d = static_cast(pGuest->m_pGuestDnD->d_ptr); const ComObjPtr &p = d->p; DnDGuestResponse *pResp = d->response(); if (pResp == NULL) return VERR_INVALID_PARAMETER; int rc = VINF_SUCCESS; switch (u32Function) { case DragAndDropSvc::GUEST_DND_HG_ACK_OP: { DragAndDropSvc::PVBOXDNDCBHGACKOPDATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGACKOPDATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_ACK_OP == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); pResp->setDefAction(pCBData->uAction); rc = pResp->notifyAboutGuestResponse(); break; } case DragAndDropSvc::GUEST_DND_HG_REQ_DATA: { DragAndDropSvc::PVBOXDNDCBHGREQDATADATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGREQDATADATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_REQ_DATA == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); pResp->setFormat(pCBData->pszFormat); rc = pResp->notifyAboutGuestResponse(); break; } case DragAndDropSvc::GUEST_DND_HG_EVT_PROGRESS: { DragAndDropSvc::PVBOXDNDCBHGEVTPROGRESSDATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGEVTPROGRESSDATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_EVT_PROGRESS == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); rc = pResp->setProgress(pCBData->uPercentage, pCBData->uState, pCBData->rc); break; } #ifdef VBOX_WITH_DRAG_AND_DROP_GH case DragAndDropSvc::GUEST_DND_GH_ACK_PENDING: { DragAndDropSvc::PVBOXDNDCBGHACKPENDINGDATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBGHACKPENDINGDATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_ACK_PENDING == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); pResp->setFormat(pCBData->pszFormat); pResp->setDefAction(pCBData->uDefAction); pResp->setAllActions(pCBData->uAllActions); rc = pResp->notifyAboutGuestResponse(); break; } case DragAndDropSvc::GUEST_DND_GH_SND_DATA: { DragAndDropSvc::PVBOXDNDCBSNDDATADATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBSNDDATADATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_SND_DATA == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); uint32_t cbCurSize = 0; rc = pResp->dataAdd(pCBData->pvData, pCBData->cbData, &cbCurSize); if (RT_SUCCESS(rc)) { uint32_t cbTotalSize = pCBData->cbAllSize; unsigned int cPercentage; if (!cbTotalSize) /* Watch out for division by zero. */ cPercentage = 100; else cPercentage = cbCurSize * 100.0 / cbTotalSize; /** @todo Don't use anonymous enums. */ uint32_t uState = DragAndDropSvc::DND_PROGRESS_RUNNING; if ( pCBData->cbAllSize == cbCurSize /* Empty data? Should not happen, but anyway ... */ || !pCBData->cbAllSize) { uState = DragAndDropSvc::DND_PROGRESS_COMPLETE; } rc = pResp->setProgress(cPercentage, uState); } /* Todo: for now we instantly confirm the cancel. Check if the * guest should first clean up stuff itself and than really confirm * the cancel request by an extra message. */ if (rc == VERR_CANCELLED) pResp->setProgress(100, DragAndDropSvc::DND_PROGRESS_CANCELLED); break; } case DragAndDropSvc::GUEST_DND_GH_EVT_ERROR: { DragAndDropSvc::PVBOXDNDCBEVTERRORDATA pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBEVTERRORDATA) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_EVT_ERROR == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); /* Cleanup */ pResp->reset(); rc = pResp->setProgress(100, DragAndDropSvc::DND_PROGRESS_ERROR, pCBData->rc); break; } #endif /* VBOX_WITH_DRAG_AND_DROP_GH */ default: rc = VERR_NOT_SUPPORTED; /* Tell the guest. */ break; } LogFlowFunc(("Returning rc=%Rrc\n", rc)); return rc; } #endif /* VBOX_WITH_DRAG_AND_DROP */