VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/VBoxClient/draganddrop.cpp@ 59083

Last change on this file since 59083 was 58945, checked in by vboxsync, 9 years ago

Additions/x11/DnD: uninitialized variable

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 114.1 KB
Line 
1/* $Id: draganddrop.cpp 58945 2015-12-02 08:38:06Z vboxsync $ */
2/** @file
3 * X11 guest client - Drag and drop implementation.
4 */
5
6/*
7 * Copyright (C) 2011-2015 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 <X11/Xlib.h>
19#include <X11/Xutil.h>
20#include <X11/Xatom.h>
21#ifdef VBOX_DND_WITH_XTEST
22# include <X11/extensions/XTest.h>
23#endif
24
25#include <iprt/asm.h>
26#include <iprt/buildconfig.h>
27#include <iprt/critsect.h>
28#include <iprt/thread.h>
29#include <iprt/time.h>
30
31#include <iprt/cpp/mtlist.h>
32#include <iprt/cpp/ministring.h>
33
34#include <limits.h>
35
36#ifdef LOG_GROUP
37# undef LOG_GROUP
38#endif
39#define LOG_GROUP LOG_GROUP_GUEST_DND
40#include <VBox/log.h>
41#include <VBox/VBoxGuestLib.h>
42
43#include "VBox/HostServices/DragAndDropSvc.h"
44#include "VBoxClient.h"
45
46/* Enable this define to see the proxy window(s) when debugging
47 * their behavior. Don't have this enabled in release builds! */
48#ifdef DEBUG
49//# define VBOX_DND_DEBUG_WND
50#endif
51
52/**
53 * For X11 guest Xdnd is used. See http://www.acc.umu.se/~vatten/XDND.html for
54 * a walk trough.
55 *
56 * Host -> Guest:
57 * For X11 this means mainly forwarding all the events from HGCM to the
58 * appropriate X11 events. There exists a proxy window, which is invisible and
59 * used for all the X11 communication. On a HGCM Enter event, we set our proxy
60 * window as XdndSelection owner with the given mime-types. On every HGCM move
61 * event, we move the X11 mouse cursor to the new position and query for the
62 * window below that position. Depending on if it is XdndAware, a new window or
63 * a known window, we send the appropriate X11 messages to it. On HGCM drop, we
64 * send a XdndDrop message to the current window and wait for a X11
65 * SelectionMessage from the target window. Because we didn't have the data in
66 * the requested mime-type, yet, we save that message and ask the host for the
67 * data. When the data is successfully received from the host, we put the data
68 * as a property to the window and send a X11 SelectionNotify event to the
69 * target window.
70 *
71 * Guest -> Host:
72 * This is a lot more trickery than H->G. When a pending event from HGCM
73 * arrives, we ask if there currently is an owner of the XdndSelection
74 * property. If so, our proxy window is shown (1x1, but without backing store)
75 * and some mouse event is triggered. This should be followed by an XdndEnter
76 * event send to the proxy window. From this event we can fetch the necessary
77 * info of the MIME types and allowed actions and send this back to the host.
78 * On a drop request from the host, we query for the selection and should get
79 * the data in the specified mime-type. This data is send back to the host.
80 * After that we send a XdndLeave event to the source window.
81 *
82 ** @todo Cancelling (e.g. with ESC key) doesn't work.
83 ** @todo INCR (incremental transfers) support.
84 ** @todo Really check for the Xdnd version and the supported features.
85 ** @todo Either get rid of the xHelpers class or properly unify the code with the drag instance class.
86 */
87
88#define VBOX_XDND_VERSION (4)
89#define VBOX_MAX_XPROPERTIES (LONG_MAX-1)
90
91/**
92 * Structure for storing new X11 events and HGCM messages
93 * into a single vent queue.
94 */
95struct DnDEvent
96{
97 enum DnDEventType
98 {
99 HGCM_Type = 1,
100 X11_Type
101 };
102 DnDEventType type;
103 union
104 {
105 VBGLR3DNDHGCMEVENT hgcm;
106 XEvent x11;
107 };
108};
109
110enum XA_Type
111{
112 /* States */
113 XA_WM_STATE = 0,
114 /* Properties */
115 XA_TARGETS,
116 XA_MULTIPLE,
117 XA_INCR,
118 /* Mime Types */
119 XA_image_bmp,
120 XA_image_jpg,
121 XA_image_tiff,
122 XA_image_png,
123 XA_text_uri_list,
124 XA_text_uri,
125 XA_text_plain,
126 XA_TEXT,
127 /* Xdnd */
128 XA_XdndSelection,
129 XA_XdndAware,
130 XA_XdndEnter,
131 XA_XdndLeave,
132 XA_XdndTypeList,
133 XA_XdndActionList,
134 XA_XdndPosition,
135 XA_XdndActionCopy,
136 XA_XdndActionMove,
137 XA_XdndActionLink,
138 XA_XdndStatus,
139 XA_XdndDrop,
140 XA_XdndFinished,
141 /* Our own stop marker */
142 XA_dndstop,
143 /* End marker */
144 XA_End
145};
146
147/**
148 * Xdnd message value indexes, sorted by message type.
149 */
150typedef enum XdndMsg
151{
152 /** XdndEnter. */
153 XdndEnterTypeCount = 3, /* Maximum number of types in XdndEnter message. */
154
155 XdndEnterWindow = 0, /* Source window (sender). */
156 XdndEnterFlags, /* Version in high byte, bit 0 => more data types. */
157 XdndEnterType1, /* First available data type. */
158 XdndEnterType2, /* Second available data type. */
159 XdndEnterType3, /* Third available data type. */
160
161 XdndEnterMoreTypesFlag = 1, /* Set if there are more than XdndEnterTypeCount. */
162 XdndEnterVersionRShift = 24, /* Right shift to position version number. */
163 XdndEnterVersionMask = 0xFF, /* Mask to get version after shifting. */
164
165 /** XdndHere. */
166 XdndHereWindow = 0, /* Source window (sender). */
167 XdndHereFlags, /* Reserved. */
168 XdndHerePt, /* X + Y coordinates of mouse (root window coords). */
169 XdndHereTimeStamp, /* Timestamp for requesting data. */
170 XdndHereAction, /* Action requested by user. */
171
172 /** XdndPosition. */
173 XdndPositionWindow = 0, /* Source window (sender). */
174 XdndPositionFlags, /* Flags. */
175 XdndPositionXY, /* X/Y coordinates of the mouse position relative to the root window. */
176 XdndPositionTimeStamp, /* Time stamp for retrieving the data. */
177 XdndPositionAction, /* Action requested by the user. */
178
179 /** XdndStatus. */
180 XdndStatusWindow = 0, /* Target window (sender).*/
181 XdndStatusFlags, /* Flags returned by target. */
182 XdndStatusNoMsgXY, /* X + Y of "no msg" rectangle (root window coords). */
183 XdndStatusNoMsgWH, /* Width + height of "no msg" rectangle. */
184 XdndStatusAction, /* Action accepted by target. */
185
186 XdndStatusAcceptDropFlag = 1, /* Set if target will accept the drop. */
187 XdndStatusSendHereFlag = 2, /* Set if target wants a stream of XdndPosition. */
188
189 /** XdndLeave. */
190 XdndLeaveWindow = 0, /* Source window (sender). */
191 XdndLeaveFlags, /* Reserved. */
192
193 /** XdndDrop. */
194 XdndDropWindow = 0, /* Source window (sender). */
195 XdndDropFlags, /* Reserved. */
196 XdndDropTimeStamp, /* Timestamp for requesting data. */
197
198 /** XdndFinished. */
199 XdndFinishedWindow = 0, /* Target window (sender). */
200 XdndFinishedFlags, /* Version 5: Bit 0 is set if the current target accepted the drop. */
201 XdndFinishedAction /* Version 5: Contains the action performed by the target. */
202
203} XdndMsg;
204
205class DragAndDropService;
206
207/** List of Atoms. */
208#define VBoxDnDAtomList RTCList<Atom>
209
210/*******************************************************************************
211 *
212 * xHelpers Declaration
213 *
214 ******************************************************************************/
215
216class xHelpers
217{
218public:
219
220 static xHelpers *getInstance(Display *pDisplay = 0)
221 {
222 if (!m_pInstance)
223 {
224 AssertPtrReturn(pDisplay, NULL);
225 m_pInstance = new xHelpers(pDisplay);
226 }
227
228 return m_pInstance;
229 }
230
231 inline Display *display() const { return m_pDisplay; }
232 inline Atom xAtom(XA_Type e) const { return m_xAtoms[e]; }
233
234 inline Atom stringToxAtom(const char *pcszString) const
235 {
236 return XInternAtom(m_pDisplay, pcszString, False);
237 }
238 inline RTCString xAtomToString(Atom atom) const
239 {
240 if (atom == None) return "None";
241
242 char* pcsAtom = XGetAtomName(m_pDisplay, atom);
243 RTCString strAtom(pcsAtom);
244 XFree(pcsAtom);
245
246 return strAtom;
247 }
248
249 inline RTCString xAtomListToString(const VBoxDnDAtomList &formatList)
250 {
251 RTCString format;
252 for (size_t i = 0; i < formatList.size(); ++i)
253 format += xAtomToString(formatList.at(i)) + "\r\n";
254 return format;
255 }
256
257 RTCString xErrorToString(int xRc) const;
258 Window applicationWindowBelowCursor(Window parentWin) const;
259
260private:
261
262 xHelpers(Display *pDisplay)
263 : m_pDisplay(pDisplay)
264 {
265 /* Not all x11 atoms we use are defined in the headers. Create the
266 * additional one we need here. */
267 for (int i = 0; i < XA_End; ++i)
268 m_xAtoms[i] = XInternAtom(m_pDisplay, m_xAtomNames[i], False);
269 };
270
271 /* Private member vars */
272 static xHelpers *m_pInstance;
273 Display *m_pDisplay;
274 Atom m_xAtoms[XA_End];
275 static const char *m_xAtomNames[XA_End];
276};
277
278/* Some xHelpers convenience defines. */
279#define gX11 xHelpers::getInstance()
280#define xAtom(xa) gX11->xAtom((xa))
281#define xAtomToString(xa) gX11->xAtomToString((xa))
282
283/*******************************************************************************
284 *
285 * xHelpers Implementation
286 *
287 ******************************************************************************/
288
289xHelpers *xHelpers::m_pInstance = NULL;
290
291/* Has to be in sync with the XA_Type enum. */
292const char *xHelpers::m_xAtomNames[] =
293{
294 /* States */
295 "WM_STATE",
296 /* Properties */
297 "TARGETS",
298 "MULTIPLE",
299 "INCR",
300 /* Mime Types */
301 "image/bmp",
302 "image/jpg",
303 "image/tiff",
304 "image/png",
305 "text/uri-list",
306 "text/uri",
307 "text/plain",
308 "TEXT",
309 /* Xdnd */
310 "XdndSelection",
311 "XdndAware",
312 "XdndEnter",
313 "XdndLeave",
314 "XdndTypeList",
315 "XdndActionList",
316 "XdndPosition",
317 "XdndActionCopy",
318 "XdndActionMove",
319 "XdndActionLink",
320 "XdndStatus",
321 "XdndDrop",
322 "XdndFinished",
323 /* Our own stop marker */
324 "dndstop"
325};
326
327RTCString xHelpers::xErrorToString(int xRc) const
328{
329 switch (xRc)
330 {
331 case Success: return RTCStringFmt("%d (Success)", xRc); break;
332 case BadRequest: return RTCStringFmt("%d (BadRequest)", xRc); break;
333 case BadValue: return RTCStringFmt("%d (BadValue)", xRc); break;
334 case BadWindow: return RTCStringFmt("%d (BadWindow)", xRc); break;
335 case BadPixmap: return RTCStringFmt("%d (BadPixmap)", xRc); break;
336 case BadAtom: return RTCStringFmt("%d (BadAtom)", xRc); break;
337 case BadCursor: return RTCStringFmt("%d (BadCursor)", xRc); break;
338 case BadFont: return RTCStringFmt("%d (BadFont)", xRc); break;
339 case BadMatch: return RTCStringFmt("%d (BadMatch)", xRc); break;
340 case BadDrawable: return RTCStringFmt("%d (BadDrawable)", xRc); break;
341 case BadAccess: return RTCStringFmt("%d (BadAccess)", xRc); break;
342 case BadAlloc: return RTCStringFmt("%d (BadAlloc)", xRc); break;
343 case BadColor: return RTCStringFmt("%d (BadColor)", xRc); break;
344 case BadGC: return RTCStringFmt("%d (BadGC)", xRc); break;
345 case BadIDChoice: return RTCStringFmt("%d (BadIDChoice)", xRc); break;
346 case BadName: return RTCStringFmt("%d (BadName)", xRc); break;
347 case BadLength: return RTCStringFmt("%d (BadLength)", xRc); break;
348 case BadImplementation: return RTCStringFmt("%d (BadImplementation)", xRc); break;
349 }
350 return RTCStringFmt("%d (unknown)", xRc);
351}
352
353/** todo Make this iterative. */
354Window xHelpers::applicationWindowBelowCursor(Window wndParent) const
355{
356 /* No parent, nothing to do. */
357 if(wndParent == 0)
358 return 0;
359
360 Window wndApp = 0;
361 int cProps = -1;
362
363 /* Fetch all x11 window properties of the parent window. */
364 Atom *pProps = XListProperties(m_pDisplay, wndParent, &cProps);
365 if (cProps > 0)
366 {
367 /* We check the window for the WM_STATE property. */
368 for (int i = 0; i < cProps; ++i)
369 {
370 if (pProps[i] == xAtom(XA_WM_STATE))
371 {
372 /* Found it. */
373 wndApp = wndParent;
374 break;
375 }
376 }
377
378 /* Cleanup */
379 XFree(pProps);
380 }
381
382 if (!wndApp)
383 {
384 Window wndChild, wndTemp;
385 int tmp;
386 unsigned int utmp;
387
388 /* Query the next child window of the parent window at the current
389 * mouse position. */
390 XQueryPointer(m_pDisplay, wndParent, &wndTemp, &wndChild, &tmp, &tmp, &tmp, &tmp, &utmp);
391
392 /* Recursive call our self to dive into the child tree. */
393 wndApp = applicationWindowBelowCursor(wndChild);
394 }
395
396 return wndApp;
397}
398
399/*******************************************************************************
400 *
401 * DragInstance Declaration
402 *
403 ******************************************************************************/
404
405#ifdef DEBUG
406# define VBOX_DND_FN_DECL_LOG(x) inline x /* For LogFlowXXX logging. */
407#else
408# define VBOX_DND_FN_DECL_LOG(x) x
409#endif
410
411/** @todo Move all proxy window-related stuff into this class! Clean up this mess. */
412class VBoxDnDProxyWnd
413{
414
415public:
416
417 VBoxDnDProxyWnd(void);
418
419 virtual ~VBoxDnDProxyWnd(void);
420
421public:
422
423 int init(Display *pDisplay);
424 void destroy();
425
426 int sendFinished(Window hWndSource, uint32_t uAction);
427
428public:
429
430 Display *pDisp;
431 /** Proxy window handle. */
432 Window hWnd;
433 int iX;
434 int iY;
435 int iWidth;
436 int iHeight;
437};
438
439/**
440 * Class for handling a single drag and drop operation, that is,
441 * one source and one target at a time.
442 *
443 * For now only one DragInstance will exits when the app is running.
444 */
445class DragInstance
446{
447public:
448
449 enum State
450 {
451 Uninitialized = 0,
452 Initialized,
453 Dragging,
454 Dropped,
455 State_32BIT_Hack = 0x7fffffff
456 };
457
458 enum Mode
459 {
460 Unknown = 0,
461 HG,
462 GH,
463 Mode_32Bit_Hack = 0x7fffffff
464 };
465
466 DragInstance(Display *pDisplay, DragAndDropService *pParent);
467
468public:
469
470 int init(uint32_t u32ScreenId);
471 void uninit(void);
472 void reset(void);
473
474 /* Logging. */
475 VBOX_DND_FN_DECL_LOG(void) logInfo(const char *pszFormat, ...);
476 VBOX_DND_FN_DECL_LOG(void) logError(const char *pszFormat, ...);
477
478 /* X11 message processing. */
479 int onX11ClientMessage(const XEvent &e);
480 int onX11MotionNotify(const XEvent &e);
481 int onX11SelectionClear(const XEvent &e);
482 int onX11SelectionNotify(const XEvent &e);
483 int onX11SelectionRequest(const XEvent &e);
484 int onX11Event(const XEvent &e);
485 int waitForStatusChange(uint32_t enmState, RTMSINTERVAL uTimeoutMS = 30000);
486 bool waitForX11Msg(XEvent &evX, int iType, RTMSINTERVAL uTimeoutMS = 100);
487 bool waitForX11ClientMsg(XClientMessageEvent &evMsg, Atom aType, RTMSINTERVAL uTimeoutMS = 100);
488
489 /* Host -> Guest handling. */
490 int hgEnter(const RTCList<RTCString> &formats, uint32_t actions);
491 int hgLeave(void);
492 int hgMove(uint32_t u32xPos, uint32_t u32yPos, uint32_t uDefaultAction);
493 int hgDrop(uint32_t u32xPos, uint32_t u32yPos, uint32_t uDefaultAction);
494 int hgDataReceived(const void *pvData, uint32_t cData);
495
496#ifdef VBOX_WITH_DRAG_AND_DROP_GH
497 /* Guest -> Host handling. */
498 int ghIsDnDPending(void);
499 int ghDropped(const RTCString &strFormat, uint32_t action);
500#endif
501
502 /* X11 helpers. */
503 int mouseCursorFakeMove(void) const;
504 int mouseCursorMove(int iPosX, int iPosY) const;
505 void mouseButtonSet(Window wndDest, int rx, int ry, int iButton, bool fPress);
506 int proxyWinShow(int *piRootX = NULL, int *piRootY = NULL) const;
507 int proxyWinHide(void);
508
509 /* X11 window helpers. */
510 char *wndX11GetNameA(Window wndThis) const;
511
512 /* Xdnd protocol helpers. */
513 void wndXDnDClearActionList(Window wndThis) const;
514 void wndXDnDClearFormatList(Window wndThis) const;
515 int wndXDnDGetActionList(Window wndThis, VBoxDnDAtomList &lstActions) const;
516 int wndXDnDGetFormatList(Window wndThis, VBoxDnDAtomList &lstTypes) const;
517 int wndXDnDSetActionList(Window wndThis, const VBoxDnDAtomList &lstActions) const;
518 int wndXDnDSetFormatList(Window wndThis, Atom atmProp, const VBoxDnDAtomList &lstFormats) const;
519
520 /* Atom / HGCM formatting helpers. */
521 int toAtomList(const RTCList<RTCString> &lstFormats, VBoxDnDAtomList &lstAtoms) const;
522 int toAtomList(const void *pvData, uint32_t cbData, VBoxDnDAtomList &lstAtoms) const;
523 static Atom toAtomAction(uint32_t uAction);
524 static int toAtomActions(uint32_t uActions, VBoxDnDAtomList &lstAtoms);
525 static uint32_t toHGCMAction(Atom atom);
526 static uint32_t toHGCMActions(const VBoxDnDAtomList &actionsList);
527
528protected:
529
530 /** The instance's own DnD context. */
531 VBGLR3GUESTDNDCMDCTX m_dndCtx;
532 /** Pointer to service instance. */
533 DragAndDropService *m_pParent;
534 /** Pointer to X display operating on. */
535 Display *m_pDisplay;
536 /** X screen ID to operate on. */
537 int m_screenId;
538 /** Pointer to X screen operating on. */
539 Screen *m_pScreen;
540 /** Root window handle. */
541 Window m_wndRoot;
542 /** Proxy window. */
543 VBoxDnDProxyWnd m_wndProxy;
544 /** Current source/target window handle. */
545 Window m_wndCur;
546 /** The XDnD protocol version the current
547 * source/target window is using. */
548 long m_curVer;
549 /** List of (Atom) formats the source window supports. */
550 VBoxDnDAtomList m_lstFormats;
551 /** List of (Atom) actions the source window supports. */
552 VBoxDnDAtomList m_lstActions;
553 /** Buffer for answering the target window's selection request. */
554 void *m_pvSelReqData;
555 /** Size (in bytes) of selection request data buffer. */
556 uint32_t m_cbSelReqData;
557 /** Current operation mode. */
558 volatile uint32_t m_enmMode;
559 /** Current state of operation mode. */
560 volatile uint32_t m_enmState;
561 /** The instance's own X event queue. */
562 RTCMTList<XEvent> m_eventQueueList;
563 /** Critical section for providing serialized access to list
564 * event queue's contents. */
565 RTCRITSECT m_eventQueueCS;
566 /** Event for notifying this instance in case of a new
567 * event. */
568 RTSEMEVENT m_eventQueueEvent;
569 /** Critical section for data access. */
570 RTCRITSECT m_dataCS;
571 /** List of allowed formats. */
572 RTCList<RTCString> m_lstAllowedFormats;
573 /** Number of failed attempts by the host
574 * to query for an active drag and drop operation on the guest. */
575 uint16_t m_cFailedPendingAttempts;
576};
577
578/*******************************************************************************
579 *
580 * DragAndDropService Declaration
581 *
582 ******************************************************************************/
583
584class DragAndDropService
585{
586public:
587 DragAndDropService(void)
588 : m_pDisplay(0)
589 , m_hHGCMThread(NIL_RTTHREAD)
590 , m_hX11Thread(NIL_RTTHREAD)
591 , m_hEventSem(NIL_RTSEMEVENT)
592 , m_pCurDnD(0)
593 , m_fSrvStopping(false)
594 {}
595
596 int run(bool fDaemonised = false);
597
598private:
599
600 int dragAndDropInit(void);
601 static DECLCALLBACK(int) hgcmEventThread(RTTHREAD hThread, void *pvUser);
602 static DECLCALLBACK(int) x11EventThread(RTTHREAD hThread, void *pvUser);
603
604 /* Private member vars */
605 Display *m_pDisplay;
606
607 /** Our (thread-safe) event queue with
608 * mixed events (DnD HGCM / X11). */
609 RTCMTList<DnDEvent> m_eventQueue;
610 /** Critical section for providing serialized access to list
611 * event queue's contents. */
612 RTCRITSECT m_eventQueueCS;
613 RTTHREAD m_hHGCMThread;
614 RTTHREAD m_hX11Thread;
615 RTSEMEVENT m_hEventSem;
616 DragInstance *m_pCurDnD;
617 bool m_fSrvStopping;
618
619 friend class DragInstance;
620};
621
622/*******************************************************************************
623 *
624 * DragInstanc Implementation
625 *
626 ******************************************************************************/
627
628DragInstance::DragInstance(Display *pDisplay, DragAndDropService *pParent)
629 : m_pParent(pParent)
630 , m_pDisplay(pDisplay)
631 , m_pScreen(0)
632 , m_wndRoot(0)
633 , m_wndCur(0)
634 , m_curVer(-1)
635 , m_pvSelReqData(NULL)
636 , m_cbSelReqData(0)
637 , m_enmMode(Unknown)
638 , m_enmState(Uninitialized)
639{
640}
641
642/**
643 * Unitializes (destroys) this drag instance.
644 */
645void DragInstance::uninit(void)
646{
647 LogFlowFuncEnter();
648
649 if (m_wndProxy.hWnd != 0)
650 XDestroyWindow(m_pDisplay, m_wndProxy.hWnd);
651
652 int rc2 = VbglR3DnDDisconnect(&m_dndCtx);
653
654 if (m_pvSelReqData)
655 RTMemFree(m_pvSelReqData);
656
657 rc2 = RTSemEventDestroy(m_eventQueueEvent);
658 AssertRC(rc2);
659
660 rc2 = RTCritSectDelete(&m_eventQueueCS);
661 AssertRC(rc2);
662
663 rc2 = RTCritSectDelete(&m_dataCS);
664 AssertRC(rc2);
665}
666
667/**
668 * Resets this drag instance.
669 */
670void DragInstance::reset(void)
671{
672 LogFlowFuncEnter();
673
674 /* Hide the proxy win. */
675 proxyWinHide();
676
677 int rc2 = RTCritSectEnter(&m_dataCS);
678 if (RT_SUCCESS(rc2))
679 {
680 /* If we are currently the Xdnd selection owner, clear that. */
681 Window pWnd = XGetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection));
682 if (pWnd == m_wndProxy.hWnd)
683 XSetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection), None, CurrentTime);
684
685 /* Clear any other DnD specific data on the proxy window. */
686 wndXDnDClearFormatList(m_wndProxy.hWnd);
687 wndXDnDClearActionList(m_wndProxy.hWnd);
688
689 /* Reset the internal state. */
690 m_lstActions.clear();
691 m_lstFormats.clear();
692 m_wndCur = 0;
693 m_curVer = -1;
694 m_enmState = Initialized;
695 m_enmMode = Unknown;
696 m_eventQueueList.clear();
697 m_cFailedPendingAttempts = 0;
698
699 /* Reset the selection request buffer. */
700 if (m_pvSelReqData)
701 {
702 RTMemFree(m_pvSelReqData);
703 m_pvSelReqData = NULL;
704
705 Assert(m_cbSelReqData);
706 m_cbSelReqData = 0;
707 }
708
709 RTCritSectLeave(&m_dataCS);
710 }
711}
712
713/**
714 * Initializes this drag instance.
715 *
716 * @return IPRT status code.
717 * @param u32ScreenID X' screen ID to use.
718 */
719int DragInstance::init(uint32_t u32ScreenID)
720{
721 int rc;
722
723 do
724 {
725 rc = VbglR3DnDConnect(&m_dndCtx);
726 if (RT_FAILURE(rc))
727 break;
728
729 rc = RTSemEventCreate(&m_eventQueueEvent);
730 if (RT_FAILURE(rc))
731 break;
732
733 rc = RTCritSectInit(&m_eventQueueCS);
734 if (RT_FAILURE(rc))
735 break;
736
737 rc = RTCritSectInit(&m_dataCS);
738 if (RT_FAILURE(rc))
739 break;
740
741 /*
742 * Enough screens configured in the x11 server?
743 */
744 if ((int)u32ScreenID > ScreenCount(m_pDisplay))
745 {
746 rc = VERR_INVALID_PARAMETER;
747 break;
748 }
749#if 0
750 /* Get the screen number from the x11 server. */
751 pDrag->screen = ScreenOfDisplay(m_pDisplay, u32ScreenId);
752 if (!pDrag->screen)
753 {
754 rc = VERR_GENERAL_FAILURE;
755 break;
756 }
757#endif
758 m_screenId = u32ScreenID;
759
760 /* Now query the corresponding root window of this screen. */
761 m_wndRoot = RootWindow(m_pDisplay, m_screenId);
762 if (!m_wndRoot)
763 {
764 rc = VERR_GENERAL_FAILURE;
765 break;
766 }
767
768 /*
769 * Create an invisible window which will act as proxy for the DnD
770 * operation. This window will be used for both the GH and HG
771 * direction.
772 */
773 XSetWindowAttributes attr;
774 RT_ZERO(attr);
775 attr.event_mask = EnterWindowMask | LeaveWindowMask
776 | ButtonMotionMask | ButtonPressMask | ButtonReleaseMask;
777 attr.override_redirect = True;
778 attr.do_not_propagate_mask = NoEventMask;
779#ifdef VBOX_DND_DEBUG_WND
780 attr.background_pixel = XWhitePixel(m_pDisplay, m_screenId);
781 attr.border_pixel = XBlackPixel(m_pDisplay, m_screenId);
782 m_wndProxy.hWnd = XCreateWindow(m_pDisplay, m_wndRoot /* Parent */,
783 100, 100, /* Position */
784 100, 100, /* Width + height */
785 2, /* Border width */
786 CopyFromParent, /* Depth */
787 InputOutput, /* Class */
788 CopyFromParent, /* Visual */
789 CWBackPixel
790 | CWBorderPixel
791 | CWOverrideRedirect
792 | CWDontPropagate, /* Value mask */
793 &attr); /* Attributes for value mask */
794#else
795 m_wndProxy.hWnd = XCreateWindow(m_pDisplay, m_wndRoot /* Parent */,
796 0, 0, /* Position */
797 1, 1, /* Width + height */
798 0, /* Border width */
799 CopyFromParent, /* Depth */
800 InputOnly, /* Class */
801 CopyFromParent, /* Visual */
802 CWOverrideRedirect | CWDontPropagate, /* Value mask */
803 &attr); /* Attributes for value mask */
804#endif
805 if (!m_wndProxy.hWnd)
806 {
807 LogRel(("DnD: Error creating proxy window\n"));
808 rc = VERR_GENERAL_FAILURE;
809 break;
810 }
811
812 rc = m_wndProxy.init(m_pDisplay);
813 if (RT_FAILURE(rc))
814 {
815 LogRel(("DnD: Error initializing proxy window, rc=%Rrc\n", rc));
816 break;
817 }
818
819#ifdef VBOX_DND_DEBUG_WND
820 XFlush(m_pDisplay);
821 XMapWindow(m_pDisplay, m_wndProxy.hWnd);
822 XRaiseWindow(m_pDisplay, m_wndProxy.hWnd);
823 XFlush(m_pDisplay);
824#endif
825 logInfo("Proxy window=0x%x, root window=0x%x ...\n", m_wndProxy.hWnd, m_wndRoot);
826
827 /* Set the window's name for easier lookup. */
828 XStoreName(m_pDisplay, m_wndProxy.hWnd, "VBoxClientWndDnD");
829
830 /* Make the new window Xdnd aware. */
831 Atom ver = VBOX_XDND_VERSION;
832 XChangeProperty(m_pDisplay, m_wndProxy.hWnd, xAtom(XA_XdndAware), XA_ATOM, 32, PropModeReplace,
833 reinterpret_cast<unsigned char*>(&ver), 1);
834 } while (0);
835
836 if (RT_SUCCESS(rc))
837 {
838 reset();
839 }
840 else
841 logError("Initializing drag instance for screen %RU32 failed with rc=%Rrc\n", u32ScreenID, rc);
842
843 LogFlowFuncLeaveRC(rc);
844 return rc;
845}
846
847/**
848 * Logs an error message to the (release) logging instance.
849 *
850 * @param pszFormat Format string to log.
851 */
852VBOX_DND_FN_DECL_LOG(void) DragInstance::logError(const char *pszFormat, ...)
853{
854 va_list args;
855 va_start(args, pszFormat);
856 char *psz = NULL;
857 RTStrAPrintfV(&psz, pszFormat, args);
858 va_end(args);
859
860 AssertPtr(psz);
861 LogFlowFunc(("%s", psz));
862 LogRel(("DnD: %s", psz));
863
864 RTStrFree(psz);
865}
866
867/**
868 * Logs an info message to the (release) logging instance.
869 *
870 * @param pszFormat Format string to log.
871 */
872VBOX_DND_FN_DECL_LOG(void) DragInstance::logInfo(const char *pszFormat, ...)
873{
874 va_list args;
875 va_start(args, pszFormat);
876 char *psz = NULL;
877 RTStrAPrintfV(&psz, pszFormat, args);
878 va_end(args);
879
880 AssertPtr(psz);
881 LogFlowFunc(("%s", psz));
882 LogRel2(("DnD: %s", psz));
883
884 RTStrFree(psz);
885}
886
887/**
888 * Callback handler for a generic client message from a window.
889 *
890 * @return IPRT status code.
891 * @param e X11 event to handle.
892 */
893int DragInstance::onX11ClientMessage(const XEvent &e)
894{
895 AssertReturn(e.type == ClientMessage, VERR_INVALID_PARAMETER);
896
897 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
898 LogFlowThisFunc(("Event wnd=%#x, msg=%s\n", e.xclient.window, xAtomToString(e.xclient.message_type).c_str()));
899
900 int rc = VINF_SUCCESS;
901
902 switch (m_enmMode)
903 {
904 case HG:
905 {
906 /*
907 * Client messages are used to inform us about the status of a XdndAware
908 * window, in response of some events we send to them.
909 */
910 if ( e.xclient.message_type == xAtom(XA_XdndStatus)
911 && m_wndCur == static_cast<Window>(e.xclient.data.l[XdndStatusWindow]))
912 {
913 bool fAcceptDrop = ASMBitTest (&e.xclient.data.l[XdndStatusFlags], 0); /* Does the target accept the drop? */
914 bool fWantsPosition = ASMBitTest (&e.xclient.data.l[XdndStatusFlags], 1); /* Does the target want XdndPosition messages? */
915 RTCString strActions = xAtomToString( e.xclient.data.l[XdndStatusAction]);
916
917 char *pszWndName = wndX11GetNameA(e.xclient.data.l[XdndStatusWindow]);
918 AssertPtr(pszWndName);
919
920 /*
921 * The XdndStatus message tell us if the window will accept the DnD
922 * event and with which action. We immediately send this info down to
923 * the host as a response of a previous DnD message.
924 */
925 LogFlowThisFunc(("XA_XdndStatus: wnd=%#x ('%s'), fAcceptDrop=%RTbool, fWantsPosition=%RTbool, strActions=%s\n",
926 e.xclient.data.l[XdndStatusWindow], pszWndName, fAcceptDrop, fWantsPosition, strActions.c_str()));
927
928 RTStrFree(pszWndName);
929
930 uint16_t x = RT_HI_U16((uint32_t)e.xclient.data.l[XdndStatusNoMsgXY]);
931 uint16_t y = RT_LO_U16((uint32_t)e.xclient.data.l[XdndStatusNoMsgXY]);
932 uint16_t w = RT_HI_U16((uint32_t)e.xclient.data.l[XdndStatusNoMsgWH]);
933 uint16_t h = RT_LO_U16((uint32_t)e.xclient.data.l[XdndStatusNoMsgWH]);
934 LogFlowThisFunc(("\tReported dead area: x=%RU16, y=%RU16, w=%RU16, h=%RU16\n", x, y, w, h));
935
936 uint32_t uAction = DND_IGNORE_ACTION; /* Default is ignoring. */
937 /** @todo Compare this with the allowed actions. */
938 if (fAcceptDrop)
939 uAction = toHGCMAction(static_cast<Atom>(e.xclient.data.l[XdndStatusAction]));
940
941 rc = VbglR3DnDHGSendAckOp(&m_dndCtx, uAction);
942 }
943 else if (e.xclient.message_type == xAtom(XA_XdndFinished))
944 {
945 bool fSucceeded = ASMBitTest(&e.xclient.data.l[XdndFinishedFlags], 0);
946
947 char *pszWndName = wndX11GetNameA(e.xclient.data.l[XdndFinishedWindow]);
948 AssertPtr(pszWndName);
949
950 /* This message is sent on an un/successful DnD drop request. */
951 LogFlowThisFunc(("XA_XdndFinished: wnd=%#x ('%s'), success=%RTbool, action=%s\n",
952 e.xclient.data.l[XdndFinishedWindow], pszWndName, fSucceeded,
953 xAtomToString(e.xclient.data.l[XdndFinishedAction]).c_str()));
954
955 RTStrFree(pszWndName);
956
957 reset();
958 }
959 else
960 {
961 char *pszWndName = wndX11GetNameA(e.xclient.data.l[0]);
962 AssertPtr(pszWndName);
963 LogFlowThisFunc(("Unhandled: wnd=%#x ('%s'), msg=%s\n",
964 e.xclient.data.l[0], pszWndName, xAtomToString(e.xclient.message_type).c_str()));
965 RTStrFree(pszWndName);
966
967 rc = VERR_NOT_SUPPORTED;
968 }
969
970 break;
971 }
972
973 case Unknown: /* Mode not set (yet). */
974 case GH:
975 {
976 /*
977 * This message marks the beginning of a new drag and drop
978 * operation on the guest.
979 */
980 if (e.xclient.message_type == xAtom(XA_XdndEnter))
981 {
982 LogFlowFunc(("XA_XdndEnter\n"));
983
984 /*
985 * Get the window which currently has the XA_XdndSelection
986 * bit set.
987 */
988 Window wndSelection = XGetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection));
989
990 char *pszWndName = wndX11GetNameA(wndSelection);
991 AssertPtr(pszWndName);
992 LogFlowThisFunc(("wndSelection=%#x ('%s'), wndProxy=%#x\n", wndSelection, pszWndName, m_wndProxy.hWnd));
993 RTStrFree(pszWndName);
994
995 mouseButtonSet(m_wndProxy.hWnd, -1, -1, 1, true /* fPress */);
996
997 /*
998 * Update our state and the window handle to process.
999 */
1000 int rc2 = RTCritSectEnter(&m_dataCS);
1001 if (RT_SUCCESS(rc2))
1002 {
1003 m_wndCur = wndSelection;
1004 m_curVer = e.xclient.data.l[XdndEnterFlags] >> XdndEnterVersionRShift;
1005 Assert(m_wndCur == (Window)e.xclient.data.l[XdndEnterWindow]); /* Source window. */
1006#ifdef DEBUG
1007 XWindowAttributes xwa;
1008 XGetWindowAttributes(m_pDisplay, m_wndCur, &xwa);
1009 LogFlowThisFunc(("wndCur=%#x, x=%d, y=%d, width=%d, height=%d\n", m_wndCur, xwa.x, xwa.y, xwa.width, xwa.height));
1010#endif
1011 /*
1012 * Retrieve supported formats.
1013 */
1014
1015 /* Check if the MIME types are in the message itself or if we need
1016 * to fetch the XdndTypeList property from the window. */
1017 bool fMoreTypes = e.xclient.data.l[XdndEnterFlags] & XdndEnterMoreTypesFlag;
1018 LogFlowThisFunc(("XdndVer=%d, fMoreTypes=%RTbool\n", m_curVer, fMoreTypes));
1019 if (!fMoreTypes)
1020 {
1021 /* Only up to 3 format types supported. */
1022 /* Start with index 2 (first item). */
1023 for (int i = 2; i < 5; i++)
1024 {
1025 LogFlowThisFunc(("\t%s\n", gX11->xAtomToString(e.xclient.data.l[i]).c_str()));
1026 m_lstFormats.append(e.xclient.data.l[i]);
1027 }
1028 }
1029 else
1030 {
1031 /* More than 3 format types supported. */
1032 rc = wndXDnDGetFormatList(wndSelection, m_lstFormats);
1033 }
1034
1035 /*
1036 * Retrieve supported actions.
1037 */
1038 if (RT_SUCCESS(rc))
1039 {
1040 if (m_curVer >= 2) /* More than one action allowed since protocol version 2. */
1041 {
1042 rc = wndXDnDGetActionList(wndSelection, m_lstActions);
1043 }
1044 else /* Only "copy" action allowed on legacy applications. */
1045 m_lstActions.append(XA_XdndActionCopy);
1046 }
1047
1048 if (RT_SUCCESS(rc))
1049 {
1050 m_enmMode = GH;
1051 m_enmState = Dragging;
1052 }
1053
1054 RTCritSectLeave(&m_dataCS);
1055 }
1056 }
1057 else if ( e.xclient.message_type == xAtom(XA_XdndPosition)
1058 && m_wndCur == static_cast<Window>(e.xclient.data.l[XdndPositionWindow]))
1059 {
1060 if (m_enmState != Dragging) /* Wrong mode? Bail out. */
1061 {
1062 reset();
1063 break;
1064 }
1065
1066 int32_t lPos = e.xclient.data.l[XdndPositionXY];
1067 Atom atmAction = m_curVer >= 2 /* Actions other than "copy" or only supported since protocol version 2. */
1068 ? e.xclient.data.l[XdndPositionAction] : xAtom(XA_XdndActionCopy);
1069
1070 LogFlowThisFunc(("XA_XdndPosition: wndProxy=%#x, wndCur=%#x, x=%RI32, y=%RI32, strAction=%s\n",
1071 m_wndProxy.hWnd, m_wndCur, RT_HIWORD(lPos), RT_LOWORD(lPos),
1072 xAtomToString(atmAction).c_str()));
1073
1074 bool fAcceptDrop = true;
1075
1076 /* Reply with a XdndStatus message to tell the source whether
1077 * the data can be dropped or not. */
1078 XClientMessageEvent m;
1079 RT_ZERO(m);
1080 m.type = ClientMessage;
1081 m.display = m_pDisplay;
1082 m.window = e.xclient.data.l[XdndPositionWindow];
1083 m.message_type = xAtom(XA_XdndStatus);
1084 m.format = 32;
1085 m.data.l[XdndStatusWindow] = m_wndProxy.hWnd;
1086 m.data.l[XdndStatusFlags] = fAcceptDrop ? RT_BIT(0) : 0; /* Whether to accept the drop or not. */
1087
1088 /* We don't want any new XA_XdndPosition messages while being
1089 * in our proxy window. */
1090 m.data.l[XdndStatusNoMsgXY] = RT_MAKE_U32(m_wndProxy.iY, m_wndProxy.iX);
1091 m.data.l[XdndStatusNoMsgWH] = RT_MAKE_U32(m_wndProxy.iHeight, m_wndProxy.iWidth);
1092
1093 /** @todo Handle default action! */
1094 m.data.l[XdndStatusAction] = fAcceptDrop ? toAtomAction(DND_COPY_ACTION) : None;
1095
1096 int xRc = XSendEvent(m_pDisplay, e.xclient.data.l[XdndPositionWindow],
1097 False /* Propagate */, NoEventMask, reinterpret_cast<XEvent *>(&m));
1098 if (xRc == 0)
1099 logError("Error sending position XA_XdndStatus event to current window=%#x: %s\n",
1100 m_wndCur, gX11->xErrorToString(xRc).c_str());
1101 }
1102 else if ( e.xclient.message_type == xAtom(XA_XdndLeave)
1103 && m_wndCur == static_cast<Window>(e.xclient.data.l[XdndLeaveWindow]))
1104 {
1105 LogFlowThisFunc(("XA_XdndLeave\n"));
1106 logInfo("Guest to host transfer canceled by the guest source window\n");
1107
1108 /* Start over. */
1109 reset();
1110 }
1111 else if ( e.xclient.message_type == xAtom(XA_XdndDrop)
1112 && m_wndCur == static_cast<Window>(e.xclient.data.l[XdndDropWindow]))
1113 {
1114 LogFlowThisFunc(("XA_XdndDrop\n"));
1115
1116 if (m_enmState != Dropped) /* Wrong mode? Bail out. */
1117 {
1118 /* Can occur when dragging from guest->host, but then back in to the guest again. */
1119 logInfo("Could not drop on own proxy window\n"); /* Not fatal. */
1120
1121 /* Let the source know. */
1122 rc = m_wndProxy.sendFinished(m_wndCur, DND_IGNORE_ACTION);
1123
1124 /* Start over. */
1125 reset();
1126 break;
1127 }
1128
1129 m_eventQueueList.append(e);
1130 rc = RTSemEventSignal(m_eventQueueEvent);
1131 }
1132 else
1133 {
1134 logInfo("Unhandled event from wnd=%#x, msg=%s\n", e.xclient.window, xAtomToString(e.xclient.message_type).c_str());
1135
1136 /* Let the source know. */
1137 rc = m_wndProxy.sendFinished(m_wndCur, DND_IGNORE_ACTION);
1138
1139 /* Start over. */
1140 reset();
1141 }
1142 break;
1143 }
1144
1145 default:
1146 {
1147 AssertMsgFailed(("Drag and drop mode not implemented: %RU32\n", m_enmMode));
1148 rc = VERR_NOT_IMPLEMENTED;
1149 break;
1150 }
1151 }
1152
1153 LogFlowThisFunc(("Returning rc=%Rrc\n", rc));
1154 return rc;
1155}
1156
1157int DragInstance::onX11MotionNotify(const XEvent &e)
1158{
1159 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1160
1161 return VINF_SUCCESS;
1162}
1163
1164/**
1165 * Callback handler for being notified if some other window now
1166 * is the owner of the current selection.
1167 *
1168 * @return IPRT status code.
1169 * @param e X11 event to handle.
1170 *
1171 * @remark
1172 */
1173int DragInstance::onX11SelectionClear(const XEvent &e)
1174{
1175 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1176
1177 return VINF_SUCCESS;
1178}
1179
1180/**
1181 * Callback handler for a XDnD selection notify from a window. This is needed
1182 * to let the us know if a certain window has drag'n drop data to share with us,
1183 * e.g. our proxy window.
1184 *
1185 * @return IPRT status code.
1186 * @param e X11 event to handle.
1187 */
1188int DragInstance::onX11SelectionNotify(const XEvent &e)
1189{
1190 AssertReturn(e.type == SelectionNotify, VERR_INVALID_PARAMETER);
1191
1192 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1193
1194 int rc;
1195
1196 switch (m_enmMode)
1197 {
1198 case GH:
1199 {
1200 if (m_enmState == Dropped)
1201 {
1202 m_eventQueueList.append(e);
1203 rc = RTSemEventSignal(m_eventQueueEvent);
1204 }
1205 else
1206 rc = VERR_WRONG_ORDER;
1207 break;
1208 }
1209
1210 default:
1211 {
1212 LogFlowThisFunc(("Unhandled: wnd=%#x, msg=%s\n",
1213 e.xclient.data.l[0], xAtomToString(e.xclient.message_type).c_str()));
1214 rc = VERR_INVALID_STATE;
1215 break;
1216 }
1217 }
1218
1219 LogFlowThisFunc(("Returning rc=%Rrc\n", rc));
1220 return rc;
1221}
1222
1223/**
1224 * Callback handler for a XDnD selection request from a window. This is needed
1225 * to retrieve the data required to complete the actual drag'n drop operation.
1226 *
1227 * @returns IPRT status code.
1228 * @param e X11 event to handle.
1229 */
1230int DragInstance::onX11SelectionRequest(const XEvent &e)
1231{
1232 AssertReturn(e.type == SelectionRequest, VERR_INVALID_PARAMETER);
1233
1234 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1235 LogFlowThisFunc(("Event owner=%#x, requestor=%#x, selection=%s, target=%s, prop=%s, time=%u\n",
1236 e.xselectionrequest.owner,
1237 e.xselectionrequest.requestor,
1238 xAtomToString(e.xselectionrequest.selection).c_str(),
1239 xAtomToString(e.xselectionrequest.target).c_str(),
1240 xAtomToString(e.xselectionrequest.property).c_str(),
1241 e.xselectionrequest.time));
1242 int rc;
1243
1244 switch (m_enmMode)
1245 {
1246 case HG:
1247 {
1248 rc = VINF_SUCCESS;
1249
1250 char *pszWndName = wndX11GetNameA(e.xselectionrequest.requestor);
1251 AssertPtr(pszWndName);
1252
1253 /*
1254 * Start by creating a refusal selection notify message.
1255 * That way we only need to care for the success case.
1256 */
1257
1258 XEvent s;
1259 RT_ZERO(s);
1260 s.xselection.type = SelectionNotify;
1261 s.xselection.display = e.xselectionrequest.display;
1262 s.xselection.requestor = e.xselectionrequest.requestor;
1263 s.xselection.selection = e.xselectionrequest.selection;
1264 s.xselection.target = e.xselectionrequest.target;
1265 s.xselection.property = None; /* "None" means refusal. */
1266 s.xselection.time = e.xselectionrequest.time;
1267
1268 const XSelectionRequestEvent *pReq = &e.xselectionrequest;
1269
1270#ifdef DEBUG
1271 LogFlowFunc(("Supported formats:\n"));
1272 for (size_t i = 0; i < m_lstFormats.size(); i++)
1273 LogFlowFunc(("\t%s\n", xAtomToString(m_lstFormats.at(i)).c_str()));
1274#endif
1275 /* Is the requestor asking for the possible MIME types? */
1276 if (pReq->target == xAtom(XA_TARGETS))
1277 {
1278 logInfo("Target window %#x ('%s') asking for target list\n", e.xselectionrequest.requestor, pszWndName);
1279
1280 /* If so, set the window property with the formats on the requestor
1281 * window. */
1282 rc = wndXDnDSetFormatList(pReq->requestor, pReq->property, m_lstFormats);
1283 if (RT_SUCCESS(rc))
1284 s.xselection.property = pReq->property;
1285 }
1286 /* Is the requestor asking for a specific MIME type (we support)? */
1287 else if (m_lstFormats.contains(pReq->target))
1288 {
1289 logInfo("Target window %#x ('%s') is asking for data as '%s'\n",
1290 pReq->requestor, pszWndName, xAtomToString(pReq->target).c_str());
1291
1292 /* Did we not drop our stuff to the guest yet? Bail out. */
1293 if (m_enmState != Dropped)
1294 {
1295 LogFlowThisFunc(("Wrong state (%RU32), refusing request\n", m_enmState));
1296 }
1297 /* Did we not store the requestor's initial selection request yet? Then do so now. */
1298 else
1299 {
1300 /* Get the data format the requestor wants from us. */
1301 RTCString strFormat = xAtomToString(pReq->target);
1302 Assert(strFormat.isNotEmpty());
1303 logInfo("Target window=%#x requested data from host as '%s', rc=%Rrc\n",
1304 pReq->requestor, strFormat.c_str(), rc);
1305
1306 /* Make a copy of the MIME data to be passed back. The X server will be become
1307 * the new owner of that data, so no deletion needed. */
1308 /** @todo Do we need to do some more conversion here? XConvertSelection? */
1309 void *pvData = RTMemDup(m_pvSelReqData, m_cbSelReqData);
1310 uint32_t cbData = m_cbSelReqData;
1311
1312 /* Always return the requested property. */
1313 s.xselection.property = pReq->property;
1314
1315 /* Note: Always seems to return BadRequest. Seems fine. */
1316 int xRc = XChangeProperty(s.xselection.display, s.xselection.requestor, s.xselection.property,
1317 s.xselection.target, 8, PropModeReplace,
1318 reinterpret_cast<const unsigned char*>(pvData), cbData);
1319
1320 LogFlowFunc(("Changing property '%s' (target '%s') of window=0x%x: %s\n",
1321 xAtomToString(pReq->property).c_str(),
1322 xAtomToString(pReq->target).c_str(),
1323 pReq->requestor,
1324 gX11->xErrorToString(xRc).c_str()));
1325 }
1326 }
1327 /* Anything else. */
1328 else
1329 {
1330 logError("Refusing unknown command/format '%s' of wnd=%#x ('%s')\n",
1331 xAtomToString(e.xselectionrequest.target).c_str(), pReq->requestor, pszWndName);
1332 rc = VERR_NOT_SUPPORTED;
1333 }
1334
1335 LogFlowThisFunc(("Offering type '%s', property '%s' to wnd=%#x ...\n",
1336 xAtomToString(pReq->target).c_str(),
1337 xAtomToString(pReq->property).c_str(), pReq->requestor));
1338
1339 int xRc = XSendEvent(pReq->display, pReq->requestor, True /* Propagate */, 0, &s);
1340 if (xRc == 0)
1341 logError("Error sending SelectionNotify(1) event to wnd=%#x: %s\n", pReq->requestor,
1342 gX11->xErrorToString(xRc).c_str());
1343 XFlush(pReq->display);
1344
1345 if (pszWndName)
1346 RTStrFree(pszWndName);
1347 break;
1348 }
1349
1350 default:
1351 rc = VERR_INVALID_STATE;
1352 break;
1353 }
1354
1355 LogFlowThisFunc(("Returning rc=%Rrc\n", rc));
1356 return rc;
1357}
1358
1359/**
1360 * Handles X11 events, called by x11EventThread.
1361 *
1362 * @returns IPRT status code.
1363 * @param e X11 event to handle.
1364 */
1365int DragInstance::onX11Event(const XEvent &e)
1366{
1367 int rc;
1368
1369 LogFlowThisFunc(("X11 event, type=%d\n", e.type));
1370 switch (e.type)
1371 {
1372 /*
1373 * This can happen if a guest->host drag operation
1374 * goes back from the host to the guest. This is not what
1375 * we want and thus resetting everything.
1376 */
1377 case ButtonPress:
1378 case ButtonRelease:
1379 LogFlowThisFunc(("Mouse button press/release\n"));
1380 rc = VINF_SUCCESS;
1381
1382 reset();
1383 break;
1384
1385 case ClientMessage:
1386 rc = onX11ClientMessage(e);
1387 break;
1388
1389 case SelectionClear:
1390 rc = onX11SelectionClear(e);
1391 break;
1392
1393 case SelectionNotify:
1394 rc = onX11SelectionNotify(e);
1395 break;
1396
1397 case SelectionRequest:
1398 rc = onX11SelectionRequest(e);
1399 break;
1400
1401 case MotionNotify:
1402 rc = onX11MotionNotify(e);
1403 break;
1404
1405 default:
1406 rc = VERR_NOT_IMPLEMENTED;
1407 break;
1408 }
1409
1410 LogFlowThisFunc(("rc=%Rrc\n", rc));
1411 return rc;
1412}
1413
1414int DragInstance::waitForStatusChange(uint32_t enmState, RTMSINTERVAL uTimeoutMS /* = 30000 */)
1415{
1416 const uint64_t uiStart = RTTimeMilliTS();
1417 volatile uint32_t enmCurState;
1418
1419 int rc = VERR_TIMEOUT;
1420
1421 LogFlowFunc(("enmState=%RU32, uTimeoutMS=%RU32\n", enmState, uTimeoutMS));
1422
1423 do
1424 {
1425 enmCurState = ASMAtomicReadU32(&m_enmState);
1426 if (enmCurState == enmState)
1427 {
1428 rc = VINF_SUCCESS;
1429 break;
1430 }
1431 }
1432 while (RTTimeMilliTS() - uiStart < uTimeoutMS);
1433
1434 LogFlowThisFunc(("Returning %Rrc\n", rc));
1435 return rc;
1436}
1437
1438#ifdef VBOX_WITH_DRAG_AND_DROP_GH
1439/**
1440 * Waits for an X11 event of a specific type.
1441 *
1442 * @returns IPRT status code.
1443 * @param evX Reference where to store the event into.
1444 * @param iType Event type to wait for.
1445 * @param uTimeoutMS Timeout (in ms) to wait for the event.
1446 */
1447bool DragInstance::waitForX11Msg(XEvent &evX, int iType, RTMSINTERVAL uTimeoutMS /* = 100 */)
1448{
1449 LogFlowThisFunc(("iType=%d, uTimeoutMS=%RU32, cEventQueue=%zu\n", iType, uTimeoutMS, m_eventQueueList.size()));
1450
1451 bool fFound = false;
1452 const uint64_t uiStart = RTTimeMilliTS();
1453
1454 do
1455 {
1456 /* Check if there is a client message in the queue. */
1457 for (size_t i = 0; i < m_eventQueueList.size(); i++)
1458 {
1459 int rc2 = RTCritSectEnter(&m_eventQueueCS);
1460 if (RT_SUCCESS(rc2))
1461 {
1462 XEvent e = m_eventQueueList.at(i);
1463
1464 fFound = e.type == iType;
1465 if (fFound)
1466 {
1467 m_eventQueueList.removeAt(i);
1468 evX = e;
1469 }
1470
1471 rc2 = RTCritSectLeave(&m_eventQueueCS);
1472 AssertRC(rc2);
1473
1474 if (fFound)
1475 break;
1476 }
1477 }
1478
1479 if (fFound)
1480 break;
1481
1482 int rc2 = RTSemEventWait(m_eventQueueEvent, 25 /* ms */);
1483 if ( RT_FAILURE(rc2)
1484 && rc2 != VERR_TIMEOUT)
1485 {
1486 LogFlowFunc(("Waiting failed with rc=%Rrc\n", rc2));
1487 break;
1488 }
1489 }
1490 while (RTTimeMilliTS() - uiStart < uTimeoutMS);
1491
1492 LogFlowThisFunc(("Returning fFound=%RTbool, msRuntime=%RU64\n", fFound, RTTimeMilliTS() - uiStart));
1493 return fFound;
1494}
1495
1496/**
1497 * Waits for an X11 client message of a specific type.
1498 *
1499 * @returns IPRT status code.
1500 * @param evMsg Reference where to store the event into.
1501 * @param aType Event type to wait for.
1502 * @param uTimeoutMS Timeout (in ms) to wait for the event.
1503 */
1504bool DragInstance::waitForX11ClientMsg(XClientMessageEvent &evMsg, Atom aType,
1505 RTMSINTERVAL uTimeoutMS /* = 100 */)
1506{
1507 LogFlowThisFunc(("aType=%s, uTimeoutMS=%RU32, cEventQueue=%zu\n",
1508 xAtomToString(aType).c_str(), uTimeoutMS, m_eventQueueList.size()));
1509
1510 bool fFound = false;
1511 const uint64_t uiStart = RTTimeMilliTS();
1512 do
1513 {
1514 /* Check if there is a client message in the queue. */
1515 for (size_t i = 0; i < m_eventQueueList.size(); i++)
1516 {
1517 int rc2 = RTCritSectEnter(&m_eventQueueCS);
1518 if (RT_SUCCESS(rc2))
1519 {
1520 XEvent e = m_eventQueueList.at(i);
1521 if ( e.type == ClientMessage
1522 && e.xclient.message_type == aType)
1523 {
1524 m_eventQueueList.removeAt(i);
1525 evMsg = e.xclient;
1526
1527 fFound = true;
1528 }
1529
1530 if (e.type == ClientMessage)
1531 {
1532 LogFlowThisFunc(("Client message: Type=%ld (%s)\n",
1533 e.xclient.message_type, xAtomToString(e.xclient.message_type).c_str()));
1534 }
1535 else
1536 LogFlowThisFunc(("X message: Type=%d\n", e.type));
1537
1538 rc2 = RTCritSectLeave(&m_eventQueueCS);
1539 AssertRC(rc2);
1540
1541 if (fFound)
1542 break;
1543 }
1544 }
1545
1546 if (fFound)
1547 break;
1548
1549 int rc2 = RTSemEventWait(m_eventQueueEvent, 25 /* ms */);
1550 if ( RT_FAILURE(rc2)
1551 && rc2 != VERR_TIMEOUT)
1552 {
1553 LogFlowFunc(("Waiting failed with rc=%Rrc\n", rc2));
1554 break;
1555 }
1556 }
1557 while (RTTimeMilliTS() - uiStart < uTimeoutMS);
1558
1559 LogFlowThisFunc(("Returning fFound=%RTbool, msRuntime=%RU64\n", fFound, RTTimeMilliTS() - uiStart));
1560 return fFound;
1561}
1562#endif /* VBOX_WITH_DRAG_AND_DROP_GH */
1563
1564/*
1565 * Host -> Guest
1566 */
1567
1568/**
1569 * Host -> Guest: Event signalling that the host's (mouse) cursor just entered the VM's (guest's) display
1570 * area.
1571 *
1572 * @returns IPRT status code.
1573 * @param lstFormats List of supported formats from the host.
1574 * @param uActions (ORed) List of supported actions from the host.
1575 */
1576int DragInstance::hgEnter(const RTCList<RTCString> &lstFormats, uint32_t uActions)
1577{
1578 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1579
1580 if (m_enmMode != Unknown)
1581 return VERR_INVALID_STATE;
1582
1583 reset();
1584
1585#ifdef DEBUG
1586 LogFlowThisFunc(("uActions=0x%x, lstFormats=%zu: ", uActions, lstFormats.size()));
1587 for (size_t i = 0; i < lstFormats.size(); ++i)
1588 LogFlow(("'%s' ", lstFormats.at(i).c_str()));
1589 LogFlow(("\n"));
1590#endif
1591
1592 int rc;
1593
1594 do
1595 {
1596 rc = toAtomList(lstFormats, m_lstFormats);
1597 if (RT_FAILURE(rc))
1598 break;
1599
1600 /* If we have more than 3 formats we have to use the type list extension. */
1601 if (m_lstFormats.size() > 3)
1602 {
1603 rc = wndXDnDSetFormatList(m_wndProxy.hWnd, xAtom(XA_XdndTypeList), m_lstFormats);
1604 if (RT_FAILURE(rc))
1605 break;
1606 }
1607
1608 /* Announce the possible actions. */
1609 VBoxDnDAtomList lstActions;
1610 rc = toAtomActions(uActions, lstActions);
1611 if (RT_FAILURE(rc))
1612 break;
1613 rc = wndXDnDSetActionList(m_wndProxy.hWnd, lstActions);
1614
1615 /* Set the DnD selection owner to our window. */
1616 /** @todo Don't use CurrentTime -- according to ICCCM section 2.1. */
1617 XSetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection), m_wndProxy.hWnd, CurrentTime);
1618
1619 m_enmMode = HG;
1620 m_enmState = Dragging;
1621
1622 } while (0);
1623
1624 LogFlowFuncLeaveRC(rc);
1625 return rc;
1626}
1627
1628/**
1629 * Host -> Guest: Event signalling that the host's (mouse) cursor has left the VM's (guest's)
1630 * display area.
1631 */
1632int DragInstance::hgLeave(void)
1633{
1634 if (m_enmMode == HG) /* Only reset if in the right operation mode. */
1635 reset();
1636
1637 return VINF_SUCCESS;
1638}
1639
1640/**
1641 * Host -> Guest: Event signalling that the host's (mouse) cursor has been moved within the VM's
1642 * (guest's) display area.
1643 *
1644 * @returns IPRT status code.
1645 * @param u32xPos Relative X position within the guest's display area.
1646 * @param u32yPos Relative Y position within the guest's display area.
1647 * @param uDefaultAction Default action the host wants to perform on the guest
1648 * as soon as the operation successfully finishes.
1649 */
1650int DragInstance::hgMove(uint32_t u32xPos, uint32_t u32yPos, uint32_t uDefaultAction)
1651{
1652 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1653 LogFlowThisFunc(("u32xPos=%RU32, u32yPos=%RU32, uAction=%RU32\n", u32xPos, u32yPos, uDefaultAction));
1654
1655 if ( m_enmMode != HG
1656 || m_enmState != Dragging)
1657 {
1658 return VERR_INVALID_STATE;
1659 }
1660
1661 int rc = VINF_SUCCESS;
1662 int xRc = Success;
1663
1664 /* Move the mouse cursor within the guest. */
1665 mouseCursorMove(u32xPos, u32yPos);
1666
1667 long newVer = -1; /* This means the current window is _not_ XdndAware. */
1668
1669 /* Search for the application window below the cursor. */
1670 Window wndCursor = gX11->applicationWindowBelowCursor(m_wndRoot);
1671 if (wndCursor != None)
1672 {
1673 /* Temp stuff for the XGetWindowProperty call. */
1674 Atom atmp;
1675 int fmt;
1676 unsigned long cItems, cbRemaining;
1677 unsigned char *pcData = NULL;
1678
1679 /* Query the XdndAware property from the window. We are interested in
1680 * the version and if it is XdndAware at all. */
1681 xRc = XGetWindowProperty(m_pDisplay, wndCursor, xAtom(XA_XdndAware),
1682 0, 2, False, AnyPropertyType,
1683 &atmp, &fmt, &cItems, &cbRemaining, &pcData);
1684 if (xRc != Success)
1685 {
1686 logError("Error getting properties of cursor window=%#x: %s\n", wndCursor, gX11->xErrorToString(xRc).c_str());
1687 }
1688 else
1689 {
1690 if (pcData == NULL || fmt != 32 || cItems != 1)
1691 {
1692 /** @todo Do we need to deal with this? */
1693 logError("Wrong window properties for window %#x: pcData=%#x, iFmt=%d, cItems=%ul\n",
1694 wndCursor, pcData, fmt, cItems);
1695 }
1696 else
1697 {
1698 /* Get the current window's Xdnd version. */
1699 newVer = reinterpret_cast<long *>(pcData)[0];
1700 }
1701
1702 XFree(pcData);
1703 }
1704 }
1705
1706#ifdef DEBUG
1707 char *pszNameCursor = wndX11GetNameA(wndCursor);
1708 AssertPtr(pszNameCursor);
1709 char *pszNameCur = wndX11GetNameA(m_wndCur);
1710 AssertPtr(pszNameCur);
1711
1712 LogFlowThisFunc(("wndCursor=%x ('%s', Xdnd version %ld), wndCur=%x ('%s', Xdnd version %ld)\n",
1713 wndCursor, pszNameCursor, newVer, m_wndCur, pszNameCur, m_curVer));
1714
1715 RTStrFree(pszNameCursor);
1716 RTStrFree(pszNameCur);
1717#endif
1718
1719 if ( wndCursor != m_wndCur
1720 && m_curVer != -1)
1721 {
1722 LogFlowThisFunc(("XA_XdndLeave: window=%#x\n", m_wndCur));
1723
1724 char *pszWndName = wndX11GetNameA(m_wndCur);
1725 AssertPtr(pszWndName);
1726 logInfo("Left old window %#x ('%s'), Xdnd version=%ld\n", m_wndCur, pszWndName, newVer);
1727 RTStrFree(pszWndName);
1728
1729 /* We left the current XdndAware window. Announce this to the current indow. */
1730 XClientMessageEvent m;
1731 RT_ZERO(m);
1732 m.type = ClientMessage;
1733 m.display = m_pDisplay;
1734 m.window = m_wndCur;
1735 m.message_type = xAtom(XA_XdndLeave);
1736 m.format = 32;
1737 m.data.l[XdndLeaveWindow] = m_wndProxy.hWnd;
1738
1739 xRc = XSendEvent(m_pDisplay, m_wndCur, False, NoEventMask, reinterpret_cast<XEvent*>(&m));
1740 if (xRc == 0)
1741 logError("Error sending XA_XdndLeave event to old window=%#x: %s\n", m_wndCur, gX11->xErrorToString(xRc).c_str());
1742
1743 /* Reset our current window. */
1744 m_wndCur = 0;
1745 m_curVer = -1;
1746 }
1747
1748 /*
1749 * Do we have a new Xdnd-aware window which now is under the cursor?
1750 */
1751 if ( wndCursor != m_wndCur
1752 && newVer != -1)
1753 {
1754 LogFlowThisFunc(("XA_XdndEnter: window=%#x\n", wndCursor));
1755
1756 char *pszWndName = wndX11GetNameA(wndCursor);
1757 AssertPtr(pszWndName);
1758 logInfo("Entered new window %#x ('%s'), supports Xdnd version=%ld\n", wndCursor, pszWndName, newVer);
1759 RTStrFree(pszWndName);
1760
1761 /*
1762 * We enter a new window. Announce the XdndEnter event to the new
1763 * window. The first three mime types are attached to the event (the
1764 * others could be requested by the XdndTypeList property from the
1765 * window itself).
1766 */
1767 XClientMessageEvent m;
1768 RT_ZERO(m);
1769 m.type = ClientMessage;
1770 m.display = m_pDisplay;
1771 m.window = wndCursor;
1772 m.message_type = xAtom(XA_XdndEnter);
1773 m.format = 32;
1774 m.data.l[XdndEnterWindow] = m_wndProxy.hWnd;
1775 m.data.l[XdndEnterFlags] = RT_MAKE_U32_FROM_U8(
1776 /* Bit 0 is set if the source supports more than three data types. */
1777 m_lstFormats.size() > 3 ? RT_BIT(0) : 0,
1778 /* Reserved for future use. */
1779 0, 0,
1780 /* Protocol version to use. */
1781 RT_MIN(VBOX_XDND_VERSION, newVer));
1782 m.data.l[XdndEnterType1] = m_lstFormats.value(0, None); /* First data type to use. */
1783 m.data.l[XdndEnterType2] = m_lstFormats.value(1, None); /* Second data type to use. */
1784 m.data.l[XdndEnterType3] = m_lstFormats.value(2, None); /* Third data type to use. */
1785
1786 xRc = XSendEvent(m_pDisplay, wndCursor, False, NoEventMask, reinterpret_cast<XEvent*>(&m));
1787 if (xRc == 0)
1788 logError("Error sending XA_XdndEnter event to window=%#x: %s\n", wndCursor, gX11->xErrorToString(xRc).c_str());
1789 }
1790
1791 if (newVer != -1)
1792 {
1793 Assert(wndCursor != None);
1794
1795 LogFlowThisFunc(("XA_XdndPosition: xPos=%RU32, yPos=%RU32 to window=%#x\n", u32xPos, u32yPos, wndCursor));
1796
1797 /*
1798 * Send a XdndPosition event with the proposed action to the guest.
1799 */
1800 Atom pa = toAtomAction(uDefaultAction);
1801 LogFlowThisFunc(("strAction=%s\n", xAtomToString(pa).c_str()));
1802
1803 XClientMessageEvent m;
1804 RT_ZERO(m);
1805 m.type = ClientMessage;
1806 m.display = m_pDisplay;
1807 m.window = wndCursor;
1808 m.message_type = xAtom(XA_XdndPosition);
1809 m.format = 32;
1810 m.data.l[XdndPositionWindow] = m_wndProxy.hWnd; /* X window ID of source window. */
1811 m.data.l[XdndPositionXY] = RT_MAKE_U32(u32yPos, u32xPos); /* Cursor coordinates relative to the root window. */
1812 m.data.l[XdndPositionTimeStamp] = CurrentTime; /* Timestamp for retrieving data. */
1813 m.data.l[XdndPositionAction] = pa; /* Actions requested by the user. */
1814
1815 xRc = XSendEvent(m_pDisplay, wndCursor, False, NoEventMask, reinterpret_cast<XEvent*>(&m));
1816 if (xRc == 0)
1817 logError("Error sending XA_XdndPosition event to current window=%#x: %s\n", wndCursor, gX11->xErrorToString(xRc).c_str());
1818 }
1819
1820 if (newVer == -1)
1821 {
1822 /* No window to process, so send a ignore ack event to the host. */
1823 rc = VbglR3DnDHGSendAckOp(&m_dndCtx, DND_IGNORE_ACTION);
1824 }
1825 else
1826 {
1827 Assert(wndCursor != None);
1828
1829 m_wndCur = wndCursor;
1830 m_curVer = newVer;
1831 }
1832
1833 LogFlowFuncLeaveRC(rc);
1834 return rc;
1835}
1836
1837/**
1838 * Host -> Guest: Event signalling that the host has dropped the data over the VM (guest) window.
1839 *
1840 * @returns IPRT status code.
1841 * @param u32xPos Relative X position within the guest's display area.
1842 * @param u32yPos Relative Y position within the guest's display area.
1843 * @param uDefaultAction Default action the host wants to perform on the guest
1844 * as soon as the operation successfully finishes.
1845 */
1846int DragInstance::hgDrop(uint32_t u32xPos, uint32_t u32yPos, uint32_t uDefaultAction)
1847{
1848 LogFlowThisFunc(("wndCur=%#x, wndProxy=%#x, mode=%RU32, state=%RU32\n", m_wndCur, m_wndProxy.hWnd, m_enmMode, m_enmState));
1849 LogFlowThisFunc(("u32xPos=%RU32, u32yPos=%RU32, uAction=%RU32\n", u32xPos, u32yPos, uDefaultAction));
1850
1851 if ( m_enmMode != HG
1852 || m_enmState != Dragging)
1853 {
1854 return VERR_INVALID_STATE;
1855 }
1856
1857 /* Set the state accordingly. */
1858 m_enmState = Dropped;
1859
1860 /*
1861 * Ask the host to send the raw data, as we don't (yet) know which format
1862 * the guest exactly expects. As blocking in a SelectionRequest message turned
1863 * out to be very unreliable (e.g. with KDE apps) we request to start transferring
1864 * file/directory data (if any) here.
1865 */
1866 char szFormat[] = { "text/uri-list" };
1867
1868 int rc = VbglR3DnDHGSendReqData(&m_dndCtx, szFormat);
1869 logInfo("Drop event from host resuled in: %Rrc\n", rc);
1870
1871 LogFlowFuncLeaveRC(rc);
1872 return rc;
1873}
1874
1875/**
1876 * Host -> Guest: Event signalling that the host has finished sending drag'n drop
1877 * data to the guest for further processing.
1878 *
1879 * @returns IPRT status code.
1880 * @param pvData Pointer to (MIME) data from host.
1881 * @param cbData Size (in bytes) of data from host.
1882 */
1883int DragInstance::hgDataReceived(const void *pvData, uint32_t cbData)
1884{
1885 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1886 LogFlowThisFunc(("pvData=%p, cbData=%RU32\n", pvData, cbData));
1887
1888 if ( m_enmMode != HG
1889 || m_enmState != Dropped)
1890 {
1891 return VERR_INVALID_STATE;
1892 }
1893
1894 if ( pvData == NULL
1895 || cbData == 0)
1896 {
1897 return VERR_INVALID_PARAMETER;
1898 }
1899
1900 int rc = VINF_SUCCESS;
1901
1902 /*
1903 * At this point all data needed (including sent files/directories) should
1904 * be on the guest, so proceed working on communicating with the target window.
1905 */
1906 logInfo("Received %RU32 bytes MIME data from host\n", cbData);
1907
1908 /* Destroy any old data. */
1909 if (m_pvSelReqData)
1910 {
1911 Assert(m_cbSelReqData);
1912
1913 RTMemFree(m_pvSelReqData); /** @todo RTMemRealloc? */
1914 m_cbSelReqData = 0;
1915 }
1916
1917 /** @todo Handle incremental transfers. */
1918
1919 /* Make a copy of the data. This data later then will be used to fill into
1920 * the selection request. */
1921 if (cbData)
1922 {
1923 m_pvSelReqData = RTMemAlloc(cbData);
1924 if (!m_pvSelReqData)
1925 return VERR_NO_MEMORY;
1926
1927 memcpy(m_pvSelReqData, pvData, cbData);
1928 m_cbSelReqData = cbData;
1929 }
1930
1931 /*
1932 * Send a drop event to the current window (target).
1933 * This window in turn then will raise a SelectionRequest message to our proxy window,
1934 * which we will handle in our onX11SelectionRequest handler.
1935 *
1936 * The SelectionRequest will tell us in which format the target wants the data from the host.
1937 */
1938 XClientMessageEvent m;
1939 RT_ZERO(m);
1940 m.type = ClientMessage;
1941 m.display = m_pDisplay;
1942 m.window = m_wndCur;
1943 m.message_type = xAtom(XA_XdndDrop);
1944 m.format = 32;
1945 m.data.l[XdndDropWindow] = m_wndProxy.hWnd; /* Source window. */
1946 m.data.l[XdndDropFlags] = 0; /* Reserved for future use. */
1947 m.data.l[XdndDropTimeStamp] = CurrentTime; /* Our DnD data does not rely on any timing, so just use the current time. */
1948
1949 int xRc = XSendEvent(m_pDisplay, m_wndCur, False /* Propagate */, NoEventMask, reinterpret_cast<XEvent*>(&m));
1950 if (xRc == 0)
1951 logError("Error sending XA_XdndDrop event to window=%#x: %s\n", m_wndCur, gX11->xErrorToString(xRc).c_str());
1952 XFlush(m_pDisplay);
1953
1954 LogFlowFuncLeaveRC(rc);
1955 return rc;
1956}
1957
1958#ifdef VBOX_WITH_DRAG_AND_DROP_GH
1959/**
1960 * Guest -> Host: Event signalling that the host is asking whether there is a pending
1961 * drag event on the guest (to the host).
1962 *
1963 * @returns IPRT status code.
1964 */
1965int DragInstance::ghIsDnDPending(void)
1966{
1967 LogFlowThisFunc(("mode=%RU32, state=%RU32\n", m_enmMode, m_enmState));
1968
1969 int rc;
1970
1971 RTCString strFormats = "\r\n"; /** @todo If empty, IOCTL fails with VERR_ACCESS_DENIED. */
1972 uint32_t uDefAction = DND_IGNORE_ACTION;
1973 uint32_t uAllActions = DND_IGNORE_ACTION;
1974
1975 /* Currently in wrong mode? Bail out. */
1976 if (m_enmMode == HG)
1977 rc = VERR_INVALID_STATE;
1978 /* Message already processed successfully? */
1979 else if ( m_enmMode == GH
1980 && ( m_enmState == Dragging
1981 || m_enmState == Dropped)
1982 )
1983 {
1984 /* No need to query for the source window again. */
1985 rc = VINF_SUCCESS;
1986 }
1987 else
1988 {
1989 /* Determine the current window which currently has the XdndSelection set. */
1990 Window wndSelection = XGetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection));
1991 LogFlowThisFunc(("wndSelection=%#x, wndProxy=%#x, wndCur=%#x\n", wndSelection, m_wndProxy.hWnd, m_wndCur));
1992
1993 /* Is this another window which has a Xdnd selection and not our proxy window? */
1994 if ( wndSelection
1995 && wndSelection != m_wndCur)
1996 {
1997 char *pszWndName = wndX11GetNameA(wndSelection);
1998 AssertPtr(pszWndName);
1999 logInfo("New guest source window %#x ('%s')\n", wndSelection, pszWndName);
2000
2001 /* Start over. */
2002 reset();
2003
2004 /* Map the window on the current cursor position, which should provoke
2005 * an XdndEnter event. */
2006 rc = proxyWinShow();
2007 if (RT_SUCCESS(rc))
2008 {
2009 rc = mouseCursorFakeMove();
2010 if (RT_SUCCESS(rc))
2011 {
2012 bool fWaitFailed = false; /* Waiting for status changed failed? */
2013
2014 /* Wait until we're in "Dragging" state. */
2015 rc = waitForStatusChange(Dragging, 100 /* 100ms timeout */);
2016
2017 /*
2018 * Note: Don't wait too long here, as this mostly will make
2019 * the drag and drop experience on the host being laggy
2020 * and unresponsive.
2021 *
2022 * Instead, let the host query multiple times with 100ms
2023 * timeout each (see above) and only report an error if
2024 * the overall querying time has been exceeded.<
2025 */
2026 if (RT_SUCCESS(rc))
2027 {
2028 m_enmMode = GH;
2029 }
2030 else if (rc == VERR_TIMEOUT)
2031 {
2032 /** @todo Make m_cFailedPendingAttempts configurable. For slower window managers? */
2033 if (m_cFailedPendingAttempts++ > 50) /* Tolerate up to 5s total (100ms for each slot). */
2034 fWaitFailed = true;
2035 else
2036 rc = VINF_SUCCESS;
2037 }
2038 else if (RT_FAILURE(rc))
2039 fWaitFailed = true;
2040
2041 if (fWaitFailed)
2042 {
2043 logError("Error mapping proxy window to guest source window %#x ('%s'), rc=%Rrc\n",
2044 wndSelection, pszWndName, rc);
2045
2046 /* Reset the counter in any case. */
2047 m_cFailedPendingAttempts = 0;
2048 }
2049 }
2050 }
2051
2052 RTStrFree(pszWndName);
2053 }
2054 }
2055
2056 /*
2057 * Acknowledge to the host in any case, regardless
2058 * if something failed here or not. Be responsive.
2059 */
2060
2061 int rc2 = RTCritSectEnter(&m_dataCS);
2062 if (RT_SUCCESS(rc2))
2063 {
2064 RTCString strFormatsCur = gX11->xAtomListToString(m_lstFormats);
2065 if (!strFormatsCur.isEmpty())
2066 {
2067 strFormats = strFormatsCur;
2068 uDefAction = DND_COPY_ACTION; /** @todo Handle default action! */
2069 uAllActions = DND_COPY_ACTION; /** @todo Ditto. */
2070 uAllActions |= toHGCMActions(m_lstActions);
2071 }
2072
2073 RTCritSectLeave(&m_dataCS);
2074 }
2075
2076 rc2 = VbglR3DnDGHSendAckPending(&m_dndCtx, uDefAction, uAllActions,
2077 strFormats.c_str(), strFormats.length() + 1 /* Include termination */);
2078 LogFlowThisFunc(("uClientID=%RU32, uDefAction=0x%x, allActions=0x%x, strFormats=%s, rc=%Rrc\n",
2079 m_dndCtx.uClientID, uDefAction, uAllActions, strFormats.c_str(), rc2));
2080 if (RT_FAILURE(rc2))
2081 {
2082 logError("Error reporting pending drag and drop operation status to host: %Rrc\n", rc2);
2083 if (RT_SUCCESS(rc))
2084 rc = rc2;
2085 }
2086
2087 LogFlowFuncLeaveRC(rc);
2088 return rc;
2089}
2090
2091/**
2092 * Guest -> Host: Event signalling that the host has dropped the item(s) on the
2093 * host side.
2094 *
2095 * @returns IPRT status code.
2096 * @param strFormat Requested format to send to the host.
2097 * @param uAction Requested action to perform on the guest.
2098 */
2099int DragInstance::ghDropped(const RTCString &strFormat, uint32_t uAction)
2100{
2101 LogFlowThisFunc(("mode=%RU32, state=%RU32, strFormat=%s, uAction=%RU32\n",
2102 m_enmMode, m_enmState, strFormat.c_str(), uAction));
2103
2104 /* Currently in wrong mode? Bail out. */
2105 if ( m_enmMode == Unknown
2106 || m_enmMode == HG)
2107 {
2108 return VERR_INVALID_STATE;
2109 }
2110
2111 if ( m_enmMode == GH
2112 && m_enmState != Dragging)
2113 {
2114 return VERR_INVALID_STATE;
2115 }
2116
2117 int rc = VINF_SUCCESS;
2118
2119 m_enmState = Dropped;
2120
2121#ifdef DEBUG
2122 XWindowAttributes xwa;
2123 XGetWindowAttributes(m_pDisplay, m_wndCur, &xwa);
2124 LogFlowThisFunc(("wndProxy=%#x, wndCur=%#x, x=%d, y=%d, width=%d, height=%d\n",
2125 m_wndProxy.hWnd, m_wndCur, xwa.x, xwa.y, xwa.width, xwa.height));
2126
2127 Window wndSelection = XGetSelectionOwner(m_pDisplay, xAtom(XA_XdndSelection));
2128 LogFlowThisFunc(("wndSelection=%#x\n", wndSelection));
2129#endif
2130
2131 /* We send a fake mouse move event to the current window, cause
2132 * this should have the grab. */
2133 mouseCursorFakeMove();
2134
2135 /**
2136 * The fake button release event above should lead to a XdndDrop event from the
2137 * source window. Because of showing our proxy window, other Xdnd events can
2138 * occur before, e.g. a XdndPosition event. We are not interested
2139 * in those, so just try to get the right one.
2140 */
2141
2142 XClientMessageEvent evDnDDrop;
2143 bool fDrop = waitForX11ClientMsg(evDnDDrop, xAtom(XA_XdndDrop), 5 * 1000 /* 5s timeout */);
2144 if (fDrop)
2145 {
2146 LogFlowThisFunc(("XA_XdndDrop\n"));
2147
2148 /* Request to convert the selection in the specific format and
2149 * place it to our proxy window as property. */
2150 Assert(evDnDDrop.message_type == xAtom(XA_XdndDrop));
2151
2152 Window wndSource = evDnDDrop.data.l[XdndDropWindow]; /* Source window which has sent the message. */
2153 Assert(wndSource == m_wndCur);
2154
2155 Atom aFormat = gX11->stringToxAtom(strFormat.c_str());
2156
2157 Time tsDrop;
2158 if (m_curVer >= 1)
2159 tsDrop = evDnDDrop.data.l[XdndDropTimeStamp];
2160 else
2161 tsDrop = CurrentTime;
2162
2163 XConvertSelection(m_pDisplay, xAtom(XA_XdndSelection), aFormat, xAtom(XA_XdndSelection),
2164 m_wndProxy.hWnd, tsDrop);
2165
2166 /* Wait for the selection notify event. */
2167 XEvent evSelNotify;
2168 RT_ZERO(evSelNotify);
2169 if (waitForX11Msg(evSelNotify, SelectionNotify, 5 * 1000 /* 5s timeout */))
2170 {
2171 bool fCancel = false;
2172
2173 /* Make some paranoid checks. */
2174 if ( evSelNotify.xselection.type == SelectionNotify
2175 && evSelNotify.xselection.display == m_pDisplay
2176 && evSelNotify.xselection.selection == xAtom(XA_XdndSelection)
2177 && evSelNotify.xselection.requestor == m_wndProxy.hWnd
2178 && evSelNotify.xselection.target == aFormat)
2179 {
2180 LogFlowThisFunc(("Selection notfiy (from wnd=%#x)\n", m_wndCur));
2181
2182 Atom aPropType;
2183 int iPropFormat;
2184 unsigned long cItems, cbRemaining;
2185 unsigned char *pcData = NULL;
2186 int xRc = XGetWindowProperty(m_pDisplay, m_wndProxy.hWnd,
2187 xAtom(XA_XdndSelection) /* Property */,
2188 0 /* Offset */,
2189 VBOX_MAX_XPROPERTIES /* Length of 32-bit multiples */,
2190 True /* Delete property? */,
2191 AnyPropertyType, /* Property type */
2192 &aPropType, &iPropFormat, &cItems, &cbRemaining, &pcData);
2193 if (xRc != Success)
2194 logError("Error getting XA_XdndSelection property of proxy window=%#x: %s\n",
2195 m_wndProxy.hWnd, gX11->xErrorToString(xRc).c_str());
2196
2197 LogFlowThisFunc(("strType=%s, iPropFormat=%d, cItems=%RU32, cbRemaining=%RU32\n",
2198 gX11->xAtomToString(aPropType).c_str(), iPropFormat, cItems, cbRemaining));
2199
2200 if ( aPropType != None
2201 && pcData != NULL
2202 && iPropFormat >= 8
2203 && cItems > 0
2204 && cbRemaining == 0)
2205 {
2206 size_t cbData = cItems * (iPropFormat / 8);
2207 LogFlowThisFunc(("cbData=%zu\n", cbData));
2208
2209 /* For whatever reason some of the string MIME types are not
2210 * zero terminated. Check that and correct it when necessary,
2211 * because the guest side wants this in any case. */
2212 if ( m_lstAllowedFormats.contains(strFormat)
2213 && pcData[cbData - 1] != '\0')
2214 {
2215 unsigned char *pvDataTmp = static_cast<unsigned char*>(RTMemAlloc(cbData + 1));
2216 if (pvDataTmp)
2217 {
2218 memcpy(pvDataTmp, pcData, cbData);
2219 pvDataTmp[cbData++] = '\0';
2220
2221 rc = VbglR3DnDGHSendData(&m_dndCtx, strFormat.c_str(), pvDataTmp, cbData);
2222 RTMemFree(pvDataTmp);
2223 }
2224 else
2225 rc = VERR_NO_MEMORY;
2226 }
2227 else
2228 {
2229 /* Send the raw data to the host. */
2230 rc = VbglR3DnDGHSendData(&m_dndCtx, strFormat.c_str(), pcData, cbData);
2231 LogFlowThisFunc(("Sent strFormat=%s, rc=%Rrc\n", strFormat.c_str(), rc));
2232 }
2233
2234 if (RT_SUCCESS(rc))
2235 {
2236 rc = m_wndProxy.sendFinished(wndSource, uAction);
2237 }
2238 else
2239 fCancel = true;
2240 }
2241 else
2242 {
2243 if (aPropType == xAtom(XA_INCR))
2244 {
2245 /** @todo Support incremental transfers. */
2246 AssertMsgFailed(("Incremental transfers are not supported yet\n"));
2247
2248 logError("Incremental transfers are not supported yet\n");
2249 rc = VERR_NOT_IMPLEMENTED;
2250 }
2251 else
2252 {
2253 logError("Not supported data type: %s\n", gX11->xAtomToString(aPropType).c_str());
2254 rc = VERR_NOT_SUPPORTED;
2255 }
2256
2257 fCancel = true;
2258 }
2259
2260 if (fCancel)
2261 {
2262 logInfo("Cancelling dropping to host\n");
2263
2264 /* Cancel the operation -- inform the source window by
2265 * sending a XdndFinished message so that the source can toss the required data. */
2266 rc = m_wndProxy.sendFinished(wndSource, DND_IGNORE_ACTION);
2267 }
2268
2269 /* Cleanup. */
2270 if (pcData)
2271 XFree(pcData);
2272 }
2273 else
2274 rc = VERR_INVALID_PARAMETER;
2275 }
2276 else
2277 rc = VERR_TIMEOUT;
2278 }
2279 else
2280 rc = VERR_TIMEOUT;
2281
2282 /* Inform the host on error. */
2283 if (RT_FAILURE(rc))
2284 {
2285 int rc2 = VbglR3DnDGHSendError(&m_dndCtx, rc);
2286 LogFlowThisFunc(("Sending error to host resulted in %Rrc\n", rc2));
2287 /* This is not fatal for us, just ignore. */
2288 }
2289
2290 /* At this point, we have either successfully transfered any data or not.
2291 * So reset our internal state because we are done here for the current (ongoing)
2292 * drag and drop operation. */
2293 reset();
2294
2295 LogFlowFuncLeaveRC(rc);
2296 return rc;
2297}
2298#endif /* VBOX_WITH_DRAG_AND_DROP_GH */
2299
2300/*
2301 * Helpers
2302 */
2303
2304/**
2305 * Fakes moving the mouse cursor to provoke various drag and drop
2306 * events such as entering a target window or moving within a
2307 * source window.
2308 *
2309 * Not the most elegant and probably correct function, but does
2310 * the work for now.
2311 *
2312 * @returns IPRT status code.
2313 */
2314int DragInstance::mouseCursorFakeMove(void) const
2315{
2316 int iScreenID = XDefaultScreen(m_pDisplay);
2317 /** @todo What about multiple screens? Test this! */
2318
2319 const int iScrX = XDisplayWidth(m_pDisplay, iScreenID);
2320 const int iScrY = XDisplayHeight(m_pDisplay, iScreenID);
2321
2322 int fx, fy, rx, ry;
2323 Window wndTemp, wndChild;
2324 int wx, wy; unsigned int mask;
2325 XQueryPointer(m_pDisplay, m_wndRoot, &wndTemp, &wndChild, &rx, &ry, &wx, &wy, &mask);
2326
2327 /*
2328 * Apply some simple clipping and change the position slightly.
2329 */
2330
2331 /* FakeX */
2332 if (rx == 0) fx = 1;
2333 else if (rx == iScrX) fx = iScrX - 1;
2334 else fx = rx + 1;
2335
2336 /* FakeY */
2337 if (ry == 0) fy = 1;
2338 else if (ry == iScrY) fy = iScrY - 1;
2339 else fy = ry + 1;
2340
2341 /*
2342 * Move the cursor to trigger the wanted events.
2343 */
2344 LogFlowThisFunc(("cursorRootX=%d, cursorRootY=%d\n", fx, fy));
2345 int rc = mouseCursorMove(fx, fy);
2346 if (RT_SUCCESS(rc))
2347 {
2348 /* Move the cursor back to its original position. */
2349 rc = mouseCursorMove(rx, ry);
2350 }
2351
2352 return rc;
2353}
2354
2355/**
2356 * Moves the mouse pointer to a specific position.
2357 *
2358 * @returns IPRT status code.
2359 * @param iPosX Absolute X coordinate.
2360 * @param iPosY Absolute Y coordinate.
2361 */
2362int DragInstance::mouseCursorMove(int iPosX, int iPosY) const
2363{
2364 int iScreenID = XDefaultScreen(m_pDisplay);
2365 /** @todo What about multiple screens? Test this! */
2366
2367 const int iScrX = XDisplayWidth(m_pDisplay, iScreenID);
2368 const int iScrY = XDisplayHeight(m_pDisplay, iScreenID);
2369
2370 iPosX = RT_CLAMP(iPosX, 0, iScrX);
2371 iPosY = RT_CLAMP(iPosY, 0, iScrY);
2372
2373 LogFlowThisFunc(("iPosX=%d, iPosY=%d\n", iPosX, iPosY));
2374
2375 /* Move the guest pointer to the DnD position, so we can find the window
2376 * below that position. */
2377 XWarpPointer(m_pDisplay, None, m_wndRoot, 0, 0, 0, 0, iPosX, iPosY);
2378 return VINF_SUCCESS;
2379}
2380
2381/**
2382 * Sends a mouse button event to a specific window.
2383 *
2384 * @param wndDest Window to send the mouse button event to.
2385 * @param rx X coordinate relative to the root window's origin.
2386 * @param ry Y coordinate relative to the root window's origin.
2387 * @param iButton Mouse button to press/release.
2388 * @param fPress Whether to press or release the mouse button.
2389 */
2390void DragInstance::mouseButtonSet(Window wndDest, int rx, int ry, int iButton, bool fPress)
2391{
2392 LogFlowThisFunc(("wndDest=%#x, rx=%d, ry=%d, iBtn=%d, fPress=%RTbool\n",
2393 wndDest, rx, ry, iButton, fPress));
2394
2395#ifdef VBOX_DND_WITH_XTEST
2396 /** @todo Make this check run only once. */
2397 int ev, er, ma, mi;
2398 if (XTestQueryExtension(m_pDisplay, &ev, &er, &ma, &mi))
2399 {
2400 LogFlowThisFunc(("XText extension available\n"));
2401
2402 int xRc = XTestFakeButtonEvent(m_pDisplay, 1, fPress ? True : False, CurrentTime);
2403 if (Rc == 0)
2404 logError("Error sending XTestFakeButtonEvent event: %s\n", gX11->xErrorToString(xRc).c_str());
2405 XFlush(m_pDisplay);
2406 }
2407 else
2408 {
2409#endif
2410 LogFlowThisFunc(("Note: XText extension not available or disabled\n"));
2411
2412 unsigned int mask = 0;
2413
2414 if ( rx == -1
2415 && ry == -1)
2416 {
2417 Window wndRoot, wndChild;
2418 int wx, wy;
2419 XQueryPointer(m_pDisplay, m_wndRoot, &wndRoot, &wndChild, &rx, &ry, &wx, &wy, &mask);
2420 LogFlowThisFunc(("Mouse pointer is at root x=%d, y=%d\n", rx, ry));
2421 }
2422
2423 XButtonEvent eBtn;
2424 RT_ZERO(eBtn);
2425
2426 eBtn.display = m_pDisplay;
2427 eBtn.root = m_wndRoot;
2428 eBtn.window = wndDest;
2429 eBtn.subwindow = None;
2430 eBtn.same_screen = True;
2431 eBtn.time = CurrentTime;
2432 eBtn.button = iButton;
2433 eBtn.state = mask | (iButton == 1 ? Button1MotionMask :
2434 iButton == 2 ? Button2MotionMask :
2435 iButton == 3 ? Button3MotionMask :
2436 iButton == 4 ? Button4MotionMask :
2437 iButton == 5 ? Button5MotionMask : 0);
2438 eBtn.type = fPress ? ButtonPress : ButtonRelease;
2439 eBtn.send_event = False;
2440 eBtn.x_root = rx;
2441 eBtn.y_root = ry;
2442
2443 XTranslateCoordinates(m_pDisplay, eBtn.root, eBtn.window, eBtn.x_root, eBtn.y_root, &eBtn.x, &eBtn.y, &eBtn.subwindow);
2444 LogFlowThisFunc(("state=0x%x, x=%d, y=%d\n", eBtn.state, eBtn.x, eBtn.y));
2445
2446 int xRc = XSendEvent(m_pDisplay, wndDest, True /* fPropagate */,
2447 ButtonPressMask,
2448 reinterpret_cast<XEvent*>(&eBtn));
2449 if (xRc == 0)
2450 logError("Error sending XButtonEvent event to window=%#x: %s\n", wndDest, gX11->xErrorToString(xRc).c_str());
2451
2452 XFlush(m_pDisplay);
2453
2454#ifdef VBOX_DND_WITH_XTEST
2455 }
2456#endif
2457}
2458
2459/**
2460 * Shows the (invisible) proxy window. The proxy window is needed for intercepting
2461 * drags from the host to the guest or from the guest to the host. It acts as a proxy
2462 * between the host and the actual (UI) element on the guest OS.
2463 *
2464 * To not make it miss any actions this window gets spawned across the entire guest
2465 * screen (think of an umbrella) to (hopefully) capture everything. A proxy window
2466 * which follows the cursor would be far too slow here.
2467 *
2468 * @returns IPRT status code.
2469 * @param piRootX X coordinate relative to the root window's origin. Optional.
2470 * @param piRootY Y coordinate relative to the root window's origin. Optional.
2471 */
2472int DragInstance::proxyWinShow(int *piRootX /* = NULL */, int *piRootY /* = NULL */) const
2473{
2474 /* piRootX is optional. */
2475 /* piRootY is optional. */
2476
2477 LogFlowThisFuncEnter();
2478
2479 int rc = VINF_SUCCESS;
2480
2481#if 0
2482# ifdef VBOX_DND_WITH_XTEST
2483 XTestGrabControl(m_pDisplay, False);
2484# endif
2485#endif
2486
2487 /* Get the mouse pointer position and determine if we're on the same screen as the root window
2488 * and return the current child window beneath our mouse pointer, if any. */
2489 int iRootX, iRootY;
2490 int iChildX, iChildY;
2491 unsigned int iMask;
2492 Window wndRoot, wndChild;
2493 Bool fInRootWnd = XQueryPointer(m_pDisplay, m_wndRoot, &wndRoot, &wndChild,
2494 &iRootX, &iRootY, &iChildX, &iChildY, &iMask);
2495
2496 LogFlowThisFunc(("fInRootWnd=%RTbool, wndRoot=0x%x, wndChild=0x%x, iRootX=%d, iRootY=%d\n",
2497 RT_BOOL(fInRootWnd), wndRoot, wndChild, iRootX, iRootY));
2498
2499 if (piRootX)
2500 *piRootX = iRootX;
2501 if (piRootY)
2502 *piRootY = iRootY;
2503
2504 XSynchronize(m_pDisplay, True /* Enable sync */);
2505
2506 /* Bring our proxy window into foreground. */
2507 XMapWindow(m_pDisplay, m_wndProxy.hWnd);
2508 XRaiseWindow(m_pDisplay, m_wndProxy.hWnd);
2509
2510 /* Spawn our proxy window over the entire screen, making it an easy drop target for the host's cursor. */
2511 LogFlowThisFunc(("Proxy window x=%d, y=%d, width=%d, height=%d\n",
2512 m_wndProxy.iX, m_wndProxy.iY, m_wndProxy.iWidth, m_wndProxy.iHeight));
2513 XMoveResizeWindow(m_pDisplay, m_wndProxy.hWnd, m_wndProxy.iX, m_wndProxy.iY, m_wndProxy.iWidth, m_wndProxy.iHeight);
2514
2515 XFlush(m_pDisplay);
2516
2517 XSynchronize(m_pDisplay, False /* Disable sync */);
2518
2519#if 0
2520# ifdef VBOX_DND_WITH_XTEST
2521 XTestGrabControl(m_pDisplay, True);
2522# endif
2523#endif
2524
2525 LogFlowFuncLeaveRC(rc);
2526 return rc;
2527}
2528
2529/**
2530 * Hides the (invisible) proxy window.
2531 */
2532int DragInstance::proxyWinHide(void)
2533{
2534 LogFlowFuncEnter();
2535
2536 XUnmapWindow(m_pDisplay, m_wndProxy.hWnd);
2537 XFlush(m_pDisplay);
2538
2539 m_eventQueueList.clear();
2540
2541 return VINF_SUCCESS; /** @todo Add error checking. */
2542}
2543
2544/**
2545 * Allocates the name (title) of an X window.
2546 * The returned pointer must be freed using RTStrFree().
2547 *
2548 * @returns Pointer to the allocated window name.
2549 * @param wndThis Window to retrieve name for.
2550 *
2551 * @remark If the window title is not available, the text
2552 * "<No name>" will be returned.
2553 */
2554char *DragInstance::wndX11GetNameA(Window wndThis) const
2555{
2556 char *pszName = NULL;
2557
2558 XTextProperty propName;
2559 if (XGetWMName(m_pDisplay, wndThis, &propName))
2560 {
2561 if (propName.value)
2562 pszName = RTStrDup((char *)propName.value); /** @todo UTF8? */
2563 XFree(propName.value);
2564 }
2565
2566 if (!pszName) /* No window name found? */
2567 pszName = RTStrDup("<No name>");
2568
2569 return pszName;
2570}
2571
2572/**
2573 * Clear a window's supported/accepted actions list.
2574 *
2575 * @param wndThis Window to clear the list for.
2576 */
2577void DragInstance::wndXDnDClearActionList(Window wndThis) const
2578{
2579 XDeleteProperty(m_pDisplay, wndThis, xAtom(XA_XdndActionList));
2580}
2581
2582/**
2583 * Clear a window's supported/accepted formats list.
2584 *
2585 * @param wndThis Window to clear the list for.
2586 */
2587void DragInstance::wndXDnDClearFormatList(Window wndThis) const
2588{
2589 XDeleteProperty(m_pDisplay, wndThis, xAtom(XA_XdndTypeList));
2590}
2591
2592/**
2593 * Retrieves a window's supported/accepted XDnD actions.
2594 *
2595 * @returns IPRT status code.
2596 * @param wndThis Window to retrieve the XDnD actions for.
2597 * @param lstActions Reference to VBoxDnDAtomList to store the action into.
2598 */
2599int DragInstance::wndXDnDGetActionList(Window wndThis, VBoxDnDAtomList &lstActions) const
2600{
2601 Atom iActType = None;
2602 int iActFmt;
2603 unsigned long cItems, cbData;
2604 unsigned char *pcbData = NULL;
2605
2606 /* Fetch the possible list of actions, if this property is set. */
2607 int xRc = XGetWindowProperty(m_pDisplay, wndThis,
2608 xAtom(XA_XdndActionList),
2609 0, VBOX_MAX_XPROPERTIES,
2610 False, XA_ATOM, &iActType, &iActFmt, &cItems, &cbData, &pcbData);
2611 if (xRc != Success)
2612 {
2613 LogFlowThisFunc(("Error getting XA_XdndActionList atoms from window=%#x: %s\n",
2614 wndThis, gX11->xErrorToString(xRc).c_str()));
2615 return VERR_NOT_FOUND;
2616 }
2617
2618 LogFlowThisFunc(("wndThis=%#x, cItems=%RU32, pcbData=%p\n", wndThis, cItems, pcbData));
2619
2620 if (cItems > 0)
2621 {
2622 AssertPtr(pcbData);
2623 Atom *paData = reinterpret_cast<Atom *>(pcbData);
2624
2625 for (unsigned i = 0; i < RT_MIN(VBOX_MAX_XPROPERTIES, cItems); i++)
2626 {
2627 LogFlowThisFunc(("\t%s\n", gX11->xAtomToString(paData[i]).c_str()));
2628 lstActions.append(paData[i]);
2629 }
2630
2631 XFree(pcbData);
2632 }
2633
2634 return VINF_SUCCESS;
2635}
2636
2637/**
2638 * Retrieves a window's supported/accepted XDnD formats.
2639 *
2640 * @returns IPRT status code.
2641 * @param wndThis Window to retrieve the XDnD formats for.
2642 * @param lstTypes Reference to VBoxDnDAtomList to store the formats into.
2643 */
2644int DragInstance::wndXDnDGetFormatList(Window wndThis, VBoxDnDAtomList &lstTypes) const
2645{
2646 Atom iActType = None;
2647 int iActFmt;
2648 unsigned long cItems, cbData;
2649 unsigned char *pcbData = NULL;
2650
2651 int xRc = XGetWindowProperty(m_pDisplay, wndThis,
2652 xAtom(XA_XdndTypeList),
2653 0, VBOX_MAX_XPROPERTIES,
2654 False, XA_ATOM, &iActType, &iActFmt, &cItems, &cbData, &pcbData);
2655 if (xRc != Success)
2656 {
2657 LogFlowThisFunc(("Error getting XA_XdndTypeList atoms from window=%#x: %s\n",
2658 wndThis, gX11->xErrorToString(xRc).c_str()));
2659 return VERR_NOT_FOUND;
2660 }
2661
2662 LogFlowThisFunc(("wndThis=%#x, cItems=%RU32, pcbData=%p\n", wndThis, cItems, pcbData));
2663
2664 if (cItems > 0)
2665 {
2666 AssertPtr(pcbData);
2667 Atom *paData = reinterpret_cast<Atom *>(pcbData);
2668
2669 for (unsigned i = 0; i < RT_MIN(VBOX_MAX_XPROPERTIES, cItems); i++)
2670 {
2671 LogFlowThisFunc(("\t%s\n", gX11->xAtomToString(paData[i]).c_str()));
2672 lstTypes.append(paData[i]);
2673 }
2674
2675 XFree(pcbData);
2676 }
2677
2678 return VINF_SUCCESS;
2679}
2680
2681/**
2682 * Sets (replaces) a window's XDnD accepted/allowed actions.
2683 *
2684 * @returns IPRT status code.
2685 * @param wndThis Window to set the format list for.
2686 * @param lstActions Reference to list of XDnD actions to set.
2687 *
2688 * @remark
2689 */
2690int DragInstance::wndXDnDSetActionList(Window wndThis, const VBoxDnDAtomList &lstActions) const
2691{
2692 if (lstActions.isEmpty())
2693 return VINF_SUCCESS;
2694
2695 XChangeProperty(m_pDisplay, wndThis,
2696 xAtom(XA_XdndActionList),
2697 XA_ATOM, 32, PropModeReplace,
2698 reinterpret_cast<const unsigned char*>(lstActions.raw()),
2699 lstActions.size());
2700
2701 return VINF_SUCCESS;
2702}
2703
2704/**
2705 * Sets (replaces) a window's XDnD accepted format list.
2706 *
2707 * @returns IPRT status code.
2708 * @param wndThis Window to set the format list for.
2709 * @param atmProp Property to set.
2710 * @param lstFormats Reference to list of XDnD formats to set.
2711 */
2712int DragInstance::wndXDnDSetFormatList(Window wndThis, Atom atmProp, const VBoxDnDAtomList &lstFormats) const
2713{
2714 if (lstFormats.isEmpty())
2715 return VERR_INVALID_PARAMETER;
2716
2717 /* We support TARGETS and the data types. */
2718 VBoxDnDAtomList lstFormatsExt(lstFormats.size() + 1);
2719 lstFormatsExt.append(xAtom(XA_TARGETS));
2720 lstFormatsExt.append(lstFormats);
2721
2722 /* Add the property with the property data to the window. */
2723 XChangeProperty(m_pDisplay, wndThis, atmProp,
2724 XA_ATOM, 32, PropModeReplace,
2725 reinterpret_cast<const unsigned char*>(lstFormatsExt.raw()),
2726 lstFormatsExt.size());
2727
2728 return VINF_SUCCESS;
2729}
2730
2731/**
2732 * Converts a RTCString list to VBoxDnDAtomList list.
2733 *
2734 * @returns IPRT status code.
2735 * @param lstFormats Reference to RTCString list to convert.
2736 * @param lstAtoms Reference to VBoxDnDAtomList list to store results in.
2737 */
2738int DragInstance::toAtomList(const RTCList<RTCString> &lstFormats, VBoxDnDAtomList &lstAtoms) const
2739{
2740 for (size_t i = 0; i < lstFormats.size(); ++i)
2741 lstAtoms.append(XInternAtom(m_pDisplay, lstFormats.at(i).c_str(), False));
2742
2743 return VINF_SUCCESS;
2744}
2745
2746/**
2747 * Converts a raw-data string list to VBoxDnDAtomList list.
2748 *
2749 * @returns IPRT status code.
2750 * @param pvData Pointer to string data to convert.
2751 * @param cbData Size (in bytes) to convert.
2752 * @param lstAtoms Reference to VBoxDnDAtomList list to store results in.
2753 */
2754int DragInstance::toAtomList(const void *pvData, uint32_t cbData, VBoxDnDAtomList &lstAtoms) const
2755{
2756 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
2757 AssertReturn(cbData, VERR_INVALID_PARAMETER);
2758
2759 const char *pszStr = (char *)pvData;
2760 uint32_t cbStr = cbData;
2761
2762 int rc = VINF_SUCCESS;
2763
2764 VBoxDnDAtomList lstAtom;
2765 while (cbStr)
2766 {
2767 size_t cbSize = RTStrNLen(pszStr, cbStr);
2768
2769 /* Create a copy with max N chars, so that we are on the save side,
2770 * even if the data isn't zero terminated. */
2771 char *pszTmp = RTStrDupN(pszStr, cbSize);
2772 if (!pszTmp)
2773 {
2774 rc = VERR_NO_MEMORY;
2775 break;
2776 }
2777
2778 lstAtom.append(XInternAtom(m_pDisplay, pszTmp, False));
2779 RTStrFree(pszTmp);
2780
2781 pszStr += cbSize + 1;
2782 cbStr -= cbSize + 1;
2783 }
2784
2785 return rc;
2786}
2787
2788/**
2789 * Converts a HGCM-based drag'n drop action to a Atom-based drag'n drop action.
2790 *
2791 * @returns Converted Atom-based drag'n drop action.
2792 * @param uActions HGCM drag'n drop actions to convert.
2793 */
2794/* static */
2795Atom DragInstance::toAtomAction(uint32_t uAction)
2796{
2797 /* Ignore is None. */
2798 return (isDnDCopyAction(uAction) ? xAtom(XA_XdndActionCopy) :
2799 isDnDMoveAction(uAction) ? xAtom(XA_XdndActionMove) :
2800 isDnDLinkAction(uAction) ? xAtom(XA_XdndActionLink) :
2801 None);
2802}
2803
2804/**
2805 * Converts HGCM-based drag'n drop actions to a VBoxDnDAtomList list.
2806 *
2807 * @returns IPRT status code.
2808 * @param uActions HGCM drag'n drop actions to convert.
2809 * @param lstAtoms Reference to VBoxDnDAtomList to store actions in.
2810 */
2811/* static */
2812int DragInstance::toAtomActions(uint32_t uActions, VBoxDnDAtomList &lstAtoms)
2813{
2814 if (hasDnDCopyAction(uActions))
2815 lstAtoms.append(xAtom(XA_XdndActionCopy));
2816 if (hasDnDMoveAction(uActions))
2817 lstAtoms.append(xAtom(XA_XdndActionMove));
2818 if (hasDnDLinkAction(uActions))
2819 lstAtoms.append(xAtom(XA_XdndActionLink));
2820
2821 return VINF_SUCCESS;
2822}
2823
2824/**
2825 * Converts an Atom-based drag'n drop action to a HGCM drag'n drop action.
2826 *
2827 * @returns HGCM drag'n drop action.
2828 * @param atom Atom-based drag'n drop action to convert.
2829 */
2830/* static */
2831uint32_t DragInstance::toHGCMAction(Atom atom)
2832{
2833 uint32_t uAction = DND_IGNORE_ACTION;
2834
2835 if (atom == xAtom(XA_XdndActionCopy))
2836 uAction = DND_COPY_ACTION;
2837 else if (atom == xAtom(XA_XdndActionMove))
2838 uAction = DND_MOVE_ACTION;
2839 else if (atom == xAtom(XA_XdndActionLink))
2840 uAction = DND_LINK_ACTION;
2841
2842 return uAction;
2843}
2844
2845/**
2846 * Converts an VBoxDnDAtomList list to an HGCM action list.
2847 *
2848 * @returns ORed HGCM action list.
2849 * @param actionsList List of Atom-based actions to convert.
2850 */
2851/* static */
2852uint32_t DragInstance::toHGCMActions(const VBoxDnDAtomList &lstActions)
2853{
2854 uint32_t uActions = DND_IGNORE_ACTION;
2855
2856 for (size_t i = 0; i < lstActions.size(); i++)
2857 uActions |= toHGCMAction(lstActions.at(i));
2858
2859 return uActions;
2860}
2861
2862/*******************************************************************************
2863 * VBoxDnDProxyWnd implementation.
2864 ******************************************************************************/
2865
2866VBoxDnDProxyWnd::VBoxDnDProxyWnd(void)
2867 : pDisp(NULL)
2868 , hWnd(0)
2869 , iX(0)
2870 , iY(0)
2871 , iWidth(0)
2872 , iHeight(0)
2873{
2874
2875}
2876
2877VBoxDnDProxyWnd::~VBoxDnDProxyWnd(void)
2878{
2879 destroy();
2880}
2881
2882int VBoxDnDProxyWnd::init(Display *pDisplay)
2883{
2884 /** @todo What about multiple screens? Test this! */
2885 int iScreenID = XDefaultScreen(pDisplay);
2886
2887 iWidth = XDisplayWidth(pDisplay, iScreenID);
2888 iHeight = XDisplayHeight(pDisplay, iScreenID);
2889 pDisp = pDisplay;
2890
2891 return VINF_SUCCESS;
2892}
2893
2894void VBoxDnDProxyWnd::destroy(void)
2895{
2896
2897}
2898
2899int VBoxDnDProxyWnd::sendFinished(Window hWndSource, uint32_t uAction)
2900{
2901 /* Was the drop accepted by the host? That is, anything than ignoring. */
2902 bool fDropAccepted = uAction > DND_IGNORE_ACTION;
2903
2904 LogFlowFunc(("uAction=%RU32\n", uAction));
2905
2906 /* Confirm the result of the transfer to the target window. */
2907 XClientMessageEvent m;
2908 RT_ZERO(m);
2909 m.type = ClientMessage;
2910 m.display = pDisp;
2911 m.window = hWnd;
2912 m.message_type = xAtom(XA_XdndFinished);
2913 m.format = 32;
2914 m.data.l[XdndFinishedWindow] = hWnd; /* Target window. */
2915 m.data.l[XdndFinishedFlags] = fDropAccepted ? RT_BIT(0) : 0; /* Was the drop accepted? */
2916 m.data.l[XdndFinishedAction] = fDropAccepted ? DragInstance::toAtomAction(uAction) : None; /* Action used on accept. */
2917
2918 int xRc = XSendEvent(pDisp, hWndSource, True, NoEventMask, reinterpret_cast<XEvent*>(&m));
2919 if (xRc == 0)
2920 {
2921 LogRel(("DnD: Error sending XA_XdndFinished event to source window=%#x: %s\n",
2922 hWndSource, gX11->xErrorToString(xRc).c_str()));
2923
2924 return VERR_GENERAL_FAILURE; /** @todo Fudge. */
2925 }
2926
2927 return VINF_SUCCESS;
2928}
2929
2930/*******************************************************************************
2931 * DragAndDropService implementation.
2932 ******************************************************************************/
2933
2934/**
2935 * Main loop for the drag and drop service which does the HGCM message
2936 * processing and routing to the according drag and drop instance(s).
2937 *
2938 * @returns IPRT status code.
2939 * @param fDaemonised Whether to run in daemonized or not. Does not
2940 * apply for this service.
2941 */
2942int DragAndDropService::run(bool fDaemonised /* = false */)
2943{
2944 LogFlowThisFunc(("fDaemonised=%RTbool\n", fDaemonised));
2945
2946 int rc;
2947 do
2948 {
2949 /* Initialize drag and drop. */
2950 rc = dragAndDropInit();
2951 if (RT_FAILURE(rc))
2952 break;
2953
2954 m_pCurDnD = new DragInstance(m_pDisplay, this);
2955 if (!m_pCurDnD)
2956 {
2957 rc = VERR_NO_MEMORY;
2958 break;
2959 }
2960
2961 /* Note: For multiple screen support in VBox it is not necessary to use
2962 * another screen number than zero. Maybe in the future it will become
2963 * necessary if VBox supports multiple X11 screens. */
2964 rc = m_pCurDnD->init(0);
2965 if (RT_FAILURE(rc))
2966 break;
2967
2968 LogRel(("DnD: Started\n"));
2969 LogRel2(("DnD: %sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr()));
2970
2971 /* Enter the main event processing loop. */
2972 do
2973 {
2974 DnDEvent e;
2975 RT_ZERO(e);
2976
2977 LogFlowFunc(("Waiting for new event ...\n"));
2978 rc = RTSemEventWait(m_hEventSem, RT_INDEFINITE_WAIT);
2979 if (RT_FAILURE(rc))
2980 break;
2981
2982 AssertMsg(m_eventQueue.size(),
2983 ("Event queue is empty when it shouldn't\n"));
2984
2985 e = m_eventQueue.first();
2986 m_eventQueue.removeFirst();
2987
2988 if (e.type == DnDEvent::HGCM_Type)
2989 {
2990 LogFlowThisFunc(("HGCM event, type=%RU32\n", e.hgcm.uType));
2991 switch (e.hgcm.uType)
2992 {
2993 case DragAndDropSvc::HOST_DND_HG_EVT_ENTER:
2994 {
2995 if (e.hgcm.cbFormats)
2996 {
2997 RTCList<RTCString> lstFormats = RTCString(e.hgcm.pszFormats, e.hgcm.cbFormats - 1).split("\r\n");
2998 rc = m_pCurDnD->hgEnter(lstFormats, e.hgcm.u.a.uAllActions);
2999 /* Enter is always followed by a move event. */
3000 }
3001 else
3002 {
3003 rc = VERR_INVALID_PARAMETER;
3004 break;
3005 }
3006 /* Not breaking unconditionally is intentional. See comment above. */
3007 }
3008 case DragAndDropSvc::HOST_DND_HG_EVT_MOVE:
3009 {
3010 rc = m_pCurDnD->hgMove(e.hgcm.u.a.uXpos, e.hgcm.u.a.uYpos, e.hgcm.u.a.uDefAction);
3011 break;
3012 }
3013 case DragAndDropSvc::HOST_DND_HG_EVT_LEAVE:
3014 {
3015 rc = m_pCurDnD->hgLeave();
3016 break;
3017 }
3018 case DragAndDropSvc::HOST_DND_HG_EVT_DROPPED:
3019 {
3020 rc = m_pCurDnD->hgDrop(e.hgcm.u.a.uXpos, e.hgcm.u.a.uYpos, e.hgcm.u.a.uDefAction);
3021 break;
3022 }
3023 case DragAndDropSvc::HOST_DND_HG_SND_DATA:
3024 {
3025 rc = m_pCurDnD->hgDataReceived(e.hgcm.u.b.pvData, e.hgcm.u.b.cbData);
3026 break;
3027 }
3028#ifdef VBOX_WITH_DRAG_AND_DROP_GH
3029 case DragAndDropSvc::HOST_DND_GH_REQ_PENDING:
3030 {
3031 rc = m_pCurDnD->ghIsDnDPending();
3032 break;
3033 }
3034 case DragAndDropSvc::HOST_DND_GH_EVT_DROPPED:
3035 {
3036 rc = m_pCurDnD->ghDropped(e.hgcm.pszFormats, e.hgcm.u.a.uDefAction);
3037 break;
3038 }
3039#endif
3040 default:
3041 {
3042 m_pCurDnD->logError("Received unsupported message: %RU32\n", e.hgcm.uType);
3043 rc = VERR_NOT_SUPPORTED;
3044 break;
3045 }
3046 }
3047
3048 LogFlowFunc(("Message %RU32 processed with %Rrc\n", e.hgcm.uType, rc));
3049 if (RT_FAILURE(rc))
3050 {
3051 /* Tell the user. */
3052 m_pCurDnD->logError("Error processing message %RU32, failed with %Rrc, resetting all\n", e.hgcm.uType, rc);
3053
3054 /* If anything went wrong, do a reset and start over. */
3055 m_pCurDnD->reset();
3056 }
3057
3058 /* Some messages require cleanup. */
3059 switch (e.hgcm.uType)
3060 {
3061 case DragAndDropSvc::HOST_DND_HG_EVT_ENTER:
3062 case DragAndDropSvc::HOST_DND_HG_EVT_MOVE:
3063 case DragAndDropSvc::HOST_DND_HG_EVT_DROPPED:
3064#ifdef VBOX_WITH_DRAG_AND_DROP_GH
3065 case DragAndDropSvc::HOST_DND_GH_EVT_DROPPED:
3066#endif
3067 {
3068 if (e.hgcm.pszFormats)
3069 RTMemFree(e.hgcm.pszFormats);
3070 break;
3071 }
3072
3073 case DragAndDropSvc::HOST_DND_HG_SND_DATA:
3074 {
3075 if (e.hgcm.pszFormats)
3076 RTMemFree(e.hgcm.pszFormats);
3077 if (e.hgcm.u.b.pvData)
3078 RTMemFree(e.hgcm.u.b.pvData);
3079 break;
3080 }
3081
3082 default:
3083 break;
3084 }
3085 }
3086 else if (e.type == DnDEvent::X11_Type)
3087 {
3088 m_pCurDnD->onX11Event(e.x11);
3089 }
3090 else
3091 AssertMsgFailed(("Unknown event queue type %d\n", e.type));
3092
3093 /*
3094 * Make sure that any X11 requests have actually been sent to the
3095 * server, since we are waiting for responses using poll() on
3096 * another thread which will not automatically trigger flushing.
3097 */
3098 XFlush(m_pDisplay);
3099
3100 } while (!ASMAtomicReadBool(&m_fSrvStopping));
3101
3102 } while (0);
3103
3104 LogRel(("DnD: Stopped with rc=%Rrc\n", rc));
3105 return rc;
3106}
3107
3108/**
3109 * Initializes the drag and drop instance.
3110 *
3111 * @returns IPRT status code.
3112 */
3113int DragAndDropService::dragAndDropInit(void)
3114{
3115 /* Initialise the guest library. */
3116 int rc = VbglR3InitUser();
3117 if (RT_FAILURE(rc))
3118 VBClFatalError(("DnD: Failed to connect to the VirtualBox kernel service, rc=%Rrc\n", rc));
3119
3120 /* Connect to the x11 server. */
3121 m_pDisplay = XOpenDisplay(NULL);
3122 if (!m_pDisplay)
3123 {
3124 VBClFatalError(("DnD: Unable to connect to X server -- running in a terminal session?\n"));
3125 return VERR_NOT_FOUND;
3126 }
3127
3128 xHelpers *pHelpers = xHelpers::getInstance(m_pDisplay);
3129 if (!pHelpers)
3130 return VERR_NO_MEMORY;
3131
3132 do
3133 {
3134 rc = RTSemEventCreate(&m_hEventSem);
3135 if (RT_FAILURE(rc))
3136 break;
3137
3138 rc = RTCritSectInit(&m_eventQueueCS);
3139 if (RT_FAILURE(rc))
3140 break;
3141
3142 /* Event thread for events coming from the HGCM device. */
3143 rc = RTThreadCreate(&m_hHGCMThread, hgcmEventThread, this,
3144 0, RTTHREADTYPE_MSG_PUMP, RTTHREADFLAGS_WAITABLE,
3145 "dndHGCM");
3146 if (RT_FAILURE(rc))
3147 break;
3148
3149 /* Event thread for events coming from the x11 system. */
3150 rc = RTThreadCreate(&m_hX11Thread, x11EventThread, this,
3151 0, RTTHREADTYPE_MSG_PUMP, RTTHREADFLAGS_WAITABLE,
3152 "dndX11");
3153 } while (0);
3154
3155 /* No clean-up code for now, as we have no good way of testing it and things
3156 * should get cleaned up when the user process/X11 client exits. */
3157 if (RT_FAILURE(rc))
3158 LogRel(("DnD: Failed to start, rc=%Rrc\n", rc));
3159
3160 return rc;
3161}
3162
3163/**
3164 * Static callback function for HGCM message processing thread. An internal
3165 * message queue will be filled which then will be processed by the according
3166 * drag'n drop instance.
3167 *
3168 * @returns IPRT status code.
3169 * @param hThread Thread handle to use.
3170 * @param pvUser Pointer to DragAndDropService instance to use.
3171 */
3172/* static */
3173DECLCALLBACK(int) DragAndDropService::hgcmEventThread(RTTHREAD hThread, void *pvUser)
3174{
3175 AssertPtrReturn(pvUser, VERR_INVALID_PARAMETER);
3176 DragAndDropService *pThis = static_cast<DragAndDropService*>(pvUser);
3177 AssertPtr(pThis);
3178
3179 /* This thread has an own DnD context, e.g. an own client ID. */
3180 VBGLR3GUESTDNDCMDCTX dndCtx;
3181
3182 int rc = VbglR3DnDConnect(&dndCtx);
3183 if (RT_FAILURE(rc))
3184 LogRel(("DnD: Unable to connect to drag and drop service, rc=%Rrc\n", rc));
3185 /* Not RT_FAILURE: VINF_PERMISSION_DENIED is host service not present. */
3186 if (rc != VINF_SUCCESS)
3187 return rc;
3188
3189 /* Number of invalid messages skipped in a row. */
3190 int cMsgSkippedInvalid = 0;
3191 DnDEvent e;
3192
3193 do
3194 {
3195 RT_ZERO(e);
3196 e.type = DnDEvent::HGCM_Type;
3197
3198 /* Wait for new events. */
3199 rc = VbglR3DnDRecvNextMsg(&dndCtx, &e.hgcm);
3200 if ( RT_SUCCESS(rc)
3201 || rc == VERR_CANCELLED)
3202 {
3203 cMsgSkippedInvalid = 0; /* Reset skipped messages count. */
3204 pThis->m_eventQueue.append(e);
3205
3206 rc = RTSemEventSignal(pThis->m_hEventSem);
3207 if (RT_FAILURE(rc))
3208 break;
3209 }
3210 else
3211 {
3212 LogRel(("DnD: Processing next message failed with rc=%Rrc\n", rc));
3213
3214 /* Old(er) hosts either are broken regarding DnD support or otherwise
3215 * don't support the stuff we do on the guest side, so make sure we
3216 * don't process invalid messages forever. */
3217 if (rc == VERR_INVALID_PARAMETER)
3218 cMsgSkippedInvalid++;
3219 if (cMsgSkippedInvalid > 32)
3220 {
3221 LogRel(("DnD: Too many invalid/skipped messages from host, exiting ...\n"));
3222 break;
3223 }
3224 }
3225
3226 } while (!ASMAtomicReadBool(&pThis->m_fSrvStopping));
3227
3228 VbglR3DnDDisconnect(&dndCtx);
3229
3230 LogFlowFuncLeaveRC(rc);
3231 return rc;
3232}
3233
3234/**
3235 * Static callback function for X11 message processing thread. All X11 messages
3236 * will be directly routed to the according drag'n drop instance.
3237 *
3238 * @returns IPRT status code.
3239 * @param hThread Thread handle to use.
3240 * @param pvUser Pointer to DragAndDropService instance to use.
3241 */
3242/* static */
3243DECLCALLBACK(int) DragAndDropService::x11EventThread(RTTHREAD hThread, void *pvUser)
3244{
3245 AssertPtrReturn(pvUser, VERR_INVALID_PARAMETER);
3246 DragAndDropService *pThis = static_cast<DragAndDropService*>(pvUser);
3247 AssertPtr(pThis);
3248
3249 int rc = VINF_SUCCESS;
3250
3251 DnDEvent e;
3252 do
3253 {
3254 /*
3255 * Wait for new events. We can't use XIfEvent here, cause this locks
3256 * the window connection with a mutex and if no X11 events occurs this
3257 * blocks any other calls we made to X11. So instead check for new
3258 * events and if there are not any new one, sleep for a certain amount
3259 * of time.
3260 */
3261 if (XEventsQueued(pThis->m_pDisplay, QueuedAfterFlush) > 0)
3262 {
3263 RT_ZERO(e);
3264 e.type = DnDEvent::X11_Type;
3265
3266 /* XNextEvent will block until a new X event becomes available. */
3267 XNextEvent(pThis->m_pDisplay, &e.x11);
3268 {
3269#ifdef DEBUG
3270 switch (e.x11.type)
3271 {
3272 case ClientMessage:
3273 {
3274 XClientMessageEvent *pEvent = reinterpret_cast<XClientMessageEvent*>(&e);
3275 AssertPtr(pEvent);
3276
3277 RTCString strType = xAtomToString(pEvent->message_type);
3278 LogFlowFunc(("ClientMessage: %s from wnd=%#x\n", strType.c_str(), pEvent->window));
3279 break;
3280 }
3281
3282 default:
3283 LogFlowFunc(("Received X event type=%d\n", e.x11.type));
3284 break;
3285 }
3286#endif
3287 /* At the moment we only have one drag instance. */
3288 DragInstance *pInstance = pThis->m_pCurDnD;
3289 AssertPtr(pInstance);
3290
3291 pInstance->onX11Event(e.x11);
3292 }
3293 }
3294 else
3295 RTThreadSleep(25 /* ms */);
3296
3297 } while (!ASMAtomicReadBool(&pThis->m_fSrvStopping));
3298
3299 LogFlowFuncLeaveRC(rc);
3300 return rc;
3301}
3302
3303/** Drag and drop magic number, start of a UUID. */
3304#define DRAGANDDROPSERVICE_MAGIC 0x67c97173
3305
3306/** VBoxClient service class wrapping the logic for the service while
3307 * the main VBoxClient code provides the daemon logic needed by all services.
3308 */
3309struct DRAGANDDROPSERVICE
3310{
3311 /** The service interface. */
3312 struct VBCLSERVICE *pInterface;
3313 /** Magic number for sanity checks. */
3314 uint32_t uMagic;
3315 /** Service object. */
3316 DragAndDropService mDragAndDrop;
3317};
3318
3319static const char *getPidFilePath()
3320{
3321 return ".vboxclient-draganddrop.pid";
3322}
3323
3324static int run(struct VBCLSERVICE **ppInterface, bool fDaemonised)
3325{
3326 struct DRAGANDDROPSERVICE *pSelf = (struct DRAGANDDROPSERVICE *)ppInterface;
3327
3328 if (pSelf->uMagic != DRAGANDDROPSERVICE_MAGIC)
3329 VBClFatalError(("Bad display service object!\n"));
3330 return pSelf->mDragAndDrop.run(fDaemonised);
3331}
3332
3333static void cleanup(struct VBCLSERVICE **ppInterface)
3334{
3335 NOREF(ppInterface);
3336 VbglR3Term();
3337}
3338
3339struct VBCLSERVICE vbclDragAndDropInterface =
3340{
3341 getPidFilePath,
3342 VBClServiceDefaultHandler, /* init */
3343 run,
3344 cleanup
3345};
3346
3347/* Static factory. */
3348struct VBCLSERVICE **VBClGetDragAndDropService(void)
3349{
3350 struct DRAGANDDROPSERVICE *pService =
3351 (struct DRAGANDDROPSERVICE *)RTMemAlloc(sizeof(*pService));
3352
3353 if (!pService)
3354 VBClFatalError(("Out of memory\n"));
3355 pService->pInterface = &vbclDragAndDropInterface;
3356 pService->uMagic = DRAGANDDROPSERVICE_MAGIC;
3357 new(&pService->mDragAndDrop) DragAndDropService();
3358 return &pService->pInterface;
3359}
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