VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/GuestDnDImpl.cpp@ 49891

Last change on this file since 49891 was 49891, checked in by vboxsync, 11 years ago

Merged private draganddrop branch into trunk.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 36.0 KB
Line 
1/* $Id: GuestDnDImpl.cpp 49891 2013-12-12 20:09:20Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation: Guest Drag and Drop parts
4 */
5
6/*
7 * Copyright (C) 2011-2013 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#include "GuestImpl.h"
19#include "AutoCaller.h"
20
21#ifdef VBOX_WITH_DRAG_AND_DROP
22# include "ConsoleImpl.h"
23# include "ProgressImpl.h"
24# include "GuestDnDImpl.h"
25
26# include <VMMDev.h>
27
28# include <VBox/com/list.h>
29# include <VBox/HostServices/DragAndDropSvc.h>
30
31# ifdef LOG_GROUP
32 # undef LOG_GROUP
33# endif
34# define LOG_GROUP LOG_GROUP_GUEST_DND
35# include <VBox/log.h>
36
37# include <iprt/stream.h>
38# include <iprt/semaphore.h>
39# include <iprt/cpp/utils.h>
40
41/* How does this work:
42 *
43 * Drag and Drop is handled over the internal HGCM service for the host <->
44 * guest communication. Beside that we need to map the Drag and Drop protocols
45 * of the various OS's we support to our internal channels, this is also highly
46 * communicative in both directions. Unfortunately HGCM isn't really designed
47 * for that. Next we have to foul some of the components. This includes to
48 * trick X11 on the guest side, but also Qt needs to be tricked on the host
49 * side a little bit.
50 *
51 * The following components are involved:
52 *
53 * 1. GUI: Uses the Qt classes for Drag and Drop and mainly forward the content
54 * of it to the Main IGuest interface (see UIDnDHandler.cpp).
55 * 2. Main: Public interface for doing Drag and Drop. Also manage the IProgress
56 * interfaces for blocking the caller by showing a progress dialog (see
57 * this file).
58 * 3. HGCM service: Handle all messages from the host to the guest at once and
59 * encapsulate the internal communication details (see dndmanager.cpp and
60 * friends).
61 * 4. Guest additions: Split into the platform neutral part (see
62 * VBoxGuestR3LibDragAndDrop.cpp) and the guest OS specific parts.
63 * Receive/send message from/to the HGCM service and does all guest specific
64 * operations. Currently only X11 is supported (see draganddrop.cpp within
65 * VBoxClient).
66 *
67 * Host -> Guest:
68 * 1. There are DnD Enter, Move, Leave events which are send exactly like this
69 * to the guest. The info includes the pos, mimetypes and allowed actions.
70 * The guest has to respond with an action it would accept, so the GUI could
71 * change the cursor.
72 * 2. On drop, first a drop event is send. If this is accepted a drop data
73 * event follows. This blocks the GUI and shows some progress indicator.
74 *
75 * Guest -> Host:
76 * 1. The GUI is asking the guest if a DnD event is pending when the user moves
77 * the cursor out of the view window. If so, this returns the mimetypes and
78 * allowed actions.
79 * (2. On every mouse move this is asked again, to make sure the DnD event is
80 * still valid.)
81 * 3. On drop the host request the data from the guest. This blocks the GUI and
82 * shows some progress indicator.
83 *
84 * Some hints:
85 * m_sstrAllowedMimeTypes here in this file defines the allowed mime-types.
86 * This is necessary because we need special handling for some of the
87 * mime-types. E.g. for URI lists we need to transfer the actual dirs and
88 * files. Text EOL may to be changed. Also unknown mime-types may need special
89 * handling as well, which may lead to undefined behavior in the host/guest, if
90 * not done.
91 *
92 * Dropping of a directory, means recursively transferring _all_ the content.
93 *
94 * Directories and files are placed into the user's temporary directory on the
95 * guest (e.g. /tmp/VirtualBox Dropped Files). We can't delete them after the
96 * DnD operation, because we didn't know what the DnD target does with it. E.g.
97 * it could just be opened in place. This could lead ofc to filling up the disk
98 * within the guest. To inform the user about this, a small app could be
99 * developed which scans this directory regularly and inform the user with a
100 * tray icon hint (and maybe the possibility to clean this up instantly). The
101 * same has to be done in the G->H direction when it is implemented.
102 *
103 * Of course only regularly files are supported. Symlinks are resolved and
104 * transfered as regularly files. First we don't know if the other side support
105 * symlinks at all and second they could point to somewhere in a directory tree
106 * which not exists on the other side.
107 *
108 * The code tries to preserve the file modes of the transfered dirs/files. This
109 * is useful (and maybe necessary) for two things:
110 * 1. If a file is executable, it should be also after the transfer, so the
111 * user can just execute it, without manually tweaking the modes first.
112 * 2. If a dir/file is not accessible by group/others in the host, it shouldn't
113 * be in the guest.
114 * In any case, the user mode is always set to rwx (so that we can access it
115 * ourself, in e.g. for a cleanup case after cancel).
116 *
117 * Cancel is supported in both directions and cleans up all previous steps
118 * (thats is: deleting already transfered dirs/files).
119 *
120 * In general I propose the following changes in the VBox HGCM infrastructure
121 * for the future:
122 * - Currently it isn't really possible to send messages to the guest from the
123 * host. The host informs the guest just that there is something, the guest
124 * than has to ask which message and depending on that send the appropriate
125 * message to the host, which is filled with the right data.
126 * - There is no generic interface for sending bigger memory blocks to/from the
127 * guest. This is now done here, but I guess was also necessary for e.g.
128 * guest execution. So something generic which brake this up into smaller
129 * blocks and send it would be nice (with all the error handling and such
130 * ofc).
131 * - I developed a "protocol" for the DnD communication here. So the host and
132 * the guest have always to match in the revision. This is ofc bad, because
133 * the additions could be outdated easily. So some generic protocol number
134 * support in HGCM for asking the host and the guest of the support version,
135 * would be nice. Ofc at least the host should be able to talk to the guest,
136 * even when the version is below the host one.
137 * All this stuff would be useful for the current services, but also for future
138 * onces.
139 *
140 * Todo:
141 * - Dragging out of the guest (partly done)
142 * - ESC doesn't really work (on Windows guests it's already implemented)
143 * - transfer of URIs (that is the files and patching of the data)
144 * - testing in a multi monitor setup
145 * ... in any case it seems a little bit difficult to handle from the Qt
146 * side. Maybe also a host specific implementation becomes necessary ...
147 * this would be really worst ofc.
148 * - Win guest support (maybe there have to be done a mapping between the
149 * official mime-types and Win Clipboard formats (see QWindowsMime, for an
150 * idea), for VBox internally only mime-types should be used)
151 * - EOL handling on text (should be shared with the clipboard code)
152 * - add configuration (GH, HG, Bidirectional, None), like for the clipboard
153 * - HG->GH and GH->HG-switch: Handle the case the user drags something out of
154 * the guest and than return to the source view (or another window in the
155 * multiple guest screen scenario).
156 * - add support for more mime-types (especially images, csv)
157 * - test unusual behavior:
158 * - DnD service crash in the guest during a DnD op (e.g. crash of VBoxClient or X11)
159 * - not expected order of the events between HGCM and the guest
160 * - Security considerations: We transfer a lot of memory between the guest and
161 * the host and even allow the creation of dirs/files. Maybe there should be
162 * limits introduced to preventing DOS attacks or filling up all the memory
163 * (both in the host and the guest).
164 * - test, test, test ...
165 */
166
167class DnDGuestResponse
168{
169public:
170 DnDGuestResponse(const ComObjPtr<Guest>& pGuest);
171 ~DnDGuestResponse();
172
173 int notifyAboutGuestResponse();
174 int waitForGuestResponse();
175
176 void setDefAction(uint32_t a) { m_defAction = a; }
177 uint32_t defAction() const { return m_defAction; }
178
179 void setAllActions(uint32_t a) { m_allActions = a; }
180 uint32_t allActions() const { return m_allActions; }
181
182 void setFormat(const Utf8Str &strFormat) { m_strFormat = strFormat; }
183 Utf8Str format() const { return m_strFormat; }
184
185 int addData(void *pvData, uint32_t cbData, uint32_t *pcbCurSize);
186 void resetData();
187 void data(void **ppvData, uint32_t *pcbData) const { *ppvData = m_pvData; *pcbData = m_cbData; }
188 bool hasData() const { return m_pvData != NULL; }
189
190 int setProgress(unsigned uPercentage, uint32_t uState, int rcOp = VINF_SUCCESS);
191 HRESULT resetProgress(const ComObjPtr<Guest>& pParent);
192 HRESULT queryProgressTo(IProgress **ppProgress);
193
194private:
195 RTSEMEVENT m_EventSem;
196 uint32_t m_defAction;
197 uint32_t m_allActions;
198 Utf8Str m_strFormat;
199 void *m_pvData;
200 uint32_t m_cbData;
201
202 ComObjPtr<Guest> m_parent;
203 ComObjPtr<Progress> m_progress;
204};
205
206class GuestDnDPrivate
207{
208private:
209 /* todo: currently we only support one response. Maybe this needs to be extended at some time. */
210 GuestDnDPrivate(GuestDnD *q, const ComObjPtr<Guest>& pGuest)
211 : q_ptr(q)
212 , p(pGuest)
213 , m_pDnDResponse(new DnDGuestResponse(pGuest))
214 {}
215 ~GuestDnDPrivate() { delete m_pDnDResponse; }
216
217 DnDGuestResponse *response() const { return m_pDnDResponse; }
218
219 void adjustCoords(ULONG uScreenId, ULONG *puX, ULONG *puY) const;
220 void hostCall(uint32_t u32Function, uint32_t cParms, PVBOXHGCMSVCPARM paParms) const;
221
222 /* Static helper */
223 static RTCString toFormatString(ComSafeArrayIn(IN_BSTR, formats));
224 static void toFormatSafeArray(const RTCString &strFormats, ComSafeArrayOut(BSTR, formats));
225
226 static DragAndDropAction_T toMainAction(uint32_t uAction);
227 static void toMainActions(uint32_t uActions, ComSafeArrayOut(DragAndDropAction_T, actions));
228 static uint32_t toHGCMAction(DragAndDropAction_T action);
229 static void toHGCMActions(DragAndDropAction_T inDefAction, uint32_t *pOutDefAction, ComSafeArrayIn(DragAndDropAction_T, inAllowedActions), uint32_t *pOutAllowedActions);
230
231 /* Private q and parent pointer */
232 GuestDnD *q_ptr;
233 ComObjPtr<Guest> p;
234
235 /* Private helper members */
236 static const RTCList<RTCString> m_sstrAllowedMimeTypes;
237 DnDGuestResponse *m_pDnDResponse;
238
239 friend class GuestDnD;
240};
241
242/* What mime-types are supported by VirtualBox.
243 * Note: If you add something here, make sure you test it with all guest OS's!
244 ** @todo Make this MIME list configurable / extendable (by extra data?). Currently
245 * this is done hardcoded on every guest platform (POSIX/Windows).
246 */
247/* static */
248const RTCList<RTCString> GuestDnDPrivate::m_sstrAllowedMimeTypes = RTCList<RTCString>()
249 /* Uri's */
250 << "text/uri-list"
251 /* Text */
252 << "text/plain;charset=utf-8"
253 << "UTF8_STRING"
254 << "text/plain"
255 << "COMPOUND_TEXT"
256 << "TEXT"
257 << "STRING"
258 /* OpenOffice formates */
259 << "application/x-openoffice-embed-source-xml;windows_formatname=\"Star Embed Source (XML)\""
260 << "application/x-openoffice-drawing;windows_formatname=\"Drawing Format\"";
261
262DnDGuestResponse::DnDGuestResponse(const ComObjPtr<Guest>& pGuest)
263 : m_EventSem(NIL_RTSEMEVENT)
264 , m_defAction(0)
265 , m_allActions(0)
266 , m_pvData(0)
267 , m_cbData(0)
268 , m_parent(pGuest)
269{
270 int rc = RTSemEventCreate(&m_EventSem);
271 AssertRC(rc);
272}
273
274DnDGuestResponse::~DnDGuestResponse()
275{
276 resetData();
277 int rc = RTSemEventDestroy(m_EventSem);
278 AssertRC(rc);
279}
280
281int DnDGuestResponse::notifyAboutGuestResponse()
282{
283 return RTSemEventSignal(m_EventSem);
284}
285
286int DnDGuestResponse::waitForGuestResponse()
287{
288 int vrc = RTSemEventWait(m_EventSem, 300);
289#ifdef DEBUG_andy
290 LogFlowFunc(("rc=%Rrc\n", vrc));
291#endif
292 return vrc;
293}
294
295int DnDGuestResponse::addData(void *pvData, uint32_t cbData, uint32_t *pcbCurSize)
296{
297 int rc = VINF_SUCCESS;
298 m_pvData = RTMemRealloc(m_pvData, m_cbData + cbData);
299 if (m_pvData)
300 {
301 memcpy(&static_cast<uint8_t*>(m_pvData)[m_cbData], pvData, cbData);
302 m_cbData += cbData;
303 *pcbCurSize = m_cbData;
304 }
305 else
306 rc = VERR_NO_MEMORY;
307
308 return rc;
309}
310
311void DnDGuestResponse::resetData()
312{
313 if (m_pvData)
314 {
315 RTMemFree(m_pvData);
316 m_pvData = NULL;
317 }
318 m_cbData = 0;
319}
320
321HRESULT DnDGuestResponse::resetProgress(const ComObjPtr<Guest>& pParent)
322{
323 m_progress.setNull();
324 HRESULT rc = m_progress.createObject();
325 if (SUCCEEDED(rc))
326 {
327 rc = m_progress->init(static_cast<IGuest*>(pParent),
328 Bstr(pParent->tr("Dropping data")).raw(),
329 TRUE);
330 }
331 return rc;
332}
333
334int DnDGuestResponse::setProgress(unsigned uPercentage, uint32_t uState, int rcOp /* = VINF_SUCCESS */)
335{
336 LogFlowFunc(("uPercentage=%RU32, uState=%ld, rcOp=%Rrc\n", uPercentage, uState, rcOp));
337
338 int vrc = VINF_SUCCESS;
339 if (!m_progress.isNull())
340 {
341 BOOL fCompleted;
342 HRESULT rc = m_progress->COMGETTER(Completed)(&fCompleted);
343 if (!fCompleted)
344 {
345 if (uState == DragAndDropSvc::DND_PROGRESS_ERROR)
346 {
347 rc = m_progress->notifyComplete(E_FAIL,
348 COM_IIDOF(IGuest),
349 m_parent->getComponentName(),
350 m_parent->tr("Guest error (%Rrc)"), rcOp);
351 }
352 else if (uState == DragAndDropSvc::DND_PROGRESS_CANCELLED)
353 {
354 rc = m_progress->Cancel();
355 vrc = VERR_CANCELLED;
356 }
357 else /* uState == DragAndDropSvc::DND_PROGRESS_RUNNING */
358 {
359 rc = m_progress->SetCurrentOperationProgress(uPercentage);
360#ifndef DEBUG_andy
361 Assert(SUCCEEDED(rc));
362#endif
363 if ( uState == DragAndDropSvc::DND_PROGRESS_COMPLETE
364 || uPercentage >= 100)
365 rc = m_progress->notifyComplete(S_OK);
366 }
367#ifndef DEBUG_andy
368 Assert(SUCCEEDED(rc));
369#endif
370 }
371 }
372
373 return vrc;
374}
375
376HRESULT DnDGuestResponse::queryProgressTo(IProgress **ppProgress)
377{
378 return m_progress.queryInterfaceTo(ppProgress);
379}
380
381void GuestDnDPrivate::adjustCoords(ULONG uScreenId, ULONG *puX, ULONG *puY) const
382{
383 /* For multi-monitor support we need to add shift values to the coordinates
384 * (depending on the screen number). */
385 ComPtr<IDisplay> pDisplay;
386 HRESULT rc = p->mParent->COMGETTER(Display)(pDisplay.asOutParam());
387 if (FAILED(rc)) throw rc;
388 ComPtr<IFramebuffer> pFramebuffer;
389 LONG xShift, yShift;
390 rc = pDisplay->GetFramebuffer(uScreenId, pFramebuffer.asOutParam(), &xShift, &yShift);
391 if (FAILED(rc)) throw rc;
392 *puX += xShift;
393 *puY += yShift;
394}
395
396void GuestDnDPrivate::hostCall(uint32_t u32Function, uint32_t cParms, PVBOXHGCMSVCPARM paParms) const
397{
398 VMMDev *vmmDev = NULL;
399 {
400 /* Make sure mParent is valid, so set the read lock while using.
401 * Do not keep this lock while doing the actual call, because in the meanwhile
402 * another thread could request a write lock which would be a bad idea ... */
403 AutoReadLock alock(p COMMA_LOCKVAL_SRC_POS);
404
405 /* Forward the information to the VMM device. */
406 AssertPtr(p->mParent);
407 vmmDev = p->mParent->getVMMDev();
408 }
409
410 if (!vmmDev)
411 throw p->setError(VBOX_E_VM_ERROR,
412 p->tr("VMM device is not available (is the VM running?)"));
413
414 LogFlowFunc(("hgcmHostCall msg=%RU32, numParms=%RU32\n", u32Function, cParms));
415 int vrc = vmmDev->hgcmHostCall("VBoxDragAndDropSvc",
416 u32Function,
417 cParms, paParms);
418 if (RT_FAILURE(vrc))
419 {
420 LogFlowFunc(("hgcmHostCall error: %Rrc\n", vrc));
421 throw p->setError(VBOX_E_VM_ERROR,
422 p->tr("hgcmHostCall failed (%Rrc)"), vrc);
423 }
424}
425
426/* static */
427RTCString GuestDnDPrivate::toFormatString(ComSafeArrayIn(IN_BSTR, formats))
428{
429 const RTCList<Utf8Str> formatList(ComSafeArrayInArg(formats));
430 RTCString strFormat;
431 for (size_t i = 0; i < formatList.size(); ++i)
432 {
433 const RTCString &f = formatList.at(i);
434 /* Only keep allowed format types. */
435 if (m_sstrAllowedMimeTypes.contains(f))
436 strFormat += f + "\r\n";
437 }
438 return strFormat;
439}
440
441/* static */
442void GuestDnDPrivate::toFormatSafeArray(const RTCString &strFormats, ComSafeArrayOut(BSTR, formats))
443{
444 RTCList<RTCString> list = strFormats.split("\r\n");
445 size_t i = 0;
446 while (i < list.size())
447 {
448 /* Only keep allowed format types. */
449 if (!m_sstrAllowedMimeTypes.contains(list.at(i)))
450 list.removeAt(i);
451 else
452 ++i;
453 }
454 /* Create a safe array out of the cleaned list. */
455 com::SafeArray<BSTR> sfaFormats(list.size());
456 for (i = 0; i < list.size(); ++i)
457 {
458 const RTCString &f = list.at(i);
459 if (m_sstrAllowedMimeTypes.contains(f))
460 {
461 Bstr bstr(f);
462 bstr.detachTo(&sfaFormats[i]);
463 }
464 }
465 sfaFormats.detachTo(ComSafeArrayOutArg(formats));
466}
467
468/* static */
469uint32_t GuestDnDPrivate::toHGCMAction(DragAndDropAction_T action)
470{
471 uint32_t a = DND_IGNORE_ACTION;
472 switch (action)
473 {
474 case DragAndDropAction_Copy: a = DND_COPY_ACTION; break;
475 case DragAndDropAction_Move: a = DND_MOVE_ACTION; break;
476 case DragAndDropAction_Link: /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */
477 case DragAndDropAction_Ignore: /* Ignored */ break;
478 default: AssertMsgFailed(("Action %u not recognized!\n", action)); break;
479 }
480 return a;
481}
482
483/* static */
484void GuestDnDPrivate::toHGCMActions(DragAndDropAction_T inDefAction, uint32_t *pOutDefAction, ComSafeArrayIn(DragAndDropAction_T, inAllowedActions), uint32_t *pOutAllowedActions)
485{
486 const com::SafeArray<DragAndDropAction_T> sfaInActions(ComSafeArrayInArg(inAllowedActions));
487
488 /* Defaults */
489 *pOutDefAction = toHGCMAction(inDefAction);;
490 *pOutAllowedActions = DND_IGNORE_ACTION;
491
492 /* First convert the allowed actions to a bit array. */
493 for (size_t i = 0; i < sfaInActions.size(); ++i)
494 *pOutAllowedActions |= toHGCMAction(sfaInActions[i]);
495
496 /* Second check if the default action is a valid action and if not so try
497 * to find an allowed action. */
498 if (isDnDIgnoreAction(*pOutDefAction))
499 {
500 if (hasDnDCopyAction(*pOutAllowedActions))
501 *pOutDefAction = DND_COPY_ACTION;
502 else if (hasDnDMoveAction(*pOutAllowedActions))
503 *pOutDefAction = DND_MOVE_ACTION;
504 }
505}
506
507/* static */
508DragAndDropAction_T GuestDnDPrivate::toMainAction(uint32_t uAction)
509{
510 /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */
511 return (isDnDCopyAction(uAction) ? (DragAndDropAction_T)DragAndDropAction_Copy :
512 isDnDMoveAction(uAction) ? (DragAndDropAction_T)DragAndDropAction_Move :
513 (DragAndDropAction_T)DragAndDropAction_Ignore);
514}
515
516/* static */
517void GuestDnDPrivate::toMainActions(uint32_t uActions, ComSafeArrayOut(DragAndDropAction_T, actions))
518{
519 /* For now it doesn't seems useful to allow a link action between host & guest. Maybe later! */
520 RTCList<DragAndDropAction_T> list;
521 if (hasDnDCopyAction(uActions))
522 list.append(DragAndDropAction_Copy);
523 if (hasDnDMoveAction(uActions))
524 list.append(DragAndDropAction_Move);
525
526 com::SafeArray<DragAndDropAction_T> sfaActions(list.size());
527 for (size_t i = 0; i < list.size(); ++i)
528 sfaActions[i] = list.at(i);
529 sfaActions.detachTo(ComSafeArrayOutArg(actions));
530}
531
532GuestDnD::GuestDnD(const ComObjPtr<Guest>& pGuest)
533 : d_ptr(new GuestDnDPrivate(this, pGuest))
534{
535}
536
537GuestDnD::~GuestDnD()
538{
539 delete d_ptr;
540}
541
542HRESULT GuestDnD::dragHGEnter(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), DragAndDropAction_T *pResultAction)
543{
544 DPTR(GuestDnD);
545 const ComObjPtr<Guest> &p = d->p;
546
547 /* Default is ignoring */
548 *pResultAction = DragAndDropAction_Ignore;
549
550 /* Check & convert the drag & drop actions */
551 uint32_t uDefAction = 0;
552 uint32_t uAllowedActions = 0;
553 d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions);
554 /* If there is no usable action, ignore this request. */
555 if (isDnDIgnoreAction(uDefAction))
556 return S_OK;
557
558 /* Make a flat data string out of the mime-type list. */
559 RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats));
560 /* If there is no valid mime-type, ignore this request. */
561 if (strFormats.isEmpty())
562 return S_OK;
563
564 HRESULT rc = S_OK;
565
566 try
567 {
568 /* Adjust the coordinates in a multi-monitor setup. */
569 d->adjustCoords(uScreenId, &uX, &uY);
570
571 VBOXHGCMSVCPARM paParms[7];
572 int i = 0;
573 paParms[i++].setUInt32(uScreenId);
574 paParms[i++].setUInt32(uX);
575 paParms[i++].setUInt32(uY);
576 paParms[i++].setUInt32(uDefAction);
577 paParms[i++].setUInt32(uAllowedActions);
578 paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1);
579 paParms[i++].setUInt32(strFormats.length() + 1);
580
581 d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_ENTER,
582 i,
583 paParms);
584
585 DnDGuestResponse *pDnD = d->response();
586 /* This blocks until the request is answered (or timeout). */
587 if (pDnD->waitForGuestResponse() == VERR_TIMEOUT)
588 return S_OK;
589
590 /* Copy the response info */
591 *pResultAction = d->toMainAction(pDnD->defAction());
592 LogFlowFunc(("*pResultAction=%ld\n", *pResultAction));
593 }
594 catch (HRESULT rc2)
595 {
596 rc = rc2;
597 }
598
599 return rc;
600}
601
602HRESULT GuestDnD::dragHGMove(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), DragAndDropAction_T *pResultAction)
603{
604 DPTR(GuestDnD);
605 const ComObjPtr<Guest> &p = d->p;
606
607 /* Default is ignoring */
608 *pResultAction = DragAndDropAction_Ignore;
609
610 /* Check & convert the drag & drop actions */
611 uint32_t uDefAction = 0;
612 uint32_t uAllowedActions = 0;
613 d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions);
614 /* If there is no usable action, ignore this request. */
615 if (isDnDIgnoreAction(uDefAction))
616 return S_OK;
617
618 /* Make a flat data string out of the mime-type list. */
619 RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats));
620 /* If there is no valid mime-type, ignore this request. */
621 if (strFormats.isEmpty())
622 return S_OK;
623
624 HRESULT rc = S_OK;
625
626 try
627 {
628 /* Adjust the coordinates in a multi-monitor setup. */
629 d->adjustCoords(uScreenId, &uX, &uY);
630
631 VBOXHGCMSVCPARM paParms[7];
632 int i = 0;
633 paParms[i++].setUInt32(uScreenId);
634 paParms[i++].setUInt32(uX);
635 paParms[i++].setUInt32(uY);
636 paParms[i++].setUInt32(uDefAction);
637 paParms[i++].setUInt32(uAllowedActions);
638 paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1);
639 paParms[i++].setUInt32(strFormats.length() + 1);
640
641 d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_MOVE,
642 i,
643 paParms);
644
645 DnDGuestResponse *pDnD = d->response();
646 /* This blocks until the request is answered (or timeout). */
647 if (pDnD->waitForGuestResponse() == VERR_TIMEOUT)
648 return S_OK;
649
650 /* Copy the response info */
651 *pResultAction = d->toMainAction(pDnD->defAction());
652 LogFlowFunc(("*pResultAction=%ld\n", *pResultAction));
653 }
654 catch (HRESULT rc2)
655 {
656 rc = rc2;
657 }
658
659 return rc;
660}
661
662HRESULT GuestDnD::dragHGLeave(ULONG uScreenId)
663{
664 DPTR(GuestDnD);
665 const ComObjPtr<Guest> &p = d->p;
666
667 HRESULT rc = S_OK;
668
669 try
670 {
671 d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_LEAVE,
672 0,
673 NULL);
674
675 DnDGuestResponse *pDnD = d->response();
676 /* This blocks until the request is answered (or timeout). */
677 pDnD->waitForGuestResponse();
678 }
679 catch (HRESULT rc2)
680 {
681 rc = rc2;
682 }
683
684 return rc;
685}
686
687HRESULT GuestDnD::dragHGDrop(ULONG uScreenId, ULONG uX, ULONG uY, DragAndDropAction_T defaultAction, ComSafeArrayIn(DragAndDropAction_T, allowedActions), ComSafeArrayIn(IN_BSTR, formats), BSTR *pstrFormat, DragAndDropAction_T *pResultAction)
688{
689 DPTR(GuestDnD);
690 const ComObjPtr<Guest> &p = d->p;
691
692 /* Default is ignoring */
693 *pResultAction = DragAndDropAction_Ignore;
694
695 /* Check & convert the drag & drop actions */
696 uint32_t uDefAction = 0;
697 uint32_t uAllowedActions = 0;
698 d->toHGCMActions(defaultAction, &uDefAction, ComSafeArrayInArg(allowedActions), &uAllowedActions);
699 /* If there is no usable action, ignore this request. */
700 if (isDnDIgnoreAction(uDefAction))
701 return S_OK;
702
703 /* Make a flat data string out of the mime-type list. */
704 RTCString strFormats = d->toFormatString(ComSafeArrayInArg(formats));
705 /* If there is no valid mime-type, ignore this request. */
706 if (strFormats.isEmpty())
707 return S_OK;
708
709 HRESULT rc = S_OK;
710
711 try
712 {
713 /* Adjust the coordinates in a multi-monitor setup. */
714 d->adjustCoords(uScreenId, &uX, &uY);
715
716 VBOXHGCMSVCPARM paParms[7];
717 int i = 0;
718 paParms[i++].setUInt32(uScreenId);
719 paParms[i++].setUInt32(uX);
720 paParms[i++].setUInt32(uY);
721 paParms[i++].setUInt32(uDefAction);
722 paParms[i++].setUInt32(uAllowedActions);
723 paParms[i++].setPointer((void*)strFormats.c_str(), strFormats.length() + 1);
724 paParms[i++].setUInt32(strFormats.length() + 1);
725
726 d->hostCall(DragAndDropSvc::HOST_DND_HG_EVT_DROPPED,
727 i,
728 paParms);
729
730 DnDGuestResponse *pDnD = d->response();
731 /* This blocks until the request is answered (or timeout). */
732 if (pDnD->waitForGuestResponse() == VERR_TIMEOUT)
733 return S_OK;
734
735 /* Copy the response info */
736 *pResultAction = d->toMainAction(pDnD->defAction());
737 Bstr(pDnD->format()).cloneTo(pstrFormat);
738
739 LogFlowFunc(("*pResultAction=%ld\n", *pResultAction));
740 }
741 catch (HRESULT rc2)
742 {
743 rc = rc2;
744 }
745
746 return rc;
747}
748
749HRESULT GuestDnD::dragHGPutData(ULONG uScreenId, IN_BSTR bstrFormat, ComSafeArrayIn(BYTE, data), IProgress **ppProgress)
750{
751 DPTR(GuestDnD);
752 const ComObjPtr<Guest> &p = d->p;
753
754 HRESULT rc = S_OK;
755
756 try
757 {
758 Utf8Str strFormat(bstrFormat);
759 com::SafeArray<BYTE> sfaData(ComSafeArrayInArg(data));
760
761 VBOXHGCMSVCPARM paParms[5];
762 int i = 0;
763 paParms[i++].setUInt32(uScreenId);
764 paParms[i++].setPointer((void*)strFormat.c_str(), (uint32_t)strFormat.length() + 1);
765 paParms[i++].setUInt32((uint32_t)strFormat.length() + 1);
766 paParms[i++].setPointer((void*)sfaData.raw(), (uint32_t)sfaData.size());
767 paParms[i++].setUInt32((uint32_t)sfaData.size());
768
769 DnDGuestResponse *pDnD = d->response();
770 /* Reset any old progress status. */
771 pDnD->resetProgress(p);
772
773 d->hostCall(DragAndDropSvc::HOST_DND_HG_SND_DATA,
774 i,
775 paParms);
776
777 /* Query the progress object to the caller. */
778 pDnD->queryProgressTo(ppProgress);
779 }
780 catch (HRESULT rc2)
781 {
782 rc = rc2;
783 }
784
785 return rc;
786}
787
788# ifdef VBOX_WITH_DRAG_AND_DROP_GH
789HRESULT GuestDnD::dragGHPending(ULONG uScreenId, ComSafeArrayOut(BSTR, formats), ComSafeArrayOut(DragAndDropAction_T, allowedActions), DragAndDropAction_T *pDefaultAction)
790{
791 DPTR(GuestDnD);
792 const ComObjPtr<Guest> &p = d->p;
793
794 /* Default is ignoring */
795 *pDefaultAction = DragAndDropAction_Ignore;
796
797 HRESULT rc = S_OK;
798
799 try
800 {
801 VBOXHGCMSVCPARM paParms[1];
802 int i = 0;
803 paParms[i++].setUInt32(uScreenId);
804
805 d->hostCall(DragAndDropSvc::HOST_DND_GH_REQ_PENDING,
806 i,
807 paParms);
808
809 DnDGuestResponse *pDnD = d->response();
810 /* This blocks until the request is answered (or timeout). */
811 if (pDnD->waitForGuestResponse() == VERR_TIMEOUT)
812 return S_OK;
813
814 if (isDnDIgnoreAction(pDnD->defAction()))
815 return S_OK;
816
817 /* Fetch the default action to use. */
818 *pDefaultAction = d->toMainAction(pDnD->defAction());
819 /* Convert the formats strings to a vector of strings. */
820 d->toFormatSafeArray(pDnD->format(), ComSafeArrayOutArg(formats));
821 /* Convert the action bit field to a vector of actions. */
822 d->toMainActions(pDnD->allActions(), ComSafeArrayOutArg(allowedActions));
823 }
824 catch (HRESULT rc2)
825 {
826 rc = rc2;
827 }
828
829 return rc;
830}
831
832HRESULT GuestDnD::dragGHDropped(IN_BSTR bstrFormat, DragAndDropAction_T action, IProgress **ppProgress)
833{
834 DPTR(GuestDnD);
835 const ComObjPtr<Guest> &p = d->p;
836
837 Utf8Str strFormat(bstrFormat);
838 HRESULT rc = S_OK;
839
840 uint32_t uAction = d->toHGCMAction(action);
841 /* If there is no usable action, ignore this request. */
842 if (isDnDIgnoreAction(uAction))
843 return S_OK;
844
845 try
846 {
847 VBOXHGCMSVCPARM paParms[3];
848 int i = 0;
849 paParms[i++].setPointer((void*)strFormat.c_str(), (uint32_t)strFormat.length() + 1);
850 paParms[i++].setUInt32((uint32_t)strFormat.length() + 1);
851 paParms[i++].setUInt32(uAction);
852
853 DnDGuestResponse *pDnD = d->response();
854 /* Reset any old data and the progress status. */
855 pDnD->resetData();
856 pDnD->resetProgress(p);
857
858 d->hostCall(DragAndDropSvc::HOST_DND_GH_EVT_DROPPED,
859 i,
860 paParms);
861
862 /* Query the progress object to the caller. */
863 pDnD->queryProgressTo(ppProgress);
864 }
865 catch (HRESULT rc2)
866 {
867 rc = rc2;
868 }
869
870 return rc;
871}
872
873HRESULT GuestDnD::dragGHGetData(ComSafeArrayOut(BYTE, data))
874{
875 DPTR(GuestDnD);
876 const ComObjPtr<Guest> &p = d->p;
877
878 HRESULT rc = S_OK;
879
880 DnDGuestResponse *pDnD = d->response();
881 /* Is there data at all? */
882 if (pDnD->hasData())
883 {
884 /* Copy the data into an safe array of bytes. */
885 void *pvData = 0;
886 uint32_t cbData = 0;
887 pDnD->data(&pvData, &cbData);
888 com::SafeArray<BYTE> sfaData(cbData);
889 memcpy(sfaData.raw(), pvData, cbData);
890 sfaData.detachTo(ComSafeArrayOutArg(data));
891 /* Delete the data. */
892 pDnD->resetData();
893 }
894 else
895 rc = VBOX_E_INVALID_OBJECT_STATE;
896
897 return rc;
898}
899
900# endif /* VBOX_WITH_DRAG_AND_DROP_GH */
901
902DECLCALLBACK(int) GuestDnD::notifyGuestDragAndDropEvent(void *pvExtension, uint32_t u32Function, void *pvParms, uint32_t cbParms)
903{
904 LogFlowFunc(("pvExtension=%p, u32Function=%RU32, pvParms=%p, cbParms=%RU32\n",
905 pvExtension, u32Function, pvParms, cbParms));
906
907 ComObjPtr<Guest> pGuest = reinterpret_cast<Guest*>(pvExtension);
908 if (!pGuest->m_pGuestDnD)
909 return VINF_SUCCESS;
910
911 GuestDnDPrivate *d = static_cast<GuestDnDPrivate*>(pGuest->m_pGuestDnD->d_ptr);
912 const ComObjPtr<Guest> &p = d->p;
913
914 DnDGuestResponse *pResp = d->response();
915 if (pResp == NULL)
916 return VERR_INVALID_PARAMETER;
917
918 int rc = VINF_SUCCESS;
919 switch (u32Function)
920 {
921 case DragAndDropSvc::GUEST_DND_HG_ACK_OP:
922 {
923 DragAndDropSvc::PVBOXDNDCBHGACKOPDATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBHGACKOPDATA>(pvParms);
924 AssertPtr(pCBData);
925 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGACKOPDATA) == cbParms, VERR_INVALID_PARAMETER);
926 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_ACK_OP == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
927 pResp->setDefAction(pCBData->uAction);
928 rc = pResp->notifyAboutGuestResponse();
929 break;
930 }
931 case DragAndDropSvc::GUEST_DND_HG_REQ_DATA:
932 {
933 DragAndDropSvc::PVBOXDNDCBHGREQDATADATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBHGREQDATADATA>(pvParms);
934 AssertPtr(pCBData);
935 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGREQDATADATA) == cbParms, VERR_INVALID_PARAMETER);
936 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_REQ_DATA == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
937 pResp->setFormat(pCBData->pszFormat);
938 rc = pResp->notifyAboutGuestResponse();
939 break;
940 }
941 case DragAndDropSvc::GUEST_DND_HG_EVT_PROGRESS:
942 {
943 DragAndDropSvc::PVBOXDNDCBHGEVTPROGRESSDATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBHGEVTPROGRESSDATA>(pvParms);
944 AssertPtr(pCBData);
945 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBHGEVTPROGRESSDATA) == cbParms, VERR_INVALID_PARAMETER);
946 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_HG_EVT_PROGRESS == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
947 rc = pResp->setProgress(pCBData->uPercentage, pCBData->uState, pCBData->rc);
948 break;
949 }
950#ifdef VBOX_WITH_DRAG_AND_DROP_GH
951 case DragAndDropSvc::GUEST_DND_GH_ACK_PENDING:
952 {
953 DragAndDropSvc::PVBOXDNDCBGHACKPENDINGDATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBGHACKPENDINGDATA>(pvParms);
954 AssertPtr(pCBData);
955 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBGHACKPENDINGDATA) == cbParms, VERR_INVALID_PARAMETER);
956 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_ACK_PENDING == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
957 pResp->setFormat(pCBData->pszFormat);
958 pResp->setDefAction(pCBData->uDefAction);
959 pResp->setAllActions(pCBData->uAllActions);
960 rc = pResp->notifyAboutGuestResponse();
961 break;
962 }
963 case DragAndDropSvc::GUEST_DND_GH_SND_DATA:
964 {
965 DragAndDropSvc::PVBOXDNDCBSNDDATADATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBSNDDATADATA>(pvParms);
966 AssertPtr(pCBData);
967 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBSNDDATADATA) == cbParms, VERR_INVALID_PARAMETER);
968 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_SND_DATA == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
969 uint32_t cbCurSize = 0;
970 pResp->addData(pCBData->pvData, pCBData->cbData, &cbCurSize);
971 rc = pResp->setProgress(100.0 / pCBData->cbAllSize * cbCurSize, (pCBData->cbAllSize == cbCurSize ? DragAndDropSvc::DND_PROGRESS_COMPLETE : DragAndDropSvc::DND_PROGRESS_RUNNING));
972 /* Todo: for now we instantly confirm the cancel. Check if the
973 * guest should first clean up stuff itself and than really confirm
974 * the cancel request by an extra message. */
975 if (rc == VERR_CANCELLED)
976 pResp->setProgress(100, DragAndDropSvc::DND_PROGRESS_CANCELLED);
977 break;
978 }
979 case DragAndDropSvc::GUEST_DND_GH_EVT_ERROR:
980 {
981 DragAndDropSvc::PVBOXDNDCBEVTERRORDATA pCBData = reinterpret_cast<DragAndDropSvc::PVBOXDNDCBEVTERRORDATA>(pvParms);
982 AssertPtr(pCBData);
983 AssertReturn(sizeof(DragAndDropSvc::VBOXDNDCBEVTERRORDATA) == cbParms, VERR_INVALID_PARAMETER);
984 AssertReturn(DragAndDropSvc::CB_MAGIC_DND_GH_EVT_ERROR == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER);
985 /* Cleanup */
986 pResp->resetData();
987 rc = pResp->setProgress(100, DragAndDropSvc::DND_PROGRESS_ERROR, pCBData->rc);
988 break;
989 }
990#endif /* VBOX_WITH_DRAG_AND_DROP_GH */
991 default:
992 AssertMsgFailedReturn(("Function %RU32 not supported\n", u32Function), VERR_NOT_SUPPORTED);
993 break;
994 }
995
996 LogFlowFunc(("Returning rc=%Rrc\n", rc));
997 return rc;
998}
999
1000#endif /* VBOX_WITH_DRAG_AND_DROP */
1001
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