VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedFolders/mappings.cpp@ 58600

Last change on this file since 58600 was 56962, checked in by vboxsync, 9 years ago

HostServices/SharedFolders: enable guest to host path conversion

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.2 KB
Line 
1/** @file
2 * Shared Folders: Mappings support.
3 */
4
5/*
6 * Copyright (C) 2006-2012 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#ifdef UNITTEST
18# include "testcase/tstSharedFolderService.h"
19#endif
20
21#include "mappings.h"
22#include <iprt/alloc.h>
23#include <iprt/assert.h>
24#include <iprt/string.h>
25
26#ifdef UNITTEST
27# include "teststubs.h"
28#endif
29
30/* Shared folders order in the saved state and in the FolderMapping can differ.
31 * So a translation array of root handle is needed.
32 */
33
34static MAPPING FolderMapping[SHFL_MAX_MAPPINGS];
35static SHFLROOT aIndexFromRoot[SHFL_MAX_MAPPINGS];
36
37void vbsfMappingInit(void)
38{
39 unsigned root;
40
41 for (root = 0; root < RT_ELEMENTS(aIndexFromRoot); root++)
42 {
43 aIndexFromRoot[root] = SHFL_ROOT_NIL;
44 }
45}
46
47int vbsfMappingLoaded(const PMAPPING pLoadedMapping, SHFLROOT root)
48{
49 /* Mapping loaded from the saved state with the index. Which means
50 * the guest uses the iMapping as root handle for this folder.
51 * Check whether there is the same mapping in FolderMapping and
52 * update the aIndexFromRoot.
53 *
54 * Also update the mapping properties, which were lost: cMappings.
55 */
56 if (root >= SHFL_MAX_MAPPINGS)
57 {
58 return VERR_INVALID_PARAMETER;
59 }
60
61 SHFLROOT i;
62 for (i = 0; i < RT_ELEMENTS(FolderMapping); i++)
63 {
64 MAPPING *pMapping = &FolderMapping[i];
65
66 /* Equal? */
67 if ( pLoadedMapping->fValid == pMapping->fValid
68 && ShflStringSizeOfBuffer(pLoadedMapping->pMapName) == ShflStringSizeOfBuffer(pMapping->pMapName)
69 && memcmp(pLoadedMapping->pMapName, pMapping->pMapName, ShflStringSizeOfBuffer(pMapping->pMapName)) == 0)
70 {
71 /* Actual index is i. */
72 aIndexFromRoot[root] = i;
73
74 /* Update the mapping properties. */
75 pMapping->cMappings = pLoadedMapping->cMappings;
76
77 return VINF_SUCCESS;
78 }
79 }
80
81 return VERR_INVALID_PARAMETER;
82}
83
84MAPPING *vbsfMappingGetByRoot(SHFLROOT root)
85{
86 if (root < RT_ELEMENTS(aIndexFromRoot))
87 {
88 SHFLROOT iMapping = aIndexFromRoot[root];
89
90 if ( iMapping != SHFL_ROOT_NIL
91 && iMapping < RT_ELEMENTS(FolderMapping))
92 {
93 return &FolderMapping[iMapping];
94 }
95 }
96
97 return NULL;
98}
99
100static SHFLROOT vbsfMappingGetRootFromIndex(SHFLROOT iMapping)
101{
102 unsigned root;
103
104 for (root = 0; root < RT_ELEMENTS(aIndexFromRoot); root++)
105 {
106 if (iMapping == aIndexFromRoot[root])
107 {
108 return root;
109 }
110 }
111
112 return SHFL_ROOT_NIL;
113}
114
115static MAPPING *vbsfMappingGetByName (PRTUTF16 pwszName, SHFLROOT *pRoot)
116{
117 unsigned i;
118
119 for (i=0; i<SHFL_MAX_MAPPINGS; i++)
120 {
121 if (FolderMapping[i].fValid == true)
122 {
123 if (!RTUtf16LocaleICmp(FolderMapping[i].pMapName->String.ucs2, pwszName))
124 {
125 SHFLROOT root = vbsfMappingGetRootFromIndex(i);
126
127 if (root != SHFL_ROOT_NIL)
128 {
129 if (pRoot)
130 {
131 *pRoot = root;
132 }
133 return &FolderMapping[i];
134 }
135 else
136 {
137 AssertFailed();
138 }
139 }
140 }
141 }
142
143 return NULL;
144}
145
146static void vbsfRootHandleAdd(SHFLROOT iMapping)
147{
148 unsigned root;
149
150 for (root = 0; root < RT_ELEMENTS(aIndexFromRoot); root++)
151 {
152 if (aIndexFromRoot[root] == SHFL_ROOT_NIL)
153 {
154 aIndexFromRoot[root] = iMapping;
155 return;
156 }
157 }
158
159 AssertFailed();
160}
161
162static void vbsfRootHandleRemove(SHFLROOT iMapping)
163{
164 unsigned root;
165
166 for (root = 0; root < RT_ELEMENTS(aIndexFromRoot); root++)
167 {
168 if (aIndexFromRoot[root] == iMapping)
169 {
170 aIndexFromRoot[root] = SHFL_ROOT_NIL;
171 return;
172 }
173 }
174
175 AssertFailed();
176}
177
178
179
180#ifdef UNITTEST
181/** Unit test the SHFL_FN_ADD_MAPPING API. Located here as a form of API
182 * documentation. */
183void testMappingsAdd(RTTEST hTest)
184{
185 /* If the number or types of parameters are wrong the API should fail. */
186 testMappingsAddBadParameters(hTest);
187 /* Add tests as required... */
188}
189#endif
190/*
191 * We are always executed from one specific HGCM thread. So thread safe.
192 */
193int vbsfMappingsAdd(PSHFLSTRING pFolderName, PSHFLSTRING pMapName,
194 bool fWritable, bool fAutoMount, bool fSymlinksCreate, bool fMissing)
195{
196 unsigned i;
197
198 Assert(pFolderName && pMapName);
199
200 Log(("vbsfMappingsAdd %ls\n", pMapName->String.ucs2));
201
202 /* check for duplicates */
203 for (i=0; i<SHFL_MAX_MAPPINGS; i++)
204 {
205 if (FolderMapping[i].fValid == true)
206 {
207 if (!RTUtf16LocaleICmp(FolderMapping[i].pMapName->String.ucs2, pMapName->String.ucs2))
208 {
209 AssertMsgFailed(("vbsfMappingsAdd: %ls mapping already exists!!\n", pMapName->String.ucs2));
210 return VERR_ALREADY_EXISTS;
211 }
212 }
213 }
214
215 for (i=0; i<SHFL_MAX_MAPPINGS; i++)
216 {
217 if (FolderMapping[i].fValid == false)
218 {
219 int rc = RTUtf16ToUtf8(pFolderName->String.ucs2, &FolderMapping[i].pszFolderName);
220 AssertRCReturn(rc, rc);
221
222 FolderMapping[i].pMapName = (PSHFLSTRING)RTMemAlloc(ShflStringSizeOfBuffer(pMapName));
223 if (!FolderMapping[i].pMapName)
224 {
225 RTStrFree(FolderMapping[i].pszFolderName);
226 AssertFailed();
227 return VERR_NO_MEMORY;
228 }
229
230 FolderMapping[i].pMapName->u16Length = pMapName->u16Length;
231 FolderMapping[i].pMapName->u16Size = pMapName->u16Size;
232 memcpy(FolderMapping[i].pMapName->String.ucs2, pMapName->String.ucs2, pMapName->u16Size);
233
234 FolderMapping[i].fValid = true;
235 FolderMapping[i].cMappings = 0;
236 FolderMapping[i].fWritable = fWritable;
237 FolderMapping[i].fAutoMount = fAutoMount;
238 FolderMapping[i].fSymlinksCreate = fSymlinksCreate;
239 FolderMapping[i].fMissing = fMissing;
240
241 /* Check if the host file system is case sensitive */
242 RTFSPROPERTIES prop;
243 char *pszAsciiRoot;
244
245 rc = RTStrUtf8ToCurrentCP(&pszAsciiRoot, FolderMapping[i].pszFolderName);
246 if (RT_SUCCESS(rc))
247 {
248 rc = RTFsQueryProperties(pszAsciiRoot, &prop);
249 AssertRC(rc);
250 RTStrFree(pszAsciiRoot);
251 }
252
253 FolderMapping[i].fHostCaseSensitive = RT_SUCCESS(rc) ? prop.fCaseSensitive : false;
254 vbsfRootHandleAdd(i);
255 break;
256 }
257 }
258 if (i == SHFL_MAX_MAPPINGS)
259 {
260 AssertMsgFailed(("vbsfMappingsAdd: no more room to add mapping %ls to %ls!!\n", pFolderName->String.ucs2, pMapName->String.ucs2));
261 return VERR_TOO_MUCH_DATA;
262 }
263
264 Log(("vbsfMappingsAdd: added mapping %ls to %ls\n", pFolderName->String.ucs2, pMapName->String.ucs2));
265 return VINF_SUCCESS;
266}
267
268#ifdef UNITTEST
269/** Unit test the SHFL_FN_REMOVE_MAPPING API. Located here as a form of API
270 * documentation. */
271void testMappingsRemove(RTTEST hTest)
272{
273 /* If the number or types of parameters are wrong the API should fail. */
274 testMappingsRemoveBadParameters(hTest);
275 /* Add tests as required... */
276}
277#endif
278int vbsfMappingsRemove(PSHFLSTRING pMapName)
279{
280 unsigned i;
281
282 Assert(pMapName);
283
284 Log(("vbsfMappingsRemove %ls\n", pMapName->String.ucs2));
285 for (i=0; i<SHFL_MAX_MAPPINGS; i++)
286 {
287 if (FolderMapping[i].fValid == true)
288 {
289 if (!RTUtf16LocaleICmp(FolderMapping[i].pMapName->String.ucs2, pMapName->String.ucs2))
290 {
291 if (FolderMapping[i].cMappings != 0)
292 {
293 Log(("vbsfMappingsRemove: trying to remove active share %ls\n", pMapName->String.ucs2));
294 return VERR_PERMISSION_DENIED;
295 }
296
297 RTStrFree(FolderMapping[i].pszFolderName);
298 RTMemFree(FolderMapping[i].pMapName);
299 FolderMapping[i].pszFolderName = NULL;
300 FolderMapping[i].pMapName = NULL;
301 FolderMapping[i].fValid = false;
302 vbsfRootHandleRemove(i);
303 break;
304 }
305 }
306 }
307
308 if (i == SHFL_MAX_MAPPINGS)
309 {
310 AssertMsgFailed(("vbsfMappingsRemove: mapping %ls not found!!!!\n", pMapName->String.ucs2));
311 return VERR_FILE_NOT_FOUND;
312 }
313 Log(("vbsfMappingsRemove: mapping %ls removed\n", pMapName->String.ucs2));
314 return VINF_SUCCESS;
315}
316
317const char* vbsfMappingsQueryHostRoot(SHFLROOT root)
318{
319 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
320 AssertReturn(pFolderMapping, NULL);
321 if (pFolderMapping->fMissing)
322 return NULL;
323 return pFolderMapping->pszFolderName;
324}
325
326int vbsfMappingsQueryHostRootEx(SHFLROOT hRoot, const char **ppszRoot, uint32_t *pcbRootLen)
327{
328 MAPPING *pFolderMapping = vbsfMappingGetByRoot(hRoot);
329 AssertReturn(pFolderMapping, VERR_INVALID_PARAMETER);
330 if (pFolderMapping->fMissing)
331 return VERR_NOT_FOUND;
332 if ( pFolderMapping->pszFolderName == NULL
333 || pFolderMapping->pszFolderName[0] == 0)
334 return VERR_NOT_FOUND;
335 *ppszRoot = pFolderMapping->pszFolderName;
336 *pcbRootLen = strlen(pFolderMapping->pszFolderName);
337 return VINF_SUCCESS;
338}
339
340bool vbsfIsGuestMappingCaseSensitive(SHFLROOT root)
341{
342 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
343 AssertReturn(pFolderMapping, false);
344 return pFolderMapping->fGuestCaseSensitive;
345}
346
347bool vbsfIsHostMappingCaseSensitive(SHFLROOT root)
348{
349 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
350 AssertReturn(pFolderMapping, false);
351 return pFolderMapping->fHostCaseSensitive;
352}
353
354#ifdef UNITTEST
355/** Unit test the SHFL_FN_QUERY_MAPPINGS API. Located here as a form of API
356 * documentation (or should it better be inline in include/VBox/shflsvc.h?) */
357void testMappingsQuery(RTTEST hTest)
358{
359 /* The API should return all mappings if we provide enough buffers. */
360 testMappingsQuerySimple(hTest);
361 /* If we provide too few buffers that should be signalled correctly. */
362 testMappingsQueryTooFewBuffers(hTest);
363 /* The SHFL_MF_AUTOMOUNT flag means return only auto-mounted mappings. */
364 testMappingsQueryAutoMount(hTest);
365 /* The mappings return array must have numberOfMappings entries. */
366 testMappingsQueryArrayWrongSize(hTest);
367}
368#endif
369/**
370 * Note: If pMappings / *pcMappings is smaller than the actual amount of mappings
371 * that *could* have been returned *pcMappings contains the required buffer size
372 * so that the caller can retry the operation if wanted.
373 */
374int vbsfMappingsQuery(PSHFLCLIENTDATA pClient, PSHFLMAPPING pMappings, uint32_t *pcMappings)
375{
376 int rc = VINF_SUCCESS;
377
378 uint32_t cMappings = 0; /* Will contain actual valid mappings. */
379 uint32_t idx = 0; /* Current index in mappings buffer. */
380
381 LogFlow(("vbsfMappingsQuery: pClient = %p, pMappings = %p, pcMappings = %p, *pcMappings = %d\n",
382 pClient, pMappings, pcMappings, *pcMappings));
383
384 for (uint32_t i = 0; i < SHFL_MAX_MAPPINGS; i++)
385 {
386 MAPPING *pFolderMapping = vbsfMappingGetByRoot(i);
387 if ( pFolderMapping != NULL
388 && pFolderMapping->fValid == true)
389 {
390 if (idx < *pcMappings)
391 {
392 /* Skip mappings which are not marked for auto-mounting if
393 * the SHFL_MF_AUTOMOUNT flag ist set. */
394 if ( (pClient->fu32Flags & SHFL_MF_AUTOMOUNT)
395 && !pFolderMapping->fAutoMount)
396 continue;
397
398 pMappings[idx].u32Status = SHFL_MS_NEW;
399 pMappings[idx].root = i;
400 idx++;
401 }
402 cMappings++;
403 }
404 }
405
406 /* Return actual number of mappings, regardless whether the handed in
407 * mapping buffer was big enough. */
408 *pcMappings = cMappings;
409
410 LogFlow(("vbsfMappingsQuery: return rc = %Rrc\n", rc));
411 return rc;
412}
413
414#ifdef UNITTEST
415/** Unit test the SHFL_FN_QUERY_MAP_NAME API. Located here as a form of API
416 * documentation. */
417void testMappingsQueryName(RTTEST hTest)
418{
419 /* If we query an valid mapping it should be returned. */
420 testMappingsQueryNameValid(hTest);
421 /* If we query an invalid mapping that should be signalled. */
422 testMappingsQueryNameInvalid(hTest);
423 /* If we pass in a bad string buffer that should be detected. */
424 testMappingsQueryNameBadBuffer(hTest);
425}
426#endif
427int vbsfMappingsQueryName(PSHFLCLIENTDATA pClient, SHFLROOT root, SHFLSTRING *pString)
428{
429 int rc = VINF_SUCCESS;
430
431 LogFlow(("vbsfMappingsQuery: pClient = %p, root = %d, *pString = %p\n",
432 pClient, root, pString));
433
434 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
435 if (pFolderMapping == NULL)
436 {
437 return VERR_INVALID_PARAMETER;
438 }
439
440 if (BIT_FLAG(pClient->fu32Flags, SHFL_CF_UTF8))
441 {
442 /* Not implemented. */
443 AssertFailed();
444 return VERR_INVALID_PARAMETER;
445 }
446
447 if (pFolderMapping->fValid == true)
448 {
449 if (pString->u16Size < pFolderMapping->pMapName->u16Size)
450 {
451 Log(("vbsfMappingsQuery: passed string too short (%d < %d bytes)!\n",
452 pString->u16Size, pFolderMapping->pMapName->u16Size));
453 rc = VERR_INVALID_PARAMETER;
454 }
455 else
456 {
457 pString->u16Length = pFolderMapping->pMapName->u16Length;
458 memcpy(pString->String.ucs2, pFolderMapping->pMapName->String.ucs2,
459 pFolderMapping->pMapName->u16Size);
460 }
461 }
462 else
463 rc = VERR_FILE_NOT_FOUND;
464
465 LogFlow(("vbsfMappingsQuery:Name return rc = %Rrc\n", rc));
466
467 return rc;
468}
469
470int vbsfMappingsQueryWritable(PSHFLCLIENTDATA pClient, SHFLROOT root, bool *fWritable)
471{
472 int rc = VINF_SUCCESS;
473
474 LogFlow(("vbsfMappingsQueryWritable: pClient = %p, root = %d\n", pClient, root));
475
476 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
477 AssertReturn(pFolderMapping, VERR_INVALID_PARAMETER);
478
479 if ( pFolderMapping->fValid
480 && !pFolderMapping->fMissing)
481 *fWritable = pFolderMapping->fWritable;
482 else
483 rc = VERR_FILE_NOT_FOUND;
484
485 LogFlow(("vbsfMappingsQuery:Writable return rc = %Rrc\n", rc));
486
487 return rc;
488}
489
490int vbsfMappingsQueryAutoMount(PSHFLCLIENTDATA pClient, SHFLROOT root, bool *fAutoMount)
491{
492 int rc = VINF_SUCCESS;
493
494 LogFlow(("vbsfMappingsQueryAutoMount: pClient = %p, root = %d\n", pClient, root));
495
496 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
497 AssertReturn(pFolderMapping, VERR_INVALID_PARAMETER);
498
499 if (pFolderMapping->fValid == true)
500 *fAutoMount = pFolderMapping->fAutoMount;
501 else
502 rc = VERR_FILE_NOT_FOUND;
503
504 LogFlow(("vbsfMappingsQueryAutoMount:Writable return rc = %Rrc\n", rc));
505
506 return rc;
507}
508
509int vbsfMappingsQuerySymlinksCreate(PSHFLCLIENTDATA pClient, SHFLROOT root, bool *fSymlinksCreate)
510{
511 int rc = VINF_SUCCESS;
512
513 LogFlow(("vbsfMappingsQueryAutoMount: pClient = %p, root = %d\n", pClient, root));
514
515 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
516 AssertReturn(pFolderMapping, VERR_INVALID_PARAMETER);
517
518 if (pFolderMapping->fValid == true)
519 *fSymlinksCreate = pFolderMapping->fSymlinksCreate;
520 else
521 rc = VERR_FILE_NOT_FOUND;
522
523 LogFlow(("vbsfMappingsQueryAutoMount:SymlinksCreate return rc = %Rrc\n", rc));
524
525 return rc;
526}
527
528#ifdef UNITTEST
529/** Unit test the SHFL_FN_MAP_FOLDER API. Located here as a form of API
530 * documentation. */
531void testMapFolder(RTTEST hTest)
532{
533 /* If we try to map a valid name we should get the root. */
534 testMapFolderValid(hTest);
535 /* If we try to map a valid name we should get VERR_FILE_NOT_FOUND. */
536 testMapFolderInvalid(hTest);
537 /* If we map a folder twice we can unmap it twice.
538 * Currently unmapping too often is only asserted but not signalled. */
539 testMapFolderTwice(hTest);
540 /* The delimiter should be converted in e.g. file delete operations. */
541 testMapFolderDelimiter(hTest);
542 /* Test case sensitive mapping by opening a file with the wrong case. */
543 testMapFolderCaseSensitive(hTest);
544 /* Test case insensitive mapping by opening a file with the wrong case. */
545 testMapFolderCaseInsensitive(hTest);
546 /* If the number or types of parameters are wrong the API should fail. */
547 testMapFolderBadParameters(hTest);
548}
549#endif
550int vbsfMapFolder(PSHFLCLIENTDATA pClient, PSHFLSTRING pszMapName,
551 RTUTF16 wcDelimiter, bool fCaseSensitive, SHFLROOT *pRoot)
552{
553 MAPPING *pFolderMapping = NULL;
554
555 if (BIT_FLAG(pClient->fu32Flags, SHFL_CF_UTF8))
556 {
557 Log(("vbsfMapFolder %s\n", pszMapName->String.utf8));
558 }
559 else
560 {
561 Log(("vbsfMapFolder %ls\n", pszMapName->String.ucs2));
562 }
563
564 AssertMsgReturn(wcDelimiter == '/' || wcDelimiter == '\\',
565 ("Invalid path delimiter: %#x\n", wcDelimiter),
566 VERR_INVALID_PARAMETER);
567 if (pClient->PathDelimiter == 0)
568 {
569 pClient->PathDelimiter = wcDelimiter;
570 }
571 else
572 {
573 AssertMsgReturn(wcDelimiter == pClient->PathDelimiter,
574 ("wcDelimiter=%#x PathDelimiter=%#x", wcDelimiter, pClient->PathDelimiter),
575 VERR_INVALID_PARAMETER);
576 }
577
578 if (BIT_FLAG(pClient->fu32Flags, SHFL_CF_UTF8))
579 {
580 int rc;
581 PRTUTF16 utf16Name;
582
583 rc = RTStrToUtf16 ((const char *) pszMapName->String.utf8, &utf16Name);
584 if (RT_FAILURE (rc))
585 return rc;
586
587 pFolderMapping = vbsfMappingGetByName(utf16Name, pRoot);
588 RTUtf16Free (utf16Name);
589 }
590 else
591 {
592 pFolderMapping = vbsfMappingGetByName(pszMapName->String.ucs2, pRoot);
593 }
594
595 if (!pFolderMapping)
596 {
597 return VERR_FILE_NOT_FOUND;
598 }
599
600 pFolderMapping->cMappings++;
601 Assert(pFolderMapping->cMappings == 1 || pFolderMapping->fGuestCaseSensitive == fCaseSensitive);
602 pFolderMapping->fGuestCaseSensitive = fCaseSensitive;
603 return VINF_SUCCESS;
604}
605
606#ifdef UNITTEST
607/** Unit test the SHFL_FN_UNMAP_FOLDER API. Located here as a form of API
608 * documentation. */
609void testUnmapFolder(RTTEST hTest)
610{
611 /* Unmapping a mapped folder should succeed.
612 * If the folder is not mapped this is only asserted, not signalled. */
613 testUnmapFolderValid(hTest);
614 /* Unmapping a non-existant root should fail. */
615 testUnmapFolderInvalid(hTest);
616 /* If the number or types of parameters are wrong the API should fail. */
617 testUnmapFolderBadParameters(hTest);
618}
619#endif
620int vbsfUnmapFolder(PSHFLCLIENTDATA pClient, SHFLROOT root)
621{
622 int rc = VINF_SUCCESS;
623
624 MAPPING *pFolderMapping = vbsfMappingGetByRoot(root);
625 if (pFolderMapping == NULL)
626 {
627 AssertFailed();
628 return VERR_FILE_NOT_FOUND;
629 }
630
631 Assert(pFolderMapping->fValid == true && pFolderMapping->cMappings > 0);
632 if (pFolderMapping->cMappings > 0)
633 pFolderMapping->cMappings--;
634
635 Log(("vbsfUnmapFolder\n"));
636 return rc;
637}
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