VirtualBox

source: vbox/trunk/src/VBox/GuestHost/DragAndDrop/DnDURIObject.cpp@ 74946

Last change on this file since 74946 was 74714, checked in by vboxsync, 6 years ago

DnD: Cleaned up DnDURIObject API / internal state handling.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.0 KB
Line 
1/* $Id: DnDURIObject.cpp 74714 2018-10-09 11:50:22Z vboxsync $ */
2/** @file
3 * DnD: URI object class. For handling creation/reading/writing to files and directories
4 * on host or guest side.
5 */
6
7/*
8 * Copyright (C) 2014-2018 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19
20/*********************************************************************************************************************************
21* Header Files *
22*********************************************************************************************************************************/
23
24#include <iprt/dir.h>
25#include <iprt/file.h>
26#include <iprt/fs.h>
27#include <iprt/path.h>
28#include <iprt/uri.h>
29
30#ifdef LOG_GROUP
31 #undef LOG_GROUP
32#endif
33#define LOG_GROUP LOG_GROUP_GUEST_DND
34#include <VBox/log.h>
35
36#include <VBox/GuestHost/DragAndDrop.h>
37
38DnDURIObject::DnDURIObject(void)
39 : m_enmType(Type_Unknown)
40 , m_enmView(View_Unknown)
41 , m_fIsOpen(false)
42{
43 RT_ZERO(u);
44}
45
46DnDURIObject::DnDURIObject(Type enmType,
47 const RTCString &strSrcPathAbs /* = 0 */,
48 const RTCString &strDstPathAbs /* = 0 */)
49 : m_enmType(enmType)
50 , m_enmView(View_Unknown)
51 , m_strSrcPathAbs(strSrcPathAbs)
52 , m_strTgtPathAbs(strDstPathAbs)
53 , m_fIsOpen(false)
54{
55 RT_ZERO(u);
56}
57
58DnDURIObject::~DnDURIObject(void)
59{
60 closeInternal();
61}
62
63/**
64 * Closes the object's internal handles (to files / ...).
65 *
66 */
67void DnDURIObject::closeInternal(void)
68{
69 LogFlowThisFuncEnter();
70
71 if (!m_fIsOpen)
72 return;
73
74 switch (m_enmType)
75 {
76 case Type_File:
77 {
78 RTFileClose(u.File.hFile);
79 u.File.hFile = NIL_RTFILE;
80 RT_ZERO(u.File.objInfo);
81 break;
82 }
83
84 case Type_Directory:
85 {
86 RTDirClose(u.Dir.hDir);
87 u.Dir.hDir = NIL_RTDIR;
88 RT_ZERO(u.Dir.objInfo);
89 break;
90 }
91
92 default:
93 break;
94 }
95
96 m_fIsOpen = false;
97}
98
99/**
100 * Closes the object.
101 * This also closes the internal handles associated with the object (to files / ...).
102 */
103void DnDURIObject::Close(void)
104{
105 closeInternal();
106}
107
108/**
109 * Returns the directory / file mode of the object.
110 *
111 * @return File / directory mode.
112 */
113RTFMODE DnDURIObject::GetMode(void) const
114{
115 switch (m_enmType)
116 {
117 case Type_File:
118 return u.File.objInfo.Attr.fMode;
119
120 case Type_Directory:
121 return u.Dir.objInfo.Attr.fMode;
122
123 default:
124 break;
125 }
126
127 AssertFailed();
128 return 0;
129}
130
131/**
132 * Returns the bytes already processed (read / written).
133 *
134 * Note: Only applies if the object is of type DnDURIObject::Type_File.
135 *
136 * @return Bytes already processed (read / written).
137 */
138uint64_t DnDURIObject::GetProcessed(void) const
139{
140 if (m_enmType == Type_File)
141 return u.File.cbProcessed;
142
143 return 0;
144}
145
146/**
147 * Returns the file's logical size (in bytes).
148 *
149 * Note: Only applies if the object is of type DnDURIObject::Type_File.
150 *
151 * @return The file's logical size (in bytes).
152 */
153uint64_t DnDURIObject::GetSize(void) const
154{
155 if (m_enmType == Type_File)
156 return u.File.cbToProcess;
157
158 return 0;
159}
160
161/**
162 * Returns whether the processing of the object is complete or not.
163 * For file objects this means that all bytes have been processed.
164 *
165 * @return True if complete, False if not.
166 */
167bool DnDURIObject::IsComplete(void) const
168{
169 bool fComplete;
170
171 switch (m_enmType)
172 {
173 case Type_File:
174 Assert(u.File.cbProcessed <= u.File.cbToProcess);
175 fComplete = u.File.cbProcessed == u.File.cbToProcess;
176 break;
177
178 case Type_Directory:
179 fComplete = true;
180 break;
181
182 default:
183 fComplete = true;
184 break;
185 }
186
187 return fComplete;
188}
189
190/**
191 * Returns whether the object is in an open state or not.
192 */
193bool DnDURIObject::IsOpen(void) const
194{
195 return m_fIsOpen;
196}
197
198/**
199 * (Re-)Opens the object with a specific view, open and file mode.
200 *
201 * @return IPRT status code.
202 * @param enmView View to use for opening the object.
203 * @param fOpen File open flags to use.
204 * @param fMode File mode to use.
205 */
206int DnDURIObject::Open(View enmView, uint64_t fOpen /* = 0 */, RTFMODE fMode /* = 0 */)
207{
208 return OpenEx( enmView == View_Source
209 ? m_strSrcPathAbs : m_strTgtPathAbs
210 , enmView, fOpen, fMode, DNDURIOBJECT_FLAGS_NONE);
211}
212
213/**
214 * Open the object with a specific file type, and, depending on the type, specifying additional parameters.
215 *
216 * @return IPRT status code.
217 * @param strPathAbs Absolute path of the object (file / directory / ...).
218 * @param enmView View of the object.
219 * @param fOpen Open mode to use; only valid for file objects.
220 * @param fMode File mode to use; only valid for file objects.
221 * @param fFlags Additional DnD URI object flags.
222 */
223int DnDURIObject::OpenEx(const RTCString &strPathAbs, View enmView,
224 uint64_t fOpen /* = 0 */, RTFMODE fMode /* = 0 */, DNDURIOBJECTFLAGS fFlags /* = DNDURIOBJECT_FLAGS_NONE */)
225{
226 AssertReturn(!(fFlags & ~DNDURIOBJECT_FLAGS_VALID_MASK), VERR_INVALID_FLAGS);
227 RT_NOREF1(fFlags);
228
229 if (m_fIsOpen)
230 return VINF_SUCCESS;
231
232 int rc = VINF_SUCCESS;
233
234 switch (enmView)
235 {
236 case View_Source:
237 m_strSrcPathAbs = strPathAbs;
238 break;
239
240 case View_Target:
241 m_strTgtPathAbs = strPathAbs;
242 break;
243
244 default:
245 rc = VERR_NOT_IMPLEMENTED;
246 break;
247 }
248
249 if ( RT_SUCCESS(rc)
250 && fOpen) /* Opening mode specified? */
251 {
252 LogFlowThisFunc(("strPath=%s, enmView=%RU32, fOpen=0x%x, fMode=0x%x, fFlags=0x%x\n",
253 strPathAbs.c_str(), enmView, fOpen, fMode, fFlags));
254 switch (m_enmType)
255 {
256 case Type_File:
257 {
258 /*
259 * Open files on the source with RTFILE_O_DENY_WRITE to prevent races
260 * where the OS writes to the file while the destination side transfers
261 * it over.
262 */
263 LogFlowThisFunc(("Opening ...\n"));
264 rc = RTFileOpen(&u.File.hFile, strPathAbs.c_str(), fOpen);
265 if (RT_SUCCESS(rc))
266 {
267 if ( (fOpen & RTFILE_O_WRITE) /* Only set the file mode on write. */
268 && fMode /* Some file mode to set specified? */)
269 {
270 rc = RTFileSetMode(u.File.hFile, fMode);
271 }
272 else if (fOpen & RTFILE_O_READ)
273 {
274 rc = queryInfoInternal(enmView);
275 }
276 }
277
278 if (RT_SUCCESS(rc))
279 {
280 LogFlowThisFunc(("File cbObject=%RU64, fMode=0x%x\n",
281 u.File.objInfo.cbObject, u.File.objInfo.Attr.fMode));
282 u.File.cbToProcess = u.File.objInfo.cbObject;
283 u.File.cbProcessed = 0;
284 }
285
286 break;
287 }
288
289 case Type_Directory:
290 {
291 rc = RTDirOpen(&u.Dir.hDir, strPathAbs.c_str());
292 if (RT_SUCCESS(rc))
293 rc = queryInfoInternal(enmView);
294 break;
295 }
296
297 default:
298 rc = VERR_NOT_IMPLEMENTED;
299 break;
300 }
301 }
302
303 if (RT_SUCCESS(rc))
304 {
305 m_enmView = enmView;
306 m_fIsOpen = true;
307 }
308
309 LogFlowFuncLeaveRC(rc);
310 return rc;
311}
312
313/**
314 * Queries information about the object using a specific view, internal version.
315 *
316 * @return IPRT status code.
317 * @param enmView View to use for querying information.
318 */
319int DnDURIObject::queryInfoInternal(View enmView)
320{
321 RT_NOREF(enmView);
322
323 int rc;
324
325 switch (m_enmType)
326 {
327 case Type_File:
328 rc = RTFileQueryInfo(u.File.hFile, &u.File.objInfo, RTFSOBJATTRADD_NOTHING);
329 break;
330
331 case Type_Directory:
332 rc = RTDirQueryInfo(u.Dir.hDir, &u.Dir.objInfo, RTFSOBJATTRADD_NOTHING);
333 break;
334
335 default:
336 rc = VERR_NOT_IMPLEMENTED;
337 break;
338 }
339
340 return rc;
341}
342
343/**
344 * Queries information about the object using a specific view.
345 *
346 * @return IPRT status code.
347 * @param enmView View to use for querying information.
348 */
349int DnDURIObject::QueryInfo(View enmView)
350{
351 return queryInfoInternal(enmView);
352}
353
354/**
355 * Rebases an absolute URI path from an old path base to a new path base.
356 * This function is needed in order to transform path from the source side to the target side.
357 *
358 * @return IPRT status code.
359 * @param strPathAbs Absolute URI path to rebase.
360 * @param strBaseOld Old base path to rebase from.
361 * @param strBaseNew New base path to rebase to.
362 *
363 ** @todo Put this into an own class like DnDURIPath : public RTCString?
364 */
365/* static */
366int DnDURIObject::RebaseURIPath(RTCString &strPathAbs,
367 const RTCString &strBaseOld /* = "" */,
368 const RTCString &strBaseNew /* = "" */)
369{
370 char *pszPath = RTUriFilePath(strPathAbs.c_str());
371 if (!pszPath) /* No URI? */
372 pszPath = RTStrDup(strPathAbs.c_str());
373
374 int rc;
375
376 if (pszPath)
377 {
378 const char *pszPathStart = pszPath;
379 const char *pszBaseOld = strBaseOld.c_str();
380 if ( pszBaseOld
381 && RTPathStartsWith(pszPath, pszBaseOld))
382 {
383 pszPathStart += strlen(pszBaseOld);
384 }
385
386 rc = VINF_SUCCESS;
387
388 if (RT_SUCCESS(rc))
389 {
390 char *pszPathNew = RTPathJoinA(strBaseNew.c_str(), pszPathStart);
391 if (pszPathNew)
392 {
393 char *pszPathURI = RTUriCreate("file" /* pszScheme */, "/" /* pszAuthority */,
394 pszPathNew /* pszPath */,
395 NULL /* pszQuery */, NULL /* pszFragment */);
396 if (pszPathURI)
397 {
398 LogFlowFunc(("Rebasing \"%s\" to \"%s\"\n", strPathAbs.c_str(), pszPathURI));
399
400 strPathAbs = RTCString(pszPathURI) + "\r\n";
401 RTStrFree(pszPathURI);
402 }
403 else
404 rc = VERR_INVALID_PARAMETER;
405
406 RTStrFree(pszPathNew);
407 }
408 else
409 rc = VERR_NO_MEMORY;
410 }
411
412 RTStrFree(pszPath);
413 }
414 else
415 rc = VERR_NO_MEMORY;
416
417 return rc;
418}
419
420/**
421 * Reads data from the object. Only applies to files objects.
422 *
423 * @return IPRT status code.
424 * @param pvBuf Buffer where to store the read data.
425 * @param cbBuf Size (in bytes) of the buffer.
426 * @param pcbRead Pointer where to store how many bytes were read. Optional.
427 */
428int DnDURIObject::Read(void *pvBuf, size_t cbBuf, uint32_t *pcbRead)
429{
430 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
431 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
432 /* pcbRead is optional. */
433
434 AssertMsgReturn(m_fIsOpen, ("Object not in open state\n"), VERR_INVALID_STATE);
435 AssertMsgReturn(m_enmView == View_Source, ("Cannot write to an object which is not in target view\n"),
436 VERR_INVALID_STATE);
437
438 size_t cbRead = 0;
439
440 int rc;
441 switch (m_enmType)
442 {
443 case Type_File:
444 {
445 rc = RTFileRead(u.File.hFile, pvBuf, cbBuf, &cbRead);
446 if (RT_SUCCESS(rc))
447 {
448 u.File.cbProcessed += cbRead;
449 Assert(u.File.cbProcessed <= u.File.cbToProcess);
450
451 /* End of file reached or error occurred? */
452 if ( u.File.cbToProcess
453 && u.File.cbProcessed == u.File.cbToProcess)
454 {
455 rc = VINF_EOF;
456 }
457 }
458 break;
459 }
460
461 case Type_Directory:
462 {
463 rc = VINF_SUCCESS;
464 break;
465 }
466
467 default:
468 rc = VERR_NOT_IMPLEMENTED;
469 break;
470 }
471
472 if (RT_SUCCESS(rc))
473 {
474 if (pcbRead)
475 *pcbRead = (uint32_t)cbRead;
476 }
477
478 LogFlowFunc(("Returning strSourcePath=%s, cbRead=%zu, rc=%Rrc\n", m_strSrcPathAbs.c_str(), cbRead, rc));
479 return rc;
480}
481
482/**
483 * Resets the object's state and closes all related handles.
484 */
485void DnDURIObject::Reset(void)
486{
487 LogFlowThisFuncEnter();
488
489 Close();
490
491 m_enmType = Type_Unknown;
492 m_enmView = View_Unknown;
493 m_strSrcPathAbs = "";
494 m_strTgtPathAbs = "";
495
496 RT_ZERO(u);
497}
498
499/**
500 * Sets the bytes to process by the object.
501 *
502 * Note: Only applies if the object is of type DnDURIObject::Type_File.
503 *
504 * @return IPRT return code.
505 * @param cbSize Size (in bytes) to process.
506 */
507int DnDURIObject::SetSize(uint64_t cbSize)
508{
509 AssertReturn(m_enmType == Type_File, VERR_INVALID_PARAMETER);
510
511 /** @todo Implement sparse file support here. */
512
513 u.File.cbToProcess = cbSize;
514 return VINF_SUCCESS;
515}
516
517/**
518 * Writes data to an object. Only applies to file objects.
519 *
520 * @return IPRT status code.
521 * @param pvBuf Buffer of data to write.
522 * @param cbBuf Size (in bytes) of data to write.
523 * @param pcbWritten Pointer where to store how many bytes were written. Optional.
524 */
525int DnDURIObject::Write(const void *pvBuf, size_t cbBuf, uint32_t *pcbWritten)
526{
527 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
528 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
529 /* pcbWritten is optional. */
530
531 AssertMsgReturn(m_fIsOpen, ("Object not in open state\n"), VERR_INVALID_STATE);
532 AssertMsgReturn(m_enmView == View_Target, ("Cannot write to an object which is not in target view\n"),
533 VERR_INVALID_STATE);
534
535 size_t cbWritten = 0;
536
537 int rc;
538 switch (m_enmType)
539 {
540 case Type_File:
541 {
542 rc = RTFileWrite(u.File.hFile, pvBuf, cbBuf, &cbWritten);
543 if (RT_SUCCESS(rc))
544 u.File.cbProcessed += cbWritten;
545 break;
546 }
547
548 case Type_Directory:
549 {
550 rc = VINF_SUCCESS;
551 break;
552 }
553
554 default:
555 rc = VERR_NOT_IMPLEMENTED;
556 break;
557 }
558
559 if (RT_SUCCESS(rc))
560 {
561 if (pcbWritten)
562 *pcbWritten = (uint32_t)cbWritten;
563 }
564
565 LogFlowThisFunc(("Returning strSourcePathAbs=%s, cbWritten=%zu, rc=%Rrc\n", m_strSrcPathAbs.c_str(), cbWritten, rc));
566 return rc;
567}
568
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