VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuestLib/VBoxGuestR3LibDragAndDrop.cpp@ 62842

Last change on this file since 62842 was 62470, checked in by vboxsync, 9 years ago

Additions: scm

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 76.6 KB
Line 
1/* $Id: VBoxGuestR3LibDragAndDrop.cpp 62470 2016-07-22 18:02:56Z vboxsync $ */
2/** @file
3 * VBoxGuestR3Lib - Ring-3 Support Library for VirtualBox guest additions, Drag & Drop.
4 */
5
6/*
7 * Copyright (C) 2011-2016 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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/path.h>
32#include <iprt/dir.h>
33#include <iprt/file.h>
34#include <iprt/uri.h>
35#include <iprt/thread.h>
36
37#include <iprt/cpp/list.h>
38#include <iprt/cpp/ministring.h>
39
40#ifdef LOG_GROUP
41 #undef LOG_GROUP
42#endif
43#define LOG_GROUP LOG_GROUP_GUEST_DND
44#include <VBox/log.h>
45
46#include <VBox/VBoxGuestLib.h>
47#include <VBox/GuestHost/DragAndDrop.h>
48#include <VBox/HostServices/DragAndDropSvc.h>
49
50using namespace DragAndDropSvc;
51
52#include "VBGLR3Internal.h"
53
54
55/*********************************************************************************************************************************
56* Forward declarations *
57*********************************************************************************************************************************/
58
59VBGLR3DECL(int) VbglR3DnDHGSendProgress(PVBGLR3GUESTDNDCMDCTX pCtx, uint32_t uStatus, uint8_t uPercent, int rcErr);
60
61
62/*********************************************************************************************************************************
63* Private internal functions *
64*********************************************************************************************************************************/
65
66static int vbglR3DnDGetNextMsgType(PVBGLR3GUESTDNDCMDCTX pCtx, uint32_t *puMsg, uint32_t *pcParms, bool fWait)
67{
68 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
69 AssertPtrReturn(puMsg, VERR_INVALID_POINTER);
70 AssertPtrReturn(pcParms, VERR_INVALID_POINTER);
71
72 VBOXDNDNEXTMSGMSG Msg;
73 RT_ZERO(Msg);
74 Msg.hdr.result = VERR_WRONG_ORDER;
75 Msg.hdr.u32ClientID = pCtx->uClientID;
76 Msg.hdr.u32Function = GUEST_DND_GET_NEXT_HOST_MSG;
77 Msg.hdr.cParms = 3;
78
79 Msg.uMsg.SetUInt32(0);
80 Msg.cParms.SetUInt32(0);
81 Msg.fBlock.SetUInt32(fWait ? 1 : 0);
82
83 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
84 if (RT_SUCCESS(rc))
85 {
86 rc = Msg.hdr.result;
87 if (RT_SUCCESS(rc))
88 {
89 rc = Msg.uMsg.GetUInt32(puMsg); AssertRC(rc);
90 rc = Msg.cParms.GetUInt32(pcParms); AssertRC(rc);
91 }
92 }
93
94 return rc;
95}
96
97/** @todo r=andy Clean up the parameter list. */
98static int vbglR3DnDHGRecvAction(PVBGLR3GUESTDNDCMDCTX pCtx,
99 uint32_t uMsg,
100 uint32_t *puScreenId,
101 uint32_t *puX,
102 uint32_t *puY,
103 uint32_t *puDefAction,
104 uint32_t *puAllActions,
105 char *pszFormats,
106 uint32_t cbFormats,
107 uint32_t *pcbFormatsRecv)
108{
109 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
110 AssertPtrReturn(puScreenId, VERR_INVALID_POINTER);
111 AssertPtrReturn(puX, VERR_INVALID_POINTER);
112 AssertPtrReturn(puY, VERR_INVALID_POINTER);
113 AssertPtrReturn(puDefAction, VERR_INVALID_POINTER);
114 AssertPtrReturn(puAllActions, VERR_INVALID_POINTER);
115 AssertPtrReturn(pszFormats, VERR_INVALID_POINTER);
116 AssertReturn(cbFormats, VERR_INVALID_PARAMETER);
117 AssertPtrReturn(pcbFormatsRecv, VERR_INVALID_POINTER);
118
119 VBOXDNDHGACTIONMSG Msg;
120 RT_ZERO(Msg);
121 Msg.hdr.result = VERR_WRONG_ORDER;
122 Msg.hdr.u32ClientID = pCtx->uClientID;
123 Msg.hdr.u32Function = uMsg;
124
125 if (pCtx->uProtocol < 3)
126 {
127 Msg.hdr.cParms = 7;
128
129 Msg.u.v1.uScreenId.SetUInt32(0);
130 Msg.u.v1.uX.SetUInt32(0);
131 Msg.u.v1.uY.SetUInt32(0);
132 Msg.u.v1.uDefAction.SetUInt32(0);
133 Msg.u.v1.uAllActions.SetUInt32(0);
134 Msg.u.v1.pvFormats.SetPtr(pszFormats, cbFormats);
135 Msg.u.v1.cbFormats.SetUInt32(0);
136 }
137 else
138 {
139 Msg.hdr.cParms = 8;
140
141 Msg.u.v3.uContext.SetUInt32(0);
142 Msg.u.v3.uScreenId.SetUInt32(0);
143 Msg.u.v3.uX.SetUInt32(0);
144 Msg.u.v3.uY.SetUInt32(0);
145 Msg.u.v3.uDefAction.SetUInt32(0);
146 Msg.u.v3.uAllActions.SetUInt32(0);
147 Msg.u.v3.pvFormats.SetPtr(pszFormats, cbFormats);
148 Msg.u.v3.cbFormats.SetUInt32(0);
149 }
150
151 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
152 if (RT_SUCCESS(rc))
153 {
154 rc = Msg.hdr.result;
155 if (RT_SUCCESS(rc))
156 {
157 if (pCtx->uProtocol < 3)
158 {
159 rc = Msg.u.v1.uScreenId.GetUInt32(puScreenId); AssertRC(rc);
160 rc = Msg.u.v1.uX.GetUInt32(puX); AssertRC(rc);
161 rc = Msg.u.v1.uY.GetUInt32(puY); AssertRC(rc);
162 rc = Msg.u.v1.uDefAction.GetUInt32(puDefAction); AssertRC(rc);
163 rc = Msg.u.v1.uAllActions.GetUInt32(puAllActions); AssertRC(rc);
164 rc = Msg.u.v1.cbFormats.GetUInt32(pcbFormatsRecv); AssertRC(rc);
165 }
166 else
167 {
168 /** @todo Context ID not used yet. */
169 rc = Msg.u.v3.uScreenId.GetUInt32(puScreenId); AssertRC(rc);
170 rc = Msg.u.v3.uX.GetUInt32(puX); AssertRC(rc);
171 rc = Msg.u.v3.uY.GetUInt32(puY); AssertRC(rc);
172 rc = Msg.u.v3.uDefAction.GetUInt32(puDefAction); AssertRC(rc);
173 rc = Msg.u.v3.uAllActions.GetUInt32(puAllActions); AssertRC(rc);
174 rc = Msg.u.v3.cbFormats.GetUInt32(pcbFormatsRecv); AssertRC(rc);
175 }
176
177 AssertReturn(cbFormats >= *pcbFormatsRecv, VERR_TOO_MUCH_DATA);
178 }
179 }
180
181 return rc;
182}
183
184static int vbglR3DnDHGRecvLeave(PVBGLR3GUESTDNDCMDCTX pCtx)
185{
186 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
187
188 VBOXDNDHGLEAVEMSG Msg;
189 RT_ZERO(Msg);
190 Msg.hdr.u32ClientID = pCtx->uClientID;
191 Msg.hdr.u32Function = HOST_DND_HG_EVT_LEAVE;
192
193 if (pCtx->uProtocol < 3)
194 {
195 Msg.hdr.cParms = 0;
196 }
197 else
198 {
199 Msg.hdr.cParms = 1;
200
201 /** @todo Context ID not used yet. */
202 Msg.u.v3.uContext.SetUInt32(0);
203 }
204
205 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
206 if (RT_SUCCESS(rc))
207 rc = Msg.hdr.result;
208
209 return rc;
210}
211
212static int vbglR3DnDHGRecvCancel(PVBGLR3GUESTDNDCMDCTX pCtx)
213{
214 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
215
216 VBOXDNDHGCANCELMSG Msg;
217 RT_ZERO(Msg);
218 Msg.hdr.result = VERR_WRONG_ORDER;
219 Msg.hdr.u32ClientID = pCtx->uClientID;
220 Msg.hdr.u32Function = HOST_DND_HG_EVT_CANCEL;
221
222 if (pCtx->uProtocol < 3)
223 {
224 Msg.hdr.cParms = 0;
225 }
226 else
227 {
228 Msg.hdr.cParms = 1;
229
230 /** @todo Context ID not used yet. */
231 Msg.u.v3.uContext.SetUInt32(0);
232 }
233
234 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
235 if (RT_SUCCESS(rc))
236 rc = Msg.hdr.result;
237
238 return rc;
239}
240
241static int vbglR3DnDHGRecvDir(PVBGLR3GUESTDNDCMDCTX pCtx,
242 char *pszDirname,
243 uint32_t cbDirname,
244 uint32_t *pcbDirnameRecv,
245 uint32_t *pfMode)
246{
247 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
248 AssertPtrReturn(pszDirname, VERR_INVALID_POINTER);
249 AssertReturn(cbDirname, VERR_INVALID_PARAMETER);
250 AssertPtrReturn(pcbDirnameRecv, VERR_INVALID_POINTER);
251 AssertPtrReturn(pfMode, VERR_INVALID_POINTER);
252
253 VBOXDNDHGSENDDIRMSG Msg;
254 RT_ZERO(Msg);
255 Msg.hdr.result = VERR_WRONG_ORDER;
256 Msg.hdr.u32ClientID = pCtx->uClientID;
257 Msg.hdr.u32Function = HOST_DND_HG_SND_DIR;
258
259 if (pCtx->uProtocol < 3)
260 {
261 Msg.hdr.cParms = 3;
262
263 Msg.u.v1.pvName.SetPtr(pszDirname, cbDirname);
264 Msg.u.v1.cbName.SetUInt32(cbDirname);
265 Msg.u.v1.fMode.SetUInt32(0);
266 }
267 else
268 {
269 Msg.hdr.cParms = 4;
270
271 /** @todo Context ID not used yet. */
272 Msg.u.v3.uContext.SetUInt32(0);
273 Msg.u.v3.pvName.SetPtr(pszDirname, cbDirname);
274 Msg.u.v3.cbName.SetUInt32(cbDirname);
275 Msg.u.v3.fMode.SetUInt32(0);
276 }
277
278 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
279 if (RT_SUCCESS(rc))
280 {
281 rc = Msg.hdr.result;
282 if (RT_SUCCESS(Msg.hdr.result))
283 {
284 if (pCtx->uProtocol < 3)
285 {
286 rc = Msg.u.v1.cbName.GetUInt32(pcbDirnameRecv); AssertRC(rc);
287 rc = Msg.u.v1.fMode.GetUInt32(pfMode); AssertRC(rc);
288 }
289 else
290 {
291 /** @todo Context ID not used yet. */
292 rc = Msg.u.v3.cbName.GetUInt32(pcbDirnameRecv); AssertRC(rc);
293 rc = Msg.u.v3.fMode.GetUInt32(pfMode); AssertRC(rc);
294 }
295
296 AssertReturn(cbDirname >= *pcbDirnameRecv, VERR_TOO_MUCH_DATA);
297 }
298 }
299
300 return rc;
301}
302
303static int vbglR3DnDHGRecvFileData(PVBGLR3GUESTDNDCMDCTX pCtx,
304 char *pszFilename,
305 uint32_t cbFilename,
306 uint32_t *pcbFilenameRecv,
307 void *pvData,
308 uint32_t cbData,
309 uint32_t *pcbDataRecv,
310 uint32_t *pfMode)
311{
312 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
313 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
314 AssertReturn(cbFilename, VERR_INVALID_PARAMETER);
315 AssertPtrReturn(pcbFilenameRecv, VERR_INVALID_POINTER);
316 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
317 AssertReturn(cbData, VERR_INVALID_PARAMETER);
318 AssertPtrReturn(pcbDataRecv, VERR_INVALID_POINTER);
319 AssertPtrReturn(pfMode, VERR_INVALID_POINTER);
320
321 VBOXDNDHGSENDFILEDATAMSG Msg;
322 RT_ZERO(Msg);
323 Msg.hdr.result = VERR_WRONG_ORDER;
324 Msg.hdr.u32ClientID = pCtx->uClientID;
325 Msg.hdr.u32Function = HOST_DND_HG_SND_FILE_DATA;
326
327 if (pCtx->uProtocol <= 1)
328 {
329 Msg.u.v1.pvName.SetPtr(pszFilename, cbFilename);
330 Msg.u.v1.cbName.SetUInt32(0);
331 Msg.u.v1.pvData.SetPtr(pvData, cbData);
332 Msg.u.v1.cbData.SetUInt32(0);
333 Msg.u.v1.fMode.SetUInt32(0);
334
335 Msg.hdr.cParms = 5;
336 }
337 else if (pCtx->uProtocol == 2)
338 {
339 Msg.u.v2.uContext.SetUInt32(0);
340 Msg.u.v2.pvData.SetPtr(pvData, cbData);
341 Msg.u.v2.cbData.SetUInt32(cbData);
342
343 Msg.hdr.cParms = 3;
344 }
345 else if (pCtx->uProtocol >= 3)
346 {
347 Msg.u.v3.uContext.SetUInt32(0);
348 Msg.u.v3.pvData.SetPtr(pvData, cbData);
349 Msg.u.v3.cbData.SetUInt32(0);
350 Msg.u.v3.pvChecksum.SetPtr(NULL, 0);
351 Msg.u.v3.cbChecksum.SetUInt32(0);
352
353 Msg.hdr.cParms = 5;
354 }
355 else
356 AssertMsgFailed(("Protocol %RU32 not implemented\n", pCtx->uProtocol));
357
358 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
359 if (RT_SUCCESS(rc))
360 {
361 rc = Msg.hdr.result;
362 if (RT_SUCCESS(rc))
363 {
364 if (pCtx->uProtocol <= 1)
365 {
366 rc = Msg.u.v1.cbName.GetUInt32(pcbFilenameRecv); AssertRC(rc);
367 rc = Msg.u.v1.cbData.GetUInt32(pcbDataRecv); AssertRC(rc);
368 rc = Msg.u.v1.fMode.GetUInt32(pfMode); AssertRC(rc);
369
370 AssertReturn(cbFilename >= *pcbFilenameRecv, VERR_TOO_MUCH_DATA);
371 AssertReturn(cbData >= *pcbDataRecv, VERR_TOO_MUCH_DATA);
372 }
373 else if (pCtx->uProtocol == 2)
374 {
375 /** @todo Context ID not used yet. */
376 rc = Msg.u.v2.cbData.GetUInt32(pcbDataRecv); AssertRC(rc);
377 AssertReturn(cbData >= *pcbDataRecv, VERR_TOO_MUCH_DATA);
378 }
379 else if (pCtx->uProtocol >= 3)
380 {
381 /** @todo Context ID not used yet. */
382 rc = Msg.u.v3.cbData.GetUInt32(pcbDataRecv); AssertRC(rc);
383 AssertReturn(cbData >= *pcbDataRecv, VERR_TOO_MUCH_DATA);
384 /** @todo Add checksum support. */
385 }
386 else
387 AssertMsgFailed(("Protocol %RU32 not implemented\n", pCtx->uProtocol));
388 }
389 }
390
391 return rc;
392}
393
394static int vbglR3DnDHGRecvFileHdr(PVBGLR3GUESTDNDCMDCTX pCtx,
395 char *pszFilename,
396 uint32_t cbFilename,
397 uint32_t *puFlags,
398 uint32_t *pfMode,
399 uint64_t *pcbTotal)
400{
401 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
402 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
403 AssertReturn(cbFilename, VERR_INVALID_PARAMETER);
404 AssertPtrReturn(puFlags, VERR_INVALID_POINTER);
405 AssertPtrReturn(pfMode, VERR_INVALID_POINTER);
406 AssertReturn(pcbTotal, VERR_INVALID_POINTER);
407
408 VBOXDNDHGSENDFILEHDRMSG Msg;
409 RT_ZERO(Msg);
410 Msg.hdr.result = VERR_WRONG_ORDER;
411 Msg.hdr.u32ClientID = pCtx->uClientID;
412 Msg.hdr.u32Function = HOST_DND_HG_SND_FILE_HDR;
413
414 int rc;
415
416 if (pCtx->uProtocol <= 1)
417 {
418 rc = VERR_NOT_SUPPORTED;
419 }
420 else
421 {
422 Msg.hdr.cParms = 6;
423
424 Msg.uContext.SetUInt32(0); /** @todo Not used yet. */
425 Msg.pvName.SetPtr(pszFilename, cbFilename);
426 Msg.cbName.SetUInt32(cbFilename);
427 Msg.uFlags.SetUInt32(0);
428 Msg.fMode.SetUInt32(0);
429 Msg.cbTotal.SetUInt64(0);
430
431 rc = VINF_SUCCESS;
432 }
433
434 if (RT_SUCCESS(rc))
435 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
436 if (RT_SUCCESS(rc))
437 {
438 rc = Msg.hdr.result;
439 if (RT_SUCCESS(rc))
440 {
441 /** @todo Get context ID. */
442 rc = Msg.uFlags.GetUInt32(puFlags); AssertRC(rc);
443 rc = Msg.fMode.GetUInt32(pfMode); AssertRC(rc);
444 rc = Msg.cbTotal.GetUInt64(pcbTotal); AssertRC(rc);
445 }
446 }
447
448 return rc;
449}
450
451static int vbglR3DnDHGRecvURIData(PVBGLR3GUESTDNDCMDCTX pCtx, PVBOXDNDSNDDATAHDR pDataHdr, DnDDroppedFiles *pDroppedFiles)
452{
453 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
454 AssertPtrReturn(pDataHdr, VERR_INVALID_POINTER);
455 AssertPtrReturn(pDroppedFiles, VERR_INVALID_POINTER);
456
457 /* Only count the raw data minus the already received meta data. */
458 Assert(pDataHdr->cbTotal >= pDataHdr->cbMeta);
459 uint64_t cbToRecvBytes = pDataHdr->cbTotal - pDataHdr->cbMeta;
460 uint64_t cToRecvObjs = pDataHdr->cObjects;
461
462 LogFlowFunc(("cbToRecvBytes=%RU64, cToRecvObjs=%RU64, (cbTotal=%RU64, cbMeta=%RU32)\n",
463 cbToRecvBytes, cToRecvObjs, pDataHdr->cbTotal, pDataHdr->cbMeta));
464
465 /*
466 * Only do accounting for protocol v3 and up.
467 * The older protocols did not have any data accounting available, so
468 * we simply tried to receive as much data as available and bail out.
469 */
470 const bool fDoAccounting = pCtx->uProtocol >= 3;
471
472 /* Anything to do at all? */
473 if (fDoAccounting)
474 {
475 if ( !cbToRecvBytes
476 && !cToRecvObjs)
477 {
478 return VINF_SUCCESS;
479 }
480 }
481
482 /*
483 * Allocate temporary chunk buffer.
484 */
485 uint32_t cbChunkMax = pCtx->cbMaxChunkSize;
486 void *pvChunk = RTMemAlloc(cbChunkMax);
487 if (!pvChunk)
488 return VERR_NO_MEMORY;
489 uint32_t cbChunkRead = 0;
490
491 uint64_t cbFileSize = 0; /* Total file size (in bytes). */
492 uint64_t cbFileWritten = 0; /* Written bytes. */
493
494 /*
495 * Create and query the (unique) drop target directory in the user's temporary directory.
496 */
497 int rc = pDroppedFiles->OpenTemp(0 /* fFlags */);
498 if (RT_FAILURE(rc))
499 {
500 RTMemFree(pvChunk);
501 return VERR_NO_MEMORY;
502 }
503
504 const char *pszDropDir = pDroppedFiles->GetDirAbs();
505 AssertPtr(pszDropDir);
506
507 /*
508 * Enter the main loop of retieving files + directories.
509 */
510 DnDURIObject objFile(DnDURIObject::File);
511
512 char szPathName[RTPATH_MAX] = { 0 };
513 uint32_t cbPathName = 0;
514 uint32_t fFlags = 0;
515 uint32_t fMode = 0;
516
517 /*
518 * Only wait for new incoming commands for protocol v3 and up.
519 * The older protocols did not have any data accounting available, so
520 * we simply tried to receive as much data as available and bail out.
521 */
522 const bool fWait = pCtx->uProtocol >= 3;
523
524 do
525 {
526 LogFlowFunc(("Wating for new message ...\n"));
527
528 uint32_t uNextMsg;
529 uint32_t cNextParms;
530 rc = vbglR3DnDGetNextMsgType(pCtx, &uNextMsg, &cNextParms, fWait);
531 if (RT_SUCCESS(rc))
532 {
533 LogFlowFunc(("uNextMsg=%RU32, cNextParms=%RU32\n", uNextMsg, cNextParms));
534
535 switch (uNextMsg)
536 {
537 case HOST_DND_HG_SND_DIR:
538 {
539 rc = vbglR3DnDHGRecvDir(pCtx,
540 szPathName,
541 sizeof(szPathName),
542 &cbPathName,
543 &fMode);
544 LogFlowFunc(("HOST_DND_HG_SND_DIR pszPathName=%s, cbPathName=%RU32, fMode=0x%x, rc=%Rrc\n",
545 szPathName, cbPathName, fMode, rc));
546
547 char *pszPathAbs = RTPathJoinA(pszDropDir, szPathName);
548 if (pszPathAbs)
549 {
550#ifdef RT_OS_WINDOWS
551 uint32_t fCreationMode = (fMode & RTFS_DOS_MASK) | RTFS_DOS_NT_NORMAL;
552#else
553 uint32_t fCreationMode = (fMode & RTFS_UNIX_MASK) | RTFS_UNIX_IRWXU;
554#endif
555 rc = RTDirCreate(pszPathAbs, fCreationMode, 0);
556 if (RT_SUCCESS(rc))
557 rc = pDroppedFiles->AddDir(pszPathAbs);
558
559 if ( RT_SUCCESS(rc)
560 && fDoAccounting)
561 {
562 Assert(cToRecvObjs);
563 cToRecvObjs--;
564 }
565
566 RTStrFree(pszPathAbs);
567 }
568 else
569 rc = VERR_NO_MEMORY;
570 break;
571 }
572 case HOST_DND_HG_SND_FILE_HDR:
573 case HOST_DND_HG_SND_FILE_DATA:
574 {
575 if (uNextMsg == HOST_DND_HG_SND_FILE_HDR)
576 {
577 rc = vbglR3DnDHGRecvFileHdr(pCtx,
578 szPathName,
579 sizeof(szPathName),
580 &fFlags,
581 &fMode,
582 &cbFileSize);
583 LogFlowFunc(("HOST_DND_HG_SND_FILE_HDR: "
584 "szPathName=%s, fFlags=0x%x, fMode=0x%x, cbFileSize=%RU64, rc=%Rrc\n",
585 szPathName, fFlags, fMode, cbFileSize, rc));
586 }
587 else
588 {
589 rc = vbglR3DnDHGRecvFileData(pCtx,
590 szPathName,
591 sizeof(szPathName),
592 &cbPathName,
593 pvChunk,
594 cbChunkMax,
595 &cbChunkRead,
596 &fMode);
597
598 LogFlowFunc(("HOST_DND_HG_SND_FILE_DATA: "
599 "szPathName=%s, cbPathName=%RU32, cbChunkRead=%RU32, fMode=0x%x, rc=%Rrc\n",
600 szPathName, cbPathName, cbChunkRead, fMode, rc));
601 }
602
603 if ( RT_SUCCESS(rc)
604 && ( uNextMsg == HOST_DND_HG_SND_FILE_HDR
605 /* Protocol v1 always sends the file name, so opening the file every time. */
606 || pCtx->uProtocol <= 1)
607 )
608 {
609 char *pszPathAbs = RTPathJoinA(pszDropDir, szPathName);
610 if (pszPathAbs)
611 {
612 LogFlowFunc(("Opening pszPathName=%s, cbPathName=%RU32, fMode=0x%x, cbFileSize=%zu\n",
613 szPathName, cbPathName, fMode, cbFileSize));
614
615 uint64_t fOpen = RTFILE_O_WRITE | RTFILE_O_DENY_WRITE;
616 if (pCtx->uProtocol <= 1)
617 fOpen |= RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND;
618 else
619 fOpen |= RTFILE_O_CREATE_REPLACE;
620
621 /* Is there already a file open, e.g. in transfer? */
622 if (!objFile.IsOpen())
623 {
624 RTCString strPathAbs(pszPathAbs);
625#ifdef RT_OS_WINDOWS
626 uint32_t fCreationMode = (fMode & RTFS_DOS_MASK) | RTFS_DOS_NT_NORMAL;
627#else
628 uint32_t fCreationMode = (fMode & RTFS_UNIX_MASK) | RTFS_UNIX_IRUSR | RTFS_UNIX_IWUSR;
629#endif
630 rc = objFile.OpenEx(strPathAbs, DnDURIObject::File, DnDURIObject::Target, fOpen, fCreationMode);
631 if (RT_SUCCESS(rc))
632 {
633 rc = pDroppedFiles->AddFile(strPathAbs.c_str());
634 if (RT_SUCCESS(rc))
635 {
636 cbFileWritten = 0;
637
638 if (pCtx->uProtocol >= 2) /* Set the expected file size. */
639 objFile.SetSize(cbFileSize);
640 }
641 }
642 }
643 else
644 {
645 AssertMsgFailed(("ObjType=%RU32, Proto=%RU32\n", objFile.GetType(), pCtx->uProtocol));
646 rc = VERR_WRONG_ORDER;
647 }
648
649 RTStrFree(pszPathAbs);
650 }
651 else
652 rc = VERR_NO_MEMORY;
653 }
654
655 if ( RT_SUCCESS(rc)
656 && uNextMsg == HOST_DND_HG_SND_FILE_DATA)
657 {
658 uint32_t cbChunkWritten;
659 rc = objFile.Write(pvChunk, cbChunkRead, &cbChunkWritten);
660 if (RT_SUCCESS(rc))
661 {
662 LogFlowFunc(("HOST_DND_HG_SND_FILE_DATA "
663 "cbChunkRead=%RU32, cbChunkWritten=%RU32, cbFileWritten=%RU64 cbFileSize=%RU64\n",
664 cbChunkRead, cbChunkWritten, cbFileWritten + cbChunkWritten, cbFileSize));
665
666 cbFileWritten += cbChunkWritten;
667
668 if (fDoAccounting)
669 {
670 Assert(cbChunkRead <= cbToRecvBytes);
671 cbToRecvBytes -= cbChunkRead;
672 }
673 }
674 }
675
676 bool fClose = false;
677 if (pCtx->uProtocol >= 2)
678 {
679 /* Data transfer complete? Close the file. */
680 fClose = objFile.IsComplete();
681 if ( fClose
682 && fDoAccounting)
683 {
684 Assert(cToRecvObjs);
685 cToRecvObjs--;
686 }
687
688 /* Only since protocol v2 we know the file size upfront. */
689 Assert(cbFileWritten <= cbFileSize);
690 }
691 else
692 fClose = true; /* Always close the file after each chunk. */
693
694 if (fClose)
695 {
696 LogFlowFunc(("Closing file\n"));
697 objFile.Close();
698 }
699
700 break;
701 }
702 case HOST_DND_HG_EVT_CANCEL:
703 {
704 rc = vbglR3DnDHGRecvCancel(pCtx);
705 if (RT_SUCCESS(rc))
706 rc = VERR_CANCELLED;
707 break;
708 }
709 default:
710 {
711 LogFlowFunc(("Message %RU32 not supported\n", uNextMsg));
712 rc = VERR_NOT_SUPPORTED;
713 break;
714 }
715 }
716 }
717
718 if (RT_FAILURE(rc))
719 break;
720
721 if (fDoAccounting)
722 {
723 LogFlowFunc(("cbToRecvBytes=%RU64, cToRecvObjs=%RU64\n", cbToRecvBytes, cToRecvObjs));
724 if ( !cbToRecvBytes
725 && !cToRecvObjs)
726 {
727 break;
728 }
729 }
730
731 } while (RT_SUCCESS(rc));
732
733 LogFlowFunc(("Loop ended with %Rrc\n", rc));
734
735 /* All URI data processed? */
736 if (rc == VERR_NO_DATA)
737 rc = VINF_SUCCESS;
738
739 /* Delete temp buffer again. */
740 if (pvChunk)
741 RTMemFree(pvChunk);
742
743 /* Cleanup on failure or if the user has canceled the operation or
744 * something else went wrong. */
745 if (RT_FAILURE(rc))
746 {
747 objFile.Close();
748 pDroppedFiles->Rollback();
749 }
750 else
751 {
752 /** @todo Compare the URI list with the dirs/files we really transferred. */
753 /** @todo Implement checksum verification, if any. */
754 }
755
756 /*
757 * Close the dropped files directory.
758 * Don't try to remove it here, however, as the files are being needed
759 * by the client's drag'n drop operation lateron.
760 */
761 int rc2 = pDroppedFiles->Reset(false /* fRemoveDropDir */);
762 if (RT_FAILURE(rc2)) /* Not fatal, don't report back to host. */
763 LogFlowFunc(("Closing dropped files directory failed with %Rrc\n", rc2));
764
765 LogFlowFuncLeaveRC(rc);
766 return rc;
767}
768
769static int vbglR3DnDHGRecvDataRaw(PVBGLR3GUESTDNDCMDCTX pCtx, PVBOXDNDSNDDATAHDR pDataHdr,
770 void *pvData, uint32_t cbData, uint32_t *pcbDataRecv)
771{
772 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
773 AssertPtrReturn(pDataHdr, VERR_INVALID_POINTER);
774 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
775 AssertReturn(cbData, VERR_INVALID_PARAMETER);
776 /* pcbDataRecv is optional. */
777
778 int rc;
779
780 LogFlowFunc(("pvDate=%p, cbData=%RU32\n", pvData, cbData));
781
782 VBOXDNDHGSENDDATAMSG Msg;
783 RT_ZERO(Msg);
784 Msg.hdr.result = VERR_WRONG_ORDER;
785 Msg.hdr.u32ClientID = pCtx->uClientID;
786 Msg.hdr.u32Function = HOST_DND_HG_SND_DATA;
787
788 do
789 {
790 uint32_t cbDataRecv;
791
792 if (pCtx->uProtocol < 3)
793 {
794 Msg.hdr.cParms = 5;
795
796 Msg.u.v1.uScreenId.SetUInt32(0);
797 Msg.u.v1.pvFormat.SetPtr(pDataHdr->pvMetaFmt, pDataHdr->cbMetaFmt);
798 Msg.u.v1.cbFormat.SetUInt32(0);
799 Msg.u.v1.pvData.SetPtr(pvData, cbData);
800 Msg.u.v1.cbData.SetUInt32(0);
801
802 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
803 if (RT_SUCCESS(rc))
804 {
805 rc = Msg.hdr.result;
806 if ( RT_SUCCESS(rc)
807 || rc == VERR_BUFFER_OVERFLOW)
808 {
809 rc = Msg.u.v1.uScreenId.GetUInt32(&pDataHdr->uScreenId);
810 AssertRC(rc);
811
812 /*
813 * In case of VERR_BUFFER_OVERFLOW get the data sizes required
814 * for the format + data blocks.
815 */
816 uint32_t cbFormatRecv;
817 rc = Msg.u.v1.cbFormat.GetUInt32(&cbFormatRecv);
818 AssertRC(rc);
819 if (cbFormatRecv >= pDataHdr->cbMetaFmt)
820 {
821 rc = VERR_TOO_MUCH_DATA;
822 break;
823 }
824
825 rc = Msg.u.v1.cbData.GetUInt32(&cbDataRecv);
826 AssertRC(rc);
827 if (cbDataRecv >= pDataHdr->cbMeta)
828 {
829 rc = VERR_TOO_MUCH_DATA;
830 break;
831 }
832
833 pDataHdr->cbMetaFmt = cbFormatRecv;
834 }
835 }
836
837 if (RT_FAILURE(rc))
838 break;
839 }
840 else /* Protocol v3 and up. */
841 {
842 Msg.hdr.cParms = 5;
843
844 Msg.u.v3.uContext.SetUInt32(0);
845 Msg.u.v3.pvData.SetPtr(pvData, cbData);
846 Msg.u.v3.cbData.SetUInt32(0);
847 Msg.u.v3.pvChecksum.SetPtr(NULL, 0);
848 Msg.u.v3.cbChecksum.SetUInt32(0);
849
850 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
851 if (RT_SUCCESS(rc))
852 {
853 rc = Msg.hdr.result;
854 if (RT_SUCCESS(rc))
855 {
856 rc = Msg.u.v3.cbData.GetUInt32(&cbDataRecv);
857 AssertRC(rc);
858 }
859
860 /** @todo Use checksum for validating the received data. */
861 }
862
863 if (RT_FAILURE(rc))
864 break;
865 }
866
867 if (pcbDataRecv)
868 *pcbDataRecv = cbDataRecv;
869
870 } while (0);
871
872 LogFlowFuncLeaveRC(rc);
873 return rc;
874}
875
876static int vbglR3DnDHGRecvDataHdr(PVBGLR3GUESTDNDCMDCTX pCtx, PVBOXDNDSNDDATAHDR pDataHdr)
877{
878 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
879 AssertPtrReturn(pDataHdr, VERR_INVALID_POINTER);
880
881 Assert(pCtx->uProtocol >= 3); /* Only for protocol v3 and up. */
882
883 VBOXDNDHGSENDDATAHDRMSG Msg;
884 RT_ZERO(Msg);
885 Msg.hdr.result = VERR_WRONG_ORDER;
886 Msg.hdr.u32ClientID = pCtx->uClientID;
887 Msg.hdr.u32Function = HOST_DND_HG_SND_DATA_HDR;
888 Msg.hdr.cParms = 12;
889
890 Msg.uContext.SetUInt32(0);
891 Msg.uFlags.SetUInt32(0);
892 Msg.uScreenId.SetUInt32(0);
893 Msg.cbTotal.SetUInt64(0);
894 Msg.cbMeta.SetUInt32(0);
895 Msg.pvMetaFmt.SetPtr(pDataHdr->pvMetaFmt, pDataHdr->cbMetaFmt);
896 Msg.cbMetaFmt.SetUInt32(0);
897 Msg.cObjects.SetUInt64(0);
898 Msg.enmCompression.SetUInt32(0);
899 Msg.enmChecksumType.SetUInt32(0);
900 Msg.pvChecksum.SetPtr(pDataHdr->pvChecksum, pDataHdr->cbChecksum);
901 Msg.cbChecksum.SetUInt32(0);
902
903 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
904 if (RT_SUCCESS(rc))
905 {
906 rc = Msg.hdr.result;
907 if (RT_SUCCESS(rc))
908 {
909 /* Msg.uContext not needed here. */
910 Msg.uFlags.GetUInt32(&pDataHdr->uFlags);
911 Msg.uScreenId.GetUInt32(&pDataHdr->uScreenId);
912 Msg.cbTotal.GetUInt64(&pDataHdr->cbTotal);
913 Msg.cbMeta.GetUInt32(&pDataHdr->cbMeta);
914 Msg.cbMetaFmt.GetUInt32(&pDataHdr->cbMetaFmt);
915 Msg.cObjects.GetUInt64(&pDataHdr->cObjects);
916 Msg.enmCompression.GetUInt32(&pDataHdr->enmCompression);
917 Msg.enmChecksumType.GetUInt32((uint32_t *)&pDataHdr->enmChecksumType);
918 Msg.cbChecksum.GetUInt32(&pDataHdr->cbChecksum);
919 }
920 }
921
922 LogFlowFuncLeaveRC(rc);
923 return rc;
924}
925
926/** @todo Deprecated function; will be removed. */
927static int vbglR3DnDHGRecvMoreData(PVBGLR3GUESTDNDCMDCTX pCtx,
928 void *pvData, uint32_t cbData, uint32_t *pcbDataRecv)
929{
930 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
931 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
932 AssertReturn(cbData, VERR_INVALID_PARAMETER);
933 AssertPtrReturn(pcbDataRecv, VERR_INVALID_POINTER);
934
935 VBOXDNDHGSENDMOREDATAMSG Msg;
936 RT_ZERO(Msg);
937 Msg.hdr.result = VERR_WRONG_ORDER;
938 Msg.hdr.u32ClientID = pCtx->uClientID;
939 Msg.hdr.u32Function = HOST_DND_HG_SND_MORE_DATA;
940 Msg.hdr.cParms = 2;
941
942 Msg.pvData.SetPtr(pvData, cbData);
943 Msg.cbData.SetUInt32(0);
944
945 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
946 if (RT_SUCCESS(rc))
947 {
948 rc = Msg.hdr.result;
949 if ( RT_SUCCESS(rc)
950 || rc == VERR_BUFFER_OVERFLOW)
951 {
952 rc = Msg.cbData.GetUInt32(pcbDataRecv); AssertRC(rc);
953 AssertReturn(cbData >= *pcbDataRecv, VERR_TOO_MUCH_DATA);
954 }
955 }
956 return rc;
957}
958
959static int vbglR3DnDHGRecvDataLoop(PVBGLR3GUESTDNDCMDCTX pCtx, PVBOXDNDSNDDATAHDR pDataHdr,
960 void **ppvData, uint64_t *pcbData)
961{
962 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
963 AssertPtrReturn(pDataHdr, VERR_INVALID_POINTER);
964 AssertPtrReturn(ppvData, VERR_INVALID_POINTER);
965 AssertPtrReturn(pcbData, VERR_INVALID_POINTER);
966
967 int rc;
968 uint32_t cbDataRecv;
969
970 LogFlowFuncEnter();
971
972 if (pCtx->uProtocol < 3)
973 {
974 uint64_t cbDataTmp = pCtx->cbMaxChunkSize;
975 void *pvDataTmp = RTMemAlloc(cbDataTmp);
976
977 if (!cbDataTmp)
978 return VERR_NO_MEMORY;
979
980 /**
981 * Protocols < v3 contain the header information in every HOST_DND_HG_SND_DATA
982 * message, so do the actual retrieving immediately.
983 *
984 * Also, the initial implementation used VERR_BUFFER_OVERFLOW as a return code to
985 * indicate that there will be more data coming in after the initial data chunk. There
986 * was no way of telling the total data size upfront (in form of a header or some such),
987 * so also handle this case to not break backwards compatibility.
988 */
989 rc = vbglR3DnDHGRecvDataRaw(pCtx, pDataHdr, pvDataTmp, pCtx->cbMaxChunkSize, &cbDataRecv);
990
991 /* See comment above. */
992 while (rc == VERR_BUFFER_OVERFLOW)
993 {
994 uint32_t uNextMsg;
995 uint32_t cNextParms;
996 rc = vbglR3DnDGetNextMsgType(pCtx, &uNextMsg, &cNextParms, false /* fBlock */);
997 if (RT_SUCCESS(rc))
998 {
999 switch(uNextMsg)
1000 {
1001 case HOST_DND_HG_SND_MORE_DATA:
1002 {
1003 /** @todo r=andy Don't use reallocate here; can go wrong with *really* big URI lists.
1004 * Instead send as many URI entries as possible per chunk and add those entries
1005 * to our to-process list for immediata processing. Repeat the step after processing then. */
1006 LogFlowFunc(("HOST_DND_HG_SND_MORE_DATA cbDataTotal: %RU64 -> %RU64\n",
1007 cbDataTmp, cbDataTmp + pCtx->cbMaxChunkSize));
1008 void *pvDataNew = RTMemRealloc(pvDataTmp, cbDataTmp + pCtx->cbMaxChunkSize);
1009 if (!pvDataNew)
1010 {
1011 rc = VERR_NO_MEMORY;
1012 break;
1013 }
1014
1015 pvDataTmp = pvDataNew;
1016
1017 uint8_t *pvDataOff = (uint8_t *)pvDataTmp + cbDataTmp;
1018 rc = vbglR3DnDHGRecvMoreData(pCtx, pvDataOff, pCtx->cbMaxChunkSize, &cbDataRecv);
1019 if ( RT_SUCCESS(rc)
1020 || rc == VERR_BUFFER_OVERFLOW) /* Still can return VERR_BUFFER_OVERFLOW. */
1021 {
1022 cbDataTmp += cbDataRecv;
1023 }
1024 break;
1025 }
1026 case HOST_DND_HG_EVT_CANCEL:
1027 default:
1028 {
1029 rc = vbglR3DnDHGRecvCancel(pCtx);
1030 if (RT_SUCCESS(rc))
1031 rc = VERR_CANCELLED;
1032 break;
1033 }
1034 }
1035 }
1036 }
1037
1038 if (RT_SUCCESS(rc))
1039 {
1040 /* There was no way of telling the total data size upfront
1041 * (in form of a header or some such), so set the total data size here. */
1042 pDataHdr->cbTotal = cbDataTmp;
1043
1044 *ppvData = pvDataTmp;
1045 *pcbData = cbDataTmp;
1046 }
1047 else
1048 RTMemFree(pvDataTmp);
1049 }
1050 else /* Protocol v3 and up. */
1051 {
1052 rc = vbglR3DnDHGRecvDataHdr(pCtx, pDataHdr);
1053 if (RT_SUCCESS(rc))
1054 {
1055 LogFlowFunc(("cbTotal=%RU64, cbMeta=%RU32\n", pDataHdr->cbTotal, pDataHdr->cbMeta));
1056 if (pDataHdr->cbMeta)
1057 {
1058 uint64_t cbDataTmp = 0;
1059 void *pvDataTmp = RTMemAlloc(pDataHdr->cbMeta);
1060 if (!pvDataTmp)
1061 rc = VERR_NO_MEMORY;
1062
1063 if (RT_SUCCESS(rc))
1064 {
1065 uint8_t *pvDataOff = (uint8_t *)pvDataTmp;
1066 while (cbDataTmp < pDataHdr->cbMeta)
1067 {
1068 rc = vbglR3DnDHGRecvDataRaw(pCtx, pDataHdr,
1069 pvDataOff, RT_MIN(pDataHdr->cbMeta - cbDataTmp, pCtx->cbMaxChunkSize),
1070 &cbDataRecv);
1071 if (RT_SUCCESS(rc))
1072 {
1073 LogFlowFunc(("cbDataRecv=%RU32, cbDataTmp=%RU64\n", cbDataRecv, cbDataTmp));
1074 Assert(cbDataTmp + cbDataRecv <= pDataHdr->cbMeta);
1075 cbDataTmp += cbDataRecv;
1076 pvDataOff += cbDataRecv;
1077 }
1078 else
1079 break;
1080 }
1081
1082 if (RT_SUCCESS(rc))
1083 {
1084 Assert(cbDataTmp == pDataHdr->cbMeta);
1085
1086 LogFlowFunc(("Received %RU64 bytes of data\n", cbDataTmp));
1087
1088 *ppvData = pvDataTmp;
1089 *pcbData = cbDataTmp;
1090 }
1091 else
1092 RTMemFree(pvDataTmp);
1093 }
1094 }
1095 else
1096 {
1097 *ppvData = NULL;
1098 *pcbData = 0;
1099 }
1100 }
1101 }
1102
1103 LogFlowFuncLeaveRC(rc);
1104 return rc;
1105}
1106
1107/** @todo Replace the parameters (except pCtx) with PVBOXDNDSNDDATAHDR. Later. */
1108/** @todo Hand in the DnDURIList + DnDDroppedFiles objects so that this function
1109 * can fill it directly instead of passing huge blobs of data around. */
1110static int vbglR3DnDHGRecvDataMain(PVBGLR3GUESTDNDCMDCTX pCtx,
1111 uint32_t *puScreenId,
1112 char **ppszFormat,
1113 uint32_t *pcbFormat,
1114 void **ppvData,
1115 uint32_t *pcbData)
1116{
1117 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1118 AssertPtrReturn(puScreenId, VERR_INVALID_POINTER);
1119 AssertPtrReturn(ppszFormat, VERR_INVALID_POINTER);
1120 AssertPtrReturn(pcbFormat, VERR_INVALID_POINTER);
1121 AssertPtrReturn(ppvData, VERR_INVALID_POINTER);
1122 AssertPtrReturn(pcbData, VERR_INVALID_POINTER);
1123
1124 VBOXDNDDATAHDR dataHdr; /** @todo See todo above. */
1125 RT_ZERO(dataHdr);
1126
1127 dataHdr.cbMetaFmt = _64K; /** @todo Make this configurable? */
1128 dataHdr.pvMetaFmt = RTMemAlloc(dataHdr.cbMetaFmt);
1129 if (!dataHdr.pvMetaFmt)
1130 return VERR_NO_MEMORY;
1131
1132 DnDURIList lstURI;
1133 DnDDroppedFiles droppedFiles;
1134
1135 void *pvData; /** @todo See todo above. */
1136 uint64_t cbData; /** @todo See todo above. */
1137
1138 int rc = vbglR3DnDHGRecvDataLoop(pCtx, &dataHdr, &pvData, &cbData);
1139 if (RT_SUCCESS(rc))
1140 {
1141 /**
1142 * Check if this is an URI event. If so, let VbglR3 do all the actual
1143 * data transfer + file/directory creation internally without letting
1144 * the caller know.
1145 *
1146 * This keeps the actual (guest OS-)dependent client (like VBoxClient /
1147 * VBoxTray) small by not having too much redundant code.
1148 */
1149 Assert(dataHdr.cbMetaFmt);
1150 AssertPtr(dataHdr.pvMetaFmt);
1151 if (DnDMIMEHasFileURLs((char *)dataHdr.pvMetaFmt, dataHdr.cbMetaFmt))
1152 {
1153 AssertPtr(pvData);
1154 Assert(cbData);
1155 rc = lstURI.RootFromURIData(pvData, cbData, 0 /* fFlags */);
1156 if (RT_SUCCESS(rc))
1157 rc = vbglR3DnDHGRecvURIData(pCtx, &dataHdr, &droppedFiles);
1158
1159 if (RT_SUCCESS(rc)) /** @todo Remove this block as soon as we hand in DnDURIList. */
1160 {
1161 if (pvData)
1162 {
1163 /* Reuse data buffer to fill in the transformed URI file list. */
1164 RTMemFree(pvData);
1165 pvData = NULL;
1166 }
1167
1168 RTCString strData = lstURI.RootToString(droppedFiles.GetDirAbs());
1169 Assert(!strData.isEmpty());
1170
1171 cbData = strData.length() + 1;
1172 LogFlowFunc(("URI list has %zu bytes\n", cbData));
1173
1174 pvData = RTMemAlloc(cbData);
1175 if (pvData)
1176 {
1177 memcpy(pvData, strData.c_str(), cbData);
1178 }
1179 else
1180 rc = VERR_NO_MEMORY;
1181 }
1182 }
1183 else /* Raw data. */
1184 {
1185 const uint32_t cbDataRaw = dataHdr.cbMetaFmt;
1186 if (cbData >= cbDataRaw)
1187 {
1188 if (cbDataRaw)
1189 memcpy(pvData, dataHdr.pvMetaFmt, cbDataRaw);
1190 cbData = cbDataRaw;
1191 }
1192 else
1193 rc = VERR_BUFFER_OVERFLOW;
1194 }
1195 }
1196
1197 if ( RT_FAILURE(rc)
1198 && rc != VERR_CANCELLED)
1199 {
1200 if (dataHdr.pvMetaFmt)
1201 RTMemFree(dataHdr.pvMetaFmt);
1202 if (pvData)
1203 RTMemFree(pvData);
1204
1205 int rc2 = VbglR3DnDHGSendProgress(pCtx, DND_PROGRESS_ERROR, 100 /* Percent */, rc);
1206 if (RT_FAILURE(rc2))
1207 LogFlowFunc(("Unable to send progress error %Rrc to host: %Rrc\n", rc, rc2));
1208 }
1209 else if (RT_SUCCESS(rc))
1210 {
1211 *ppszFormat = (char *)dataHdr.pvMetaFmt;
1212 *pcbFormat = dataHdr.cbMetaFmt;
1213 *ppvData = pvData;
1214 *pcbData = cbData;
1215 }
1216
1217 LogFlowFuncLeaveRC(rc);
1218 return rc;
1219}
1220
1221static int vbglR3DnDGHRecvPending(PVBGLR3GUESTDNDCMDCTX pCtx, uint32_t *puScreenId)
1222{
1223 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1224 AssertPtrReturn(puScreenId, VERR_INVALID_POINTER);
1225
1226 VBOXDNDGHREQPENDINGMSG Msg;
1227 RT_ZERO(Msg);
1228 Msg.hdr.result = VERR_WRONG_ORDER;
1229 Msg.hdr.u32ClientID = pCtx->uClientID;
1230 Msg.hdr.u32Function = HOST_DND_GH_REQ_PENDING;
1231
1232 if (pCtx->uProtocol < 3)
1233 {
1234 Msg.hdr.cParms = 1;
1235
1236 Msg.u.v1.uScreenId.SetUInt32(0);
1237 }
1238 else
1239 {
1240 Msg.hdr.cParms = 2;
1241
1242 /** @todo Context ID not used yet. */
1243 Msg.u.v3.uContext.SetUInt32(0);
1244 Msg.u.v3.uScreenId.SetUInt32(0);
1245 }
1246
1247 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1248 if (RT_SUCCESS(rc))
1249 {
1250 rc = Msg.hdr.result;
1251 if (RT_SUCCESS(rc))
1252 {
1253 if (pCtx->uProtocol < 3)
1254 {
1255 rc = Msg.u.v1.uScreenId.GetUInt32(puScreenId); AssertRC(rc);
1256 }
1257 else
1258 {
1259 /** @todo Context ID not used yet. */
1260 rc = Msg.u.v3.uContext.GetUInt32(puScreenId); AssertRC(rc);
1261 }
1262 }
1263 }
1264
1265 return rc;
1266}
1267
1268static int vbglR3DnDGHRecvDropped(PVBGLR3GUESTDNDCMDCTX pCtx,
1269 char *pszFormat,
1270 uint32_t cbFormat,
1271 uint32_t *pcbFormatRecv,
1272 uint32_t *puAction)
1273{
1274 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1275 AssertPtrReturn(pszFormat, VERR_INVALID_POINTER);
1276 AssertReturn(cbFormat, VERR_INVALID_PARAMETER);
1277 AssertPtrReturn(pcbFormatRecv, VERR_INVALID_POINTER);
1278 AssertPtrReturn(puAction, VERR_INVALID_POINTER);
1279
1280 VBOXDNDGHDROPPEDMSG Msg;
1281 RT_ZERO(Msg);
1282 Msg.hdr.result = VERR_WRONG_ORDER;
1283 Msg.hdr.u32ClientID = pCtx->uClientID;
1284 Msg.hdr.u32Function = HOST_DND_GH_EVT_DROPPED;
1285
1286 if (pCtx->uProtocol < 3)
1287 {
1288 Msg.hdr.cParms = 3;
1289
1290 Msg.u.v1.pvFormat.SetPtr(pszFormat, cbFormat);
1291 Msg.u.v1.cbFormat.SetUInt32(0);
1292 Msg.u.v1.uAction.SetUInt32(0);
1293 }
1294 else
1295 {
1296 Msg.hdr.cParms = 4;
1297
1298 Msg.u.v3.uContext.SetUInt32(0);
1299 Msg.u.v3.pvFormat.SetPtr(pszFormat, cbFormat);
1300 Msg.u.v3.cbFormat.SetUInt32(0);
1301 Msg.u.v3.uAction.SetUInt32(0);
1302 }
1303
1304 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1305 if (RT_SUCCESS(rc))
1306 {
1307 rc = Msg.hdr.result;
1308 if (RT_SUCCESS(rc))
1309 {
1310 if (pCtx->uProtocol < 3)
1311 {
1312 rc = Msg.u.v1.cbFormat.GetUInt32(pcbFormatRecv); AssertRC(rc);
1313 rc = Msg.u.v1.uAction.GetUInt32(puAction); AssertRC(rc);
1314 }
1315 else
1316 {
1317 /** @todo Context ID not used yet. */
1318 rc = Msg.u.v3.cbFormat.GetUInt32(pcbFormatRecv); AssertRC(rc);
1319 rc = Msg.u.v3.uAction.GetUInt32(puAction); AssertRC(rc);
1320 }
1321
1322 AssertReturn(cbFormat >= *pcbFormatRecv, VERR_TOO_MUCH_DATA);
1323 }
1324 }
1325
1326 return rc;
1327}
1328
1329
1330/*********************************************************************************************************************************
1331* Public functions *
1332*********************************************************************************************************************************/
1333
1334VBGLR3DECL(int) VbglR3DnDConnect(PVBGLR3GUESTDNDCMDCTX pCtx)
1335{
1336 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1337
1338 /* Initialize header */
1339 VBoxGuestHGCMConnectInfo Info;
1340 RT_ZERO(Info.Loc.u);
1341 Info.result = VERR_WRONG_ORDER;
1342 Info.u32ClientID = UINT32_MAX; /* try make valgrind shut up. */
1343 Info.Loc.type = VMMDevHGCMLoc_LocalHost_Existing;
1344
1345 int rc = RTStrCopy(Info.Loc.u.host.achName, sizeof(Info.Loc.u.host.achName), "VBoxDragAndDropSvc");
1346 if (RT_FAILURE(rc))
1347 return rc;
1348
1349 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CONNECT, &Info, sizeof(Info));
1350 if (RT_SUCCESS(rc))
1351 {
1352 rc = Info.result;
1353 if (RT_SUCCESS(rc))
1354 {
1355 /* Set the default protocol version to use. */
1356 pCtx->uProtocol = 3;
1357
1358 Assert(Info.u32ClientID);
1359 pCtx->uClientID = Info.u32ClientID;
1360 }
1361 }
1362
1363 if (RT_FAILURE(rc))
1364 return rc;
1365
1366 /*
1367 * Get the VM's session ID.
1368 * This is not fatal in case we're running with an ancient VBox version.
1369 */
1370 pCtx->uSessionID = 0;
1371 int rc2 = VbglR3GetSessionId(&pCtx->uSessionID);
1372 LogFlowFunc(("uSessionID=%RU64, rc=%Rrc\n", pCtx->uSessionID, rc2));
1373
1374 /*
1375 * Check if the host is >= VBox 5.0 which in case supports GUEST_DND_CONNECT.
1376 */
1377 bool fSupportsConnectReq = false;
1378 if (RT_SUCCESS(rc))
1379 {
1380 /* The guest property service might not be available. Not fatal. */
1381 uint32_t uGuestPropSvcClientID;
1382 rc2 = VbglR3GuestPropConnect(&uGuestPropSvcClientID);
1383 if (RT_SUCCESS(rc2))
1384 {
1385 char *pszHostVersion;
1386 rc2 = VbglR3GuestPropReadValueAlloc(uGuestPropSvcClientID, "/VirtualBox/HostInfo/VBoxVer", &pszHostVersion);
1387 if (RT_SUCCESS(rc2))
1388 {
1389 fSupportsConnectReq = RTStrVersionCompare(pszHostVersion, "5.0") >= 0;
1390 LogFlowFunc(("pszHostVersion=%s, fSupportsConnectReq=%RTbool\n", pszHostVersion, fSupportsConnectReq));
1391 VbglR3GuestPropReadValueFree(pszHostVersion);
1392 }
1393
1394 VbglR3GuestPropDisconnect(uGuestPropSvcClientID);
1395 }
1396
1397 if (RT_FAILURE(rc2))
1398 LogFlowFunc(("Retrieving host version failed with rc=%Rrc\n", rc2));
1399 }
1400
1401 if (fSupportsConnectReq)
1402 {
1403 /*
1404 * Try sending the connect message to tell the protocol version to use.
1405 * Note: This might fail when the Guest Additions run on an older VBox host (< VBox 5.0) which
1406 * does not implement this command.
1407 */
1408 VBOXDNDCONNECTMSG Msg;
1409 RT_ZERO(Msg);
1410 Msg.hdr.result = VERR_WRONG_ORDER;
1411 Msg.hdr.u32ClientID = pCtx->uClientID;
1412 Msg.hdr.u32Function = GUEST_DND_CONNECT;
1413
1414 if (pCtx->uProtocol < 3)
1415 {
1416 Msg.hdr.cParms = 2;
1417
1418 Msg.u.v2.uProtocol.SetUInt32(pCtx->uProtocol);
1419 Msg.u.v2.uFlags.SetUInt32(0); /* Unused at the moment. */
1420 }
1421 else
1422 {
1423 Msg.hdr.cParms = 3;
1424
1425 /** @todo Context ID not used yet. */
1426 Msg.u.v3.uContext.SetUInt32(0);
1427 Msg.u.v3.uProtocol.SetUInt32(pCtx->uProtocol);
1428 Msg.u.v3.uFlags.SetUInt32(0); /* Unused at the moment. */
1429 }
1430
1431 rc2 = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1432 if (RT_SUCCESS(rc2))
1433 rc2 = Msg.hdr.result; /* Not fatal. */
1434
1435 if (RT_FAILURE(rc2))
1436 fSupportsConnectReq = false;
1437
1438 LogFlowFunc(("Connection request ended with rc=%Rrc\n", rc2));
1439 }
1440
1441 /* GUEST_DND_CONNECT not supported; play safe here and just use protocol v1. */
1442 if (!fSupportsConnectReq)
1443 pCtx->uProtocol = 1; /* Fall back to protocol version 1 (< VBox 5.0). */
1444
1445 pCtx->cbMaxChunkSize = _64K; /** @todo Use a scratch buffer on the heap? */
1446
1447 LogFlowFunc(("uClient=%RU32, uProtocol=%RU32, rc=%Rrc\n", pCtx->uClientID, pCtx->uProtocol, rc));
1448 return rc;
1449}
1450
1451VBGLR3DECL(int) VbglR3DnDDisconnect(PVBGLR3GUESTDNDCMDCTX pCtx)
1452{
1453 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1454
1455 VBoxGuestHGCMDisconnectInfo Info;
1456 Info.result = VERR_WRONG_ORDER;
1457 Info.u32ClientID = pCtx->uClientID;
1458
1459 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_DISCONNECT, &Info, sizeof(Info));
1460 if (RT_SUCCESS(rc))
1461 rc = Info.result;
1462
1463 return rc;
1464}
1465
1466VBGLR3DECL(int) VbglR3DnDRecvNextMsg(PVBGLR3GUESTDNDCMDCTX pCtx, CPVBGLR3DNDHGCMEVENT pEvent)
1467{
1468 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1469 AssertPtrReturn(pEvent, VERR_INVALID_POINTER);
1470
1471 uint32_t uMsg = 0;
1472 uint32_t uNumParms = 0;
1473
1474 const uint32_t cbDataMax = pCtx->cbMaxChunkSize;
1475 const uint32_t cbFormatMax = pCtx->cbMaxChunkSize;
1476
1477 int rc = vbglR3DnDGetNextMsgType(pCtx, &uMsg, &uNumParms, true /* fWait */);
1478 if (RT_SUCCESS(rc))
1479 {
1480 /* Check for VM session change. */
1481 uint64_t uSessionID;
1482 int rc2 = VbglR3GetSessionId(&uSessionID);
1483 if ( RT_SUCCESS(rc2)
1484 && (uSessionID != pCtx->uSessionID))
1485 {
1486 LogFlowFunc(("VM session ID changed to %RU64, doing reconnect\n", uSessionID));
1487
1488 /* Try a reconnect to the DnD service. */
1489 rc2 = VbglR3DnDDisconnect(pCtx);
1490 AssertRC(rc2);
1491 rc2 = VbglR3DnDConnect(pCtx);
1492 AssertRC(rc2);
1493
1494 /* At this point we continue processing the messsages with the new client ID. */
1495 }
1496 }
1497
1498 if (RT_SUCCESS(rc))
1499 {
1500 pEvent->uType = uMsg;
1501
1502 switch(uMsg)
1503 {
1504 case HOST_DND_HG_EVT_ENTER:
1505 case HOST_DND_HG_EVT_MOVE:
1506 case HOST_DND_HG_EVT_DROPPED:
1507 {
1508 pEvent->pszFormats = static_cast<char*>(RTMemAlloc(cbFormatMax));
1509 if (!pEvent->pszFormats)
1510 rc = VERR_NO_MEMORY;
1511
1512 if (RT_SUCCESS(rc))
1513 rc = vbglR3DnDHGRecvAction(pCtx,
1514 uMsg,
1515 &pEvent->uScreenId,
1516 &pEvent->u.a.uXpos,
1517 &pEvent->u.a.uYpos,
1518 &pEvent->u.a.uDefAction,
1519 &pEvent->u.a.uAllActions,
1520 pEvent->pszFormats,
1521 cbFormatMax,
1522 &pEvent->cbFormats);
1523 break;
1524 }
1525 case HOST_DND_HG_EVT_LEAVE:
1526 {
1527 rc = vbglR3DnDHGRecvLeave(pCtx);
1528 break;
1529 }
1530 case HOST_DND_HG_SND_DATA:
1531 /* Protocol v1 + v2: Also contains the header data.
1532 * Note: Fall through is intentional. */
1533 case HOST_DND_HG_SND_DATA_HDR:
1534 {
1535 rc = vbglR3DnDHGRecvDataMain(pCtx,
1536 /* Screen ID */
1537 &pEvent->uScreenId,
1538 /* Format */
1539 &pEvent->pszFormats,
1540 &pEvent->cbFormats,
1541 /* Data */
1542 &pEvent->u.b.pvData,
1543 &pEvent->u.b.cbData);
1544 break;
1545 }
1546 case HOST_DND_HG_SND_MORE_DATA:
1547 case HOST_DND_HG_SND_DIR:
1548 case HOST_DND_HG_SND_FILE_DATA:
1549 {
1550 /*
1551 * All messages in this case are handled internally
1552 * by vbglR3DnDHGRecvDataMain() and must be specified
1553 * by a preceding HOST_DND_HG_SND_DATA or HOST_DND_HG_SND_DATA_HDR
1554 * calls.
1555 */
1556 rc = VERR_WRONG_ORDER;
1557 break;
1558 }
1559 case HOST_DND_HG_EVT_CANCEL:
1560 {
1561 rc = vbglR3DnDHGRecvCancel(pCtx);
1562 break;
1563 }
1564#ifdef VBOX_WITH_DRAG_AND_DROP_GH
1565 case HOST_DND_GH_REQ_PENDING:
1566 {
1567 rc = vbglR3DnDGHRecvPending(pCtx, &pEvent->uScreenId);
1568 break;
1569 }
1570 case HOST_DND_GH_EVT_DROPPED:
1571 {
1572 pEvent->pszFormats = static_cast<char*>(RTMemAlloc(cbFormatMax));
1573 if (!pEvent->pszFormats)
1574 rc = VERR_NO_MEMORY;
1575
1576 if (RT_SUCCESS(rc))
1577 rc = vbglR3DnDGHRecvDropped(pCtx,
1578 pEvent->pszFormats,
1579 cbFormatMax,
1580 &pEvent->cbFormats,
1581 &pEvent->u.a.uDefAction);
1582 break;
1583 }
1584#endif
1585 default:
1586 {
1587 rc = VERR_NOT_SUPPORTED;
1588 break;
1589 }
1590 }
1591 }
1592
1593 if (RT_FAILURE(rc))
1594 LogFlowFunc(("Returning error %Rrc\n", rc));
1595 return rc;
1596}
1597
1598VBGLR3DECL(int) VbglR3DnDHGSendAckOp(PVBGLR3GUESTDNDCMDCTX pCtx, uint32_t uAction)
1599{
1600 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1601
1602 VBOXDNDHGACKOPMSG Msg;
1603 RT_ZERO(Msg);
1604 Msg.hdr.result = VERR_WRONG_ORDER;
1605 Msg.hdr.u32ClientID = pCtx->uClientID;
1606 Msg.hdr.u32Function = GUEST_DND_HG_ACK_OP;
1607
1608 LogFlowFunc(("uProto=%RU32\n", pCtx->uProtocol));
1609
1610 if (pCtx->uProtocol < 3)
1611 {
1612 Msg.hdr.cParms = 1;
1613
1614 Msg.u.v1.uAction.SetUInt32(uAction);
1615 }
1616 else
1617 {
1618 Msg.hdr.cParms = 2;
1619
1620 /** @todo Context ID not used yet. */
1621 Msg.u.v3.uContext.SetUInt32(0);
1622 Msg.u.v3.uAction.SetUInt32(uAction);
1623 }
1624
1625 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1626 if (RT_SUCCESS(rc))
1627 rc = Msg.hdr.result;
1628
1629 return rc;
1630}
1631
1632VBGLR3DECL(int) VbglR3DnDHGSendReqData(PVBGLR3GUESTDNDCMDCTX pCtx, const char* pcszFormat)
1633{
1634 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1635 AssertPtrReturn(pcszFormat, VERR_INVALID_POINTER);
1636
1637 VBOXDNDHGREQDATAMSG Msg;
1638 RT_ZERO(Msg);
1639 Msg.hdr.result = VERR_WRONG_ORDER;
1640 Msg.hdr.u32ClientID = pCtx->uClientID;
1641 Msg.hdr.u32Function = GUEST_DND_HG_REQ_DATA;
1642
1643 if (!RTStrIsValidEncoding(pcszFormat))
1644 return VERR_INVALID_PARAMETER;
1645
1646 const uint32_t cbFormat = (uint32_t)strlen(pcszFormat) + 1; /* Include termination */
1647
1648 if (pCtx->uProtocol < 3)
1649 {
1650 Msg.hdr.cParms = 1;
1651
1652 Msg.u.v1.pvFormat.SetPtr((void*)pcszFormat, cbFormat);
1653 }
1654 else
1655 {
1656 Msg.hdr.cParms = 3;
1657
1658 /** @todo Context ID not used yet. */
1659 Msg.u.v3.uContext.SetUInt32(0);
1660 Msg.u.v3.pvFormat.SetPtr((void*)pcszFormat, cbFormat);
1661 Msg.u.v3.cbFormat.SetUInt32(cbFormat);
1662 }
1663
1664 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1665 if (RT_SUCCESS(rc))
1666 rc = Msg.hdr.result;
1667
1668 return rc;
1669}
1670
1671VBGLR3DECL(int) VbglR3DnDHGSendProgress(PVBGLR3GUESTDNDCMDCTX pCtx, uint32_t uStatus, uint8_t uPercent, int rcErr)
1672{
1673 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1674 AssertReturn(uStatus > DND_PROGRESS_UNKNOWN, VERR_INVALID_PARAMETER);
1675
1676 VBOXDNDHGEVTPROGRESSMSG Msg;
1677 RT_ZERO(Msg);
1678 Msg.hdr.result = VERR_WRONG_ORDER;
1679 Msg.hdr.u32ClientID = pCtx->uClientID;
1680 Msg.hdr.u32Function = uStatus;
1681
1682 if (pCtx->uProtocol < 3)
1683 {
1684 Msg.hdr.cParms = 3;
1685
1686 Msg.u.v1.uStatus.SetUInt32(uStatus);
1687 Msg.u.v1.uPercent.SetUInt32(uPercent);
1688 Msg.u.v1.rc.SetUInt32((uint32_t)rcErr); /* uint32_t vs. int. */
1689 }
1690 else
1691 {
1692 Msg.hdr.cParms = 4;
1693
1694 /** @todo Context ID not used yet. */
1695 Msg.u.v3.uContext.SetUInt32(0);
1696 Msg.u.v3.uStatus.SetUInt32(uStatus);
1697 Msg.u.v3.uPercent.SetUInt32(uPercent);
1698 Msg.u.v3.rc.SetUInt32((uint32_t)rcErr); /* uint32_t vs. int. */
1699 }
1700
1701 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1702 if (RT_SUCCESS(rc))
1703 rc = Msg.hdr.result;
1704
1705 return rc;
1706}
1707
1708#ifdef VBOX_WITH_DRAG_AND_DROP_GH
1709VBGLR3DECL(int) VbglR3DnDGHSendAckPending(PVBGLR3GUESTDNDCMDCTX pCtx,
1710 uint32_t uDefAction, uint32_t uAllActions, const char* pcszFormats, uint32_t cbFormats)
1711{
1712 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1713 AssertPtrReturn(pcszFormats, VERR_INVALID_POINTER);
1714 AssertReturn(cbFormats, VERR_INVALID_PARAMETER);
1715
1716 if (!RTStrIsValidEncoding(pcszFormats))
1717 return VERR_INVALID_UTF8_ENCODING;
1718
1719 VBOXDNDGHACKPENDINGMSG Msg;
1720 RT_ZERO(Msg);
1721 Msg.hdr.result = VERR_WRONG_ORDER;
1722 Msg.hdr.u32ClientID = pCtx->uClientID;
1723 Msg.hdr.u32Function = GUEST_DND_GH_ACK_PENDING;
1724
1725 if (pCtx->uProtocol < 3)
1726 {
1727 Msg.hdr.cParms = 3;
1728
1729 Msg.u.v1.uDefAction.SetUInt32(uDefAction);
1730 Msg.u.v1.uAllActions.SetUInt32(uAllActions);
1731 Msg.u.v1.pvFormats.SetPtr((void*)pcszFormats, cbFormats);
1732 }
1733 else
1734 {
1735 Msg.hdr.cParms = 5;
1736
1737 /** @todo Context ID not used yet. */
1738 Msg.u.v3.uContext.SetUInt32(0);
1739 Msg.u.v3.uDefAction.SetUInt32(uDefAction);
1740 Msg.u.v3.uAllActions.SetUInt32(uAllActions);
1741 Msg.u.v3.pvFormats.SetPtr((void*)pcszFormats, cbFormats);
1742 Msg.u.v3.cbFormats.SetUInt32(cbFormats);
1743 }
1744
1745 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1746 if (RT_SUCCESS(rc))
1747 rc = Msg.hdr.result;
1748
1749 return rc;
1750}
1751
1752static int vbglR3DnDGHSendDataInternal(PVBGLR3GUESTDNDCMDCTX pCtx,
1753 void *pvData, uint64_t cbData, PVBOXDNDSNDDATAHDR pDataHdr)
1754{
1755 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1756 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
1757 AssertReturn(cbData, VERR_INVALID_PARAMETER);
1758 /* cbAdditionalData is optional. */
1759 /* pDataHdr is optional in protocols < v3. */
1760
1761 int rc = VINF_SUCCESS;
1762
1763 /* For protocol v3 and up we need to send the data header first. */
1764 if (pCtx->uProtocol >= 3)
1765 {
1766 AssertPtrReturn(pDataHdr, VERR_INVALID_POINTER);
1767
1768 VBOXDNDGHSENDDATAHDRMSG Msg;
1769 RT_ZERO(Msg);
1770 Msg.hdr.result = VERR_WRONG_ORDER;
1771 Msg.hdr.u32ClientID = pCtx->uClientID;
1772 Msg.hdr.u32Function = GUEST_DND_GH_SND_DATA_HDR;
1773 Msg.hdr.cParms = 12;
1774
1775 Msg.uContext.SetUInt32(0); /** @todo Not used yet. */
1776 Msg.uFlags.SetUInt32(0); /** @todo Not used yet. */
1777 Msg.uScreenId.SetUInt32(0); /** @todo Not used for guest->host (yet). */
1778 Msg.cbTotal.SetUInt64(pDataHdr->cbTotal);
1779 Msg.cbMeta.SetUInt32(pDataHdr->cbMeta);
1780 Msg.pvMetaFmt.SetPtr(pDataHdr->pvMetaFmt, pDataHdr->cbMetaFmt);
1781 Msg.cbMetaFmt.SetUInt32(pDataHdr->cbMetaFmt);
1782 Msg.cObjects.SetUInt64(pDataHdr->cObjects);
1783 Msg.enmCompression.SetUInt32(0); /** @todo Not used yet. */
1784 Msg.enmChecksumType.SetUInt32(RTDIGESTTYPE_INVALID); /** @todo Not used yet. */
1785 Msg.pvChecksum.SetPtr(NULL, 0); /** @todo Not used yet. */
1786 Msg.cbChecksum.SetUInt32(0); /** @todo Not used yet. */
1787
1788 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1789 if (RT_SUCCESS(rc))
1790 rc = Msg.hdr.result;
1791
1792 LogFlowFunc(("cbTotal=%RU64, cbMeta=%RU32, cObjects=%RU64, rc=%Rrc\n",
1793 pDataHdr->cbTotal, pDataHdr->cbMeta, pDataHdr->cObjects, rc));
1794 }
1795
1796 if (RT_SUCCESS(rc))
1797 {
1798 VBOXDNDGHSENDDATAMSG Msg;
1799 RT_ZERO(Msg);
1800 Msg.hdr.result = VERR_WRONG_ORDER;
1801 Msg.hdr.u32ClientID = pCtx->uClientID;
1802 Msg.hdr.u32Function = GUEST_DND_GH_SND_DATA;
1803
1804 if (pCtx->uProtocol >= 3)
1805 {
1806 Msg.hdr.cParms = 5;
1807
1808 Msg.u.v3.uContext.SetUInt32(0); /** @todo Not used yet. */
1809 Msg.u.v3.pvChecksum.SetPtr(NULL, 0); /** @todo Not used yet. */
1810 Msg.u.v3.cbChecksum.SetUInt32(0); /** @todo Not used yet. */
1811 }
1812 else
1813 {
1814 Msg.hdr.cParms = 2;
1815
1816 /* Total amount of bytes to send (meta data + all directory/file objects). */
1817 /* Note: Only supports uint32_t, so this is *not* a typo. */
1818 Msg.u.v1.cbTotalBytes.SetUInt32((uint32_t)pDataHdr->cbTotal);
1819 }
1820
1821 uint32_t cbCurChunk;
1822 const uint32_t cbMaxChunk = pCtx->cbMaxChunkSize;
1823 uint32_t cbSent = 0;
1824
1825 HGCMFunctionParameter *pParm = (pCtx->uProtocol >= 3)
1826 ? &Msg.u.v3.pvData
1827 : &Msg.u.v1.pvData;
1828 while (cbSent < cbData)
1829 {
1830 cbCurChunk = RT_MIN(cbData - cbSent, cbMaxChunk);
1831 pParm->SetPtr(static_cast<uint8_t *>(pvData) + cbSent, cbCurChunk);
1832 if (pCtx->uProtocol > 2)
1833 Msg.u.v3.cbData.SetUInt32(cbCurChunk);
1834
1835 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1836 if (RT_SUCCESS(rc))
1837 rc = Msg.hdr.result;
1838
1839 if (RT_FAILURE(rc))
1840 break;
1841
1842 cbSent += cbCurChunk;
1843 }
1844
1845 LogFlowFunc(("cbMaxChunk=%RU32, cbData=%RU64, cbSent=%RU32, rc=%Rrc\n",
1846 cbMaxChunk, cbData, cbSent, rc));
1847
1848 if (RT_SUCCESS(rc))
1849 Assert(cbSent == cbData);
1850 }
1851
1852 LogFlowFuncLeaveRC(rc);
1853 return rc;
1854}
1855
1856static int vbglR3DnDGHSendDir(PVBGLR3GUESTDNDCMDCTX pCtx, DnDURIObject *pObj)
1857{
1858 AssertPtrReturn(pObj, VERR_INVALID_POINTER);
1859 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1860 AssertReturn(pObj->GetType() == DnDURIObject::Directory, VERR_INVALID_PARAMETER);
1861
1862 RTCString strPath = pObj->GetDestPath();
1863 LogFlowFunc(("strDir=%s (%zu), fMode=0x%x\n",
1864 strPath.c_str(), strPath.length(), pObj->GetMode()));
1865
1866 if (strPath.length() > RTPATH_MAX)
1867 return VERR_INVALID_PARAMETER;
1868
1869 const uint32_t cbPath = (uint32_t)strPath.length() + 1; /* Include termination. */
1870
1871 VBOXDNDGHSENDDIRMSG Msg;
1872 RT_ZERO(Msg);
1873 Msg.hdr.result = VERR_WRONG_ORDER;
1874 Msg.hdr.u32ClientID = pCtx->uClientID;
1875 Msg.hdr.u32Function = GUEST_DND_GH_SND_DIR;
1876
1877 if (pCtx->uProtocol < 3)
1878 {
1879 Msg.hdr.cParms = 3;
1880
1881 Msg.u.v1.pvName.SetPtr((void *)strPath.c_str(), (uint32_t)cbPath);
1882 Msg.u.v1.cbName.SetUInt32((uint32_t)cbPath);
1883 Msg.u.v1.fMode.SetUInt32(pObj->GetMode());
1884 }
1885 else
1886 {
1887 Msg.hdr.cParms = 4;
1888
1889 /** @todo Context ID not used yet. */
1890 Msg.u.v3.uContext.SetUInt32(0);
1891 Msg.u.v3.pvName.SetPtr((void *)strPath.c_str(), (uint32_t)cbPath);
1892 Msg.u.v3.cbName.SetUInt32((uint32_t)cbPath);
1893 Msg.u.v3.fMode.SetUInt32(pObj->GetMode());
1894 }
1895
1896 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
1897 if (RT_SUCCESS(rc))
1898 rc = Msg.hdr.result;
1899
1900 LogFlowFuncLeaveRC(rc);
1901 return rc;
1902}
1903
1904static int vbglR3DnDGHSendFile(PVBGLR3GUESTDNDCMDCTX pCtx, DnDURIObject *pObj)
1905{
1906 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
1907 AssertPtrReturn(pObj, VERR_INVALID_POINTER);
1908 AssertReturn(pObj->GetType() == DnDURIObject::File, VERR_INVALID_PARAMETER);
1909 AssertReturn(pObj->IsOpen(), VERR_INVALID_STATE);
1910
1911 uint32_t cbBuf = _64K; /** @todo Make this configurable? */
1912 void *pvBuf = RTMemAlloc(cbBuf); /** @todo Make this buffer part of PVBGLR3GUESTDNDCMDCTX? */
1913 if (!pvBuf)
1914 return VERR_NO_MEMORY;
1915
1916 int rc;
1917
1918 RTCString strPath = pObj->GetDestPath();
1919
1920 LogFlowFunc(("strFile=%s (%zu), cbSize=%RU64, fMode=0x%x\n", strPath.c_str(), strPath.length(),
1921 pObj->GetSize(), pObj->GetMode()));
1922 LogFlowFunc(("uProtocol=%RU32, uClientID=%RU32\n", pCtx->uProtocol, pCtx->uClientID));
1923
1924 if (pCtx->uProtocol >= 2) /* Protocol version 2 and up sends a file header first. */
1925 {
1926 VBOXDNDGHSENDFILEHDRMSG MsgHdr;
1927 RT_ZERO(MsgHdr);
1928 MsgHdr.hdr.result = VERR_WRONG_ORDER;
1929 MsgHdr.hdr.u32ClientID = pCtx->uClientID;
1930 MsgHdr.hdr.u32Function = GUEST_DND_GH_SND_FILE_HDR;
1931 MsgHdr.hdr.cParms = 6;
1932
1933 MsgHdr.uContext.SetUInt32(0); /* Context ID; unused at the moment. */
1934 MsgHdr.pvName.SetPtr((void *)strPath.c_str(), (uint32_t)(strPath.length() + 1));
1935 MsgHdr.cbName.SetUInt32((uint32_t)(strPath.length() + 1));
1936 MsgHdr.uFlags.SetUInt32(0); /* Flags; unused at the moment. */
1937 MsgHdr.fMode.SetUInt32(pObj->GetMode()); /* File mode */
1938 MsgHdr.cbTotal.SetUInt64(pObj->GetSize()); /* File size (in bytes). */
1939
1940 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(MsgHdr)), &MsgHdr, sizeof(MsgHdr));
1941 if (RT_SUCCESS(rc))
1942 rc = MsgHdr.hdr.result;
1943
1944 LogFlowFunc(("Sending file header resulted in %Rrc\n", rc));
1945 }
1946 else
1947 rc = VINF_SUCCESS;
1948
1949 if (RT_SUCCESS(rc))
1950 {
1951 /*
1952 * Send the actual file data, chunk by chunk.
1953 */
1954 VBOXDNDGHSENDFILEDATAMSG Msg;
1955 RT_ZERO(Msg);
1956 Msg.hdr.result = VERR_WRONG_ORDER;
1957 Msg.hdr.u32ClientID = pCtx->uClientID;
1958 Msg.hdr.u32Function = GUEST_DND_GH_SND_FILE_DATA;
1959
1960 switch (pCtx->uProtocol)
1961 {
1962 case 3:
1963 {
1964 Msg.hdr.cParms = 5;
1965
1966 Msg.u.v3.uContext.SetUInt32(0);
1967 Msg.u.v3.pvChecksum.SetPtr(NULL, 0);
1968 Msg.u.v3.cbChecksum.SetUInt32(0);
1969 break;
1970 }
1971
1972 case 2:
1973 {
1974 Msg.hdr.cParms = 3;
1975
1976 Msg.u.v2.uContext.SetUInt32(0);
1977 break;
1978 }
1979
1980 default: /* Protocol v1 */
1981 {
1982 Msg.hdr.cParms = 5;
1983
1984 Msg.u.v1.pvName.SetPtr((void *)strPath.c_str(), (uint32_t)(strPath.length() + 1));
1985 Msg.u.v1.cbName.SetUInt32((uint32_t)(strPath.length() + 1));
1986 Msg.u.v1.fMode.SetUInt32(pObj->GetMode());
1987 break;
1988 }
1989 }
1990
1991 uint64_t cbToReadTotal = pObj->GetSize();
1992 uint64_t cbWrittenTotal = 0;
1993 while (cbToReadTotal)
1994 {
1995 uint32_t cbToRead = RT_MIN(cbToReadTotal, cbBuf);
1996 uint32_t cbRead = 0;
1997 if (cbToRead)
1998 rc = pObj->Read(pvBuf, cbToRead, &cbRead);
1999
2000 LogFlowFunc(("cbToReadTotal=%RU64, cbToRead=%RU32, cbRead=%RU32, rc=%Rrc\n",
2001 cbToReadTotal, cbToRead, cbRead, rc));
2002
2003 if ( RT_SUCCESS(rc)
2004 && cbRead)
2005 {
2006 switch (pCtx->uProtocol)
2007 {
2008 case 3:
2009 {
2010 Msg.u.v3.pvData.SetPtr(pvBuf, cbRead);
2011 Msg.u.v3.cbData.SetUInt32(cbRead);
2012 /** @todo Calculate + set checksums. */
2013 break;
2014 }
2015
2016 case 2:
2017 {
2018 Msg.u.v2.pvData.SetPtr(pvBuf, cbRead);
2019 Msg.u.v2.cbData.SetUInt32(cbRead);
2020 break;
2021 }
2022
2023 default:
2024 {
2025 Msg.u.v1.pvData.SetPtr(pvBuf, cbRead);
2026 Msg.u.v1.cbData.SetUInt32(cbRead);
2027 break;
2028 }
2029 }
2030
2031 rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
2032 if (RT_SUCCESS(rc))
2033 rc = Msg.hdr.result;
2034 }
2035
2036 if (RT_FAILURE(rc))
2037 {
2038 LogFlowFunc(("Reading failed with rc=%Rrc\n", rc));
2039 break;
2040 }
2041
2042 Assert(cbRead <= cbToReadTotal);
2043 cbToReadTotal -= cbRead;
2044 cbWrittenTotal += cbRead;
2045
2046 LogFlowFunc(("%RU64/%RU64 -- %RU8%%\n", cbWrittenTotal, pObj->GetSize(), cbWrittenTotal * 100 / pObj->GetSize()));
2047 };
2048 }
2049
2050 RTMemFree(pvBuf);
2051
2052 LogFlowFuncLeaveRC(rc);
2053 return rc;
2054}
2055
2056static int vbglR3DnDGHSendURIObject(PVBGLR3GUESTDNDCMDCTX pCtx, DnDURIObject *pObj)
2057{
2058 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
2059 AssertPtrReturn(pObj, VERR_INVALID_POINTER);
2060
2061 int rc;
2062
2063 switch (pObj->GetType())
2064 {
2065 case DnDURIObject::Directory:
2066 rc = vbglR3DnDGHSendDir(pCtx, pObj);
2067 break;
2068
2069 case DnDURIObject::File:
2070 rc = vbglR3DnDGHSendFile(pCtx, pObj);
2071 break;
2072
2073 default:
2074 AssertMsgFailed(("Object type %ld not implemented\n", pObj->GetType()));
2075 rc = VERR_NOT_IMPLEMENTED;
2076 break;
2077 }
2078
2079 return rc;
2080}
2081
2082static int vbglR3DnDGHSendRawData(PVBGLR3GUESTDNDCMDCTX pCtx, void *pvData, size_t cbData)
2083{
2084 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
2085 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
2086 /* cbData can be 0. */
2087
2088 VBOXDNDDATAHDR dataHdr;
2089 RT_ZERO(dataHdr);
2090
2091 /* For raw data only the total size is required to be specified. */
2092 dataHdr.cbTotal = cbData;
2093
2094 return vbglR3DnDGHSendDataInternal(pCtx, pvData, cbData, &dataHdr);
2095}
2096
2097static int vbglR3DnDGHSendURIData(PVBGLR3GUESTDNDCMDCTX pCtx, const void *pvData, size_t cbData)
2098{
2099 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
2100 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
2101 AssertReturn(cbData, VERR_INVALID_PARAMETER);
2102
2103 RTCList<RTCString> lstPaths =
2104 RTCString((const char *)pvData, cbData).split("\r\n");
2105
2106 /** @todo Add symlink support (DNDURILIST_FLAGS_RESOLVE_SYMLINKS) here. */
2107 /** @todo Add lazy loading (DNDURILIST_FLAGS_LAZY) here. */
2108 uint32_t fFlags = DNDURILIST_FLAGS_KEEP_OPEN;
2109
2110 DnDURIList lstURI;
2111 int rc = lstURI.AppendURIPathsFromList(lstPaths, fFlags);
2112 if (RT_SUCCESS(rc))
2113 {
2114 /*
2115 * Send the (meta) data; in case of URIs it's the (non-recursive) file/directory
2116 * URI list the host needs to know upfront to set up the drag'n drop operation.
2117 */
2118 RTCString strRootDest = lstURI.RootToString();
2119 if (strRootDest.isNotEmpty())
2120 {
2121 void *pvURIList = (void *)strRootDest.c_str(); /* URI root list. */
2122 uint32_t cbURLIist = (uint32_t)strRootDest.length() + 1; /* Include string termination. */
2123
2124 /* The total size also contains the size of the meta data. */
2125 uint64_t cbTotal = cbURLIist;
2126 cbTotal += lstURI.TotalBytes();
2127
2128 /* We're going to send an URI list in text format. */
2129 const char szMetaFmt[] = "text/uri-list";
2130 const uint32_t cbMetaFmt = (uint32_t)strlen(szMetaFmt) + 1; /* Include termination. */
2131
2132 VBOXDNDDATAHDR dataHdr;
2133 dataHdr.uFlags = 0; /* Flags not used yet. */
2134 dataHdr.cbTotal = cbTotal;
2135 dataHdr.cbMeta = cbURLIist;
2136 dataHdr.pvMetaFmt = (void *)szMetaFmt;
2137 dataHdr.cbMetaFmt = cbMetaFmt;
2138 dataHdr.cObjects = lstURI.TotalCount();
2139
2140 rc = vbglR3DnDGHSendDataInternal(pCtx,
2141 pvURIList, cbURLIist, &dataHdr);
2142 }
2143 else
2144 rc = VERR_INVALID_PARAMETER;
2145 }
2146
2147 if (RT_SUCCESS(rc))
2148 {
2149 while (!lstURI.IsEmpty())
2150 {
2151 DnDURIObject *pNextObj = lstURI.First();
2152
2153 rc = vbglR3DnDGHSendURIObject(pCtx, pNextObj);
2154 if (RT_FAILURE(rc))
2155 break;
2156
2157 lstURI.RemoveFirst();
2158 }
2159 }
2160
2161 return rc;
2162}
2163
2164VBGLR3DECL(int) VbglR3DnDGHSendData(PVBGLR3GUESTDNDCMDCTX pCtx, const char *pszFormat, void *pvData, uint32_t cbData)
2165{
2166 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
2167 AssertPtrReturn(pszFormat, VERR_INVALID_POINTER);
2168 AssertPtrReturn(pvData, VERR_INVALID_POINTER);
2169 AssertReturn(cbData, VERR_INVALID_PARAMETER);
2170
2171 LogFlowFunc(("pszFormat=%s, pvData=%p, cbData=%RU32\n", pszFormat, pvData, cbData));
2172
2173 int rc;
2174 if (DnDMIMEHasFileURLs(pszFormat, strlen(pszFormat)))
2175 {
2176 /* Send file data. */
2177 rc = vbglR3DnDGHSendURIData(pCtx, pvData, cbData);
2178 }
2179 else
2180 rc = vbglR3DnDGHSendRawData(pCtx, pvData, cbData);
2181
2182 if (RT_FAILURE(rc))
2183 {
2184 int rc2 = VbglR3DnDGHSendError(pCtx, rc);
2185 if (RT_FAILURE(rc2))
2186 LogFlowFunc(("Unable to send error (%Rrc) to host, rc=%Rrc\n", rc, rc2));
2187 }
2188
2189 return rc;
2190}
2191
2192VBGLR3DECL(int) VbglR3DnDGHSendError(PVBGLR3GUESTDNDCMDCTX pCtx, int rcErr)
2193{
2194 AssertPtrReturn(pCtx, VERR_INVALID_POINTER);
2195
2196 VBOXDNDGHEVTERRORMSG Msg;
2197 RT_ZERO(Msg);
2198 Msg.hdr.result = VERR_WRONG_ORDER;
2199 Msg.hdr.u32ClientID = pCtx->uClientID;
2200 Msg.hdr.u32Function = GUEST_DND_GH_EVT_ERROR;
2201
2202 if (pCtx->uProtocol < 3)
2203 {
2204 Msg.hdr.cParms = 1;
2205
2206 Msg.u.v1.rc.SetUInt32((uint32_t)rcErr); /* uint32_t vs. int. */
2207 }
2208 else
2209 {
2210 Msg.hdr.cParms = 2;
2211
2212 /** @todo Context ID not used yet. */
2213 Msg.u.v3.uContext.SetUInt32(0);
2214 Msg.u.v3.rc.SetUInt32((uint32_t)rcErr); /* uint32_t vs. int. */
2215 }
2216
2217 int rc = vbglR3DoIOCtl(VBOXGUEST_IOCTL_HGCM_CALL(sizeof(Msg)), &Msg, sizeof(Msg));
2218 if (RT_SUCCESS(rc))
2219 {
2220 if (RT_FAILURE(Msg.hdr.result))
2221 {
2222 LogFlowFunc(("Sending error %Rrc failed with rc=%Rrc\n", rcErr, Msg.hdr.result));
2223
2224 /* Never return an error if the host did not accept the error at
2225 * the current time. This can be due to the host not having any appropriate
2226 * callbacks set which would handle that error. */
2227 rc = VINF_SUCCESS;
2228 }
2229 }
2230
2231 return rc;
2232}
2233#endif /* VBOX_WITH_DRAG_AND_DROP_GH */
2234
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