VirtualBox

source: vbox/trunk/src/VBox/Main/include/GuestDnDPrivate.h@ 56720

Last change on this file since 56720 was 56653, checked in by vboxsync, 10 years ago

DnD: Some renaming, documentation, logging.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.9 KB
Line 
1/* $Id: GuestDnDPrivate.h 56653 2015-06-26 08:31:29Z vboxsync $ */
2/** @file
3 * Private guest drag and drop code, used by GuestDnDTarget +
4 * GuestDnDSource.
5 */
6
7/*
8 * Copyright (C) 2011-2015 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#ifndef ____H_GUESTDNDPRIVATE
20#define ____H_GUESTDNDPRIVATE
21
22#include <iprt/dir.h>
23#include <iprt/file.h>
24
25#include "VBox/hgcmsvc.h" /* For PVBOXHGCMSVCPARM. */
26#include "VBox/GuestHost/DragAndDrop.h"
27
28/**
29 * Forward prototype declarations.
30 */
31class Guest;
32class GuestDnDBase;
33class GuestDnDResponse;
34class GuestDnDSource;
35class GuestDnDTarget;
36class Progress;
37
38class GuestDnDCallbackEvent
39{
40public:
41
42 GuestDnDCallbackEvent(void)
43 : mSemEvent(NIL_RTSEMEVENT)
44 , mRc(VINF_SUCCESS) { }
45
46 virtual ~GuestDnDCallbackEvent(void);
47
48public:
49
50 int Reset(void);
51
52 int Notify(int rc = VINF_SUCCESS);
53
54 int Result(void) const { return mRc; }
55
56 int Wait(RTMSINTERVAL msTimeout);
57
58protected:
59
60 /** Event semaphore to notify on error/completion. */
61 RTSEMEVENT mSemEvent;
62 /** Callback result. */
63 int mRc;
64};
65
66/**
67 * Structure for keeping the (URI) data to be sent/received.
68 */
69typedef struct GuestDnDData
70{
71 GuestDnDData(void)
72 : cbToProcess(0)
73 , cbProcessed(0) { }
74
75 void Reset(void)
76 {
77 vecData.clear();
78 cbToProcess = 0;
79 cbProcessed = 0;
80 }
81
82 /** Array (vector) of guest DnD data. This might be an URI list, according
83 * to the format being set. */
84 std::vector<BYTE> vecData;
85 /** Overall size (in bytes) of data to send. */
86 uint64_t cbToProcess;
87 /** Overall size (in bytes) of processed file data. */
88 uint64_t cbProcessed;
89
90} GuestDnDData;
91
92/**
93 * Structure for keeping around URI (list) data.
94 */
95typedef struct GuestDnDURIData
96{
97 GuestDnDURIData(void)
98 : pvScratchBuf(NULL)
99 , cbScratchBuf(0) { }
100
101 virtual ~GuestDnDURIData(void)
102 {
103 Reset();
104 }
105
106 void Reset(void)
107 {
108 lstURI.Clear();
109#if 0 /* Currently the scratch buffer will be maintained elswewhere. */
110 if (pvScratchBuf)
111 {
112 RTMemFree(pvScratchBuf);
113 pvScratchBuf = NULL;
114 }
115 cbScratchBuf = 0;
116#else
117 pvScratchBuf = NULL;
118 cbScratchBuf = 0;
119#endif
120 }
121
122 DNDDIRDROPPEDFILES mDropDir;
123 /** (Non-recursive) List of root URI objects to receive. */
124 DnDURIList lstURI;
125 /** Current object to receive. */
126 DnDURIObject objURI;
127 /** Pointer to an optional scratch buffer to use for
128 * doing the actual chunk transfers. */
129 void *pvScratchBuf;
130 /** Size (in bytes) of scratch buffer. */
131 size_t cbScratchBuf;
132
133} GuestDnDURIData;
134
135/**
136 * Context structure for sending data to the guest.
137 */
138typedef struct SENDDATACTX
139{
140 /** Pointer to guest target class this context belongs to. */
141 GuestDnDTarget *mpTarget;
142 /** Pointer to guest response class this context belongs to. */
143 GuestDnDResponse *mpResp;
144 /** Flag indicating whether a file transfer is active and
145 * initiated by the host. */
146 bool mIsActive;
147 /** Target (VM) screen ID. */
148 uint32_t mScreenID;
149 /** Drag'n drop format requested by the guest. */
150 com::Utf8Str mFmtReq;
151 /** Drag'n drop data to send.
152 * This can be arbitrary data or an URI list. */
153 GuestDnDData mData;
154 /** URI data structure. */
155 GuestDnDURIData mURI;
156 /** Callback event to use. */
157 GuestDnDCallbackEvent mCallback;
158
159} SENDDATACTX, *PSENDDATACTX;
160
161/**
162 * Context structure for receiving data from the guest.
163 */
164typedef struct RECVDATACTX
165{
166 /** Pointer to guest source class this context belongs to. */
167 GuestDnDSource *mpSource;
168 /** Pointer to guest response class this context belongs to. */
169 GuestDnDResponse *mpResp;
170 /** Flag indicating whether a file transfer is active and
171 * initiated by the host. */
172 bool mIsActive;
173 /** Drag'n drop format to send. */
174 com::Utf8Str mFormat;
175 /** Desired drop action to perform on the host.
176 * Needed to tell the guest if data has to be
177 * deleted e.g. when moving instead of copying. */
178 uint32_t mAction;
179 /** Drag'n drop received from the guest.
180 * This can be arbitrary data or an URI list. */
181 GuestDnDData mData;
182 /** URI data structure. */
183 GuestDnDURIData mURI;
184 /** Callback event to use. */
185 GuestDnDCallbackEvent mCallback;
186
187} RECVDATACTX, *PRECVDATACTX;
188
189/**
190 * Simple structure for a buffered guest DnD message.
191 */
192class GuestDnDMsg
193{
194public:
195
196 GuestDnDMsg(void)
197 : uMsg(0)
198 , cParms(0)
199 , cParmsAlloc(0)
200 , paParms(NULL) { }
201
202 virtual ~GuestDnDMsg(void)
203 {
204 if (paParms)
205 {
206 /* Remove deep copies. */
207 for (uint32_t i = 0; i < cParms; i++)
208 {
209 if ( paParms[i].type == VBOX_HGCM_SVC_PARM_PTR
210 && paParms[i].u.pointer.size)
211 {
212 AssertPtr(paParms[i].u.pointer.addr);
213 RTMemFree(paParms[i].u.pointer.addr);
214 }
215 }
216
217 RTMemFree(paParms);
218 }
219 }
220
221public:
222
223 PVBOXHGCMSVCPARM getNextParam(void)
224 {
225 if (cParms >= cParmsAlloc)
226 {
227 paParms = (PVBOXHGCMSVCPARM)RTMemRealloc(paParms, (cParmsAlloc + 4) * sizeof(VBOXHGCMSVCPARM));
228 if (!paParms)
229 throw VERR_NO_MEMORY;
230 RT_BZERO(&paParms[cParmsAlloc], 4 * sizeof(VBOXHGCMSVCPARM));
231 cParmsAlloc += 4;
232 }
233
234 return &paParms[cParms++];
235 }
236
237 uint32_t getCount(void) const { return cParms; }
238 PVBOXHGCMSVCPARM getParms(void) const { return paParms; }
239 uint32_t getType(void) const { return uMsg; }
240
241 int setNextPointer(void *pvBuf, uint32_t cbBuf)
242 {
243 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
244 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
245
246 PVBOXHGCMSVCPARM pParm = getNextParam();
247 if (!pParm)
248 return VERR_NO_MEMORY;
249
250 void *pvTmp = RTMemDup(pvBuf, cbBuf);
251 if (!pvTmp)
252 {
253 RTMemFree(pParm);
254 return VERR_NO_MEMORY;
255 }
256
257 pParm->setPointer(pvTmp, cbBuf);
258 return VINF_SUCCESS;
259 }
260
261 int setNextString(const char *pszString)
262 {
263 PVBOXHGCMSVCPARM pParm = getNextParam();
264 if (!pParm)
265 return VERR_NO_MEMORY;
266
267 char *pszTemp = RTStrDup(pszString);
268 if (!pszTemp)
269 {
270 RTMemFree(pParm);
271 return VERR_NO_MEMORY;
272 }
273
274 pParm->setString(pszTemp);
275 return VINF_SUCCESS;
276 }
277
278 int setNextUInt32(uint32_t u32Val)
279 {
280 PVBOXHGCMSVCPARM pParm = getNextParam();
281 if (!pParm)
282 return VERR_NO_MEMORY;
283
284 pParm->setUInt32(u32Val);
285 return VINF_SUCCESS;
286 }
287
288 int setNextUInt64(uint64_t u64Val)
289 {
290 PVBOXHGCMSVCPARM pParm = getNextParam();
291 if (!pParm)
292 return VERR_NO_MEMORY;
293
294 pParm->setUInt64(u64Val);
295 return VINF_SUCCESS;
296 }
297
298 void setType(uint32_t uMsgType) { uMsg = uMsgType; }
299
300protected:
301
302 /** Message type. */
303 uint32_t uMsg;
304 /** Message parameters. */
305 uint32_t cParms;
306 /** Size of array. */
307 uint32_t cParmsAlloc;
308 /** Array of HGCM parameters */
309 PVBOXHGCMSVCPARM paParms;
310};
311
312/** Guest DnD callback function definition. */
313typedef DECLCALLBACKPTR(int, PFNGUESTDNDCALLBACK) (uint32_t uMsg, void *pvParms, size_t cbParms, void *pvUser);
314
315/**
316 * Structure for keeping a guest DnD callback.
317 * Each callback can handle one HGCM message, however, multiple HGCM messages can be registered
318 * to the same callback (function).
319 */
320typedef struct GuestDnDCallback
321{
322 GuestDnDCallback(void)
323 : uMessgage(0)
324 , pfnCallback(NULL)
325 , pvUser(NULL) { }
326
327 GuestDnDCallback(PFNGUESTDNDCALLBACK pvCB, uint32_t uMsg, void *pvUsr = NULL)
328 : uMessgage(uMsg)
329 , pfnCallback(pvCB)
330 , pvUser(pvUsr) { }
331
332 /** The HGCM message ID to handle. */
333 uint32_t uMessgage;
334 /** Pointer to callback function. */
335 PFNGUESTDNDCALLBACK pfnCallback;
336 /** Pointer to user-supplied data. */
337 void *pvUser;
338
339} GuestDnDCallback;
340
341/** Contains registered callback pointers for specific HGCM message types. */
342typedef std::map<uint32_t, GuestDnDCallback> GuestDnDCallbackMap;
343
344class GuestDnDResponse
345{
346
347public:
348
349 GuestDnDResponse(const ComObjPtr<Guest>& pGuest);
350 virtual ~GuestDnDResponse(void);
351
352public:
353
354 int notifyAboutGuestResponse(void) const;
355 int waitForGuestResponse(RTMSINTERVAL msTimeout = 500) const;
356
357 void setAllActions(uint32_t a) { m_allActions = a; }
358 uint32_t allActions(void) const { return m_allActions; }
359
360 void setDefAction(uint32_t a) { m_defAction = a; }
361 uint32_t defAction(void) const { return m_defAction; }
362
363 void setFmtReq(const Utf8Str &strFormat) { m_strFmtReq = strFormat; }
364 Utf8Str fmtReq(void) const { return m_strFmtReq; }
365
366 void reset(void);
367
368 bool isProgressCanceled(void) const;
369 int setCallback(uint32_t uMsg, PFNGUESTDNDCALLBACK pfnCallback, void *pvUser = NULL);
370 int setProgress(unsigned uPercentage, uint32_t uState, int rcOp = VINF_SUCCESS, const Utf8Str &strMsg = "");
371 HRESULT resetProgress(const ComObjPtr<Guest>& pParent);
372 HRESULT queryProgressTo(IProgress **ppProgress);
373
374public:
375
376 /** @name HGCM callback handling.
377 @{ */
378 int onDispatch(uint32_t u32Function, void *pvParms, uint32_t cbParms);
379 /** @} */
380
381protected:
382
383 /** Pointer to context this class is tied to. */
384 void *m_pvCtx;
385 /** Event for waiting for response. */
386 RTSEMEVENT m_EventSem;
387 /** Default action to perform in case of a
388 * successful drop. */
389 uint32_t m_defAction;
390 /** Actions supported by the guest in case of
391 * a successful drop. */
392 uint32_t m_allActions;
393 /** Format requested by the guest. */
394 Utf8Str m_strFmtReq;
395 /** Pointer to IGuest parent object. */
396 ComObjPtr<Guest> m_parent;
397 /** Pointer to associated progress object. Optional. */
398 ComObjPtr<Progress> m_progress;
399 /** Callback map. */
400 GuestDnDCallbackMap m_mapCallbacks;
401};
402
403/**
404 * Private singleton class for the guest's DnD
405 * implementation. Can't be instanciated directly, only via
406 * the factory pattern.
407 */
408class GuestDnD
409{
410public:
411
412 static GuestDnD *createInstance(const ComObjPtr<Guest>& pGuest)
413 {
414 Assert(NULL == GuestDnD::s_pInstance);
415 GuestDnD::s_pInstance = new GuestDnD(pGuest);
416 return GuestDnD::s_pInstance;
417 }
418
419 static void destroyInstance(void)
420 {
421 if (GuestDnD::s_pInstance)
422 {
423 delete GuestDnD::s_pInstance;
424 GuestDnD::s_pInstance = NULL;
425 }
426 }
427
428 static inline GuestDnD *getInstance(void)
429 {
430 AssertPtr(GuestDnD::s_pInstance);
431 return GuestDnD::s_pInstance;
432 }
433
434protected:
435
436 GuestDnD(const ComObjPtr<Guest>& pGuest);
437 virtual ~GuestDnD(void);
438
439public:
440
441 /** @name Public helper functions.
442 * @{ */
443 int adjustScreenCoordinates(ULONG uScreenId, ULONG *puX, ULONG *puY) const;
444 int hostCall(uint32_t u32Function, uint32_t cParms, PVBOXHGCMSVCPARM paParms) const;
445 GuestDnDResponse *response(void) { return m_pResponse; }
446 std::vector<com::Utf8Str> defaultFormats(void) const { return m_strDefaultFormats; }
447 /** @} */
448
449public:
450
451 /** @name Static low-level HGCM callback handler.
452 * @{ */
453 static DECLCALLBACK(int) notifyDnDDispatcher(void *pvExtension, uint32_t u32Function, void *pvParms, uint32_t cbParms);
454 /** @} */
455
456 /** @name Static helper methods.
457 * @{ */
458 static com::Utf8Str toFormatString(const std::vector<com::Utf8Str> &lstSupportedFormats, const std::vector<com::Utf8Str> &lstWantedFormats);
459 static void toFormatVector(const std::vector<com::Utf8Str> &lstSupportedFormats, const com::Utf8Str &strFormats, std::vector<com::Utf8Str> &vecformats);
460 static DnDAction_T toMainAction(uint32_t uAction);
461 static void toMainActions(uint32_t uActions, std::vector<DnDAction_T> &vecActions);
462 static uint32_t toHGCMAction(DnDAction_T enmAction);
463 static void toHGCMActions(DnDAction_T enmDefAction, uint32_t *puDefAction, const std::vector<DnDAction_T> vecAllowedActions, uint32_t *puAllowedActions);
464 /** @} */
465
466protected:
467
468 /** @name Singleton properties.
469 * @{ */
470 /** List of supported default MIME/Content-type formats. */
471 std::vector<com::Utf8Str> m_strDefaultFormats;
472 /** Pointer to guest implementation. */
473 const ComObjPtr<Guest> m_pGuest;
474 /** The current (last) response from the guest. At the
475 * moment we only support only response a time (ARQ-style). */
476 GuestDnDResponse *m_pResponse;
477 /** @} */
478
479private:
480
481 /** Staic pointer to singleton instance. */
482 static GuestDnD *s_pInstance;
483};
484
485/** Access to the GuestDnD's singleton instance. */
486#define GuestDnDInst() GuestDnD::getInstance()
487
488/** List of pointers to guest DnD Messages. */
489typedef std::list<GuestDnDMsg *> GuestDnDMsgList;
490
491/**
492 * IDnDBase class implementation for sharing code between
493 * IGuestDnDSource and IGuestDnDTarget implementation.
494 */
495class GuestDnDBase
496{
497protected:
498
499 GuestDnDBase(void);
500
501protected:
502
503 /** Shared (internal) IDnDBase method implementations.
504 * @{ */
505 HRESULT i_isFormatSupported(const com::Utf8Str &aFormat, BOOL *aSupported);
506 HRESULT i_getFormats(std::vector<com::Utf8Str> &aFormats);
507 HRESULT i_addFormats(const std::vector<com::Utf8Str> &aFormats);
508 HRESULT i_removeFormats(const std::vector<com::Utf8Str> &aFormats);
509
510 HRESULT i_getProtocolVersion(ULONG *puVersion);
511 /** @} */
512
513protected:
514
515 int getProtocolVersion(uint32_t *puVersion);
516
517 /** @name Functions for handling a simple host HGCM message queue.
518 * @{ */
519 int msgQueueAdd(GuestDnDMsg *pMsg);
520 GuestDnDMsg *msgQueueGetNext(void);
521 void msgQueueRemoveNext(void);
522 void msgQueueClear(void);
523 /** @} */
524
525 int sendCancel(void);
526 int waitForEvent(RTMSINTERVAL msTimeout, GuestDnDCallbackEvent &Event, GuestDnDResponse *pResp);
527
528protected:
529
530 /** @name Public attributes (through getters/setters).
531 * @{ */
532 /** Pointer to guest implementation. */
533 const ComObjPtr<Guest> m_pGuest;
534 /** List of supported MIME/Content-type formats. */
535 std::vector<com::Utf8Str> m_vecFmtSup;
536 /** List of offered/compatible MIME/Content-type formats to the
537 * counterpart. */
538 std::vector<com::Utf8Str> m_vecFmtOff;
539 /** @} */
540
541 struct
542 {
543 /** Flag indicating whether a drop operation currently
544 * is in progress or not. */
545 bool mfTransferIsPending;
546 /** The DnD protocol version to use, depending on the
547 * installed Guest Additions. */
548 uint32_t mProtocolVersion;
549 /** Outgoing message queue. */
550 GuestDnDMsgList mListOutgoing;
551 } mDataBase;
552};
553#endif /* ____H_GUESTDNDPRIVATE */
554
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