VirtualBox

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

Last change on this file since 61334 was 59838, checked in by vboxsync, 9 years ago

DnD: DnDURIObject.cpp: Added another initialization for m_fOpen.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.5 KB
Line 
1/* $Id: DnDURIObject.cpp 59838 2016-02-26 10:33:53Z 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-2016 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_Type(Unknown)
40 , m_fOpen(false)
41 , m_fMode(0)
42 , m_cbSize(0)
43 , m_cbProcessed(0)
44{
45 RT_ZERO(u);
46}
47
48DnDURIObject::DnDURIObject(Type type,
49 const RTCString &strSrcPath /* = 0 */,
50 const RTCString &strDstPath /* = 0 */,
51 uint32_t fMode /* = 0 */, uint64_t cbSize /* = 0 */)
52 : m_Type(type)
53 , m_strSrcPath(strSrcPath)
54 , m_strTgtPath(strDstPath)
55 , m_fOpen(false)
56 , m_fMode(fMode)
57 , m_cbSize(cbSize)
58 , m_cbProcessed(0)
59{
60}
61
62DnDURIObject::~DnDURIObject(void)
63{
64 closeInternal();
65}
66
67void DnDURIObject::closeInternal(void)
68{
69 LogFlowThisFuncEnter();
70
71 if (!m_fOpen)
72 return;
73
74 switch (m_Type)
75 {
76 case File:
77 {
78 RTFileClose(u.m_hFile);
79 u.m_hFile = NIL_RTFILE;
80 break;
81 }
82
83 case Directory:
84 break;
85
86 default:
87 break;
88 }
89
90 m_fOpen = false;
91}
92
93void DnDURIObject::Close(void)
94{
95 closeInternal();
96}
97
98bool DnDURIObject::IsComplete(void) const
99{
100 bool fComplete;
101
102 switch (m_Type)
103 {
104 case File:
105 Assert(m_cbProcessed <= m_cbSize);
106 fComplete = m_cbProcessed == m_cbSize;
107 break;
108
109 case Directory:
110 fComplete = true;
111 break;
112
113 default:
114 fComplete = true;
115 break;
116 }
117
118 return fComplete;
119}
120
121bool DnDURIObject::IsOpen(void) const
122{
123 return m_fOpen;
124}
125
126int DnDURIObject::Open(Dest enmDest, uint64_t fOpen /* = 0 */, uint32_t fMode /* = 0 */)
127{
128 return OpenEx( enmDest == Source
129 ? m_strSrcPath : m_strTgtPath
130 , m_Type, enmDest, fOpen, fMode, 0 /* fFlags */);
131}
132
133int DnDURIObject::OpenEx(const RTCString &strPath, Type enmType, Dest enmDest,
134 uint64_t fOpen /* = 0 */, uint32_t fMode /* = 0 */, uint32_t fFlags /* = 0 */)
135{
136 int rc = VINF_SUCCESS;
137
138 switch (enmDest)
139 {
140 case Source:
141 m_strSrcPath = strPath;
142 break;
143
144 case Target:
145 m_strTgtPath = strPath;
146 break;
147
148 default:
149 rc = VERR_NOT_IMPLEMENTED;
150 break;
151 }
152
153 if ( RT_SUCCESS(rc)
154 && fOpen) /* Opening mode specified? */
155 {
156 LogFlowThisFunc(("enmType=%RU32, strPath=%s, fOpen=0x%x, enmType=%RU32, enmDest=%RU32\n",
157 enmType, strPath.c_str(), fOpen, enmType, enmDest));
158 switch (enmType)
159 {
160 case File:
161 {
162 if (!m_fOpen)
163 {
164 /*
165 * Open files on the source with RTFILE_O_DENY_WRITE to prevent races
166 * where the OS writes to the file while the destination side transfers
167 * it over.
168 */
169 LogFlowThisFunc(("Opening ...\n"));
170 rc = RTFileOpen(&u.m_hFile, strPath.c_str(), fOpen);
171 if (RT_SUCCESS(rc))
172 rc = RTFileGetSize(u.m_hFile, &m_cbSize);
173
174 if (RT_SUCCESS(rc))
175 {
176 if ( (fOpen & RTFILE_O_WRITE) /* Only set the file mode on write. */
177 && fMode /* Some file mode to set specified? */)
178 {
179 rc = RTFileSetMode(u.m_hFile, fMode);
180 if (RT_SUCCESS(rc))
181 m_fMode = fMode;
182 }
183 else if (fOpen & RTFILE_O_READ)
184 {
185#if 0 /** @todo Enable this as soon as RTFileGetMode is implemented. */
186 rc = RTFileGetMode(u.m_hFile, &m_fMode);
187#else
188 RTFSOBJINFO ObjInfo;
189 rc = RTFileQueryInfo(u.m_hFile, &ObjInfo, RTFSOBJATTRADD_NOTHING);
190 if (RT_SUCCESS(rc))
191 m_fMode = ObjInfo.Attr.fMode;
192#endif
193 }
194 }
195
196 if (RT_SUCCESS(rc))
197 {
198 LogFlowThisFunc(("cbSize=%RU64, fMode=0x%x\n", m_cbSize, m_fMode));
199 m_cbProcessed = 0;
200 }
201 }
202 else
203 rc = VINF_SUCCESS;
204
205 break;
206 }
207
208 case Directory:
209 rc = VINF_SUCCESS;
210 break;
211
212 default:
213 rc = VERR_NOT_IMPLEMENTED;
214 break;
215 }
216 }
217
218 if (RT_SUCCESS(rc))
219 {
220 m_Type = enmType;
221 m_fOpen = true;
222 }
223
224 LogFlowFuncLeaveRC(rc);
225 return rc;
226}
227
228/* static */
229/** @todo Put this into an own class like DnDURIPath : public RTCString? */
230int DnDURIObject::RebaseURIPath(RTCString &strPath,
231 const RTCString &strBaseOld /* = "" */,
232 const RTCString &strBaseNew /* = "" */)
233{
234 char *pszPath = RTUriFilePath(strPath.c_str());
235 if (!pszPath) /* No URI? */
236 pszPath = RTStrDup(strPath.c_str());
237
238 int rc;
239
240 if (pszPath)
241 {
242 const char *pszPathStart = pszPath;
243 const char *pszBaseOld = strBaseOld.c_str();
244 if ( pszBaseOld
245 && RTPathStartsWith(pszPath, pszBaseOld))
246 {
247 pszPathStart += strlen(pszBaseOld);
248 }
249
250 rc = VINF_SUCCESS;
251
252 if (RT_SUCCESS(rc))
253 {
254 char *pszPathNew = RTPathJoinA(strBaseNew.c_str(), pszPathStart);
255 if (pszPathNew)
256 {
257 char *pszPathURI = RTUriCreate("file" /* pszScheme */, "/" /* pszAuthority */,
258 pszPathNew /* pszPath */,
259 NULL /* pszQuery */, NULL /* pszFragment */);
260 if (pszPathURI)
261 {
262 LogFlowFunc(("Rebasing \"%s\" to \"%s\"\n", strPath.c_str(), pszPathURI));
263
264 strPath = RTCString(pszPathURI) + "\r\n";
265 RTStrFree(pszPathURI);
266 }
267 else
268 rc = VERR_INVALID_PARAMETER;
269
270 RTStrFree(pszPathNew);
271 }
272 else
273 rc = VERR_NO_MEMORY;
274 }
275
276 RTStrFree(pszPath);
277 }
278 else
279 rc = VERR_NO_MEMORY;
280
281 return rc;
282}
283
284int DnDURIObject::Read(void *pvBuf, size_t cbBuf, uint32_t *pcbRead)
285{
286 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
287 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
288 /* pcbRead is optional. */
289
290 size_t cbRead = 0;
291
292 int rc;
293 switch (m_Type)
294 {
295 case File:
296 {
297 rc = OpenEx(m_strSrcPath, File, Source,
298 /* Use some sensible defaults. */
299 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE, 0 /* fFlags */);
300 if (RT_SUCCESS(rc))
301 {
302 rc = RTFileRead(u.m_hFile, pvBuf, cbBuf, &cbRead);
303 if (RT_SUCCESS(rc))
304 {
305 m_cbProcessed += cbRead;
306 Assert(m_cbProcessed <= m_cbSize);
307
308 /* End of file reached or error occurred? */
309 if ( m_cbSize
310 && m_cbProcessed == m_cbSize)
311 {
312 rc = VINF_EOF;
313 }
314 }
315 }
316
317 break;
318 }
319
320 case Directory:
321 {
322 rc = VINF_SUCCESS;
323 break;
324 }
325
326 default:
327 rc = VERR_NOT_IMPLEMENTED;
328 break;
329 }
330
331 if (RT_SUCCESS(rc))
332 {
333 if (pcbRead)
334 *pcbRead = (uint32_t)cbRead;
335 }
336
337 LogFlowFunc(("Returning strSourcePath=%s, cbRead=%zu, rc=%Rrc\n", m_strSrcPath.c_str(), cbRead, rc));
338 return rc;
339}
340
341void DnDURIObject::Reset(void)
342{
343 LogFlowThisFuncEnter();
344
345 Close();
346
347 m_Type = Unknown;
348 m_strSrcPath = "";
349 m_strTgtPath = "";
350 m_fMode = 0;
351 m_cbSize = 0;
352 m_cbProcessed = 0;
353}
354
355int DnDURIObject::Write(const void *pvBuf, size_t cbBuf, uint32_t *pcbWritten)
356{
357 AssertPtrReturn(pvBuf, VERR_INVALID_POINTER);
358 AssertReturn(cbBuf, VERR_INVALID_PARAMETER);
359 /* pcbWritten is optional. */
360
361 size_t cbWritten = 0;
362
363 int rc;
364 switch (m_Type)
365 {
366 case File:
367 {
368 rc = OpenEx(m_strTgtPath, File, Target,
369 /* Use some sensible defaults. */
370 RTFILE_O_OPEN_CREATE | RTFILE_O_DENY_WRITE | RTFILE_O_WRITE, 0 /* fFlags */);
371 if (RT_SUCCESS(rc))
372 {
373 rc = RTFileWrite(u.m_hFile, pvBuf, cbBuf, &cbWritten);
374 if (RT_SUCCESS(rc))
375 m_cbProcessed += cbWritten;
376 }
377 break;
378 }
379
380 case Directory:
381 {
382 rc = VINF_SUCCESS;
383 break;
384 }
385
386 default:
387 rc = VERR_NOT_IMPLEMENTED;
388 break;
389 }
390
391 if (RT_SUCCESS(rc))
392 {
393 if (pcbWritten)
394 *pcbWritten = (uint32_t)cbWritten;
395 }
396
397 LogFlowThisFunc(("Returning strSourcePath=%s, cbWritten=%zu, rc=%Rrc\n", m_strSrcPath.c_str(), cbWritten, rc));
398 return rc;
399}
400
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