VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/GuestDnDTargetImpl.cpp@ 76690

Last change on this file since 76690 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 53.7 KB
Line 
1/* $Id: GuestDnDTargetImpl.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * VBox Console COM Class implementation - Guest drag'n drop target.
4 */
5
6/*
7 * Copyright (C) 2014-2019 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
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_GUEST_DND //LOG_GROUP_MAIN_GUESTDNDTARGET
23#include "LoggingNew.h"
24
25#include "GuestImpl.h"
26#include "GuestDnDTargetImpl.h"
27#include "ConsoleImpl.h"
28
29#include "Global.h"
30#include "AutoCaller.h"
31#include "ThreadTask.h"
32
33#include <algorithm> /* For std::find(). */
34
35#include <iprt/asm.h>
36#include <iprt/file.h>
37#include <iprt/dir.h>
38#include <iprt/path.h>
39#include <iprt/uri.h>
40#include <iprt/cpp/utils.h> /* For unconst(). */
41
42#include <VBox/com/array.h>
43
44#include <VBox/GuestHost/DragAndDrop.h>
45#include <VBox/HostServices/Service.h>
46
47
48/**
49 * Base class for a target task.
50 */
51class GuestDnDTargetTask : public ThreadTask
52{
53public:
54
55 GuestDnDTargetTask(GuestDnDTarget *pTarget)
56 : ThreadTask("GenericGuestDnDTargetTask")
57 , mTarget(pTarget)
58 , mRC(VINF_SUCCESS) { }
59
60 virtual ~GuestDnDTargetTask(void) { }
61
62 int getRC(void) const { return mRC; }
63 bool isOk(void) const { return RT_SUCCESS(mRC); }
64 const ComObjPtr<GuestDnDTarget> &getTarget(void) const { return mTarget; }
65
66protected:
67
68 const ComObjPtr<GuestDnDTarget> mTarget;
69 int mRC;
70};
71
72/**
73 * Task structure for sending data to a target using
74 * a worker thread.
75 */
76class SendDataTask : public GuestDnDTargetTask
77{
78public:
79
80 SendDataTask(GuestDnDTarget *pTarget, PSENDDATACTX pCtx)
81 : GuestDnDTargetTask(pTarget),
82 mpCtx(pCtx)
83 {
84 m_strTaskName = "dndTgtSndData";
85 }
86
87 void handler()
88 {
89 GuestDnDTarget::i_sendDataThreadTask(this);
90 }
91
92 virtual ~SendDataTask(void)
93 {
94 if (mpCtx)
95 {
96 delete mpCtx;
97 mpCtx = NULL;
98 }
99 }
100
101
102 PSENDDATACTX getCtx(void) { return mpCtx; }
103
104protected:
105
106 /** Pointer to send data context. */
107 PSENDDATACTX mpCtx;
108};
109
110// constructor / destructor
111/////////////////////////////////////////////////////////////////////////////
112
113DEFINE_EMPTY_CTOR_DTOR(GuestDnDTarget)
114
115HRESULT GuestDnDTarget::FinalConstruct(void)
116{
117 /* Set the maximum block size our guests can handle to 64K. This always has
118 * been hardcoded until now. */
119 /* Note: Never ever rely on information from the guest; the host dictates what and
120 * how to do something, so try to negogiate a sensible value here later. */
121 mData.mcbBlockSize = _64K; /** @todo Make this configurable. */
122
123 LogFlowThisFunc(("\n"));
124 return BaseFinalConstruct();
125}
126
127void GuestDnDTarget::FinalRelease(void)
128{
129 LogFlowThisFuncEnter();
130 uninit();
131 BaseFinalRelease();
132 LogFlowThisFuncLeave();
133}
134
135// public initializer/uninitializer for internal purposes only
136/////////////////////////////////////////////////////////////////////////////
137
138int GuestDnDTarget::init(const ComObjPtr<Guest>& pGuest)
139{
140 LogFlowThisFuncEnter();
141
142 /* Enclose the state transition NotReady->InInit->Ready. */
143 AutoInitSpan autoInitSpan(this);
144 AssertReturn(autoInitSpan.isOk(), E_FAIL);
145
146 unconst(m_pGuest) = pGuest;
147
148 /* Confirm a successful initialization when it's the case. */
149 autoInitSpan.setSucceeded();
150
151 return VINF_SUCCESS;
152}
153
154/**
155 * Uninitializes the instance.
156 * Called from FinalRelease().
157 */
158void GuestDnDTarget::uninit(void)
159{
160 LogFlowThisFunc(("\n"));
161
162 /* Enclose the state transition Ready->InUninit->NotReady. */
163 AutoUninitSpan autoUninitSpan(this);
164 if (autoUninitSpan.uninitDone())
165 return;
166}
167
168// implementation of wrapped IDnDBase methods.
169/////////////////////////////////////////////////////////////////////////////
170
171HRESULT GuestDnDTarget::isFormatSupported(const com::Utf8Str &aFormat, BOOL *aSupported)
172{
173#if !defined(VBOX_WITH_DRAG_AND_DROP)
174 ReturnComNotImplemented();
175#else /* VBOX_WITH_DRAG_AND_DROP */
176
177 AutoCaller autoCaller(this);
178 if (FAILED(autoCaller.rc())) return autoCaller.rc();
179
180 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
181
182 return GuestDnDBase::i_isFormatSupported(aFormat, aSupported);
183#endif /* VBOX_WITH_DRAG_AND_DROP */
184}
185
186HRESULT GuestDnDTarget::getFormats(GuestDnDMIMEList &aFormats)
187{
188#if !defined(VBOX_WITH_DRAG_AND_DROP)
189 ReturnComNotImplemented();
190#else /* VBOX_WITH_DRAG_AND_DROP */
191
192 AutoCaller autoCaller(this);
193 if (FAILED(autoCaller.rc())) return autoCaller.rc();
194
195 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
196
197 return GuestDnDBase::i_getFormats(aFormats);
198#endif /* VBOX_WITH_DRAG_AND_DROP */
199}
200
201HRESULT GuestDnDTarget::addFormats(const GuestDnDMIMEList &aFormats)
202{
203#if !defined(VBOX_WITH_DRAG_AND_DROP)
204 ReturnComNotImplemented();
205#else /* VBOX_WITH_DRAG_AND_DROP */
206
207 AutoCaller autoCaller(this);
208 if (FAILED(autoCaller.rc())) return autoCaller.rc();
209
210 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
211
212 return GuestDnDBase::i_addFormats(aFormats);
213#endif /* VBOX_WITH_DRAG_AND_DROP */
214}
215
216HRESULT GuestDnDTarget::removeFormats(const GuestDnDMIMEList &aFormats)
217{
218#if !defined(VBOX_WITH_DRAG_AND_DROP)
219 ReturnComNotImplemented();
220#else /* VBOX_WITH_DRAG_AND_DROP */
221
222 AutoCaller autoCaller(this);
223 if (FAILED(autoCaller.rc())) return autoCaller.rc();
224
225 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
226
227 return GuestDnDBase::i_removeFormats(aFormats);
228#endif /* VBOX_WITH_DRAG_AND_DROP */
229}
230
231HRESULT GuestDnDTarget::getProtocolVersion(ULONG *aProtocolVersion)
232{
233#if !defined(VBOX_WITH_DRAG_AND_DROP)
234 ReturnComNotImplemented();
235#else /* VBOX_WITH_DRAG_AND_DROP */
236
237 AutoCaller autoCaller(this);
238 if (FAILED(autoCaller.rc())) return autoCaller.rc();
239
240 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
241
242 return GuestDnDBase::i_getProtocolVersion(aProtocolVersion);
243#endif /* VBOX_WITH_DRAG_AND_DROP */
244}
245
246// implementation of wrapped IDnDTarget methods.
247/////////////////////////////////////////////////////////////////////////////
248
249HRESULT GuestDnDTarget::enter(ULONG aScreenId, ULONG aX, ULONG aY,
250 DnDAction_T aDefaultAction,
251 const std::vector<DnDAction_T> &aAllowedActions,
252 const GuestDnDMIMEList &aFormats,
253 DnDAction_T *aResultAction)
254{
255#if !defined(VBOX_WITH_DRAG_AND_DROP)
256 ReturnComNotImplemented();
257#else /* VBOX_WITH_DRAG_AND_DROP */
258
259 /* Input validation. */
260 if (aDefaultAction == DnDAction_Ignore)
261 return setError(E_INVALIDARG, tr("No default action specified"));
262 if (!aAllowedActions.size())
263 return setError(E_INVALIDARG, tr("Number of allowed actions is empty"));
264 if (!aFormats.size())
265 return setError(E_INVALIDARG, tr("Number of supported formats is empty"));
266
267 AutoCaller autoCaller(this);
268 if (FAILED(autoCaller.rc())) return autoCaller.rc();
269
270 /* Determine guest DnD protocol to use. */
271 GuestDnDBase::getProtocolVersion(&mDataBase.m_uProtocolVersion);
272
273 /* Default action is ignoring. */
274 DnDAction_T resAction = DnDAction_Ignore;
275
276 /* Check & convert the drag & drop actions. */
277 VBOXDNDACTION dndActionDefault = 0;
278 VBOXDNDACTIONLIST dndActionListAllowed = 0;
279 GuestDnD::toHGCMActions(aDefaultAction, &dndActionDefault,
280 aAllowedActions, &dndActionListAllowed);
281
282 /* If there is no usable action, ignore this request. */
283 if (isDnDIgnoreAction(dndActionDefault))
284 return S_OK;
285
286 /*
287 * Make a flat data string out of the supported format list.
288 * In the GuestDnDTarget case the source formats are from the host,
289 * as GuestDnDTarget acts as a source for the guest.
290 */
291 Utf8Str strFormats = GuestDnD::toFormatString(GuestDnD::toFilteredFormatList(m_lstFmtSupported, aFormats));
292 if (strFormats.isEmpty())
293 return setError(E_INVALIDARG, tr("No or not supported format(s) specified"));
294 const uint32_t cbFormats = (uint32_t)strFormats.length() + 1; /* Include terminating zero. */
295
296 LogRel2(("DnD: Offered formats to guest:\n"));
297 RTCList<RTCString> lstFormats = strFormats.split("\r\n");
298 for (size_t i = 0; i < lstFormats.size(); i++)
299 LogRel2(("DnD: \t%s\n", lstFormats[i].c_str()));
300
301 /* Save the formats offered to the guest. This is needed to later
302 * decide what to do with the data when sending stuff to the guest. */
303 m_lstFmtOffered = aFormats;
304 Assert(m_lstFmtOffered.size());
305
306 HRESULT hr = S_OK;
307
308 /* Adjust the coordinates in a multi-monitor setup. */
309 int rc = GuestDnDInst()->adjustScreenCoordinates(aScreenId, &aX, &aY);
310 if (RT_SUCCESS(rc))
311 {
312 GuestDnDMsg Msg;
313 Msg.setType(HOST_DND_HG_EVT_ENTER);
314 if (mDataBase.m_uProtocolVersion >= 3)
315 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
316 Msg.setNextUInt32(aScreenId);
317 Msg.setNextUInt32(aX);
318 Msg.setNextUInt32(aY);
319 Msg.setNextUInt32(dndActionDefault);
320 Msg.setNextUInt32(dndActionListAllowed);
321 Msg.setNextPointer((void *)strFormats.c_str(), cbFormats);
322 Msg.setNextUInt32(cbFormats);
323
324 rc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
325 if (RT_SUCCESS(rc))
326 {
327 GuestDnDResponse *pResp = GuestDnDInst()->response();
328 if (pResp && RT_SUCCESS(pResp->waitForGuestResponse()))
329 resAction = GuestDnD::toMainAction(pResp->getActionDefault());
330 }
331 }
332
333 if (RT_FAILURE(rc))
334 hr = VBOX_E_IPRT_ERROR;
335
336 if (SUCCEEDED(hr))
337 {
338 if (aResultAction)
339 *aResultAction = resAction;
340 }
341
342 LogFlowFunc(("hr=%Rhrc, resAction=%ld\n", hr, resAction));
343 return hr;
344#endif /* VBOX_WITH_DRAG_AND_DROP */
345}
346
347HRESULT GuestDnDTarget::move(ULONG aScreenId, ULONG aX, ULONG aY,
348 DnDAction_T aDefaultAction,
349 const std::vector<DnDAction_T> &aAllowedActions,
350 const GuestDnDMIMEList &aFormats,
351 DnDAction_T *aResultAction)
352{
353#if !defined(VBOX_WITH_DRAG_AND_DROP)
354 ReturnComNotImplemented();
355#else /* VBOX_WITH_DRAG_AND_DROP */
356
357 /* Input validation. */
358
359 AutoCaller autoCaller(this);
360 if (FAILED(autoCaller.rc())) return autoCaller.rc();
361
362 /* Default action is ignoring. */
363 DnDAction_T resAction = DnDAction_Ignore;
364
365 /* Check & convert the drag & drop actions. */
366 VBOXDNDACTION dndActionDefault = 0;
367 VBOXDNDACTIONLIST dndActionListAllowed = 0;
368 GuestDnD::toHGCMActions(aDefaultAction, &dndActionDefault,
369 aAllowedActions, &dndActionListAllowed);
370
371 /* If there is no usable action, ignore this request. */
372 if (isDnDIgnoreAction(dndActionDefault))
373 return S_OK;
374
375 /*
376 * Make a flat data string out of the supported format list.
377 * In the GuestDnDTarget case the source formats are from the host,
378 * as GuestDnDTarget acts as a source for the guest.
379 */
380 Utf8Str strFormats = GuestDnD::toFormatString(GuestDnD::toFilteredFormatList(m_lstFmtSupported, aFormats));
381 if (strFormats.isEmpty())
382 return setError(E_INVALIDARG, tr("No or not supported format(s) specified"));
383 const uint32_t cbFormats = (uint32_t)strFormats.length() + 1; /* Include terminating zero. */
384
385 HRESULT hr = S_OK;
386
387 int rc = GuestDnDInst()->adjustScreenCoordinates(aScreenId, &aX, &aY);
388 if (RT_SUCCESS(rc))
389 {
390 GuestDnDMsg Msg;
391 Msg.setType(HOST_DND_HG_EVT_MOVE);
392 if (mDataBase.m_uProtocolVersion >= 3)
393 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
394 Msg.setNextUInt32(aScreenId);
395 Msg.setNextUInt32(aX);
396 Msg.setNextUInt32(aY);
397 Msg.setNextUInt32(dndActionDefault);
398 Msg.setNextUInt32(dndActionListAllowed);
399 Msg.setNextPointer((void *)strFormats.c_str(), cbFormats);
400 Msg.setNextUInt32(cbFormats);
401
402 rc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
403 if (RT_SUCCESS(rc))
404 {
405 GuestDnDResponse *pResp = GuestDnDInst()->response();
406 if (pResp && RT_SUCCESS(pResp->waitForGuestResponse()))
407 resAction = GuestDnD::toMainAction(pResp->getActionDefault());
408 }
409 }
410
411 if (RT_FAILURE(rc))
412 hr = VBOX_E_IPRT_ERROR;
413
414 if (SUCCEEDED(hr))
415 {
416 if (aResultAction)
417 *aResultAction = resAction;
418 }
419
420 LogFlowFunc(("hr=%Rhrc, *pResultAction=%ld\n", hr, resAction));
421 return hr;
422#endif /* VBOX_WITH_DRAG_AND_DROP */
423}
424
425HRESULT GuestDnDTarget::leave(ULONG uScreenId)
426{
427 RT_NOREF(uScreenId);
428#if !defined(VBOX_WITH_DRAG_AND_DROP)
429 ReturnComNotImplemented();
430#else /* VBOX_WITH_DRAG_AND_DROP */
431
432 AutoCaller autoCaller(this);
433 if (FAILED(autoCaller.rc())) return autoCaller.rc();
434
435 HRESULT hr = S_OK;
436
437 GuestDnDMsg Msg;
438 Msg.setType(HOST_DND_HG_EVT_LEAVE);
439 if (mDataBase.m_uProtocolVersion >= 3)
440 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
441
442 int rc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
443 if (RT_SUCCESS(rc))
444 {
445 GuestDnDResponse *pResp = GuestDnDInst()->response();
446 if (pResp)
447 pResp->waitForGuestResponse();
448 }
449
450 if (RT_FAILURE(rc))
451 hr = VBOX_E_IPRT_ERROR;
452
453 LogFlowFunc(("hr=%Rhrc\n", hr));
454 return hr;
455#endif /* VBOX_WITH_DRAG_AND_DROP */
456}
457
458HRESULT GuestDnDTarget::drop(ULONG aScreenId, ULONG aX, ULONG aY,
459 DnDAction_T aDefaultAction,
460 const std::vector<DnDAction_T> &aAllowedActions,
461 const GuestDnDMIMEList &aFormats,
462 com::Utf8Str &aFormat,
463 DnDAction_T *aResultAction)
464{
465#if !defined(VBOX_WITH_DRAG_AND_DROP)
466 ReturnComNotImplemented();
467#else /* VBOX_WITH_DRAG_AND_DROP */
468
469 if (aDefaultAction == DnDAction_Ignore)
470 return setError(E_INVALIDARG, tr("Invalid default action specified"));
471 if (!aAllowedActions.size())
472 return setError(E_INVALIDARG, tr("Invalid allowed actions specified"));
473 if (!aFormats.size())
474 return setError(E_INVALIDARG, tr("No drop format(s) specified"));
475 /* aResultAction is optional. */
476
477 AutoCaller autoCaller(this);
478 if (FAILED(autoCaller.rc())) return autoCaller.rc();
479
480 /* Default action is ignoring. */
481 DnDAction_T resAction = DnDAction_Ignore;
482
483 /* Check & convert the drag & drop actions to HGCM codes. */
484 VBOXDNDACTION dndActionDefault = VBOX_DND_ACTION_IGNORE;
485 VBOXDNDACTIONLIST dndActionListAllowed = 0;
486 GuestDnD::toHGCMActions(aDefaultAction, &dndActionDefault,
487 aAllowedActions, &dndActionListAllowed);
488
489 /* If there is no usable action, ignore this request. */
490 if (isDnDIgnoreAction(dndActionDefault))
491 {
492 aFormat = "";
493 if (aResultAction)
494 *aResultAction = DnDAction_Ignore;
495 return S_OK;
496 }
497
498 /*
499 * Make a flat data string out of the supported format list.
500 * In the GuestDnDTarget case the source formats are from the host,
501 * as GuestDnDTarget acts as a source for the guest.
502 */
503 Utf8Str strFormats = GuestDnD::toFormatString(GuestDnD::toFilteredFormatList(m_lstFmtSupported, aFormats));
504 if (strFormats.isEmpty())
505 return setError(E_INVALIDARG, tr("No or not supported format(s) specified"));
506 const uint32_t cbFormats = (uint32_t)strFormats.length() + 1; /* Include terminating zero. */
507
508 /* Adjust the coordinates in a multi-monitor setup. */
509 HRESULT hr = GuestDnDInst()->adjustScreenCoordinates(aScreenId, &aX, &aY);
510 if (SUCCEEDED(hr))
511 {
512 GuestDnDMsg Msg;
513 Msg.setType(HOST_DND_HG_EVT_DROPPED);
514 if (mDataBase.m_uProtocolVersion >= 3)
515 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
516 Msg.setNextUInt32(aScreenId);
517 Msg.setNextUInt32(aX);
518 Msg.setNextUInt32(aY);
519 Msg.setNextUInt32(dndActionDefault);
520 Msg.setNextUInt32(dndActionListAllowed);
521 Msg.setNextPointer((void*)strFormats.c_str(), cbFormats);
522 Msg.setNextUInt32(cbFormats);
523
524 int vrc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
525 if (RT_SUCCESS(vrc))
526 {
527 GuestDnDResponse *pResp = GuestDnDInst()->response();
528 AssertPtr(pResp);
529
530 vrc = pResp->waitForGuestResponse();
531 if (RT_SUCCESS(vrc))
532 {
533 resAction = GuestDnD::toMainAction(pResp->getActionDefault());
534
535 GuestDnDMIMEList lstFormats = pResp->formats();
536 if (lstFormats.size() == 1) /* Exactly one format to use specified? */
537 {
538 aFormat = lstFormats.at(0);
539 LogFlowFunc(("resFormat=%s, resAction=%RU32\n", aFormat.c_str(), pResp->getActionDefault()));
540 }
541 else
542 /** @todo r=bird: This isn't an IPRT error, is it? */
543 hr = setError(VBOX_E_IPRT_ERROR, tr("Guest returned invalid drop formats (%zu formats)"), lstFormats.size());
544 }
545 else
546 hr = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("Waiting for response of dropped event failed (%Rrc)"), vrc);
547 }
548 else
549 hr = setErrorBoth(VBOX_E_IPRT_ERROR, vrc, tr("Sending dropped event to guest failed (%Rrc)"), vrc);
550 }
551 else
552 hr = setError(hr, tr("Retrieving drop coordinates failed"));
553
554 if (SUCCEEDED(hr))
555 {
556 if (aResultAction)
557 *aResultAction = resAction;
558 }
559
560 LogFlowFunc(("Returning hr=%Rhrc\n", hr));
561 return hr;
562#endif /* VBOX_WITH_DRAG_AND_DROP */
563}
564
565/* static */
566void GuestDnDTarget::i_sendDataThreadTask(SendDataTask *pTask)
567{
568 LogFlowFunc(("pTask=%p\n", pTask));
569 AssertPtrReturnVoid(pTask);
570
571 const ComObjPtr<GuestDnDTarget> pThis(pTask->getTarget());
572 Assert(!pThis.isNull());
573
574 AutoCaller autoCaller(pThis);
575 if (FAILED(autoCaller.rc()))
576 return;
577
578 int vrc = pThis->i_sendData(pTask->getCtx(), RT_INDEFINITE_WAIT /* msTimeout */);
579 if (RT_FAILURE(vrc)) /* In case we missed some error handling within i_sendData(). */
580 {
581 AssertFailed();
582 LogRel(("DnD: Sending data to guest failed with %Rrc\n", vrc));
583 }
584
585 AutoWriteLock alock(pThis COMMA_LOCKVAL_SRC_POS);
586
587 Assert(pThis->mDataBase.m_cTransfersPending);
588 if (pThis->mDataBase.m_cTransfersPending)
589 pThis->mDataBase.m_cTransfersPending--;
590
591 LogFlowFunc(("pTarget=%p, vrc=%Rrc (ignored)\n", (GuestDnDTarget *)pThis, vrc));
592}
593
594/**
595 * Initiates a data transfer from the host to the guest. The source is the host whereas the target is the
596 * guest in this case.
597 *
598 * @return HRESULT
599 * @param aScreenId
600 * @param aFormat
601 * @param aData
602 * @param aProgress
603 */
604HRESULT GuestDnDTarget::sendData(ULONG aScreenId, const com::Utf8Str &aFormat, const std::vector<BYTE> &aData,
605 ComPtr<IProgress> &aProgress)
606{
607#if !defined(VBOX_WITH_DRAG_AND_DROP)
608 ReturnComNotImplemented();
609#else /* VBOX_WITH_DRAG_AND_DROP */
610
611 AutoCaller autoCaller(this);
612 if (FAILED(autoCaller.rc())) return autoCaller.rc();
613
614 /* Input validation. */
615 if (RT_UNLIKELY((aFormat.c_str()) == NULL || *(aFormat.c_str()) == '\0'))
616 return setError(E_INVALIDARG, tr("No data format specified"));
617 if (RT_UNLIKELY(!aData.size()))
618 return setError(E_INVALIDARG, tr("No data to send specified"));
619
620 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
621
622 /* At the moment we only support one transfer at a time. */
623 if (mDataBase.m_cTransfersPending)
624 return setError(E_INVALIDARG, tr("Another drop operation already is in progress"));
625
626 /* Ditto. */
627 GuestDnDResponse *pResp = GuestDnDInst()->response();
628 AssertPtr(pResp);
629
630 HRESULT hr = pResp->resetProgress(m_pGuest);
631 if (FAILED(hr))
632 return hr;
633
634 SendDataTask *pTask = NULL;
635 PSENDDATACTX pSendCtx = NULL;
636
637 try
638 {
639 //pSendCtx is passed into SendDataTask where one is deleted in destructor
640 pSendCtx = new SENDDATACTX;
641 RT_BZERO(pSendCtx, sizeof(SENDDATACTX));
642
643 pSendCtx->mpTarget = this;
644 pSendCtx->mpResp = pResp;
645 pSendCtx->mScreenID = aScreenId;
646 pSendCtx->mFmtReq = aFormat;
647 pSendCtx->mData.getMeta().add(aData);
648
649 /* pTask is responsible for deletion of pSendCtx after creating */
650 pTask = new SendDataTask(this, pSendCtx);
651 if (!pTask->isOk())
652 {
653 delete pTask;
654 LogRel2(("DnD: Could not create SendDataTask object \n"));
655 throw hr = E_FAIL;
656 }
657
658 /* This function delete pTask in case of exceptions,
659 * so there is no need in the call of delete operator. */
660 hr = pTask->createThreadWithType(RTTHREADTYPE_MAIN_WORKER);
661
662 }
663 catch (std::bad_alloc &)
664 {
665 hr = setError(E_OUTOFMEMORY);
666 }
667 catch (...)
668 {
669 LogRel2(("DnD: Could not create thread for data sending task\n"));
670 hr = E_FAIL;
671 }
672
673 if (SUCCEEDED(hr))
674 {
675 mDataBase.m_cTransfersPending++;
676
677 hr = pResp->queryProgressTo(aProgress.asOutParam());
678 ComAssertComRC(hr);
679
680 /* Note: pTask is now owned by the worker thread. */
681 }
682 else
683 hr = setError(hr, tr("Starting thread for GuestDnDTarget::i_sendDataThread (%Rhrc)"), hr);
684
685 LogFlowFunc(("Returning hr=%Rhrc\n", hr));
686 return hr;
687#endif /* VBOX_WITH_DRAG_AND_DROP */
688}
689
690int GuestDnDTarget::i_cancelOperation(void)
691{
692 /** @todo Check for pending cancel requests. */
693
694#if 0 /** @todo Later. */
695 /* Cancel any outstanding waits for guest responses first. */
696 if (pResp)
697 pResp->notifyAboutGuestResponse();
698#endif
699
700 LogFlowFunc(("Cancelling operation, telling guest ...\n"));
701
702 GuestDnDMsg Msg;
703 Msg.setType(HOST_DND_HG_EVT_CANCEL);
704 if (mDataBase.m_uProtocolVersion >= 3)
705 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
706
707 return GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
708}
709
710/* static */
711Utf8Str GuestDnDTarget::i_guestErrorToString(int guestRc)
712{
713 Utf8Str strError;
714
715 switch (guestRc)
716 {
717 case VERR_ACCESS_DENIED:
718 strError += Utf8StrFmt(tr("For one or more guest files or directories selected for transferring to the host your guest "
719 "user does not have the appropriate access rights for. Please make sure that all selected "
720 "elements can be accessed and that your guest user has the appropriate rights"));
721 break;
722
723 case VERR_NOT_FOUND:
724 /* Should not happen due to file locking on the guest, but anyway ... */
725 strError += Utf8StrFmt(tr("One or more guest files or directories selected for transferring to the host were not"
726 "found on the guest anymore. This can be the case if the guest files were moved and/or"
727 "altered while the drag and drop operation was in progress"));
728 break;
729
730 case VERR_SHARING_VIOLATION:
731 strError += Utf8StrFmt(tr("One or more guest files or directories selected for transferring to the host were locked. "
732 "Please make sure that all selected elements can be accessed and that your guest user has "
733 "the appropriate rights"));
734 break;
735
736 case VERR_TIMEOUT:
737 strError += Utf8StrFmt(tr("The guest was not able to process the drag and drop data within time"));
738 break;
739
740 default:
741 strError += Utf8StrFmt(tr("Drag and drop error from guest (%Rrc)"), guestRc);
742 break;
743 }
744
745 return strError;
746}
747
748/* static */
749Utf8Str GuestDnDTarget::i_hostErrorToString(int hostRc)
750{
751 Utf8Str strError;
752
753 switch (hostRc)
754 {
755 case VERR_ACCESS_DENIED:
756 strError += Utf8StrFmt(tr("For one or more host files or directories selected for transferring to the guest your host "
757 "user does not have the appropriate access rights for. Please make sure that all selected "
758 "elements can be accessed and that your host user has the appropriate rights."));
759 break;
760
761 case VERR_NOT_FOUND:
762 /* Should not happen due to file locking on the host, but anyway ... */
763 strError += Utf8StrFmt(tr("One or more host files or directories selected for transferring to the host were not"
764 "found on the host anymore. This can be the case if the host files were moved and/or"
765 "altered while the drag and drop operation was in progress."));
766 break;
767
768 case VERR_SHARING_VIOLATION:
769 strError += Utf8StrFmt(tr("One or more host files or directories selected for transferring to the guest were locked. "
770 "Please make sure that all selected elements can be accessed and that your host user has "
771 "the appropriate rights."));
772 break;
773
774 default:
775 strError += Utf8StrFmt(tr("Drag and drop error from host (%Rrc)"), hostRc);
776 break;
777 }
778
779 return strError;
780}
781
782/**
783 * @returns VBox status code that the caller ignores. Not sure if that's
784 * intentional or not.
785 */
786int GuestDnDTarget::i_sendData(PSENDDATACTX pCtx, RTMSINTERVAL msTimeout)
787{
788 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
789
790 /* Is this context already in sending state? */
791 if (ASMAtomicReadBool(&pCtx->mIsActive))
792 return VERR_WRONG_ORDER;
793 ASMAtomicWriteBool(&pCtx->mIsActive, true);
794
795 /* Clear all remaining outgoing messages. */
796 mDataBase.m_lstMsgOut.clear();
797
798 /**
799 * Do we need to build up a file tree?
800 * Note: The decision whether we need to build up a file tree and sending
801 * actual file data only depends on the actual formats offered by this target.
802 * If the guest does not want an URI list ("text/uri-list") but text ("TEXT" and
803 * friends) instead, still send the data over to the guest -- the file as such still
804 * is needed on the guest in this case, as the guest then just wants a simple path
805 * instead of an URI list (pointing to a file on the guest itself).
806 *
807 ** @todo Support more than one format; add a format<->function handler concept. Later. */
808 int rc;
809 bool fHasURIList = std::find(m_lstFmtOffered.begin(),
810 m_lstFmtOffered.end(), "text/uri-list") != m_lstFmtOffered.end();
811 if (fHasURIList)
812 {
813 rc = i_sendURIData(pCtx, msTimeout);
814 }
815 else
816 {
817 rc = i_sendRawData(pCtx, msTimeout);
818 }
819
820 ASMAtomicWriteBool(&pCtx->mIsActive, false);
821
822 LogFlowFuncLeaveRC(rc);
823 return rc;
824}
825
826int GuestDnDTarget::i_sendDataBody(PSENDDATACTX pCtx, GuestDnDData *pData)
827{
828 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
829 AssertPtrReturn(pData, VERR_INVALID_POINTER);
830
831 /** @todo Add support for multiple HOST_DND_HG_SND_DATA messages in case of more than 64K data! */
832 if (pData->getMeta().getSize() > _64K)
833 return VERR_NOT_IMPLEMENTED;
834
835 GuestDnDMsg Msg;
836
837 LogFlowFunc(("cbFmt=%RU32, cbMeta=%RU32, cbChksum=%RU32\n",
838 pData->getFmtSize(), pData->getMeta().getSize(), pData->getChkSumSize()));
839
840 Msg.setType(HOST_DND_HG_SND_DATA);
841 if (mDataBase.m_uProtocolVersion < 3)
842 {
843 Msg.setNextUInt32(pCtx->mScreenID); /* uScreenId */
844 Msg.setNextPointer(pData->getFmtMutable(), pData->getFmtSize()); /* pvFormat */
845 Msg.setNextUInt32(pData->getFmtSize()); /* cbFormat */
846 Msg.setNextPointer(pData->getMeta().getDataMutable(), pData->getMeta().getSize()); /* pvData */
847 /* Fill in the current data block size to send.
848 * Note: Only supports uint32_t. */
849 Msg.setNextUInt32((uint32_t)pData->getMeta().getSize()); /* cbData */
850 }
851 else
852 {
853 Msg.setNextUInt32(0); /** @todo ContextID not used yet. */
854 Msg.setNextPointer(pData->getMeta().getDataMutable(), pData->getMeta().getSize()); /* pvData */
855 Msg.setNextUInt32(pData->getMeta().getSize()); /* cbData */
856 Msg.setNextPointer(pData->getChkSumMutable(), pData->getChkSumSize()); /** @todo pvChecksum; not used yet. */
857 Msg.setNextUInt32(pData->getChkSumSize()); /** @todo cbChecksum; not used yet. */
858 }
859
860 int rc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
861 if (RT_SUCCESS(rc))
862 rc = updateProgress(pData, pCtx->mpResp, pData->getMeta().getSize());
863
864 LogFlowFuncLeaveRC(rc);
865 return rc;
866}
867
868int GuestDnDTarget::i_sendDataHeader(PSENDDATACTX pCtx, GuestDnDData *pData, GuestDnDURIData *pURIData /* = NULL */)
869{
870 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
871 AssertPtrReturn(pData, VERR_INVALID_POINTER);
872 /* pURIData is optional. */
873
874 GuestDnDMsg Msg;
875
876 Msg.setType(HOST_DND_HG_SND_DATA_HDR);
877
878 Msg.setNextUInt32(0); /** @todo uContext; not used yet. */
879 Msg.setNextUInt32(0); /** @todo uFlags; not used yet. */
880 Msg.setNextUInt32(pCtx->mScreenID); /* uScreen */
881 Msg.setNextUInt64(pData->getTotal()); /* cbTotal */
882 Msg.setNextUInt32(pData->getMeta().getSize()); /* cbMeta*/
883 Msg.setNextPointer(pData->getFmtMutable(), pData->getFmtSize()); /* pvMetaFmt */
884 Msg.setNextUInt32(pData->getFmtSize()); /* cbMetaFmt */
885 Msg.setNextUInt64(pURIData ? pURIData->getObjToProcess() : 0); /* cObjects */
886 Msg.setNextUInt32(0); /** @todo enmCompression; not used yet. */
887 Msg.setNextUInt32(0); /** @todo enmChecksumType; not used yet. */
888 Msg.setNextPointer(NULL, 0); /** @todo pvChecksum; not used yet. */
889 Msg.setNextUInt32(0); /** @todo cbChecksum; not used yet. */
890
891 int rc = GuestDnDInst()->hostCall(Msg.getType(), Msg.getCount(), Msg.getParms());
892
893 LogFlowFuncLeaveRC(rc);
894 return rc;
895}
896
897int GuestDnDTarget::i_sendDirectory(PSENDDATACTX pCtx, GuestDnDURIObjCtx *pObjCtx, GuestDnDMsg *pMsg)
898{
899 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
900 AssertPtrReturn(pObjCtx, VERR_INVALID_POINTER);
901 AssertPtrReturn(pMsg, VERR_INVALID_POINTER);
902
903 DnDURIObject *pObj = pObjCtx->getObj();
904 AssertPtr(pObj);
905
906 RTCString strPath = pObj->GetDestPathAbs();
907 if (strPath.isEmpty())
908 return VERR_INVALID_PARAMETER;
909 if (strPath.length() >= RTPATH_MAX) /* Note: Maximum is RTPATH_MAX on guest side. */
910 return VERR_BUFFER_OVERFLOW;
911
912 LogRel2(("DnD: Transferring host directory '%s' to guest\n", strPath.c_str()));
913
914 pMsg->setType(HOST_DND_HG_SND_DIR);
915 if (mDataBase.m_uProtocolVersion >= 3)
916 pMsg->setNextUInt32(0); /** @todo ContextID not used yet. */
917 pMsg->setNextString(strPath.c_str()); /* path */
918 pMsg->setNextUInt32((uint32_t)(strPath.length() + 1)); /* path length (maximum is RTPATH_MAX on guest side). */
919 pMsg->setNextUInt32(pObj->GetMode()); /* mode */
920
921 return VINF_SUCCESS;
922}
923
924int GuestDnDTarget::i_sendFile(PSENDDATACTX pCtx, GuestDnDURIObjCtx *pObjCtx, GuestDnDMsg *pMsg)
925{
926 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
927 AssertPtrReturn(pObjCtx, VERR_INVALID_POINTER);
928 AssertPtrReturn(pMsg, VERR_INVALID_POINTER);
929
930 DnDURIObject *pObj = pObjCtx->getObj();
931 AssertPtr(pObj);
932
933 RTCString strPathSrc = pObj->GetSourcePathAbs();
934 if (strPathSrc.isEmpty())
935 return VERR_INVALID_PARAMETER;
936
937 int rc = VINF_SUCCESS;
938
939 LogFlowFunc(("Sending file with %RU32 bytes buffer, using protocol v%RU32 ...\n",
940 mData.mcbBlockSize, mDataBase.m_uProtocolVersion));
941 LogFlowFunc(("strPathSrc=%s, fIsOpen=%RTbool, cbSize=%RU64\n", strPathSrc.c_str(), pObj->IsOpen(), pObj->GetSize()));
942
943 if (!pObj->IsOpen())
944 {
945 LogRel2(("DnD: Opening host file '%s' for transferring to guest\n", strPathSrc.c_str()));
946 rc = pObj->OpenEx(strPathSrc, DnDURIObject::View_Source,
947 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
948 if (RT_FAILURE(rc))
949 LogRel(("DnD: Opening host file '%s' failed, rc=%Rrc\n", strPathSrc.c_str(), rc));
950 }
951
952 bool fSendData = false;
953 if (RT_SUCCESS(rc))
954 {
955 if (mDataBase.m_uProtocolVersion >= 2)
956 {
957 uint32_t fState = pObjCtx->getState();
958 if (!(fState & DND_OBJCTX_STATE_HAS_HDR))
959 {
960 /*
961 * Since protocol v2 the file header and the actual file contents are
962 * separate messages, so send the file header first.
963 * The just registered callback will be called by the guest afterwards.
964 */
965 pMsg->setType(HOST_DND_HG_SND_FILE_HDR);
966 pMsg->setNextUInt32(0); /** @todo ContextID not used yet. */
967 pMsg->setNextString(pObj->GetDestPathAbs().c_str()); /* pvName */
968 pMsg->setNextUInt32((uint32_t)(pObj->GetDestPathAbs().length() + 1)); /* cbName */
969 pMsg->setNextUInt32(0); /* uFlags */
970 pMsg->setNextUInt32(pObj->GetMode()); /* fMode */
971 pMsg->setNextUInt64(pObj->GetSize()); /* uSize */
972
973 LogFlowFunc(("Sending file header ...\n"));
974 LogRel2(("DnD: Transferring host file to guest: %s (%RU64 bytes, mode 0x%x)\n",
975 strPathSrc.c_str(), pObj->GetSize(), pObj->GetMode()));
976
977 /** @todo Set progress object title to current file being transferred? */
978
979 pObjCtx->setState(fState | DND_OBJCTX_STATE_HAS_HDR);
980 }
981 else
982 {
983 /* File header was sent, so only send the actual file data. */
984 fSendData = true;
985 }
986 }
987 else /* Protocol v1. */
988 {
989 /* Always send the file data, every time. */
990 fSendData = true;
991 }
992 }
993
994 if ( RT_SUCCESS(rc)
995 && fSendData)
996 {
997 rc = i_sendFileData(pCtx, pObjCtx, pMsg);
998 }
999
1000 if (RT_FAILURE(rc))
1001 LogRel(("DnD: Sending host file to guest failed, rc=%Rrc\n", rc));
1002
1003 LogFlowFuncLeaveRC(rc);
1004 return rc;
1005}
1006
1007int GuestDnDTarget::i_sendFileData(PSENDDATACTX pCtx, GuestDnDURIObjCtx *pObjCtx, GuestDnDMsg *pMsg)
1008{
1009 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1010 AssertPtrReturn(pObjCtx, VERR_INVALID_POINTER);
1011 AssertPtrReturn(pMsg, VERR_INVALID_POINTER);
1012
1013 DnDURIObject *pObj = pObjCtx->getObj();
1014 AssertPtr(pObj);
1015
1016 AssertPtr(pCtx->mpResp);
1017
1018 /** @todo Don't allow concurrent reads per context! */
1019
1020 /*
1021 * Start sending stuff.
1022 */
1023
1024 /* Set the message type. */
1025 pMsg->setType(HOST_DND_HG_SND_FILE_DATA);
1026
1027 /* Protocol version 1 sends the file path *every* time with a new file chunk.
1028 * In protocol version 2 we only do this once with HOST_DND_HG_SND_FILE_HDR. */
1029 if (mDataBase.m_uProtocolVersion <= 1)
1030 {
1031 pMsg->setNextString(pObj->GetDestPathAbs().c_str()); /* pvName */
1032 pMsg->setNextUInt32((uint32_t)(pObj->GetDestPathAbs().length() + 1)); /* cbName */
1033 }
1034 else if (mDataBase.m_uProtocolVersion >= 2)
1035 {
1036 pMsg->setNextUInt32(0); /** @todo ContextID not used yet. */
1037 }
1038
1039 uint32_t cbRead = 0;
1040
1041 int rc = pObj->Read(pCtx->mURI.getBufferMutable(), pCtx->mURI.getBufferSize(), &cbRead);
1042 if (RT_SUCCESS(rc))
1043 {
1044 pCtx->mData.addProcessed(cbRead);
1045 LogFlowFunc(("cbBufSize=%zu, cbRead=%RU32\n", pCtx->mURI.getBufferSize(), cbRead));
1046
1047 if (mDataBase.m_uProtocolVersion <= 1)
1048 {
1049 pMsg->setNextPointer(pCtx->mURI.getBufferMutable(), cbRead); /* pvData */
1050 pMsg->setNextUInt32(cbRead); /* cbData */
1051 pMsg->setNextUInt32(pObj->GetMode()); /* fMode */
1052 }
1053 else /* Protocol v2 and up. */
1054 {
1055 pMsg->setNextPointer(pCtx->mURI.getBufferMutable(), cbRead); /* pvData */
1056 pMsg->setNextUInt32(cbRead); /* cbData */
1057
1058 if (mDataBase.m_uProtocolVersion >= 3)
1059 {
1060 /** @todo Calculate checksum. */
1061 pMsg->setNextPointer(NULL, 0); /* pvChecksum */
1062 pMsg->setNextUInt32(0); /* cbChecksum */
1063 }
1064 }
1065
1066 if (pObj->IsComplete()) /* Done reading? */
1067 {
1068 LogRel2(("DnD: Transferring file '%s' to guest complete\n", pObj->GetSourcePathAbs().c_str()));
1069 LogFlowFunc(("File '%s' complete\n", pObj->GetSourcePathAbs().c_str()));
1070
1071 /* DnDURIObject::Read() returns VINF_EOF when finished reading the entire fire,
1072 * but we don't want this here -- so just override this with VINF_SUCCESS. */
1073 rc = VINF_SUCCESS;
1074 }
1075 }
1076
1077 if (RT_FAILURE(rc))
1078 LogRel(("DnD: Reading from host file '%s' failed, rc=%Rrc\n", pObj->GetSourcePathAbs().c_str(), rc));
1079
1080 LogFlowFuncLeaveRC(rc);
1081 return rc;
1082}
1083
1084/* static */
1085DECLCALLBACK(int) GuestDnDTarget::i_sendURIDataCallback(uint32_t uMsg, void *pvParms, size_t cbParms, void *pvUser)
1086{
1087 PSENDDATACTX pCtx = (PSENDDATACTX)pvUser;
1088 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1089
1090 GuestDnDTarget *pThis = pCtx->mpTarget;
1091 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1092
1093 LogFlowFunc(("pThis=%p, uMsg=%RU32\n", pThis, uMsg));
1094
1095 int rc = VINF_SUCCESS;
1096 int rcGuest = VINF_SUCCESS; /* Contains error code from guest in case of VERR_GSTDND_GUEST_ERROR. */
1097 bool fNotify = false;
1098
1099 switch (uMsg)
1100 {
1101 case GUEST_DND_CONNECT:
1102 /* Nothing to do here (yet). */
1103 break;
1104
1105 case GUEST_DND_DISCONNECT:
1106 rc = VERR_CANCELLED;
1107 break;
1108
1109 case GUEST_DND_GET_NEXT_HOST_MSG:
1110 {
1111 PVBOXDNDCBHGGETNEXTHOSTMSG pCBData = reinterpret_cast<PVBOXDNDCBHGGETNEXTHOSTMSG>(pvParms);
1112 AssertPtr(pCBData);
1113 AssertReturn(sizeof(VBOXDNDCBHGGETNEXTHOSTMSG) == cbParms, VERR_INVALID_PARAMETER);
1114 AssertReturn(CB_MAGIC_DND_HG_GET_NEXT_HOST_MSG == pCBData->hdr.uMagic, VERR_INVALID_PARAMETER);
1115
1116 try
1117 {
1118 GuestDnDMsg *pMsg = new GuestDnDMsg();
1119
1120 rc = pThis->i_sendURIDataLoop(pCtx, pMsg);
1121 if (rc == VINF_EOF) /* Transfer complete? */
1122 {
1123 LogFlowFunc(("Last URI item processed, bailing out\n"));
1124 }
1125 else if (RT_SUCCESS(rc))
1126 {
1127 rc = pThis->msgQueueAdd(pMsg);
1128 if (RT_SUCCESS(rc)) /* Return message type & required parameter count to the guest. */
1129 {
1130 LogFlowFunc(("GUEST_DND_GET_NEXT_HOST_MSG -> %RU32 (%RU32 params)\n", pMsg->getType(), pMsg->getCount()));
1131 pCBData->uMsg = pMsg->getType();
1132 pCBData->cParms = pMsg->getCount();
1133 }
1134 }
1135
1136 if ( RT_FAILURE(rc)
1137 || rc == VINF_EOF) /* Transfer complete? */
1138 {
1139 delete pMsg;
1140 pMsg = NULL;
1141 }
1142 }
1143 catch(std::bad_alloc & /*e*/)
1144 {
1145 rc = VERR_NO_MEMORY;
1146 }
1147 break;
1148 }
1149 case GUEST_DND_GH_EVT_ERROR:
1150 {
1151 PVBOXDNDCBEVTERRORDATA pCBData = reinterpret_cast<PVBOXDNDCBEVTERRORDATA>(pvParms);
1152 AssertPtr(pCBData);
1153 AssertReturn(sizeof(VBOXDNDCBEVTERRORDATA) == cbParms, VERR_INVALID_PARAMETER);
1154 AssertReturn(CB_MAGIC_DND_GH_EVT_ERROR == pCBData->hdr.uMagic, VERR_INVALID_PARAMETER);
1155
1156 pCtx->mpResp->reset();
1157
1158 if (RT_SUCCESS(pCBData->rc))
1159 {
1160 AssertMsgFailed(("Guest has sent an error event but did not specify an actual error code\n"));
1161 pCBData->rc = VERR_GENERAL_FAILURE; /* Make sure some error is set. */
1162 }
1163
1164 rc = pCtx->mpResp->setProgress(100, DND_PROGRESS_ERROR, pCBData->rc,
1165 GuestDnDTarget::i_guestErrorToString(pCBData->rc));
1166 if (RT_SUCCESS(rc))
1167 {
1168 rc = VERR_GSTDND_GUEST_ERROR;
1169 rcGuest = pCBData->rc;
1170 }
1171 break;
1172 }
1173 case HOST_DND_HG_SND_DIR:
1174 case HOST_DND_HG_SND_FILE_HDR:
1175 case HOST_DND_HG_SND_FILE_DATA:
1176 {
1177 PVBOXDNDCBHGGETNEXTHOSTMSGDATA pCBData
1178 = reinterpret_cast<PVBOXDNDCBHGGETNEXTHOSTMSGDATA>(pvParms);
1179 AssertPtr(pCBData);
1180 AssertReturn(sizeof(VBOXDNDCBHGGETNEXTHOSTMSGDATA) == cbParms, VERR_INVALID_PARAMETER);
1181
1182 LogFlowFunc(("pCBData->uMsg=%RU32, paParms=%p, cParms=%RU32\n", pCBData->uMsg, pCBData->paParms, pCBData->cParms));
1183
1184 GuestDnDMsg *pMsg = pThis->msgQueueGetNext();
1185 if (pMsg)
1186 {
1187 /*
1188 * Sanity checks.
1189 */
1190 if ( pCBData->uMsg != uMsg
1191 || pCBData->paParms == NULL
1192 || pCBData->cParms != pMsg->getCount())
1193 {
1194 LogFlowFunc(("Current message does not match:\n"));
1195 LogFlowFunc(("\tCallback: uMsg=%RU32, cParms=%RU32, paParms=%p\n",
1196 pCBData->uMsg, pCBData->cParms, pCBData->paParms));
1197 LogFlowFunc(("\t Next: uMsg=%RU32, cParms=%RU32\n", pMsg->getType(), pMsg->getCount()));
1198
1199 /* Start over. */
1200 pThis->msgQueueClear();
1201
1202 rc = VERR_INVALID_PARAMETER;
1203 }
1204
1205 if (RT_SUCCESS(rc))
1206 {
1207 LogFlowFunc(("Returning uMsg=%RU32\n", uMsg));
1208 rc = HGCM::Message::CopyParms(pCBData->paParms, pCBData->cParms, pMsg->getParms(), pMsg->getCount(),
1209 false /* fDeepCopy */);
1210 if (RT_SUCCESS(rc))
1211 {
1212 pCBData->cParms = pMsg->getCount();
1213 pThis->msgQueueRemoveNext();
1214 }
1215 else
1216 LogFlowFunc(("Copying parameters failed with rc=%Rrc\n", rc));
1217 }
1218 }
1219 else
1220 rc = VERR_NO_DATA;
1221
1222 LogFlowFunc(("Processing next message ended with rc=%Rrc\n", rc));
1223 break;
1224 }
1225 default:
1226 rc = VERR_NOT_SUPPORTED;
1227 break;
1228 }
1229
1230 int rcToGuest = VINF_SUCCESS; /* Status which will be sent back to the guest. */
1231
1232 /*
1233 * Resolve errors.
1234 */
1235 switch (rc)
1236 {
1237 case VINF_SUCCESS:
1238 break;
1239
1240 case VINF_EOF:
1241 {
1242 LogRel2(("DnD: Transfer to guest complete\n"));
1243
1244 /* Complete operation on host side. */
1245 fNotify = true;
1246
1247 /* The guest expects VERR_NO_DATA if the transfer is complete. */
1248 rcToGuest = VERR_NO_DATA;
1249 break;
1250 }
1251
1252 case VERR_GSTDND_GUEST_ERROR:
1253 {
1254 LogRel(("DnD: Guest reported error %Rrc, aborting transfer to guest\n", rcGuest));
1255 break;
1256 }
1257
1258 case VERR_CANCELLED:
1259 {
1260 LogRel2(("DnD: Transfer to guest canceled\n"));
1261 rcToGuest = VERR_CANCELLED; /* Also cancel on guest side. */
1262 break;
1263 }
1264
1265 default:
1266 {
1267 LogRel(("DnD: Host error %Rrc occurred, aborting transfer to guest\n", rc));
1268 rcToGuest = VERR_CANCELLED; /* Also cancel on guest side. */
1269 break;
1270 }
1271 }
1272
1273 if (RT_FAILURE(rc))
1274 {
1275 /* Unregister this callback. */
1276 AssertPtr(pCtx->mpResp);
1277 int rc2 = pCtx->mpResp->setCallback(uMsg, NULL /* PFNGUESTDNDCALLBACK */);
1278 AssertRC(rc2);
1279
1280 /* Let the waiter(s) know. */
1281 fNotify = true;
1282 }
1283
1284 LogFlowFunc(("fNotify=%RTbool, rc=%Rrc, rcToGuest=%Rrc\n", fNotify, rc, rcToGuest));
1285
1286 if (fNotify)
1287 {
1288 int rc2 = pCtx->mCBEvent.Notify(rc); /** @todo Also pass guest error back? */
1289 AssertRC(rc2);
1290 }
1291
1292 LogFlowFuncLeaveRC(rc);
1293 return rcToGuest; /* Tell the guest. */
1294}
1295
1296int GuestDnDTarget::i_sendURIData(PSENDDATACTX pCtx, RTMSINTERVAL msTimeout)
1297{
1298 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1299 AssertPtr(pCtx->mpResp);
1300
1301#define REGISTER_CALLBACK(x) \
1302 do { \
1303 rc = pCtx->mpResp->setCallback(x, i_sendURIDataCallback, pCtx); \
1304 if (RT_FAILURE(rc)) \
1305 return rc; \
1306 } while (0)
1307
1308#define UNREGISTER_CALLBACK(x) \
1309 do { \
1310 int rc2 = pCtx->mpResp->setCallback(x, NULL); \
1311 AssertRC(rc2); \
1312 } while (0)
1313
1314 int rc = pCtx->mURI.init(mData.mcbBlockSize);
1315 if (RT_FAILURE(rc))
1316 return rc;
1317
1318 rc = pCtx->mCBEvent.Reset();
1319 if (RT_FAILURE(rc))
1320 return rc;
1321
1322 /*
1323 * Register callbacks.
1324 */
1325 /* Guest callbacks. */
1326 REGISTER_CALLBACK(GUEST_DND_CONNECT);
1327 REGISTER_CALLBACK(GUEST_DND_DISCONNECT);
1328 REGISTER_CALLBACK(GUEST_DND_GET_NEXT_HOST_MSG);
1329 REGISTER_CALLBACK(GUEST_DND_GH_EVT_ERROR);
1330 /* Host callbacks. */
1331 REGISTER_CALLBACK(HOST_DND_HG_SND_DIR);
1332 if (mDataBase.m_uProtocolVersion >= 2)
1333 REGISTER_CALLBACK(HOST_DND_HG_SND_FILE_HDR);
1334 REGISTER_CALLBACK(HOST_DND_HG_SND_FILE_DATA);
1335
1336 do
1337 {
1338 /*
1339 * Extract URI list from current meta data.
1340 */
1341 GuestDnDData *pData = &pCtx->mData;
1342 GuestDnDURIData *pURI = &pCtx->mURI;
1343
1344 rc = pURI->fromLocalMetaData(pData->getMeta());
1345 if (RT_FAILURE(rc))
1346 break;
1347
1348 LogFlowFunc(("URI root objects: %zu, total bytes (raw data to transfer): %zu\n",
1349 pURI->getURIList().GetRootCount(), pURI->getURIList().GetTotalBytes()));
1350
1351 /*
1352 * Set the new meta data with the URI list in it.
1353 */
1354 rc = pData->getMeta().fromURIList(pURI->getURIList());
1355 if (RT_FAILURE(rc))
1356 break;
1357
1358 /*
1359 * Set the estimated data sizes we are going to send.
1360 * The total size also contains the meta data size.
1361 */
1362 const uint32_t cbMeta = pData->getMeta().getSize();
1363 pData->setEstimatedSize(pURI->getURIList().GetTotalBytes() + cbMeta /* cbTotal */,
1364 cbMeta /* cbMeta */);
1365
1366 /*
1367 * Set the meta format.
1368 */
1369 void *pvFmt = (void *)pCtx->mFmtReq.c_str();
1370 uint32_t cbFmt = (uint32_t)pCtx->mFmtReq.length() + 1; /* Include terminating zero. */
1371
1372 pData->setFmt(pvFmt, cbFmt);
1373
1374 /*
1375 * The first message always is the data header. The meta data itself then follows
1376 * and *only* contains the root elements of an URI list.
1377 *
1378 * After the meta data we generate the messages required to send the
1379 * file/directory data itself.
1380 *
1381 * Note: Protocol < v3 use the first data message to tell what's being sent.
1382 */
1383 GuestDnDMsg Msg;
1384
1385 /*
1386 * Send the data header first.
1387 */
1388 if (mDataBase.m_uProtocolVersion >= 3)
1389 rc = i_sendDataHeader(pCtx, pData, &pCtx->mURI);
1390
1391 /*
1392 * Send the (meta) data body.
1393 */
1394 if (RT_SUCCESS(rc))
1395 rc = i_sendDataBody(pCtx, pData);
1396
1397 if (RT_SUCCESS(rc))
1398 {
1399 rc = waitForEvent(&pCtx->mCBEvent, pCtx->mpResp, msTimeout);
1400 if (RT_SUCCESS(rc))
1401 pCtx->mpResp->setProgress(100, DND_PROGRESS_COMPLETE, VINF_SUCCESS);
1402 }
1403
1404 } while (0);
1405
1406 /*
1407 * Unregister callbacks.
1408 */
1409 /* Guest callbacks. */
1410 UNREGISTER_CALLBACK(GUEST_DND_CONNECT);
1411 UNREGISTER_CALLBACK(GUEST_DND_DISCONNECT);
1412 UNREGISTER_CALLBACK(GUEST_DND_GET_NEXT_HOST_MSG);
1413 UNREGISTER_CALLBACK(GUEST_DND_GH_EVT_ERROR);
1414 /* Host callbacks. */
1415 UNREGISTER_CALLBACK(HOST_DND_HG_SND_DIR);
1416 if (mDataBase.m_uProtocolVersion >= 2)
1417 UNREGISTER_CALLBACK(HOST_DND_HG_SND_FILE_HDR);
1418 UNREGISTER_CALLBACK(HOST_DND_HG_SND_FILE_DATA);
1419
1420#undef REGISTER_CALLBACK
1421#undef UNREGISTER_CALLBACK
1422
1423 if (RT_FAILURE(rc))
1424 {
1425 if (rc == VERR_CANCELLED) /* Transfer was cancelled by the host. */
1426 {
1427 /*
1428 * Now that we've cleaned up tell the guest side to cancel.
1429 * This does not imply we're waiting for the guest to react, as the
1430 * host side never must depend on anything from the guest.
1431 */
1432 int rc2 = sendCancel();
1433 AssertRC(rc2);
1434
1435 rc2 = pCtx->mpResp->setProgress(100, DND_PROGRESS_CANCELLED, VINF_SUCCESS);
1436 AssertRC(rc2);
1437 }
1438 else if (rc != VERR_GSTDND_GUEST_ERROR) /* Guest-side error are already handled in the callback. */
1439 {
1440 int rc2 = pCtx->mpResp->setProgress(100, DND_PROGRESS_ERROR, rc,
1441 GuestDnDTarget::i_hostErrorToString(rc));
1442 AssertRC(rc2);
1443 }
1444
1445 rc = VINF_SUCCESS; /* The error was handled by the setProgress() calls above. */
1446 }
1447
1448 LogFlowFuncLeaveRC(rc);
1449 return rc;
1450}
1451
1452int GuestDnDTarget::i_sendURIDataLoop(PSENDDATACTX pCtx, GuestDnDMsg *pMsg)
1453{
1454 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1455 AssertPtrReturn(pMsg, VERR_INVALID_POINTER);
1456
1457 int rc = updateProgress(&pCtx->mData, pCtx->mpResp);
1458 AssertRC(rc);
1459
1460 if ( pCtx->mData.isComplete()
1461 && pCtx->mURI.isComplete())
1462 {
1463 return VINF_EOF;
1464 }
1465
1466 GuestDnDURIObjCtx &objCtx = pCtx->mURI.getObjCurrent();
1467 if (!objCtx.isValid())
1468 return VERR_WRONG_ORDER;
1469
1470 DnDURIObject *pCurObj = objCtx.getObj();
1471 AssertPtr(pCurObj);
1472
1473 DnDURIObject::Type enmType = pCurObj->GetType();
1474 LogRel3(("DnD: Processing: srcPath=%s, dstPath=%s, enmType=%RU32, cbSize=%RU32\n",
1475 pCurObj->GetSourcePathAbs().c_str(), pCurObj->GetDestPathAbs().c_str(),
1476 enmType, pCurObj->GetSize()));
1477
1478 if (enmType == DnDURIObject::Type_Directory)
1479 {
1480 rc = i_sendDirectory(pCtx, &objCtx, pMsg);
1481 }
1482 else if (DnDURIObject::Type_File)
1483 {
1484 rc = i_sendFile(pCtx, &objCtx, pMsg);
1485 }
1486 else
1487 {
1488 AssertMsgFailed(("enmType=%RU32 is not supported for srcPath=%s, dstPath=%s\n",
1489 enmType, pCurObj->GetSourcePathAbs().c_str(), pCurObj->GetDestPathAbs().c_str()));
1490 rc = VERR_NOT_SUPPORTED;
1491 }
1492
1493 bool fRemove = false; /* Remove current entry? */
1494 if ( pCurObj->IsComplete()
1495 || RT_FAILURE(rc))
1496 {
1497 fRemove = true;
1498 }
1499
1500 if (fRemove)
1501 {
1502 LogFlowFunc(("Removing \"%s\" from list, rc=%Rrc\n", pCurObj->GetSourcePathAbs().c_str(), rc));
1503 pCtx->mURI.removeObjCurrent();
1504 }
1505
1506 LogFlowFuncLeaveRC(rc);
1507 return rc;
1508}
1509
1510int GuestDnDTarget::i_sendRawData(PSENDDATACTX pCtx, RTMSINTERVAL msTimeout)
1511{
1512 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1513 NOREF(msTimeout);
1514
1515 GuestDnDData *pData = &pCtx->mData;
1516
1517 /** @todo At the moment we only allow sending up to 64K raw data.
1518 * For protocol v1+v2: Fix this by using HOST_DND_HG_SND_MORE_DATA.
1519 * For protocol v3 : Send another HOST_DND_HG_SND_DATA message. */
1520 if (!pData->getMeta().getSize())
1521 return VINF_SUCCESS;
1522
1523 int rc = VINF_SUCCESS;
1524
1525 /*
1526 * Send the data header first.
1527 */
1528 if (mDataBase.m_uProtocolVersion >= 3)
1529 rc = i_sendDataHeader(pCtx, pData, NULL /* URI list */);
1530
1531 /*
1532 * Send the (meta) data body.
1533 */
1534 if (RT_SUCCESS(rc))
1535 rc = i_sendDataBody(pCtx, pData);
1536
1537 int rc2;
1538 if (RT_FAILURE(rc))
1539 rc2 = pCtx->mpResp->setProgress(100, DND_PROGRESS_ERROR, rc,
1540 GuestDnDTarget::i_hostErrorToString(rc));
1541 else
1542 rc2 = pCtx->mpResp->setProgress(100, DND_PROGRESS_COMPLETE, rc);
1543 AssertRC(rc2);
1544
1545 LogFlowFuncLeaveRC(rc);
1546 return rc;
1547}
1548
1549HRESULT GuestDnDTarget::cancel(BOOL *aVeto)
1550{
1551#if !defined(VBOX_WITH_DRAG_AND_DROP)
1552 ReturnComNotImplemented();
1553#else /* VBOX_WITH_DRAG_AND_DROP */
1554
1555 int rc = i_cancelOperation();
1556
1557 if (aVeto)
1558 *aVeto = FALSE; /** @todo */
1559
1560 HRESULT hr = RT_SUCCESS(rc) ? S_OK : VBOX_E_IPRT_ERROR;
1561
1562 LogFlowFunc(("hr=%Rhrc\n", hr));
1563 return hr;
1564#endif /* VBOX_WITH_DRAG_AND_DROP */
1565}
1566
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