VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/vfs/vfschain.cpp@ 39070

Last change on this file since 39070 was 37596, checked in by vboxsync, 14 years ago

*: RTFILE becomes a pointer, RTFileOpen++ expands it's flags paramter from uint32_t to uint64_t.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.2 KB
Line 
1/* $Id: vfschain.cpp 37596 2011-06-22 19:30:06Z vboxsync $ */
2/** @file
3 * IPRT - Virtual File System, Chains.
4 */
5
6/*
7 * Copyright (C) 2010 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/vfs.h>
32#include <iprt/vfslowlevel.h>
33
34#include <iprt/asm.h>
35#include <iprt/critsect.h>
36#include <iprt/err.h>
37#include <iprt/file.h>
38#include <iprt/mem.h>
39#include <iprt/once.h>
40#include <iprt/param.h>
41#include <iprt/path.h>
42#include <iprt/semaphore.h>
43#include <iprt/string.h>
44
45#include "internal/file.h"
46#include "internal/magics.h"
47//#include "internal/vfs.h"
48
49
50
51/*******************************************************************************
52* Global Variables *
53*******************************************************************************/
54/** Init the critical section once. */
55static RTONCE g_rtVfsChainElementInitOnce;
56/** Critical section protecting g_rtVfsChainElementProviderList. */
57static RTCRITSECT g_rtVfsChainElementCritSect;
58/** List of VFS chain element providers. */
59static RTLISTNODE g_rtVfsChainElementProviderList;
60
61
62/**
63 * Initializes the globals via RTOnce.
64 *
65 * @returns IPRT status code
66 * @param pvUser1 Unused, ignored.
67 * @param pvUser2 Unused, ignored.
68 */
69static DECLCALLBACK(int) rtVfsChainElementRegisterInit(void *pvUser1, void *pvUser2)
70{
71 NOREF(pvUser1);
72 NOREF(pvUser2);
73 return RTCritSectInit(&g_rtVfsChainElementCritSect);
74}
75
76
77RTDECL(int) RTVfsChainElementRegisterProvider(PRTVFSCHAINELEMENTREG pRegRec, bool fFromCtor)
78{
79 int rc;
80
81 /*
82 * Input validation.
83 */
84 AssertPtrReturn(pRegRec, VERR_INVALID_POINTER);
85 AssertMsgReturn(pRegRec->uVersion == RTVFSCHAINELEMENTREG_VERSION, ("%#x", pRegRec->uVersion), VERR_INVALID_POINTER);
86 AssertMsgReturn(pRegRec->uEndMarker == RTVFSCHAINELEMENTREG_VERSION, ("%#zx", pRegRec->uEndMarker), VERR_INVALID_POINTER);
87 AssertReturn(pRegRec->fReserved == 0, VERR_INVALID_POINTER);
88 AssertPtrReturn(pRegRec->pszName, VERR_INVALID_POINTER);
89 AssertPtrNullReturn(pRegRec->pfnOpenVfs, VERR_INVALID_POINTER);
90 AssertPtrNullReturn(pRegRec->pfnOpenDir, VERR_INVALID_POINTER);
91 AssertPtrNullReturn(pRegRec->pfnOpenFile, VERR_INVALID_POINTER);
92 AssertPtrNullReturn(pRegRec->pfnOpenIoStream, VERR_INVALID_POINTER);
93 AssertPtrNullReturn(pRegRec->pfnOpenFsStream, VERR_INVALID_POINTER);
94
95 /*
96 * Init and take the lock.
97 */
98 if (!fFromCtor)
99 {
100 rc = RTOnce(&g_rtVfsChainElementInitOnce, rtVfsChainElementRegisterInit, NULL, NULL);
101 if (RT_FAILURE(rc))
102 return rc;
103 rc = RTCritSectEnter(&g_rtVfsChainElementCritSect);
104 if (RT_FAILURE(rc))
105 return rc;
106 }
107
108 /*
109 * Duplicate name?
110 */
111 rc = VINF_SUCCESS;
112 PRTVFSCHAINELEMENTREG pIterator, pIterNext;
113 RTListForEachSafe(&g_rtVfsChainElementProviderList, pIterator, pIterNext, RTVFSCHAINELEMENTREG, ListEntry)
114 {
115 if (!strcmp(pIterator->pszName, pRegRec->pszName))
116 {
117 AssertMsgFailed(("duplicate name '%s' old=%p new=%p\n", pIterator->pszName, pIterator, pRegRec));
118 rc = VERR_ALREADY_EXISTS;
119 break;
120 }
121 }
122
123 /*
124 * If not, append the record to the list.
125 */
126 if (RT_SUCCESS(rc))
127 RTListAppend(&g_rtVfsChainElementProviderList, &pRegRec->ListEntry);
128
129 /*
130 * Leave the lock and return.
131 */
132 if (!fFromCtor)
133 RTCritSectLeave(&g_rtVfsChainElementCritSect);
134 return rc;
135}
136
137
138/**
139 * Allocates and initializes an empty spec
140 *
141 * @returns Pointer to the spec on success, NULL on failure.
142 */
143static PRTVFSCHAINSPEC rtVfsChainSpecAlloc(void)
144{
145 PRTVFSCHAINSPEC pSpec = (PRTVFSCHAINSPEC)RTMemTmpAlloc(sizeof(*pSpec));
146 if (pSpec)
147 {
148 pSpec->iActionElement = UINT32_MAX;
149 pSpec->cElements = 0;
150 pSpec->paElements = NULL;
151 }
152 return pSpec;
153}
154
155
156/**
157 * Duplicate a spec string.
158 *
159 * This differs from RTStrDupN in that it uses RTMemTmpAlloc instead of
160 * RTMemAlloc.
161 *
162 * @returns String copy on success, NULL on failure.
163 * @param psz The string to duplicate.
164 * @param cch The number of bytes to duplicate.
165 * @param prc The status code variable to set on failure. (Leeps the
166 * code shorter. -lazy bird)
167 */
168DECLINLINE(char *) rtVfsChainSpecDupStrN(const char *psz, size_t cch, int *prc)
169{
170 char *pszCopy = (char *)RTMemTmpAlloc(cch + 1);
171 if (pszCopy)
172 {
173 if (!memchr(psz, '\\', cch))
174 {
175 /* Plain string, copy it raw. */
176 memcpy(pszCopy, psz, cch);
177 pszCopy[cch] = '\0';
178 }
179 else
180 {
181 /* Has escape sequences, must unescape it. */
182 char *pszDst = pszCopy;
183 while (cch)
184 {
185 char ch = *psz++;
186 if (ch == '\\')
187 {
188 char ch2 = psz[2];
189 if (ch2 == '(' || ch2 == ')' || ch2 == '\\' || ch2 == ',')
190 {
191 psz++;
192 ch = ch2;
193 }
194 }
195 *pszDst++ = ch;
196 }
197 *pszDst = '\0';
198 }
199 }
200 else
201 *prc = VERR_NO_TMP_MEMORY;
202 return pszCopy;
203}
204
205
206/**
207 * Adds an empty element to the chain specification.
208 *
209 * The caller is responsible for filling it the element attributes.
210 *
211 * @returns Pointer to the new element on success, NULL on failure. The
212 * pointer is only valid till the next call to this function.
213 * @param pSpec The chain specification.
214 * @param prc The status code variable to set on failure. (Leeps the
215 * code shorter. -lazy bird)
216 */
217static PRTVFSCHAINELEMSPEC rtVfsChainSpecAddElement(PRTVFSCHAINSPEC pSpec, int *prc)
218{
219 AssertPtr(pSpec);
220
221 /*
222 * Resize the element table if necessary.
223 */
224 uint32_t const iElement = pSpec->cElements;
225 if ((iElement % 32) == 0)
226 {
227 PRTVFSCHAINELEMSPEC paNew = (PRTVFSCHAINELEMSPEC)RTMemTmpAlloc((iElement + 32) * sizeof(paNew[0]));
228 if (!paNew)
229 {
230 *prc = VERR_NO_TMP_MEMORY;
231 return NULL;
232 }
233
234 memcpy(paNew, pSpec->paElements, iElement * sizeof(paNew[0]));
235 RTMemTmpFree(pSpec->paElements);
236 pSpec->paElements = paNew;
237 }
238
239 /*
240 * Initialize and add the new element.
241 */
242 PRTVFSCHAINELEMSPEC pElement = &pSpec->paElements[iElement];
243 pElement->pszProvider = NULL;
244 pElement->enmTypeIn = iElement ? pSpec->paElements[iElement - 1].enmTypeOut : RTVFSOBJTYPE_INVALID;
245 pElement->enmTypeOut = RTVFSOBJTYPE_INVALID;
246 pElement->enmAction = RTVFSCHAINACTION_INVALID;
247 pElement->cArgs = 0;
248 pElement->papszArgs = 0;
249
250 pSpec->cElements = iElement + 1;
251 return pElement;
252}
253
254
255/**
256 * Adds an argument to the element spec.
257 *
258 * @returns IPRT status code.
259 * @param pElement The element.
260 * @param psz The start of the argument string.
261 * @param cch The length of the argument string, escape
262 * sequences counted twice.
263 */
264static int rtVfsChainSpecElementAddArg(PRTVFSCHAINELEMSPEC pElement, const char *psz, size_t cch)
265{
266 uint32_t iArg = pElement->cArgs;
267 if ((iArg % 32) == 0)
268 {
269 char **papszNew = (char **)RTMemTmpAlloc((iArg + 32 + 1) * sizeof(papszNew[0]));
270 if (!papszNew)
271 return VERR_NO_TMP_MEMORY;
272 memcpy(papszNew, pElement->papszArgs, iArg * sizeof(papszNew[0]));
273 RTMemTmpFree(pElement->papszArgs);
274 pElement->papszArgs = papszNew;
275 }
276
277 int rc = VINF_SUCCESS;
278 pElement->papszArgs[iArg] = rtVfsChainSpecDupStrN(psz, cch, &rc);
279 pElement->papszArgs[iArg + 1] = NULL;
280 pElement->cArgs = iArg + 1;
281 return rc;
282}
283
284
285RTDECL(void) RTVfsChainSpecFree(PRTVFSCHAINSPEC pSpec)
286{
287 if (!pSpec)
288 return;
289
290 uint32_t i = pSpec->cElements;
291 while (i-- > 0)
292 {
293 uint32_t iArg = pSpec->paElements[i].cArgs;
294 while (iArg-- > 0)
295 RTMemTmpFree(pSpec->paElements[i].papszArgs[iArg]);
296 RTMemTmpFree(pSpec->paElements[i].papszArgs);
297 RTMemTmpFree(pSpec->paElements[i].pszProvider);
298 }
299
300 RTMemTmpFree(pSpec->paElements);
301 pSpec->paElements = NULL;
302 RTMemTmpFree(pSpec);
303}
304
305
306/**
307 * Finds the end of the argument string.
308 *
309 * @returns The offset of the end character relative to @a psz.
310 * @param psz The argument string.
311 */
312static size_t rtVfsChainSpecFindArgEnd(const char *psz)
313{
314 char ch;
315 size_t off = 0;
316 while ( (ch = psz[off]) != '\0'
317 && ch != ','
318 && ch != ')'
319 && ch != '(')
320 {
321 /* check for escape sequences. */
322 if ( ch == '\\'
323 && (psz[off+1] == '(' || psz[off+1] == ')' || psz[off+1] == '\\' || psz[off+1] == ','))
324 off++;
325 off++;
326 }
327 return off;
328}
329
330/**
331 * Look for action.
332 *
333 * @returns Action.
334 * @param pszSpec The current spec position.
335 * @param pcchAction Where to return the length of the action
336 * string.
337 */
338static RTVFSCHAINACTION rtVfsChainSpecEatAction(const char *pszSpec, size_t *pcchAction)
339{
340 switch (*pszSpec)
341 {
342 case '|':
343 *pcchAction = 1;
344 return RTVFSCHAINACTION_PASSIVE;
345 case '>':
346 *pcchAction = 1;
347 return RTVFSCHAINACTION_PUSH;
348 default:
349 *pcchAction = 0;
350 return RTVFSCHAINACTION_NONE;
351 }
352}
353
354
355RTDECL(int) RTVfsChainSpecParse(const char *pszSpec, uint32_t fFlags, RTVFSCHAINACTION enmLeadingAction,
356 RTVFSCHAINACTION enmTrailingAction,
357 PRTVFSCHAINSPEC *ppSpec, const char **ppszError)
358{
359 AssertPtrNullReturn(ppszError, VERR_INVALID_POINTER);
360 if (ppszError)
361 *ppszError = NULL;
362 AssertPtrReturn(ppSpec, VERR_INVALID_POINTER);
363 *ppSpec = NULL;
364 AssertPtrReturn(pszSpec, VERR_INVALID_POINTER);
365 AssertReturn(!(fFlags & ~RTVFSCHAIN_PF_VALID_MASK), VERR_INVALID_PARAMETER);
366 AssertReturn(enmLeadingAction > RTVFSCHAINACTION_INVALID && enmLeadingAction < RTVFSCHAINACTION_END, VERR_INVALID_PARAMETER);
367
368 /*
369 * Check the start of the specification and allocate an empty return spec.
370 */
371 if (strncmp(pszSpec, RTVFSCHAIN_SPEC_PREFIX, sizeof(RTVFSCHAIN_SPEC_PREFIX) - 1))
372 return VERR_VFS_CHAIN_NO_PREFIX;
373 pszSpec = RTStrStripL(pszSpec + sizeof(RTVFSCHAIN_SPEC_PREFIX) - 1);
374 if (!*pszSpec)
375 return VERR_VFS_CHAIN_EMPTY;
376
377 PRTVFSCHAINSPEC pSpec = rtVfsChainSpecAlloc();
378 if (!pSpec)
379 return VERR_NO_TMP_MEMORY;
380
381 /*
382 * Parse the spec one element at a time.
383 */
384 int rc = VINF_SUCCESS;
385 const char *pszSrc = pszSpec;
386 while (*pszSrc && RT_SUCCESS(rc))
387 {
388 /*
389 * Pipe or redirection action symbol, except maybe the first time.
390 * The pipe symbol may occur at the end of the spec.
391 */
392 size_t cch;
393 RTVFSCHAINACTION enmAction = rtVfsChainSpecEatAction(pszSpec, &cch);
394 if (enmAction != RTVFSCHAINACTION_NONE)
395 {
396 pszSrc = RTStrStripL(pszSrc + cch);
397 if (!*pszSrc)
398 {
399 /* Fail if the caller does not approve of a trailing pipe (all
400 other actions non-trailing). */
401 if ( enmAction != enmTrailingAction
402 && !(fFlags & RTVFSCHAIN_PF_TRAILING_ACTION_OPTIONAL))
403 rc = VERR_VFS_CHAIN_EXPECTED_ELEMENT;
404 break;
405 }
406
407 /* There can only be one real action atm. */
408 if (enmAction != RTVFSCHAINACTION_PASSIVE)
409 {
410 if (pSpec->iActionElement != UINT32_MAX)
411 {
412 rc = VERR_VFS_CHAIN_MULTIPLE_ACTIONS;
413 break;
414 }
415 pSpec->iActionElement = pSpec->cElements;
416 }
417 }
418 else if (pSpec->cElements > 0)
419 {
420 rc = VERR_VFS_CHAIN_EXPECTED_ACTION;
421 break;
422 }
423
424 /* Check the leading action. */
425 if ( pSpec->cElements == 0
426 && enmAction != enmLeadingAction
427 && !(fFlags & RTVFSCHAIN_PF_LEADING_ACTION_OPTIONAL))
428 {
429 rc = VERR_VFS_CHAIN_UNEXPECTED_ACTION_TYPE;
430 break;
431 }
432
433 /*
434 * Ok, there should be an element here so add one to the return struct.
435 */
436 PRTVFSCHAINELEMSPEC pElement = rtVfsChainSpecAddElement(pSpec, &rc);
437 if (!pElement)
438 break;
439 pElement->enmAction = enmAction;
440
441 /*
442 * First up is the VFS object type followed by a parentheses.
443 */
444 if (strncmp(pszSrc, "base", cch = 4) == 0)
445 pElement->enmTypeOut = RTVFSOBJTYPE_BASE;
446 else if (strncmp(pszSrc, "vfs", cch = 3) == 0)
447 pElement->enmTypeOut = RTVFSOBJTYPE_VFS;
448 else if (strncmp(pszSrc, "fss", cch = 3) == 0)
449 pElement->enmTypeOut = RTVFSOBJTYPE_FS_STREAM;
450 else if (strncmp(pszSrc, "ios", cch = 3) == 0)
451 pElement->enmTypeOut = RTVFSOBJTYPE_IO_STREAM;
452 else if (strncmp(pszSrc, "dir", cch = 3) == 0)
453 pElement->enmTypeOut = RTVFSOBJTYPE_DIR;
454 else if (strncmp(pszSrc, "file", cch = 4) == 0)
455 pElement->enmTypeOut = RTVFSOBJTYPE_FILE;
456 else if (strncmp(pszSrc, "sym", cch = 3) == 0)
457 pElement->enmTypeOut = RTVFSOBJTYPE_SYMLINK;
458 else
459 {
460 rc = VERR_VFS_CHAIN_UNKNOWN_TYPE;
461 break;
462 }
463 pszSrc += cch;
464
465 /* Check and skip the parentheses. */
466 if (*pszSrc != '(')
467 {
468 rc = VERR_VFS_CHAIN_EXPECTED_LEFT_PARENTHESES;
469 break;
470 }
471 pszSrc = RTStrStripL(pszSrc + 1);
472
473 /*
474 * The name of the element provider.
475 */
476 cch = rtVfsChainSpecFindArgEnd(pszSrc);
477 if (!cch)
478 {
479 rc = VERR_VFS_CHAIN_EXPECTED_PROVIDER_NAME;
480 break;
481 }
482 pElement->pszProvider = rtVfsChainSpecDupStrN(pszSrc, cch, &rc);
483 if (!pElement->pszProvider)
484 break;
485 pszSrc += cch;
486
487 /*
488 * The arguments.
489 */
490 while (*pszSrc == ',')
491 {
492 pszSrc = RTStrStripL(pszSrc + 1);
493 cch = rtVfsChainSpecFindArgEnd(pszSrc);
494 rc = rtVfsChainSpecElementAddArg(pElement, pszSrc, cch);
495 pszSrc += cch;
496 }
497
498 /* Must end with a right parentheses. */
499 if (*pszSrc != ')')
500 {
501 rc = VERR_VFS_CHAIN_EXPECTED_RIGHT_PARENTHESES;
502 break;
503 }
504 pszSrc = RTStrStripL(pszSrc + 1);
505 }
506
507 /*
508 * Cleanup and set the error indicator on failure.
509 */
510 if (RT_FAILURE(rc))
511 {
512 if (ppszError)
513 *ppszError = pszSrc;
514 RTVfsChainSpecFree(pSpec);
515 }
516 return rc;
517}
518
519
520
521
522
523RTDECL(int) RTVfsChainElementDeregisterProvider(PRTVFSCHAINELEMENTREG pRegRec, bool fFromDtor)
524{
525 /*
526 * Fend off wildlife.
527 */
528 if (pRegRec == NULL)
529 return VINF_SUCCESS;
530 AssertPtrReturn(pRegRec, VERR_INVALID_POINTER);
531 AssertMsgReturn(pRegRec->uVersion == RTVFSCHAINELEMENTREG_VERSION, ("%#x", pRegRec->uVersion), VERR_INVALID_POINTER);
532 AssertMsgReturn(pRegRec->uEndMarker == RTVFSCHAINELEMENTREG_VERSION, ("%#zx", pRegRec->uEndMarker), VERR_INVALID_POINTER);
533 AssertPtrReturn(pRegRec->pszName, VERR_INVALID_POINTER);
534
535 /*
536 * Take the lock if that's safe.
537 */
538 if (!fFromDtor)
539 RTCritSectEnter(&g_rtVfsChainElementCritSect);
540
541 /*
542 * Ok, remove it.
543 */
544 int rc = VERR_NOT_FOUND;
545 PRTVFSCHAINELEMENTREG pIterator, pIterNext;
546 RTListForEachSafe(&g_rtVfsChainElementProviderList, pIterator, pIterNext, RTVFSCHAINELEMENTREG, ListEntry)
547 {
548 if (pIterator == pRegRec)
549 {
550 RTListNodeRemove(&pRegRec->ListEntry);
551 rc = VINF_SUCCESS;
552 break;
553 }
554 }
555
556 /*
557 * Leave the lock and return.
558 */
559 if (!fFromDtor)
560 RTCritSectLeave(&g_rtVfsChainElementCritSect);
561 return rc;
562}
563
564
565RTDECL(int) RTVfsChainOpenFile(const char *pszSpec, uint64_t fOpen, PRTVFSFILE phVfsFile, const char **ppszError)
566{
567 AssertPtrReturn(pszSpec, VERR_INVALID_POINTER);
568 AssertReturn(*pszSpec != '\0', VERR_INVALID_PARAMETER);
569 AssertPtrReturn(phVfsFile, VERR_INVALID_POINTER);
570 if (ppszError)
571 *ppszError = NULL;
572
573 /*
574 * If it's not a VFS chain spec, treat it as a file.
575 */
576 int rc;
577 if (strncmp(pszSpec, RTVFSCHAIN_SPEC_PREFIX, sizeof(RTVFSCHAIN_SPEC_PREFIX) - 1))
578 {
579 RTFILE hFile;
580 rc = RTFileOpen(&hFile, pszSpec, fOpen);
581 if (RT_SUCCESS(rc))
582 {
583 RTVFSFILE hVfsFile;
584 rc = RTVfsFileFromRTFile(hFile, fOpen, false /*fLeaveOpen*/, &hVfsFile);
585 if (RT_SUCCESS(rc))
586 *phVfsFile = hVfsFile;
587 else
588 RTFileClose(hFile);
589 }
590 }
591 else
592 {
593 PRTVFSCHAINSPEC pSpec;
594 rc = RTVfsChainSpecParse(pszSpec,
595 RTVFSCHAIN_PF_NO_REAL_ACTION
596 | RTVFSCHAIN_PF_LEADING_ACTION_OPTIONAL,
597 RTVFSCHAINACTION_PASSIVE,
598 RTVFSCHAINACTION_NONE,
599 &pSpec,
600 ppszError);
601 if (RT_SUCCESS(rc))
602 {
603 /** @todo implement this when needed. */
604 rc = VERR_NOT_IMPLEMENTED;
605 RTVfsChainSpecFree(pSpec);
606 }
607 }
608 return rc;
609}
610
611
612RTDECL(int) RTVfsChainOpenIoStream(const char *pszSpec, uint64_t fOpen, PRTVFSIOSTREAM phVfsIos, const char **ppszError)
613{
614 AssertPtrReturn(pszSpec, VERR_INVALID_POINTER);
615 AssertReturn(*pszSpec != '\0', VERR_INVALID_PARAMETER);
616 AssertPtrReturn(phVfsIos, VERR_INVALID_POINTER);
617 if (ppszError)
618 *ppszError = NULL;
619
620 /*
621 * If it's not a VFS chain spec, treat it as a file.
622 */
623 int rc;
624 if (strncmp(pszSpec, RTVFSCHAIN_SPEC_PREFIX, sizeof(RTVFSCHAIN_SPEC_PREFIX) - 1))
625 {
626 RTFILE hFile;
627 rc = RTFileOpen(&hFile, pszSpec, fOpen);
628 if (RT_SUCCESS(rc))
629 {
630 RTVFSFILE hVfsFile;
631 rc = RTVfsFileFromRTFile(hFile, fOpen, false /*fLeaveOpen*/, &hVfsFile);
632 if (RT_SUCCESS(rc))
633 {
634 *phVfsIos = RTVfsFileToIoStream(hVfsFile);
635 RTVfsFileRelease(hVfsFile);
636 }
637 else
638 RTFileClose(hFile);
639 }
640 }
641 else
642 {
643 PRTVFSCHAINSPEC pSpec;
644 rc = RTVfsChainSpecParse(pszSpec,
645 RTVFSCHAIN_PF_NO_REAL_ACTION
646 | RTVFSCHAIN_PF_LEADING_ACTION_OPTIONAL,
647 RTVFSCHAINACTION_PASSIVE,
648 RTVFSCHAINACTION_NONE,
649 &pSpec,
650 ppszError);
651 if (RT_SUCCESS(rc))
652 {
653
654
655 rc = VERR_NOT_IMPLEMENTED;
656 RTVfsChainSpecFree(pSpec);
657 }
658 }
659 return rc;
660}
661
662
663
664RTDECL(bool) RTVfsChainIsSpec(const char *pszSpec)
665{
666 return pszSpec
667 && strcmp(pszSpec, RTVFSCHAIN_SPEC_PREFIX) == 0;
668}
669
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