VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/vfs/vfsbase.cpp@ 97885

Last change on this file since 97885 was 96861, checked in by vboxsync, 2 years ago

IPRT/Vfs,Main/Unattended: Detect fedora ISOs, various code cleanups. Added a parameter to RTVfsQueryLabel so we can get the primary volume ID of an ISO (required for fedora ISOs, as the joliet version is truncated). bugref:9781

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 146.4 KB
Line 
1/* $Id: vfsbase.cpp 96861 2022-09-26 10:52:54Z vboxsync $ */
2/** @file
3 * IPRT - Virtual File System, Base.
4 */
5
6/*
7 * Copyright (C) 2010-2022 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#define LOG_GROUP RTLOGGROUP_FS
42#include <iprt/vfs.h>
43#include <iprt/vfslowlevel.h>
44
45#include <iprt/asm.h>
46#include <iprt/err.h>
47#include <iprt/file.h>
48#include <iprt/log.h>
49#include <iprt/mem.h>
50#include <iprt/param.h>
51#include <iprt/path.h>
52#include <iprt/poll.h>
53#include <iprt/semaphore.h>
54#include <iprt/thread.h>
55#include <iprt/zero.h>
56
57#include "internal/file.h"
58#include "internal/fs.h"
59#include "internal/magics.h"
60#include "internal/path.h"
61//#include "internal/vfs.h"
62
63
64/*********************************************************************************************************************************
65* Defined Constants And Macros *
66*********************************************************************************************************************************/
67/** The instance data alignment. */
68#define RTVFS_INST_ALIGNMENT 16U
69
70/** The max number of symbolic links to resolve in a path. */
71#define RTVFS_MAX_LINKS 20U
72
73
74/** Asserts that the VFS base object vtable is valid. */
75#define RTVFSOBJ_ASSERT_OPS(a_pObjOps, a_enmType) \
76 do \
77 { \
78 Assert((a_pObjOps)->uVersion == RTVFSOBJOPS_VERSION); \
79 Assert((a_pObjOps)->enmType == (a_enmType) || (a_enmType) == RTVFSOBJTYPE_INVALID); \
80 AssertPtr((a_pObjOps)->pszName); \
81 Assert(*(a_pObjOps)->pszName); \
82 AssertPtr((a_pObjOps)->pfnClose); \
83 AssertPtr((a_pObjOps)->pfnQueryInfo); \
84 AssertPtrNull((a_pObjOps)->pfnQueryInfoEx); \
85 Assert((a_pObjOps)->uEndMarker == RTVFSOBJOPS_VERSION); \
86 } while (0)
87
88/** Asserts that the VFS set object vtable is valid. */
89#define RTVFSOBJSET_ASSERT_OPS(a_pSetOps, a_offObjOps) \
90 do \
91 { \
92 Assert((a_pSetOps)->uVersion == RTVFSOBJSETOPS_VERSION); \
93 Assert((a_pSetOps)->offObjOps == (a_offObjOps)); \
94 AssertPtrNull((a_pSetOps)->pfnSetMode); \
95 AssertPtrNull((a_pSetOps)->pfnSetTimes); \
96 AssertPtrNull((a_pSetOps)->pfnSetOwner); \
97 Assert((a_pSetOps)->uEndMarker == RTVFSOBJSETOPS_VERSION); \
98 } while (0)
99
100/** Asserts that the VFS directory vtable is valid. */
101#define RTVFSDIR_ASSERT_OPS(pDirOps, a_enmType) \
102 do { \
103 RTVFSOBJ_ASSERT_OPS(&(pDirOps)->Obj, a_enmType); \
104 RTVFSOBJSET_ASSERT_OPS(&(pDirOps)->ObjSet, RT_UOFFSETOF(RTVFSDIROPS, ObjSet) - RT_UOFFSETOF(RTVFSDIROPS, Obj)); \
105 Assert((pDirOps)->uVersion == RTVFSDIROPS_VERSION); \
106 Assert(!(pDirOps)->fReserved); \
107 AssertPtr((pDirOps)->pfnOpen); \
108 AssertPtrNull((pDirOps)->pfnOpenFile); \
109 AssertPtrNull((pDirOps)->pfnOpenDir); \
110 AssertPtrNull((pDirOps)->pfnCreateDir); \
111 AssertPtrNull((pDirOps)->pfnOpenSymlink); \
112 AssertPtr((pDirOps)->pfnCreateSymlink); \
113 AssertPtr((pDirOps)->pfnUnlinkEntry); \
114 AssertPtr((pDirOps)->pfnRewindDir); \
115 AssertPtr((pDirOps)->pfnReadDir); \
116 Assert((pDirOps)->uEndMarker == RTVFSDIROPS_VERSION); \
117 } while (0)
118
119/** Asserts that the VFS I/O stream vtable is valid. */
120#define RTVFSIOSTREAM_ASSERT_OPS(pIoStreamOps, a_enmType) \
121 do { \
122 RTVFSOBJ_ASSERT_OPS(&(pIoStreamOps)->Obj, a_enmType); \
123 Assert((pIoStreamOps)->uVersion == RTVFSIOSTREAMOPS_VERSION); \
124 Assert(!((pIoStreamOps)->fFeatures & ~RTVFSIOSTREAMOPS_FEAT_VALID_MASK)); \
125 AssertPtr((pIoStreamOps)->pfnRead); \
126 AssertPtrNull((pIoStreamOps)->pfnWrite); \
127 AssertPtr((pIoStreamOps)->pfnFlush); \
128 AssertPtrNull((pIoStreamOps)->pfnPollOne); \
129 AssertPtr((pIoStreamOps)->pfnTell); \
130 AssertPtrNull((pIoStreamOps)->pfnSkip); \
131 AssertPtrNull((pIoStreamOps)->pfnZeroFill); \
132 Assert((pIoStreamOps)->uEndMarker == RTVFSIOSTREAMOPS_VERSION); \
133 } while (0)
134
135/** Asserts that the VFS I/O stream vtable is valid. */
136#define RTVFSFILE_ASSERT_OPS(pFileOps, a_enmType) \
137 do { \
138 RTVFSIOSTREAM_ASSERT_OPS(&(pFileOps)->Stream, a_enmType); \
139 Assert((pFileOps)->uVersion == RTVFSFILEOPS_VERSION); \
140 Assert((pFileOps)->fReserved == 0); \
141 AssertPtr((pFileOps)->pfnSeek); \
142 AssertPtrNull((pFileOps)->pfnQuerySize); \
143 AssertPtrNull((pFileOps)->pfnSetSize); \
144 AssertPtrNull((pFileOps)->pfnQueryMaxSize); \
145 Assert((pFileOps)->uEndMarker == RTVFSFILEOPS_VERSION); \
146 } while (0)
147
148/** Asserts that the VFS symlink vtable is valid. */
149#define RTVFSSYMLINK_ASSERT_OPS(pSymlinkOps, a_enmType) \
150 do { \
151 RTVFSOBJ_ASSERT_OPS(&(pSymlinkOps)->Obj, a_enmType); \
152 RTVFSOBJSET_ASSERT_OPS(&(pSymlinkOps)->ObjSet, RT_UOFFSETOF(RTVFSSYMLINKOPS, ObjSet) - RT_UOFFSETOF(RTVFSSYMLINKOPS, Obj)); \
153 Assert((pSymlinkOps)->uVersion == RTVFSSYMLINKOPS_VERSION); \
154 Assert(!(pSymlinkOps)->fReserved); \
155 AssertPtr((pSymlinkOps)->pfnRead); \
156 Assert((pSymlinkOps)->uEndMarker == RTVFSSYMLINKOPS_VERSION); \
157 } while (0)
158
159
160/** Validates a VFS handle and returns @a rcRet if it's invalid. */
161#define RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, rcRet) \
162 do { \
163 if ((hVfs) != NIL_RTVFS) \
164 { \
165 AssertPtrReturn((hVfs), (rcRet)); \
166 AssertReturn((hVfs)->uMagic == RTVFS_MAGIC, (rcRet)); \
167 } \
168 } while (0)
169
170
171/*********************************************************************************************************************************
172* Structures and Typedefs *
173*********************************************************************************************************************************/
174/** @todo Move all this stuff to internal/vfs.h */
175
176
177/**
178 * The VFS internal lock data.
179 */
180typedef struct RTVFSLOCKINTERNAL
181{
182 /** The number of references to the this lock. */
183 uint32_t volatile cRefs;
184 /** The lock type. */
185 RTVFSLOCKTYPE enmType;
186 /** Type specific data. */
187 union
188 {
189 /** Read/Write semaphore handle. */
190 RTSEMRW hSemRW;
191 /** Fast mutex semaphore handle. */
192 RTSEMFASTMUTEX hFastMtx;
193 /** Regular mutex semaphore handle. */
194 RTSEMMUTEX hMtx;
195 } u;
196} RTVFSLOCKINTERNAL;
197
198
199/**
200 * The VFS base object handle data.
201 *
202 * All other VFS handles are derived from this one. The final handle type is
203 * indicated by RTVFSOBJOPS::enmType via the RTVFSOBJINTERNAL::pOps member.
204 */
205typedef struct RTVFSOBJINTERNAL
206{
207 /** The VFS magic (RTVFSOBJ_MAGIC). */
208 uint32_t uMagic : 31;
209 /** Set if we've got no VFS reference but still got a valid hVfs.
210 * This is hack for permanent root directory objects. */
211 uint32_t fNoVfsRef : 1;
212 /** The number of references to this VFS object. */
213 uint32_t volatile cRefs;
214 /** Pointer to the instance data. */
215 void *pvThis;
216 /** The vtable. */
217 PCRTVFSOBJOPS pOps;
218 /** The lock protecting all access to the VFS.
219 * Only valid if RTVFS_C_THREAD_SAFE is set, otherwise it is NIL_RTVFSLOCK. */
220 RTVFSLOCK hLock;
221 /** Reference back to the VFS containing this object. */
222 RTVFS hVfs;
223} RTVFSOBJINTERNAL;
224
225
226/**
227 * The VFS filesystem stream handle data.
228 *
229 * @extends RTVFSOBJINTERNAL
230 */
231typedef struct RTVFSFSSTREAMINTERNAL
232{
233 /** The VFS magic (RTVFSFSTREAM_MAGIC). */
234 uint32_t uMagic;
235 /** File open flags, at a minimum the access mask. */
236 uint32_t fFlags;
237 /** The vtable. */
238 PCRTVFSFSSTREAMOPS pOps;
239 /** The base object handle data. */
240 RTVFSOBJINTERNAL Base;
241} RTVFSFSSTREAMINTERNAL;
242
243
244/**
245 * The VFS handle data.
246 *
247 * @extends RTVFSOBJINTERNAL
248 */
249typedef struct RTVFSINTERNAL
250{
251 /** The VFS magic (RTVFS_MAGIC). */
252 uint32_t uMagic;
253 /** Creation flags (RTVFS_C_XXX). */
254 uint32_t fFlags;
255 /** The vtable. */
256 PCRTVFSOPS pOps;
257 /** The base object handle data. */
258 RTVFSOBJINTERNAL Base;
259} RTVFSINTERNAL;
260
261
262/**
263 * The VFS directory handle data.
264 *
265 * @extends RTVFSOBJINTERNAL
266 */
267typedef struct RTVFSDIRINTERNAL
268{
269 /** The VFS magic (RTVFSDIR_MAGIC). */
270 uint32_t uMagic;
271 /** Reserved for flags or something. */
272 uint32_t fReserved;
273 /** The vtable. */
274 PCRTVFSDIROPS pOps;
275 /** The base object handle data. */
276 RTVFSOBJINTERNAL Base;
277} RTVFSDIRINTERNAL;
278
279
280/**
281 * The VFS symbolic link handle data.
282 *
283 * @extends RTVFSOBJINTERNAL
284 */
285typedef struct RTVFSSYMLINKINTERNAL
286{
287 /** The VFS magic (RTVFSSYMLINK_MAGIC). */
288 uint32_t uMagic;
289 /** Reserved for flags or something. */
290 uint32_t fReserved;
291 /** The vtable. */
292 PCRTVFSSYMLINKOPS pOps;
293 /** The base object handle data. */
294 RTVFSOBJINTERNAL Base;
295} RTVFSSYMLINKINTERNAL;
296
297
298/**
299 * The VFS I/O stream handle data.
300 *
301 * This is often part of a type specific handle, like a file or pipe.
302 *
303 * @extends RTVFSOBJINTERNAL
304 */
305typedef struct RTVFSIOSTREAMINTERNAL
306{
307 /** The VFS magic (RTVFSIOSTREAM_MAGIC). */
308 uint32_t uMagic;
309 /** File open flags, at a minimum the access mask. */
310 uint32_t fFlags;
311 /** The vtable. */
312 PCRTVFSIOSTREAMOPS pOps;
313 /** The base object handle data. */
314 RTVFSOBJINTERNAL Base;
315} RTVFSIOSTREAMINTERNAL;
316
317
318/**
319 * The VFS file handle data.
320 *
321 * @extends RTVFSIOSTREAMINTERNAL
322 */
323typedef struct RTVFSFILEINTERNAL
324{
325 /** The VFS magic (RTVFSFILE_MAGIC). */
326 uint32_t uMagic;
327 /** Reserved for flags or something. */
328 uint32_t fReserved;
329 /** The vtable. */
330 PCRTVFSFILEOPS pOps;
331 /** The stream handle data. */
332 RTVFSIOSTREAMINTERNAL Stream;
333} RTVFSFILEINTERNAL;
334
335#if 0 /* later */
336
337/**
338 * The VFS pipe handle data.
339 *
340 * @extends RTVFSIOSTREAMINTERNAL
341 */
342typedef struct RTVFSPIPEINTERNAL
343{
344 /** The VFS magic (RTVFSPIPE_MAGIC). */
345 uint32_t uMagic;
346 /** Reserved for flags or something. */
347 uint32_t fReserved;
348 /** The vtable. */
349 PCRTVFSPIPEOPS pOps;
350 /** The stream handle data. */
351 RTVFSIOSTREAMINTERNAL Stream;
352} RTVFSPIPEINTERNAL;
353
354
355/**
356 * The VFS socket handle data.
357 *
358 * @extends RTVFSIOSTREAMINTERNAL
359 */
360typedef struct RTVFSSOCKETINTERNAL
361{
362 /** The VFS magic (RTVFSSOCKET_MAGIC). */
363 uint32_t uMagic;
364 /** Reserved for flags or something. */
365 uint32_t fReserved;
366 /** The vtable. */
367 PCRTVFSSOCKETOPS pOps;
368 /** The stream handle data. */
369 RTVFSIOSTREAMINTERNAL Stream;
370} RTVFSSOCKETINTERNAL;
371
372#endif /* later */
373
374
375/*********************************************************************************************************************************
376* Internal Functions *
377*********************************************************************************************************************************/
378DECLINLINE(uint32_t) rtVfsObjRelease(RTVFSOBJINTERNAL *pThis);
379static int rtVfsTraverseToParent(RTVFSINTERNAL *pThis, PRTVFSPARSEDPATH pPath, uint32_t fFlags, RTVFSDIRINTERNAL **ppVfsParentDir);
380static int rtVfsDirFollowSymlinkObjToParent(RTVFSDIRINTERNAL **ppVfsParentDir, RTVFSOBJ hVfsObj,
381 PRTVFSPARSEDPATH pPath, uint32_t fFlags);
382
383
384
385/**
386 * Translates a RTVFSOBJTYPE value into a string.
387 *
388 * @returns Pointer to readonly name.
389 * @param enmType The object type to name.
390 */
391RTDECL(const char *) RTVfsTypeName(RTVFSOBJTYPE enmType)
392{
393 switch (enmType)
394 {
395 case RTVFSOBJTYPE_INVALID: return "invalid";
396 case RTVFSOBJTYPE_BASE: return "base";
397 case RTVFSOBJTYPE_VFS: return "VFS";
398 case RTVFSOBJTYPE_FS_STREAM: return "FS stream";
399 case RTVFSOBJTYPE_IO_STREAM: return "I/O stream";
400 case RTVFSOBJTYPE_DIR: return "directory";
401 case RTVFSOBJTYPE_FILE: return "file";
402 case RTVFSOBJTYPE_SYMLINK: return "symlink";
403 case RTVFSOBJTYPE_END: return "end";
404 case RTVFSOBJTYPE_32BIT_HACK:
405 break;
406 }
407 return "unknown";
408}
409
410
411/*
412 *
413 * V F S L o c k A b s t r a c t i o n
414 * V F S L o c k A b s t r a c t i o n
415 * V F S L o c k A b s t r a c t i o n
416 *
417 *
418 */
419
420
421RTDECL(uint32_t) RTVfsLockRetain(RTVFSLOCK hLock)
422{
423 RTVFSLOCKINTERNAL *pThis = hLock;
424 AssertPtrReturn(pThis, UINT32_MAX);
425 AssertReturn(pThis->enmType > RTVFSLOCKTYPE_INVALID && pThis->enmType < RTVFSLOCKTYPE_END, UINT32_MAX);
426
427 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
428 AssertMsg(cRefs > 1 && cRefs < _1M, ("%#x %p %d\n", cRefs, pThis, pThis->enmType));
429 return cRefs;
430}
431
432
433RTDECL(uint32_t) RTVfsLockRetainDebug(RTVFSLOCK hLock, RT_SRC_POS_DECL)
434{
435 RTVFSLOCKINTERNAL *pThis = hLock;
436 AssertPtrReturn(pThis, UINT32_MAX);
437 AssertReturn(pThis->enmType > RTVFSLOCKTYPE_INVALID && pThis->enmType < RTVFSLOCKTYPE_END, UINT32_MAX);
438
439 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
440 AssertMsg(cRefs > 1 && cRefs < _1M, ("%#x %p %d\n", cRefs, pThis, pThis->enmType));
441 LogFlow(("RTVfsLockRetainDebug(%p) -> %d; caller: %s %s(%u)\n", hLock, cRefs, pszFunction, pszFile, iLine));
442 RT_SRC_POS_NOREF();
443 return cRefs;
444}
445
446
447/**
448 * Destroys a VFS lock handle.
449 *
450 * @param pThis The lock to destroy.
451 */
452static void rtVfsLockDestroy(RTVFSLOCKINTERNAL *pThis)
453{
454 switch (pThis->enmType)
455 {
456 case RTVFSLOCKTYPE_RW:
457 RTSemRWDestroy(pThis->u.hSemRW);
458 pThis->u.hSemRW = NIL_RTSEMRW;
459 break;
460
461 case RTVFSLOCKTYPE_FASTMUTEX:
462 RTSemFastMutexDestroy(pThis->u.hFastMtx);
463 pThis->u.hFastMtx = NIL_RTSEMFASTMUTEX;
464 break;
465
466 case RTVFSLOCKTYPE_MUTEX:
467 RTSemMutexDestroy(pThis->u.hMtx);
468 pThis->u.hFastMtx = NIL_RTSEMMUTEX;
469 break;
470
471 default:
472 AssertMsgFailedReturnVoid(("%p %d\n", pThis, pThis->enmType));
473 }
474
475 pThis->enmType = RTVFSLOCKTYPE_INVALID;
476 RTMemFree(pThis);
477}
478
479
480RTDECL(uint32_t) RTVfsLockRelease(RTVFSLOCK hLock)
481{
482 RTVFSLOCKINTERNAL *pThis = hLock;
483 if (pThis == NIL_RTVFSLOCK)
484 return 0;
485 AssertPtrReturn(pThis, UINT32_MAX);
486 AssertReturn(pThis->enmType > RTVFSLOCKTYPE_INVALID && pThis->enmType < RTVFSLOCKTYPE_END, UINT32_MAX);
487
488 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
489 AssertMsg(cRefs < _1M, ("%#x %p %d\n", cRefs, pThis, pThis->enmType));
490 if (cRefs == 0)
491 rtVfsLockDestroy(pThis);
492 return cRefs;
493}
494
495
496/**
497 * Creates a read/write lock.
498 *
499 * @returns IPRT status code
500 * @param phLock Where to return the lock handle.
501 */
502static int rtVfsLockCreateRW(PRTVFSLOCK phLock)
503{
504 RTVFSLOCKINTERNAL *pThis = (RTVFSLOCKINTERNAL *)RTMemAlloc(sizeof(*pThis));
505 if (!pThis)
506 return VERR_NO_MEMORY;
507
508 pThis->cRefs = 1;
509 pThis->enmType = RTVFSLOCKTYPE_RW;
510
511 int rc = RTSemRWCreate(&pThis->u.hSemRW);
512 if (RT_FAILURE(rc))
513 {
514 RTMemFree(pThis);
515 return rc;
516 }
517
518 *phLock = pThis;
519 return VINF_SUCCESS;
520}
521
522
523/**
524 * Creates a fast mutex lock.
525 *
526 * @returns IPRT status code
527 * @param phLock Where to return the lock handle.
528 */
529static int rtVfsLockCreateFastMutex(PRTVFSLOCK phLock)
530{
531 RTVFSLOCKINTERNAL *pThis = (RTVFSLOCKINTERNAL *)RTMemAlloc(sizeof(*pThis));
532 if (!pThis)
533 return VERR_NO_MEMORY;
534
535 pThis->cRefs = 1;
536 pThis->enmType = RTVFSLOCKTYPE_FASTMUTEX;
537
538 int rc = RTSemFastMutexCreate(&pThis->u.hFastMtx);
539 if (RT_FAILURE(rc))
540 {
541 RTMemFree(pThis);
542 return rc;
543 }
544
545 *phLock = pThis;
546 return VINF_SUCCESS;
547
548}
549
550
551/**
552 * Creates a mutex lock.
553 *
554 * @returns IPRT status code
555 * @param phLock Where to return the lock handle.
556 */
557static int rtVfsLockCreateMutex(PRTVFSLOCK phLock)
558{
559 RTVFSLOCKINTERNAL *pThis = (RTVFSLOCKINTERNAL *)RTMemAlloc(sizeof(*pThis));
560 if (!pThis)
561 return VERR_NO_MEMORY;
562
563 pThis->cRefs = 1;
564 pThis->enmType = RTVFSLOCKTYPE_MUTEX;
565
566 int rc = RTSemMutexCreate(&pThis->u.hMtx);
567 if (RT_FAILURE(rc))
568 {
569 RTMemFree(pThis);
570 return rc;
571 }
572
573 *phLock = pThis;
574 return VINF_SUCCESS;
575}
576
577
578/**
579 * Acquires the lock for reading.
580 *
581 * @param hLock Non-nil lock handle.
582 * @internal
583 */
584RTDECL(void) RTVfsLockAcquireReadSlow(RTVFSLOCK hLock)
585{
586 RTVFSLOCKINTERNAL *pThis = hLock;
587 int rc;
588
589 AssertPtr(pThis);
590 switch (pThis->enmType)
591 {
592 case RTVFSLOCKTYPE_RW:
593 rc = RTSemRWRequestRead(pThis->u.hSemRW, RT_INDEFINITE_WAIT);
594 AssertRC(rc);
595 break;
596
597 case RTVFSLOCKTYPE_FASTMUTEX:
598 rc = RTSemFastMutexRequest(pThis->u.hFastMtx);
599 AssertRC(rc);
600 break;
601
602 case RTVFSLOCKTYPE_MUTEX:
603 rc = RTSemMutexRequest(pThis->u.hMtx, RT_INDEFINITE_WAIT);
604 AssertRC(rc);
605 break;
606 default:
607 AssertFailed();
608 }
609}
610
611
612/**
613 * Release a lock held for reading.
614 *
615 * @param hLock Non-nil lock handle.
616 * @internal
617 */
618RTDECL(void) RTVfsLockReleaseReadSlow(RTVFSLOCK hLock)
619{
620 RTVFSLOCKINTERNAL *pThis = hLock;
621 int rc;
622
623 AssertPtr(pThis);
624 switch (pThis->enmType)
625 {
626 case RTVFSLOCKTYPE_RW:
627 rc = RTSemRWReleaseRead(pThis->u.hSemRW);
628 AssertRC(rc);
629 break;
630
631 case RTVFSLOCKTYPE_FASTMUTEX:
632 rc = RTSemFastMutexRelease(pThis->u.hFastMtx);
633 AssertRC(rc);
634 break;
635
636 case RTVFSLOCKTYPE_MUTEX:
637 rc = RTSemMutexRelease(pThis->u.hMtx);
638 AssertRC(rc);
639 break;
640 default:
641 AssertFailed();
642 }
643}
644
645
646/**
647 * Acquires the lock for writing.
648 *
649 * @param hLock Non-nil lock handle.
650 * @internal
651 */
652RTDECL(void) RTVfsLockAcquireWriteSlow(RTVFSLOCK hLock)
653{
654 RTVFSLOCKINTERNAL *pThis = hLock;
655 int rc;
656
657 AssertPtr(pThis);
658 switch (pThis->enmType)
659 {
660 case RTVFSLOCKTYPE_RW:
661 rc = RTSemRWRequestWrite(pThis->u.hSemRW, RT_INDEFINITE_WAIT);
662 AssertRC(rc);
663 break;
664
665 case RTVFSLOCKTYPE_FASTMUTEX:
666 rc = RTSemFastMutexRequest(pThis->u.hFastMtx);
667 AssertRC(rc);
668 break;
669
670 case RTVFSLOCKTYPE_MUTEX:
671 rc = RTSemMutexRequest(pThis->u.hMtx, RT_INDEFINITE_WAIT);
672 AssertRC(rc);
673 break;
674 default:
675 AssertFailed();
676 }
677}
678
679
680/**
681 * Release a lock held for writing.
682 *
683 * @param hLock Non-nil lock handle.
684 * @internal
685 */
686RTDECL(void) RTVfsLockReleaseWriteSlow(RTVFSLOCK hLock)
687{
688 RTVFSLOCKINTERNAL *pThis = hLock;
689 int rc;
690
691 AssertPtr(pThis);
692 switch (pThis->enmType)
693 {
694 case RTVFSLOCKTYPE_RW:
695 rc = RTSemRWReleaseWrite(pThis->u.hSemRW);
696 AssertRC(rc);
697 break;
698
699 case RTVFSLOCKTYPE_FASTMUTEX:
700 rc = RTSemFastMutexRelease(pThis->u.hFastMtx);
701 AssertRC(rc);
702 break;
703
704 case RTVFSLOCKTYPE_MUTEX:
705 rc = RTSemMutexRelease(pThis->u.hMtx);
706 AssertRC(rc);
707 break;
708 default:
709 AssertFailed();
710 }
711}
712
713
714
715/*
716 *
717 * B A S E O B J E C T
718 * B A S E O B J E C T
719 * B A S E O B J E C T
720 *
721 */
722
723/**
724 * Internal object retainer that asserts sanity in strict builds.
725 *
726 * @param pThis The base object handle data.
727 * @param pszCaller Where we're called from.
728 */
729DECLINLINE(void) rtVfsObjRetainVoid(RTVFSOBJINTERNAL *pThis, const char *pszCaller)
730{
731 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
732LogFlow(("rtVfsObjRetainVoid(%p/%p) -> %d; caller=%s\n", pThis, pThis->pvThis, cRefs, pszCaller)); RT_NOREF(pszCaller);
733 AssertMsg(cRefs > 1 && cRefs < _1M,
734 ("%#x %p ops=%p %s (%d); caller=%s\n", cRefs, pThis, pThis->pOps, pThis->pOps->pszName, pThis->pOps->enmType, pszCaller));
735 NOREF(cRefs);
736}
737
738
739/**
740 * Initializes the base object part of a new object.
741 *
742 * @returns IPRT status code.
743 * @param pThis Pointer to the base object part.
744 * @param pObjOps The base object vtable.
745 * @param hVfs The VFS handle to associate with.
746 * @param fNoVfsRef If set, do not retain an additional reference to
747 * @a hVfs. Permanent root dir hack.
748 * @param hLock The lock handle, pseudo handle or nil.
749 * @param pvThis Pointer to the private data.
750 */
751static int rtVfsObjInitNewObject(RTVFSOBJINTERNAL *pThis, PCRTVFSOBJOPS pObjOps, RTVFS hVfs, bool fNoVfsRef,
752 RTVFSLOCK hLock, void *pvThis)
753{
754 /*
755 * Deal with the lock first as that's the most complicated matter.
756 */
757 if (hLock != NIL_RTVFSLOCK)
758 {
759 int rc;
760 if (hLock == RTVFSLOCK_CREATE_RW)
761 {
762 rc = rtVfsLockCreateRW(&hLock);
763 AssertRCReturn(rc, rc);
764 }
765 else if (hLock == RTVFSLOCK_CREATE_FASTMUTEX)
766 {
767 rc = rtVfsLockCreateFastMutex(&hLock);
768 AssertRCReturn(rc, rc);
769 }
770 else if (hLock == RTVFSLOCK_CREATE_MUTEX)
771 {
772 rc = rtVfsLockCreateMutex(&hLock);
773 AssertRCReturn(rc, rc);
774 }
775 else
776 {
777 /*
778 * The caller specified a lock, we consume the this reference.
779 */
780 AssertPtrReturn(hLock, VERR_INVALID_HANDLE);
781 AssertReturn(hLock->enmType > RTVFSLOCKTYPE_INVALID && hLock->enmType < RTVFSLOCKTYPE_END, VERR_INVALID_HANDLE);
782 AssertReturn(hLock->cRefs > 0, VERR_INVALID_HANDLE);
783 }
784 }
785 else if (hVfs != NIL_RTVFS)
786 {
787 /*
788 * Retain a reference to the VFS lock, if there is one.
789 */
790 hLock = hVfs->Base.hLock;
791 if (hLock != NIL_RTVFSLOCK)
792 {
793 uint32_t cRefs = RTVfsLockRetain(hLock);
794 if (RT_UNLIKELY(cRefs == UINT32_MAX))
795 return VERR_INVALID_HANDLE;
796 }
797 }
798
799
800 /*
801 * Do the actual initializing.
802 */
803 pThis->uMagic = RTVFSOBJ_MAGIC;
804 pThis->fNoVfsRef = fNoVfsRef;
805 pThis->pvThis = pvThis;
806 pThis->pOps = pObjOps;
807 pThis->cRefs = 1;
808 pThis->hVfs = hVfs;
809 pThis->hLock = hLock;
810 if (hVfs != NIL_RTVFS && !fNoVfsRef)
811 rtVfsObjRetainVoid(&hVfs->Base, "rtVfsObjInitNewObject");
812
813 return VINF_SUCCESS;
814}
815
816
817RTDECL(int) RTVfsNewBaseObj(PCRTVFSOBJOPS pObjOps, size_t cbInstance, RTVFS hVfs, RTVFSLOCK hLock,
818 PRTVFSOBJ phVfsObj, void **ppvInstance)
819{
820 /*
821 * Validate the input, be extra strict in strict builds.
822 */
823 AssertPtr(pObjOps);
824 AssertReturn(pObjOps->uVersion == RTVFSOBJOPS_VERSION, VERR_VERSION_MISMATCH);
825 AssertReturn(pObjOps->uEndMarker == RTVFSOBJOPS_VERSION, VERR_VERSION_MISMATCH);
826 RTVFSOBJ_ASSERT_OPS(pObjOps, RTVFSOBJTYPE_BASE);
827 Assert(cbInstance > 0);
828 AssertPtr(ppvInstance);
829 AssertPtr(phVfsObj);
830 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
831
832 /*
833 * Allocate the handle + instance data.
834 */
835 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSOBJINTERNAL), RTVFS_INST_ALIGNMENT)
836 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
837 RTVFSOBJINTERNAL *pThis = (RTVFSOBJINTERNAL *)RTMemAllocZ(cbThis);
838 if (!pThis)
839 return VERR_NO_MEMORY;
840
841 int rc = rtVfsObjInitNewObject(pThis, pObjOps, hVfs, false /*fNoVfsRef*/, hLock,
842 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
843 if (RT_FAILURE(rc))
844 {
845 RTMemFree(pThis);
846 return rc;
847 }
848
849 *phVfsObj = pThis;
850 *ppvInstance = pThis->pvThis;
851 return VINF_SUCCESS;
852}
853
854
855RTDECL(void *) RTVfsObjToPrivate(RTVFSOBJ hVfsObj, PCRTVFSOBJOPS pObjOps)
856{
857 RTVFSOBJINTERNAL *pThis = hVfsObj;
858 AssertPtrReturn(pThis, NULL);
859 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NULL);
860 if (pThis->pOps != pObjOps)
861 return NULL;
862 return pThis->pvThis;
863}
864
865
866/**
867 * Internal object retainer that asserts sanity in strict builds.
868 *
869 * @returns The new reference count.
870 * @param pThis The base object handle data.
871 */
872DECLINLINE(uint32_t) rtVfsObjRetain(RTVFSOBJINTERNAL *pThis)
873{
874 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
875LogFlow(("rtVfsObjRetain(%p/%p) -> %d\n", pThis, pThis->pvThis, cRefs));
876 AssertMsg(cRefs > 1 && cRefs < _1M,
877 ("%#x %p ops=%p %s (%d)\n", cRefs, pThis, pThis->pOps, pThis->pOps->pszName, pThis->pOps->enmType));
878 return cRefs;
879}
880
881/**
882 * Internal object retainer that asserts sanity in strict builds.
883 *
884 * @returns The new reference count.
885 * @param pThis The base object handle data.
886 */
887DECLINLINE(uint32_t) rtVfsObjRetainDebug(RTVFSOBJINTERNAL *pThis, const char *pszApi, RT_SRC_POS_DECL)
888{
889 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
890 AssertMsg(cRefs > 1 && cRefs < _1M,
891 ("%#x %p ops=%p %s (%d)\n", cRefs, pThis, pThis->pOps, pThis->pOps->pszName, pThis->pOps->enmType));
892 LogFlow(("%s(%p/%p) -> %2d; caller: %s %s(%d) \n", pszApi, pThis, pThis->pvThis, cRefs, pszFunction, pszFile, iLine));
893 RT_SRC_POS_NOREF(); RT_NOREF(pszApi);
894 return cRefs;
895}
896
897
898#ifdef DEBUG
899# undef RTVfsObjRetain
900#endif
901RTDECL(uint32_t) RTVfsObjRetain(RTVFSOBJ hVfsObj)
902{
903 RTVFSOBJINTERNAL *pThis = hVfsObj;
904 AssertPtrReturn(pThis, UINT32_MAX);
905 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, UINT32_MAX);
906
907 return rtVfsObjRetain(pThis);
908}
909#ifdef DEBUG
910# define RTVfsObjRetain(hVfsObj) RTVfsObjRetainDebug(hVfsObj, RT_SRC_POS)
911#endif
912
913
914RTDECL(uint32_t) RTVfsObjRetainDebug(RTVFSOBJ hVfsObj, RT_SRC_POS_DECL)
915{
916 RTVFSOBJINTERNAL *pThis = hVfsObj;
917 AssertPtrReturn(pThis, UINT32_MAX);
918 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, UINT32_MAX);
919
920 return rtVfsObjRetainDebug(pThis, "RTVfsObjRetainDebug", RT_SRC_POS_ARGS);
921}
922
923
924/**
925 * Does the actual object destruction for rtVfsObjRelease().
926 *
927 * @param pThis The object to destroy.
928 */
929static void rtVfsObjDestroy(RTVFSOBJINTERNAL *pThis)
930{
931 RTVFSOBJTYPE const enmType = pThis->pOps->enmType;
932
933 /*
934 * Invalidate the object.
935 */
936 RTVfsLockAcquireWrite(pThis->hLock); /* paranoia */
937 void *pvToFree = NULL;
938 switch (enmType)
939 {
940 case RTVFSOBJTYPE_BASE:
941 pvToFree = pThis;
942 break;
943
944 case RTVFSOBJTYPE_VFS:
945 pvToFree = RT_FROM_MEMBER(pThis, RTVFSINTERNAL, Base);
946 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSINTERNAL, Base)->uMagic, RTVFS_MAGIC_DEAD);
947 break;
948
949 case RTVFSOBJTYPE_FS_STREAM:
950 pvToFree = RT_FROM_MEMBER(pThis, RTVFSFSSTREAMINTERNAL, Base);
951 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSFSSTREAMINTERNAL, Base)->uMagic, RTVFSFSSTREAM_MAGIC_DEAD);
952 break;
953
954 case RTVFSOBJTYPE_IO_STREAM:
955 pvToFree = RT_FROM_MEMBER(pThis, RTVFSIOSTREAMINTERNAL, Base);
956 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSIOSTREAMINTERNAL, Base)->uMagic, RTVFSIOSTREAM_MAGIC_DEAD);
957 break;
958
959 case RTVFSOBJTYPE_DIR:
960 pvToFree = RT_FROM_MEMBER(pThis, RTVFSDIRINTERNAL, Base);
961 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSDIRINTERNAL, Base)->uMagic, RTVFSDIR_MAGIC_DEAD);
962 break;
963
964 case RTVFSOBJTYPE_FILE:
965 pvToFree = RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream.Base);
966 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream.Base)->uMagic, RTVFSFILE_MAGIC_DEAD);
967 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSIOSTREAMINTERNAL, Base)->uMagic, RTVFSIOSTREAM_MAGIC_DEAD);
968 break;
969
970 case RTVFSOBJTYPE_SYMLINK:
971 pvToFree = RT_FROM_MEMBER(pThis, RTVFSSYMLINKINTERNAL, Base);
972 ASMAtomicWriteU32(&RT_FROM_MEMBER(pThis, RTVFSSYMLINKINTERNAL, Base)->uMagic, RTVFSSYMLINK_MAGIC_DEAD);
973 break;
974
975 case RTVFSOBJTYPE_INVALID:
976 case RTVFSOBJTYPE_END:
977 case RTVFSOBJTYPE_32BIT_HACK:
978 AssertMsgFailed(("enmType=%d ops=%p %s\n", enmType, pThis->pOps, pThis->pOps->pszName));
979 break;
980 /* no default as we want gcc warnings. */
981 }
982 pThis->uMagic = RTVFSOBJ_MAGIC_DEAD;
983 RTVfsLockReleaseWrite(pThis->hLock);
984
985 /*
986 * Close the object and free the handle.
987 */
988 int rc = pThis->pOps->pfnClose(pThis->pvThis);
989 AssertRC(rc);
990 if (pThis->hVfs != NIL_RTVFS)
991 {
992 if (!pThis->fNoVfsRef)
993 rtVfsObjRelease(&pThis->hVfs->Base);
994 pThis->hVfs = NIL_RTVFS;
995 }
996 if (pThis->hLock != NIL_RTVFSLOCK)
997 {
998 RTVfsLockRelease(pThis->hLock);
999 pThis->hLock = NIL_RTVFSLOCK;
1000 }
1001 RTMemFree(pvToFree);
1002}
1003
1004
1005/**
1006 * Internal object releaser that asserts sanity in strict builds.
1007 *
1008 * @returns The new reference count.
1009 * @param pcRefs The reference counter.
1010 */
1011DECLINLINE(uint32_t) rtVfsObjRelease(RTVFSOBJINTERNAL *pThis)
1012{
1013 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
1014 AssertMsg(cRefs < _1M, ("%#x %p ops=%p %s (%d)\n", cRefs, pThis, pThis->pOps, pThis->pOps->pszName, pThis->pOps->enmType));
1015 LogFlow(("rtVfsObjRelease(%p/%p) -> %d\n", pThis, pThis->pvThis, cRefs));
1016 if (cRefs == 0)
1017 rtVfsObjDestroy(pThis);
1018 return cRefs;
1019}
1020
1021
1022RTDECL(uint32_t) RTVfsObjRelease(RTVFSOBJ hVfsObj)
1023{
1024 RTVFSOBJINTERNAL *pThis = hVfsObj;
1025 if (pThis == NIL_RTVFSOBJ)
1026 return 0;
1027 AssertPtrReturn(pThis, UINT32_MAX);
1028 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, UINT32_MAX);
1029 return rtVfsObjRelease(pThis);
1030}
1031
1032
1033RTDECL(RTVFSOBJTYPE) RTVfsObjGetType(RTVFSOBJ hVfsObj)
1034{
1035 RTVFSOBJINTERNAL *pThis = hVfsObj;
1036 if (pThis != NIL_RTVFSOBJ)
1037 {
1038 AssertPtrReturn(pThis, RTVFSOBJTYPE_INVALID);
1039 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, RTVFSOBJTYPE_INVALID);
1040 return pThis->pOps->enmType;
1041 }
1042 return RTVFSOBJTYPE_INVALID;
1043}
1044
1045
1046RTDECL(RTVFS) RTVfsObjToVfs(RTVFSOBJ hVfsObj)
1047{
1048 RTVFSOBJINTERNAL *pThis = hVfsObj;
1049 if (pThis != NIL_RTVFSOBJ)
1050 {
1051 AssertPtrReturn(pThis, NIL_RTVFS);
1052 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFS);
1053
1054 if (pThis->pOps->enmType == RTVFSOBJTYPE_VFS)
1055 {
1056 rtVfsObjRetainVoid(pThis, "RTVfsObjToVfs");
1057 LogFlow(("RTVfsObjToVfs(%p) -> %p\n", pThis, RT_FROM_MEMBER(pThis, RTVFSINTERNAL, Base)));
1058 return RT_FROM_MEMBER(pThis, RTVFSINTERNAL, Base);
1059 }
1060 }
1061 return NIL_RTVFS;
1062}
1063
1064
1065RTDECL(RTVFSFSSTREAM) RTVfsObjToFsStream(RTVFSOBJ hVfsObj)
1066{
1067 RTVFSOBJINTERNAL *pThis = hVfsObj;
1068 if (pThis != NIL_RTVFSOBJ)
1069 {
1070 AssertPtrReturn(pThis, NIL_RTVFSFSSTREAM);
1071 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSFSSTREAM);
1072
1073 if (pThis->pOps->enmType == RTVFSOBJTYPE_FS_STREAM)
1074 {
1075 rtVfsObjRetainVoid(pThis, "RTVfsObjToFsStream");
1076 return RT_FROM_MEMBER(pThis, RTVFSFSSTREAMINTERNAL, Base);
1077 }
1078 }
1079 return NIL_RTVFSFSSTREAM;
1080}
1081
1082
1083RTDECL(RTVFSDIR) RTVfsObjToDir(RTVFSOBJ hVfsObj)
1084{
1085 RTVFSOBJINTERNAL *pThis = hVfsObj;
1086 if (pThis != NIL_RTVFSOBJ)
1087 {
1088 AssertPtrReturn(pThis, NIL_RTVFSDIR);
1089 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSDIR);
1090
1091 if (pThis->pOps->enmType == RTVFSOBJTYPE_DIR)
1092 {
1093 rtVfsObjRetainVoid(pThis, "RTVfsObjToDir");
1094 return RT_FROM_MEMBER(pThis, RTVFSDIRINTERNAL, Base);
1095 }
1096 }
1097 return NIL_RTVFSDIR;
1098}
1099
1100
1101RTDECL(RTVFSIOSTREAM) RTVfsObjToIoStream(RTVFSOBJ hVfsObj)
1102{
1103 RTVFSOBJINTERNAL *pThis = hVfsObj;
1104 if (pThis != NIL_RTVFSOBJ)
1105 {
1106 AssertPtrReturn(pThis, NIL_RTVFSIOSTREAM);
1107 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSIOSTREAM);
1108
1109 if ( pThis->pOps->enmType == RTVFSOBJTYPE_IO_STREAM
1110 || pThis->pOps->enmType == RTVFSOBJTYPE_FILE)
1111 {
1112 rtVfsObjRetainVoid(pThis, "RTVfsObjToIoStream");
1113 return RT_FROM_MEMBER(pThis, RTVFSIOSTREAMINTERNAL, Base);
1114 }
1115 }
1116 return NIL_RTVFSIOSTREAM;
1117}
1118
1119
1120RTDECL(RTVFSFILE) RTVfsObjToFile(RTVFSOBJ hVfsObj)
1121{
1122 RTVFSOBJINTERNAL *pThis = hVfsObj;
1123 if (pThis != NIL_RTVFSOBJ)
1124 {
1125 AssertPtrReturn(pThis, NIL_RTVFSFILE);
1126 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSFILE);
1127
1128 if (pThis->pOps->enmType == RTVFSOBJTYPE_FILE)
1129 {
1130 rtVfsObjRetainVoid(pThis, "RTVfsObjToFile");
1131 return RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream.Base);
1132 }
1133 }
1134 return NIL_RTVFSFILE;
1135}
1136
1137
1138RTDECL(RTVFSSYMLINK) RTVfsObjToSymlink(RTVFSOBJ hVfsObj)
1139{
1140 RTVFSOBJINTERNAL *pThis = hVfsObj;
1141 if (pThis != NIL_RTVFSOBJ)
1142 {
1143 AssertPtrReturn(pThis, NIL_RTVFSSYMLINK);
1144 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSSYMLINK);
1145
1146 if (pThis->pOps->enmType == RTVFSOBJTYPE_SYMLINK)
1147 {
1148 rtVfsObjRetainVoid(pThis, "RTVfsObjToSymlink");
1149 return RT_FROM_MEMBER(pThis, RTVFSSYMLINKINTERNAL, Base);
1150 }
1151 }
1152 return NIL_RTVFSSYMLINK;
1153}
1154
1155
1156RTDECL(RTVFSOBJ) RTVfsObjFromVfs(RTVFS hVfs)
1157{
1158 if (hVfs != NIL_RTVFS)
1159 {
1160 RTVFSOBJINTERNAL *pThis = &hVfs->Base;
1161 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1162 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1163
1164 rtVfsObjRetainVoid(pThis, "RTVfsObjFromVfs");
1165 LogFlow(("RTVfsObjFromVfs(%p) -> %p\n", hVfs, pThis));
1166 return pThis;
1167 }
1168 return NIL_RTVFSOBJ;
1169}
1170
1171
1172RTDECL(RTVFSOBJ) RTVfsObjFromFsStream(RTVFSFSSTREAM hVfsFss)
1173{
1174 if (hVfsFss != NIL_RTVFSFSSTREAM)
1175 {
1176 RTVFSOBJINTERNAL *pThis = &hVfsFss->Base;
1177 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1178 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1179
1180 rtVfsObjRetainVoid(pThis, "RTVfsObjFromFsStream");
1181 return pThis;
1182 }
1183 return NIL_RTVFSOBJ;
1184}
1185
1186
1187RTDECL(RTVFSOBJ) RTVfsObjFromDir(RTVFSDIR hVfsDir)
1188{
1189 if (hVfsDir != NIL_RTVFSDIR)
1190 {
1191 RTVFSOBJINTERNAL *pThis = &hVfsDir->Base;
1192 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1193 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1194
1195 rtVfsObjRetainVoid(pThis, "RTVfsObjFromDir");
1196 return pThis;
1197 }
1198 return NIL_RTVFSOBJ;
1199}
1200
1201
1202RTDECL(RTVFSOBJ) RTVfsObjFromIoStream(RTVFSIOSTREAM hVfsIos)
1203{
1204 if (hVfsIos != NIL_RTVFSIOSTREAM)
1205 {
1206 RTVFSOBJINTERNAL *pThis = &hVfsIos->Base;
1207 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1208 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1209
1210 rtVfsObjRetainVoid(pThis, "RTVfsObjFromIoStream");
1211 return pThis;
1212 }
1213 return NIL_RTVFSOBJ;
1214}
1215
1216
1217RTDECL(RTVFSOBJ) RTVfsObjFromFile(RTVFSFILE hVfsFile)
1218{
1219 if (hVfsFile != NIL_RTVFSFILE)
1220 {
1221 RTVFSOBJINTERNAL *pThis = &hVfsFile->Stream.Base;
1222 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1223 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1224
1225 rtVfsObjRetainVoid(pThis, "RTVfsObjFromFile");
1226 return pThis;
1227 }
1228 return NIL_RTVFSOBJ;
1229}
1230
1231
1232RTDECL(RTVFSOBJ) RTVfsObjFromSymlink(RTVFSSYMLINK hVfsSym)
1233{
1234 if (hVfsSym != NIL_RTVFSSYMLINK)
1235 {
1236 RTVFSOBJINTERNAL *pThis = &hVfsSym->Base;
1237 AssertPtrReturn(pThis, NIL_RTVFSOBJ);
1238 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, NIL_RTVFSOBJ);
1239
1240 rtVfsObjRetainVoid(pThis, "RTVfsObjFromSymlink");
1241 return pThis;
1242 }
1243 return NIL_RTVFSOBJ;
1244}
1245
1246
1247RTDECL(int) RTVfsObjOpen(RTVFS hVfs, const char *pszPath, uint64_t fFileOpen, uint32_t fObjFlags, PRTVFSOBJ phVfsObj)
1248{
1249 /*
1250 * Validate input.
1251 */
1252 RTVFSINTERNAL *pThis = hVfs;
1253 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1254 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
1255 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
1256 AssertPtrReturn(phVfsObj, VERR_INVALID_POINTER);
1257
1258 int rc = rtFileRecalcAndValidateFlags(&fFileOpen);
1259 if (RT_FAILURE(rc))
1260 return rc;
1261 AssertMsgReturn( RTPATH_F_IS_VALID(fObjFlags, RTVFSOBJ_F_VALID_MASK)
1262 && (fObjFlags & RTVFSOBJ_F_CREATE_MASK) <= RTVFSOBJ_F_CREATE_DIRECTORY,
1263 ("fObjFlags=%#x\n", fObjFlags),
1264 VERR_INVALID_FLAGS);
1265 /*
1266 * Parse the path, assume current directory is root since we've got no
1267 * caller context here.
1268 */
1269 PRTVFSPARSEDPATH pPath;
1270 rc = RTVfsParsePathA(pszPath, "/", &pPath);
1271 if (RT_SUCCESS(rc))
1272 {
1273 /*
1274 * Tranverse the path, resolving the parent node.
1275 * We'll do the symbolic link checking here with help of pfnOpen.
1276 */
1277 RTVFSDIRINTERNAL *pVfsParentDir;
1278 rc = rtVfsTraverseToParent(pThis, pPath, (fObjFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK, &pVfsParentDir);
1279 if (RT_SUCCESS(rc))
1280 {
1281
1282 /*
1283 * Do the opening. Loop if we need to follow symbolic links.
1284 */
1285 for (uint32_t cLoops = 1; ; cLoops++)
1286 {
1287 /* If we end with a directory slash, adjust open flags. */
1288 if (pPath->fDirSlash)
1289 {
1290 fObjFlags &= ~RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_OPEN_DIRECTORY;
1291 if ((fObjFlags & RTVFSOBJ_F_CREATE_MASK) != RTVFSOBJ_F_CREATE_DIRECTORY)
1292 fObjFlags = (fObjFlags & ~RTVFSOBJ_F_CREATE_MASK) | RTVFSOBJ_F_CREATE_NOTHING;
1293 }
1294 if (fObjFlags & RTPATH_F_FOLLOW_LINK)
1295 fObjFlags |= RTVFSOBJ_F_OPEN_SYMLINK;
1296
1297 /* Open it. */
1298 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
1299 RTVFSOBJ hVfsObj;
1300 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
1301 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fFileOpen, fObjFlags, &hVfsObj);
1302 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
1303 if (RT_FAILURE(rc))
1304 break;
1305
1306 /* We're done if we don't follow links or this wasn't a link. */
1307 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
1308 || RTVfsObjGetType(*phVfsObj) != RTVFSOBJTYPE_SYMLINK)
1309 {
1310 *phVfsObj = hVfsObj;
1311 break;
1312 }
1313
1314 /* Follow symbolic link. */
1315 if (cLoops < RTVFS_MAX_LINKS)
1316 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fObjFlags & RTPATH_F_MASK);
1317 else
1318 rc = VERR_TOO_MANY_SYMLINKS;
1319 RTVfsObjRelease(hVfsObj);
1320 if (RT_FAILURE(rc))
1321 break;
1322 }
1323 RTVfsDirRelease(pVfsParentDir);
1324 }
1325 RTVfsParsePathFree(pPath);
1326 }
1327 return rc;
1328}
1329
1330
1331RTDECL(int) RTVfsObjQueryInfo(RTVFSOBJ hVfsObj, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
1332{
1333 RTVFSOBJINTERNAL *pThis = hVfsObj;
1334 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1335 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, VERR_INVALID_HANDLE);
1336
1337 RTVfsLockAcquireRead(pThis->hLock);
1338 int rc = pThis->pOps->pfnQueryInfo(pThis->pvThis, pObjInfo, enmAddAttr);
1339 RTVfsLockReleaseRead(pThis->hLock);
1340 return rc;
1341}
1342
1343
1344/**
1345 * Gets the RTVFSOBJSETOPS for the given base object.
1346 *
1347 * @returns Pointer to the vtable if supported by the type, otherwise NULL.
1348 * @param pThis The base object.
1349 */
1350static PCRTVFSOBJSETOPS rtVfsObjGetSetOps(RTVFSOBJINTERNAL *pThis)
1351{
1352 switch (pThis->pOps->enmType)
1353 {
1354 case RTVFSOBJTYPE_DIR:
1355 return &RT_FROM_MEMBER(pThis, RTVFSDIRINTERNAL, Base)->pOps->ObjSet;
1356 case RTVFSOBJTYPE_FILE:
1357 return &RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream.Base)->pOps->ObjSet;
1358 case RTVFSOBJTYPE_SYMLINK:
1359 return &RT_FROM_MEMBER(pThis, RTVFSSYMLINKINTERNAL, Base)->pOps->ObjSet;
1360 default:
1361 return NULL;
1362 }
1363}
1364
1365
1366RTDECL(int) RTVfsObjSetMode(RTVFSOBJ hVfsObj, RTFMODE fMode, RTFMODE fMask)
1367{
1368 RTVFSOBJINTERNAL *pThis = hVfsObj;
1369 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1370 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, VERR_INVALID_HANDLE);
1371
1372 fMode = rtFsModeNormalize(fMode, NULL, 0, 0);
1373 if (!rtFsModeIsValid(fMode))
1374 return VERR_INVALID_PARAMETER;
1375
1376 PCRTVFSOBJSETOPS pObjSetOps = rtVfsObjGetSetOps(pThis);
1377 AssertReturn(pObjSetOps, VERR_INVALID_FUNCTION);
1378
1379 int rc;
1380 if (pObjSetOps->pfnSetMode)
1381 {
1382 RTVfsLockAcquireWrite(pThis->hLock);
1383 rc = pObjSetOps->pfnSetMode(pThis->pvThis, fMode, fMask);
1384 RTVfsLockReleaseWrite(pThis->hLock);
1385 }
1386 else
1387 rc = VERR_WRITE_PROTECT;
1388 return rc;
1389}
1390
1391
1392RTDECL(int) RTVfsObjSetTimes(RTVFSOBJ hVfsObj, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
1393 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
1394{
1395 RTVFSOBJINTERNAL *pThis = hVfsObj;
1396 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1397 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, VERR_INVALID_HANDLE);
1398
1399 AssertPtrNullReturn(pAccessTime, VERR_INVALID_POINTER);
1400 AssertPtrNullReturn(pModificationTime, VERR_INVALID_POINTER);
1401 AssertPtrNullReturn(pChangeTime, VERR_INVALID_POINTER);
1402 AssertPtrNullReturn(pBirthTime, VERR_INVALID_POINTER);
1403
1404 PCRTVFSOBJSETOPS pObjSetOps = rtVfsObjGetSetOps(pThis);
1405 AssertReturn(pObjSetOps, VERR_INVALID_FUNCTION);
1406
1407 int rc;
1408 if (pObjSetOps->pfnSetTimes)
1409 {
1410 RTVfsLockAcquireWrite(pThis->hLock);
1411 rc = pObjSetOps->pfnSetTimes(pThis->pvThis, pAccessTime, pModificationTime, pChangeTime, pBirthTime);
1412 RTVfsLockReleaseWrite(pThis->hLock);
1413 }
1414 else
1415 rc = VERR_WRITE_PROTECT;
1416 return rc;
1417}
1418
1419
1420RTDECL(int) RTVfsObjSetOwner(RTVFSOBJ hVfsObj, RTUID uid, RTGID gid)
1421{
1422 RTVFSOBJINTERNAL *pThis = hVfsObj;
1423 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1424 AssertReturn(pThis->uMagic == RTVFSOBJ_MAGIC, VERR_INVALID_HANDLE);
1425
1426 PCRTVFSOBJSETOPS pObjSetOps = rtVfsObjGetSetOps(pThis);
1427 AssertReturn(pObjSetOps, VERR_INVALID_FUNCTION);
1428
1429 int rc;
1430 if (pObjSetOps->pfnSetOwner)
1431 {
1432 RTVfsLockAcquireWrite(pThis->hLock);
1433 rc = pObjSetOps->pfnSetOwner(pThis->pvThis, uid, gid);
1434 RTVfsLockReleaseWrite(pThis->hLock);
1435 }
1436 else
1437 rc = VERR_WRITE_PROTECT;
1438 return rc;
1439}
1440
1441
1442/*
1443 *
1444 * U T I L U T I L U T I L
1445 * U T I L U T I L U T I L
1446 * U T I L U T I L U T I L
1447 *
1448 */
1449
1450
1451RTDECL(int) RTVfsParsePathAppend(PRTVFSPARSEDPATH pPath, const char *pszPath, uint16_t *piRestartComp)
1452{
1453 AssertReturn(*pszPath != '/' && *pszPath != '\\', VERR_INTERNAL_ERROR_4);
1454
1455 /* In case *piRestartComp was set higher than the number of components
1456 before making the call to this function. */
1457 if (piRestartComp && *piRestartComp + 1 >= pPath->cComponents)
1458 *piRestartComp = pPath->cComponents > 0 ? pPath->cComponents - 1 : 0;
1459
1460/** @todo The '..' handling doesn't really work wrt to symbolic links in the
1461 * path. */
1462
1463 /*
1464 * Append a slash to the destination path if necessary.
1465 */
1466 char * const pszDst = pPath->szPath;
1467 size_t offDst = pPath->cch;
1468 if (pPath->cComponents > 0)
1469 {
1470 pszDst[offDst++] = '/';
1471 if (offDst >= RTVFSPARSEDPATH_MAX)
1472 return VERR_FILENAME_TOO_LONG;
1473 }
1474 if (pPath->fAbsolute)
1475 Assert(pszDst[offDst - 1] == '/' && pszDst[0] == '/');
1476 else
1477 Assert(offDst == 0 || (pszDst[0] != '/' && pszDst[offDst - 1] == '/'));
1478
1479 /*
1480 * Parse and append the relative path.
1481 */
1482 const char *pszSrc = pszPath;
1483 pPath->fDirSlash = false;
1484 for (;;)
1485 {
1486 /* Copy until we encounter the next slash. */
1487 pPath->aoffComponents[pPath->cComponents++] = (uint16_t)offDst;
1488 for (;;)
1489 {
1490 char ch = *pszSrc++;
1491 if ( ch != '/'
1492 && ch != '\\'
1493 && ch != '\0')
1494 {
1495 pszDst[offDst++] = ch;
1496 if (offDst < RTVFSPARSEDPATH_MAX)
1497 { /* likely */ }
1498 else
1499 return VERR_FILENAME_TOO_LONG;
1500 }
1501 else
1502 {
1503 /* Deal with dot components before we processes the slash/end. */
1504 if (pszDst[offDst - 1] == '.')
1505 {
1506 if ( offDst == 1
1507 || pszDst[offDst - 2] == '/')
1508 {
1509 pPath->cComponents--;
1510 offDst = pPath->aoffComponents[pPath->cComponents];
1511 }
1512 else if ( offDst > 3
1513 && pszDst[offDst - 2] == '.'
1514 && pszDst[offDst - 3] == '/')
1515 {
1516 if ( pPath->fAbsolute
1517 || offDst < 5
1518 || pszDst[offDst - 4] != '.'
1519 || pszDst[offDst - 5] != '.'
1520 || (offDst >= 6 && pszDst[offDst - 6] != '/') )
1521 {
1522 pPath->cComponents -= pPath->cComponents > 1 ? 2 : 1;
1523 offDst = pPath->aoffComponents[pPath->cComponents];
1524 if (piRestartComp && *piRestartComp + 1 >= pPath->cComponents)
1525 *piRestartComp = pPath->cComponents > 0 ? pPath->cComponents - 1 : 0;
1526 }
1527 }
1528 }
1529
1530 if (ch != '\0')
1531 {
1532 /* Skip unnecessary slashes and check for end of path. */
1533 while ((ch = *pszSrc) == '/' || ch == '\\')
1534 pszSrc++;
1535
1536 if (ch == '\0')
1537 pPath->fDirSlash = true;
1538 }
1539
1540 if (ch == '\0')
1541 {
1542 /* Drop trailing slash unless it's the root slash. */
1543 if ( offDst > 0
1544 && pszDst[offDst - 1] == '/'
1545 && ( !pPath->fAbsolute
1546 || offDst > 1))
1547 offDst--;
1548
1549 /* Terminate the string and enter its length. */
1550 pszDst[offDst] = '\0';
1551 pszDst[offDst + 1] = '\0'; /* for aoffComponents[pPath->cComponents] */
1552 pPath->cch = (uint16_t)offDst;
1553 pPath->aoffComponents[pPath->cComponents] = (uint16_t)(offDst + 1);
1554 return VINF_SUCCESS;
1555 }
1556
1557 /* Append component separator before continuing with the next component. */
1558 if (offDst > 0 && pszDst[offDst - 1] != '/')
1559 pszDst[offDst++] = '/';
1560 if (offDst >= RTVFSPARSEDPATH_MAX)
1561 return VERR_FILENAME_TOO_LONG;
1562 break;
1563 }
1564 }
1565 }
1566}
1567
1568
1569/** @todo Replace RTVfsParsePath with RTPathParse and friends? */
1570RTDECL(int) RTVfsParsePath(PRTVFSPARSEDPATH pPath, const char *pszPath, const char *pszCwd)
1571{
1572 if (*pszPath != '/' && *pszPath != '\\')
1573 {
1574 if (pszCwd)
1575 {
1576 /*
1577 * Relative with a CWD.
1578 */
1579 int rc = RTVfsParsePath(pPath, pszCwd, NULL /*crash if pszCwd is not absolute*/);
1580 if (RT_FAILURE(rc))
1581 return rc;
1582 }
1583 else
1584 {
1585 /*
1586 * Relative.
1587 */
1588 pPath->cch = 0;
1589 pPath->cComponents = 0;
1590 pPath->fDirSlash = false;
1591 pPath->fAbsolute = false;
1592 pPath->aoffComponents[0] = 0;
1593 pPath->aoffComponents[1] = 1;
1594 pPath->szPath[0] = '\0';
1595 pPath->szPath[1] = '\0';
1596 }
1597 }
1598 else
1599 {
1600 /*
1601 * Make pszPath relative, i.e. set up pPath for the root and skip
1602 * leading slashes in pszPath before appending it.
1603 */
1604 pPath->cch = 1;
1605 pPath->cComponents = 0;
1606 pPath->fDirSlash = false;
1607 pPath->fAbsolute = true;
1608 pPath->aoffComponents[0] = 1;
1609 pPath->aoffComponents[1] = 2;
1610 pPath->szPath[0] = '/';
1611 pPath->szPath[1] = '\0';
1612 pPath->szPath[2] = '\0';
1613 while (pszPath[0] == '/' || pszPath[0] == '\\')
1614 pszPath++;
1615 if (!pszPath[0])
1616 return VINF_SUCCESS;
1617 }
1618 return RTVfsParsePathAppend(pPath, pszPath, NULL);
1619}
1620
1621
1622
1623RTDECL(int) RTVfsParsePathA(const char *pszPath, const char *pszCwd, PRTVFSPARSEDPATH *ppPath)
1624{
1625 /*
1626 * Allocate the output buffer and hand the problem to rtVfsParsePath.
1627 */
1628 int rc;
1629 PRTVFSPARSEDPATH pPath = (PRTVFSPARSEDPATH)RTMemTmpAlloc(sizeof(RTVFSPARSEDPATH));
1630 if (pPath)
1631 {
1632 rc = RTVfsParsePath(pPath, pszPath, pszCwd);
1633 if (RT_FAILURE(rc))
1634 {
1635 RTMemTmpFree(pPath);
1636 pPath = NULL;
1637 }
1638 }
1639 else
1640 rc = VERR_NO_TMP_MEMORY;
1641 *ppPath = pPath; /* always set it */
1642 return rc;
1643}
1644
1645
1646RTDECL(void) RTVfsParsePathFree(PRTVFSPARSEDPATH pPath)
1647{
1648 if (pPath)
1649 {
1650 pPath->cch = UINT16_MAX;
1651 pPath->cComponents = UINT16_MAX;
1652 pPath->aoffComponents[0] = UINT16_MAX;
1653 pPath->aoffComponents[1] = UINT16_MAX;
1654 RTMemTmpFree(pPath);
1655 }
1656}
1657
1658
1659/**
1660 * Handles a symbolic link, adding it to
1661 *
1662 * @returns IPRT status code.
1663 * @param ppCurDir The current directory variable. We change it if
1664 * the symbolic links is absolute.
1665 * @param pPath The parsed path to update.
1666 * @param iPathComponent The current path component.
1667 * @param hSymlink The symbolic link to process.
1668 */
1669static int rtVfsTraverseHandleSymlink(RTVFSDIRINTERNAL **ppCurDir, PRTVFSPARSEDPATH pPath,
1670 uint16_t iPathComponent, RTVFSSYMLINK hSymlink)
1671{
1672 /*
1673 * Read the link and append the trailing path to it.
1674 */
1675 char szPath[RTPATH_MAX];
1676 int rc = RTVfsSymlinkRead(hSymlink, szPath, sizeof(szPath) - 1);
1677 if (RT_SUCCESS(rc))
1678 {
1679 szPath[sizeof(szPath) - 1] = '\0';
1680 if (iPathComponent + 1 < pPath->cComponents)
1681 rc = RTPathAppend(szPath, sizeof(szPath), &pPath->szPath[pPath->aoffComponents[iPathComponent + 1]]);
1682 }
1683 if (RT_SUCCESS(rc))
1684 {
1685 /*
1686 * Special hack help vfsstddir.cpp deal with symbolic links.
1687 */
1688 RTVFSDIRINTERNAL *pCurDir = *ppCurDir;
1689 char *pszPath = szPath;
1690 if (pCurDir->pOps->pfnFollowAbsoluteSymlink)
1691 {
1692 size_t cchRoot = rtPathRootSpecLen(szPath);
1693 if (cchRoot > 0)
1694 {
1695 pszPath = &szPath[cchRoot];
1696 char const chSaved = *pszPath;
1697 *pszPath = '\0';
1698 RTVFSDIRINTERNAL *pVfsRootDir;
1699 RTVfsLockAcquireWrite(pCurDir->Base.hLock);
1700 rc = pCurDir->pOps->pfnFollowAbsoluteSymlink(pCurDir, szPath, &pVfsRootDir);
1701 RTVfsLockAcquireWrite(pCurDir->Base.hLock);
1702 *pszPath = chSaved;
1703 if (RT_SUCCESS(rc))
1704 {
1705 RTVfsDirRelease(pCurDir);
1706 *ppCurDir = pCurDir = pVfsRootDir;
1707 }
1708 else if (rc == VERR_PATH_IS_RELATIVE)
1709 pszPath = szPath;
1710 else
1711 return rc;
1712 }
1713 }
1714
1715 rc = RTVfsParsePath(pPath, pszPath, NULL);
1716 if (RT_SUCCESS(rc))
1717 {
1718 /*
1719 * Deal with absolute references in a VFS setup.
1720 * Note! The current approach only correctly handles this on root volumes.
1721 */
1722 if ( pPath->fAbsolute
1723 && pCurDir->Base.hVfs != NIL_RTVFS) /** @todo This needs fixing once we implement mount points. */
1724 {
1725 RTVFSINTERNAL *pVfs = pCurDir->Base.hVfs;
1726 RTVFSDIRINTERNAL *pVfsRootDir;
1727 RTVfsLockAcquireRead(pVfs->Base.hLock);
1728 rc = pVfs->pOps->pfnOpenRoot(pVfs->Base.pvThis, &pVfsRootDir);
1729 RTVfsLockReleaseRead(pVfs->Base.hLock);
1730 if (RT_SUCCESS(rc))
1731 {
1732 RTVfsDirRelease(pCurDir);
1733 *ppCurDir = pCurDir = pVfsRootDir;
1734 }
1735 else
1736 return rc;
1737 }
1738 }
1739 }
1740 else if (rc == VERR_BUFFER_OVERFLOW)
1741 rc = VERR_FILENAME_TOO_LONG;
1742 return rc == VERR_BUFFER_OVERFLOW ? VERR_FILENAME_TOO_LONG : rc;
1743}
1744
1745
1746/**
1747 * Internal worker for various open functions as well as RTVfsTraverseToParent.
1748 *
1749 *
1750 * @returns IPRT status code.
1751 * @param pThis The VFS.
1752 * @param pPath The parsed path. This may be changed as symbolic
1753 * links are processed during the path traversal. If
1754 * it contains zero components, a dummy component is
1755 * added to assist the caller.
1756 * @param fFlags RTPATH_F_XXX.
1757 * @param ppVfsParentDir Where to return the parent directory handle
1758 * (referenced).
1759 */
1760static int rtVfsDirTraverseToParent(RTVFSDIRINTERNAL *pThis, PRTVFSPARSEDPATH pPath, uint32_t fFlags,
1761 RTVFSDIRINTERNAL **ppVfsParentDir)
1762{
1763 /*
1764 * Assert sanity.
1765 */
1766 AssertPtr(pThis);
1767 Assert(pThis->uMagic == RTVFSDIR_MAGIC);
1768 Assert(pThis->Base.cRefs > 0);
1769 AssertPtr(pPath);
1770 AssertPtr(ppVfsParentDir);
1771 *ppVfsParentDir = NULL;
1772 Assert(RTPATH_F_IS_VALID(fFlags, 0));
1773
1774 /*
1775 * Start with the pThis directory.
1776 */
1777 if (RTVfsDirRetain(pThis) == UINT32_MAX)
1778 return VERR_INVALID_HANDLE;
1779 RTVFSDIRINTERNAL *pCurDir = pThis;
1780
1781 /*
1782 * Special case for traversing zero components.
1783 * We fake up a "./" in the pPath to help the caller along.
1784 */
1785 if (pPath->cComponents == 0)
1786 {
1787 pPath->fDirSlash = true;
1788 pPath->szPath[0] = '.';
1789 pPath->szPath[1] = '\0';
1790 pPath->szPath[2] = '\0';
1791 pPath->cch = 1;
1792 pPath->cComponents = 1;
1793 pPath->aoffComponents[0] = 0;
1794 pPath->aoffComponents[1] = 1;
1795 pPath->aoffComponents[2] = 1;
1796
1797 *ppVfsParentDir = pCurDir;
1798 return VINF_SUCCESS;
1799 }
1800
1801
1802 /*
1803 * The traversal loop.
1804 */
1805 int rc = VINF_SUCCESS;
1806 unsigned cLinks = 0;
1807 uint16_t iComponent = 0;
1808 for (;;)
1809 {
1810 /*
1811 * Are we done yet?
1812 */
1813 bool fFinal = iComponent + 1 >= pPath->cComponents;
1814 if (fFinal && (fFlags & RTPATH_F_ON_LINK))
1815 {
1816 *ppVfsParentDir = pCurDir;
1817 return VINF_SUCCESS;
1818 }
1819
1820 /*
1821 * Try open the next entry.
1822 */
1823 const char *pszEntry = &pPath->szPath[pPath->aoffComponents[iComponent]];
1824 char *pszEntryEnd = &pPath->szPath[pPath->aoffComponents[iComponent + 1] - 1];
1825 *pszEntryEnd = '\0';
1826 RTVFSDIR hDir = NIL_RTVFSDIR;
1827 RTVFSSYMLINK hSymlink = NIL_RTVFSSYMLINK;
1828 RTVFS hVfsMnt = NIL_RTVFS;
1829 RTVFSOBJ hVfsObj = NIL_RTVFSOBJ;
1830 if (fFinal)
1831 {
1832 RTVfsLockAcquireRead(pCurDir->Base.hLock);
1833 rc = pCurDir->pOps->pfnOpen(pCurDir->Base.pvThis, pszEntry,
1834 RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE,
1835 RTVFSOBJ_F_OPEN_SYMLINK | RTVFSOBJ_F_CREATE_NOTHING
1836 | RTVFSOBJ_F_TRAVERSAL | RTPATH_F_ON_LINK,
1837 &hVfsObj);
1838 RTVfsLockReleaseRead(pCurDir->Base.hLock);
1839 *pszEntryEnd = '\0';
1840 if (RT_FAILURE(rc))
1841 {
1842 if ( rc == VERR_PATH_NOT_FOUND
1843 || rc == VERR_FILE_NOT_FOUND
1844 || rc == VERR_IS_A_DIRECTORY
1845 || rc == VERR_IS_A_FILE
1846 || rc == VERR_IS_A_FIFO
1847 || rc == VERR_IS_A_SOCKET
1848 || rc == VERR_IS_A_CHAR_DEVICE
1849 || rc == VERR_IS_A_BLOCK_DEVICE
1850 || rc == VERR_NOT_SYMLINK)
1851 {
1852 *ppVfsParentDir = pCurDir;
1853 return VINF_SUCCESS;
1854 }
1855 break;
1856 }
1857 hSymlink = RTVfsObjToSymlink(hVfsObj);
1858 Assert(hSymlink != NIL_RTVFSSYMLINK);
1859 }
1860 else
1861 {
1862 RTVfsLockAcquireRead(pCurDir->Base.hLock);
1863 rc = pCurDir->pOps->pfnOpen(pCurDir->Base.pvThis, pszEntry,
1864 RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE,
1865 RTVFSOBJ_F_OPEN_DIRECTORY | RTVFSOBJ_F_OPEN_SYMLINK | RTVFSOBJ_F_OPEN_MOUNT
1866 | RTVFSOBJ_F_CREATE_NOTHING | RTVFSOBJ_F_TRAVERSAL | RTPATH_F_ON_LINK,
1867 &hVfsObj);
1868 RTVfsLockReleaseRead(pCurDir->Base.hLock);
1869 *pszEntryEnd = '/';
1870 if (RT_FAILURE(rc))
1871 {
1872 if (rc == VERR_FILE_NOT_FOUND)
1873 rc = VERR_PATH_NOT_FOUND;
1874 break;
1875 }
1876 hDir = RTVfsObjToDir(hVfsObj);
1877 hSymlink = RTVfsObjToSymlink(hVfsObj);
1878 hVfsMnt = RTVfsObjToVfs(hVfsObj);
1879 }
1880 Assert( (hDir != NIL_RTVFSDIR && hSymlink == NIL_RTVFSSYMLINK && hVfsMnt == NIL_RTVFS)
1881 || (hDir == NIL_RTVFSDIR && hSymlink != NIL_RTVFSSYMLINK && hVfsMnt == NIL_RTVFS)
1882 || (hDir == NIL_RTVFSDIR && hSymlink == NIL_RTVFSSYMLINK && hVfsMnt != NIL_RTVFS));
1883 RTVfsObjRelease(hVfsObj);
1884
1885 if (hDir != NIL_RTVFSDIR)
1886 {
1887 /*
1888 * Directory - advance down the path.
1889 */
1890 AssertPtr(hDir);
1891 Assert(hDir->uMagic == RTVFSDIR_MAGIC);
1892 RTVfsDirRelease(pCurDir);
1893 pCurDir = hDir;
1894 iComponent++;
1895 }
1896 else if (hSymlink != NIL_RTVFSSYMLINK)
1897 {
1898 /*
1899 * Symbolic link - deal with it and retry the current component.
1900 */
1901 AssertPtr(hSymlink);
1902 Assert(hSymlink->uMagic == RTVFSSYMLINK_MAGIC);
1903 if (fFlags & RTPATH_F_NO_SYMLINKS)
1904 {
1905 rc = VERR_SYMLINK_NOT_ALLOWED;
1906 break;
1907 }
1908 cLinks++;
1909 if (cLinks >= RTVFS_MAX_LINKS)
1910 {
1911 rc = VERR_TOO_MANY_SYMLINKS;
1912 break;
1913 }
1914 rc = rtVfsTraverseHandleSymlink(&pCurDir, pPath, iComponent, hSymlink);
1915 if (RT_FAILURE(rc))
1916 break;
1917 iComponent = 0;
1918 }
1919 else
1920 {
1921 /*
1922 * Mount point - deal with it and retry the current component.
1923 */
1924 RTVfsDirRelease(pCurDir);
1925 RTVfsLockAcquireRead(hVfsMnt->Base.hLock);
1926 rc = hVfsMnt->pOps->pfnOpenRoot(hVfsMnt->Base.pvThis, &pCurDir);
1927 RTVfsLockReleaseRead(hVfsMnt->Base.hLock);
1928 if (RT_FAILURE(rc))
1929 {
1930 pCurDir = NULL;
1931 break;
1932 }
1933 iComponent = 0;
1934 /** @todo union mounts. */
1935 }
1936 }
1937
1938 if (pCurDir)
1939 RTVfsDirRelease(pCurDir);
1940
1941 return rc;
1942}
1943
1944
1945/**
1946 * Internal worker for various open functions as well as RTVfsTraverseToParent.
1947 *
1948 * @returns IPRT status code.
1949 * @param pThis The VFS.
1950 * @param pPath The parsed path. This may be changed as symbolic
1951 * links are processed during the path traversal.
1952 * @param fFlags RTPATH_F_XXX.
1953 * @param ppVfsParentDir Where to return the parent directory handle
1954 * (referenced).
1955 */
1956static int rtVfsTraverseToParent(RTVFSINTERNAL *pThis, PRTVFSPARSEDPATH pPath, uint32_t fFlags, RTVFSDIRINTERNAL **ppVfsParentDir)
1957{
1958 /*
1959 * Assert sanity.
1960 */
1961 AssertPtr(pThis);
1962 Assert(pThis->uMagic == RTVFS_MAGIC);
1963 Assert(pThis->Base.cRefs > 0);
1964 AssertPtr(pPath);
1965 AssertPtr(ppVfsParentDir);
1966 *ppVfsParentDir = NULL;
1967 Assert(RTPATH_F_IS_VALID(fFlags, 0));
1968
1969 /*
1970 * Open the root directory and join paths with the directory traversal.
1971 */
1972 /** @todo Union mounts, traversal optimization methods, races, ++ */
1973 RTVFSDIRINTERNAL *pRootDir;
1974 RTVfsLockAcquireRead(pThis->Base.hLock);
1975 int rc = pThis->pOps->pfnOpenRoot(pThis->Base.pvThis, &pRootDir);
1976 RTVfsLockReleaseRead(pThis->Base.hLock);
1977 if (RT_SUCCESS(rc))
1978 {
1979 rc = rtVfsDirTraverseToParent(pRootDir, pPath, fFlags, ppVfsParentDir);
1980 RTVfsDirRelease(pRootDir);
1981 }
1982 return rc;
1983}
1984
1985
1986
1987/**
1988 * Follows a symbolic link object to the next parent directory.
1989 *
1990 * @returns IPRT status code
1991 * @param ppVfsParentDir Pointer to the parent directory of @a hVfsObj on
1992 * input, the parent directory of the link target on
1993 * return.
1994 * @param hVfsObj Symbolic link object handle.
1995 * @param pPath Path buffer to use parse the symbolic link target.
1996 * @param fFlags See rtVfsDirTraverseToParent.
1997 */
1998static int rtVfsDirFollowSymlinkObjToParent(RTVFSDIRINTERNAL **ppVfsParentDir, RTVFSOBJ hVfsObj,
1999 PRTVFSPARSEDPATH pPath, uint32_t fFlags)
2000{
2001 RTVFSSYMLINK hVfsSymlink = RTVfsObjToSymlink(hVfsObj);
2002 AssertReturn(hVfsSymlink != NIL_RTVFSSYMLINK, VERR_INTERNAL_ERROR_3);
2003
2004 int rc = rtVfsTraverseHandleSymlink(ppVfsParentDir, pPath, pPath->cComponents, hVfsSymlink);
2005 if (RT_SUCCESS(rc))
2006 {
2007 RTVFSDIRINTERNAL *pVfsStartDir = *ppVfsParentDir;
2008 rc = rtVfsDirTraverseToParent(pVfsStartDir, pPath, fFlags, ppVfsParentDir);
2009 RTVfsDirRelease(pVfsStartDir);
2010 }
2011
2012 RTVfsSymlinkRelease(hVfsSymlink);
2013 return rc;
2014}
2015
2016
2017
2018RTDECL(int) RTVfsUtilDummyPollOne(uint32_t fEvents, RTMSINTERVAL cMillies, bool fIntr, uint32_t *pfRetEvents)
2019{
2020 NOREF(fEvents);
2021 int rc;
2022 if (fIntr)
2023 rc = RTThreadSleep(cMillies);
2024 else
2025 {
2026 uint64_t uMsStart = RTTimeMilliTS();
2027 do
2028 rc = RTThreadSleep(cMillies);
2029 while ( rc == VERR_INTERRUPTED
2030 && !fIntr
2031 && RTTimeMilliTS() - uMsStart < cMillies);
2032 if (rc == VERR_INTERRUPTED)
2033 rc = VERR_TIMEOUT;
2034 }
2035
2036 *pfRetEvents = 0;
2037 return rc;
2038}
2039
2040
2041RTDECL(int) RTVfsUtilPumpIoStreams(RTVFSIOSTREAM hVfsIosSrc, RTVFSIOSTREAM hVfsIosDst, size_t cbBufHint)
2042{
2043 /*
2044 * Allocate a temporary buffer.
2045 */
2046 size_t cbBuf = cbBufHint;
2047 if (!cbBuf)
2048 cbBuf = _64K;
2049 else if (cbBuf < _4K)
2050 cbBuf = _4K;
2051 else if (cbBuf > _1M)
2052 cbBuf = _1M;
2053
2054 void *pvBuf = RTMemTmpAlloc(cbBuf);
2055 if (!pvBuf)
2056 {
2057 cbBuf = _4K;
2058 pvBuf = RTMemTmpAlloc(cbBuf);
2059 if (!pvBuf)
2060 return VERR_NO_TMP_MEMORY;
2061 }
2062
2063 /*
2064 * Pump loop.
2065 */
2066 int rc;
2067 for (;;)
2068 {
2069 size_t cbRead;
2070 rc = RTVfsIoStrmRead(hVfsIosSrc, pvBuf, cbBuf, true /*fBlocking*/, &cbRead);
2071 if (RT_FAILURE(rc))
2072 break;
2073 if (rc == VINF_EOF && cbRead == 0)
2074 break;
2075
2076 rc = RTVfsIoStrmWrite(hVfsIosDst, pvBuf, cbRead, true /*fBlocking*/, NULL /*cbWritten*/);
2077 if (RT_FAILURE(rc))
2078 break;
2079 }
2080
2081 RTMemTmpFree(pvBuf);
2082
2083 /*
2084 * Flush the destination stream on success to make sure we've caught
2085 * errors caused by buffering delays.
2086 */
2087 if (RT_SUCCESS(rc))
2088 rc = RTVfsIoStrmFlush(hVfsIosDst);
2089
2090 return rc;
2091}
2092
2093
2094
2095
2096
2097/*
2098 * F I L E S Y S T E M R O O T
2099 * F I L E S Y S T E M R O O T
2100 * F I L E S Y S T E M R O O T
2101 */
2102
2103
2104RTDECL(int) RTVfsNew(PCRTVFSOPS pVfsOps, size_t cbInstance, RTVFS hVfs, RTVFSLOCK hLock,
2105 PRTVFS phVfs, void **ppvInstance)
2106{
2107 /*
2108 * Validate the input, be extra strict in strict builds.
2109 */
2110 AssertPtr(pVfsOps);
2111 AssertReturn(pVfsOps->uVersion == RTVFSOPS_VERSION, VERR_VERSION_MISMATCH);
2112 AssertReturn(pVfsOps->uEndMarker == RTVFSOPS_VERSION, VERR_VERSION_MISMATCH);
2113 RTVFSOBJ_ASSERT_OPS(&pVfsOps->Obj, RTVFSOBJTYPE_VFS);
2114 Assert(cbInstance > 0);
2115 AssertPtr(ppvInstance);
2116 AssertPtr(phVfs);
2117
2118 /*
2119 * Allocate the handle + instance data.
2120 */
2121 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSINTERNAL), RTVFS_INST_ALIGNMENT)
2122 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
2123 RTVFSINTERNAL *pThis = (RTVFSINTERNAL *)RTMemAllocZ(cbThis);
2124 if (!pThis)
2125 return VERR_NO_MEMORY;
2126
2127 int rc = rtVfsObjInitNewObject(&pThis->Base, &pVfsOps->Obj, hVfs, false /*fNoVfsRef*/, hLock,
2128 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
2129 if (RT_FAILURE(rc))
2130 {
2131 RTMemFree(pThis);
2132 return rc;
2133 }
2134
2135 pThis->uMagic = RTVFS_MAGIC;
2136 pThis->pOps = pVfsOps;
2137
2138 *phVfs = pThis;
2139 *ppvInstance = pThis->Base.pvThis;
2140
2141 LogFlow(("RTVfsNew -> VINF_SUCCESS; hVfs=%p pvThis=%p\n", pThis, pThis->Base.pvThis));
2142 return VINF_SUCCESS;
2143}
2144
2145#ifdef DEBUG
2146# undef RTVfsRetain
2147#endif
2148RTDECL(uint32_t) RTVfsRetain(RTVFS hVfs)
2149{
2150 RTVFSINTERNAL *pThis = hVfs;
2151 AssertPtrReturn(pThis, UINT32_MAX);
2152 AssertReturn(pThis->uMagic == RTVFS_MAGIC, UINT32_MAX);
2153 uint32_t cRefs = rtVfsObjRetain(&pThis->Base);
2154 LogFlow(("RTVfsRetain(%p/%p) -> %d\n", pThis, pThis->Base.pvThis, cRefs));
2155 return cRefs;
2156}
2157#ifdef DEBUG
2158# define RTVfsRetain(hVfs) RTVfsRetainDebug(hVfs, RT_SRC_POS)
2159#endif
2160
2161
2162RTDECL(uint32_t) RTVfsRetainDebug(RTVFS hVfs, RT_SRC_POS_DECL)
2163{
2164 RTVFSINTERNAL *pThis = hVfs;
2165 AssertPtrReturn(pThis, UINT32_MAX);
2166 AssertReturn(pThis->uMagic == RTVFS_MAGIC, UINT32_MAX);
2167 RT_SRC_POS_NOREF();
2168 return rtVfsObjRetainDebug(&pThis->Base, "RTVfsRetainDebug", RT_SRC_POS_ARGS);
2169}
2170
2171
2172RTDECL(uint32_t) RTVfsRelease(RTVFS hVfs)
2173{
2174 RTVFSINTERNAL *pThis = hVfs;
2175 if (pThis == NIL_RTVFS)
2176 return 0;
2177 AssertPtrReturn(pThis, UINT32_MAX);
2178 AssertReturn(pThis->uMagic == RTVFS_MAGIC, UINT32_MAX);
2179#ifdef LOG_ENABLED
2180 void *pvThis = pThis->Base.pvThis;
2181#endif
2182 uint32_t cRefs = rtVfsObjRelease(&pThis->Base);
2183 Log(("RTVfsRelease(%p/%p) -> %d\n", pThis, pvThis, cRefs));
2184 return cRefs;
2185}
2186
2187
2188RTDECL(int) RTVfsOpenRoot(RTVFS hVfs, PRTVFSDIR phDir)
2189{
2190 RTVFSINTERNAL *pThis = hVfs;
2191 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2192 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
2193 AssertPtrReturn(phDir, VERR_INVALID_POINTER);
2194 *phDir = NIL_RTVFSDIR;
2195
2196 if (!pThis->pOps->pfnOpenRoot)
2197 return VERR_NOT_SUPPORTED;
2198 RTVfsLockAcquireRead(pThis->Base.hLock);
2199 int rc = pThis->pOps->pfnOpenRoot(pThis->Base.pvThis, phDir);
2200 RTVfsLockReleaseRead(pThis->Base.hLock);
2201
2202 return rc;
2203}
2204
2205
2206RTDECL(int) RTVfsQueryPathInfo(RTVFS hVfs, const char *pszPath, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr, uint32_t fFlags)
2207{
2208 RTVFSINTERNAL *pThis = hVfs;
2209 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2210 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
2211 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2212 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
2213 AssertPtrReturn(pObjInfo, VERR_INVALID_POINTER);
2214 AssertReturn(enmAddAttr >= RTFSOBJATTRADD_NOTHING && enmAddAttr <= RTFSOBJATTRADD_LAST, VERR_INVALID_PARAMETER);
2215 AssertMsgReturn(RTPATH_F_IS_VALID(fFlags, 0), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
2216
2217 /*
2218 * Parse the path, assume current directory is root since we've got no
2219 * caller context here. Then traverse to the parent directory.
2220 */
2221 PRTVFSPARSEDPATH pPath;
2222 int rc = RTVfsParsePathA(pszPath, "/", &pPath);
2223 if (RT_SUCCESS(rc))
2224 {
2225 /*
2226 * Tranverse the path, resolving the parent node.
2227 * We'll do the symbolic link checking here with help of pfnOpen/pfnQueryEntryInfo.
2228 */
2229 RTVFSDIRINTERNAL *pVfsParentDir;
2230 rc = rtVfsTraverseToParent(pThis, pPath, (fFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK, &pVfsParentDir);
2231 if (RT_SUCCESS(rc))
2232 {
2233 /*
2234 * Do the opening. Loop if we need to follow symbolic links.
2235 */
2236 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_CREATE_NOTHING;
2237 for (uint32_t cLoops = 1; ; cLoops++)
2238 {
2239 /* If we end with a directory slash, adjust open flags. */
2240 if (pPath->fDirSlash)
2241 {
2242 fObjFlags &= ~RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_OPEN_DIRECTORY;
2243 if ((fObjFlags & RTVFSOBJ_F_CREATE_MASK) != RTVFSOBJ_F_CREATE_DIRECTORY)
2244 fObjFlags = (fObjFlags & ~RTVFSOBJ_F_CREATE_MASK) | RTVFSOBJ_F_CREATE_NOTHING;
2245 }
2246 if (fObjFlags & RTPATH_F_FOLLOW_LINK)
2247 fObjFlags |= RTVFSOBJ_F_OPEN_SYMLINK;
2248
2249 /* Do the querying. If pfnQueryEntryInfo is available, we use it first,
2250 falling back on pfnOpen in case of symbolic links that needs following. */
2251 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
2252 if (pVfsParentDir->pOps->pfnQueryEntryInfo)
2253 {
2254 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
2255 rc = pVfsParentDir->pOps->pfnQueryEntryInfo(pVfsParentDir->Base.pvThis, pszEntryName, pObjInfo, enmAddAttr);
2256 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
2257 if (RT_FAILURE(rc))
2258 break;
2259 if ( !RTFS_IS_SYMLINK(pObjInfo->Attr.fMode)
2260 || !(fFlags & RTPATH_F_FOLLOW_LINK))
2261 {
2262 if ( (fObjFlags & RTVFSOBJ_F_OPEN_MASK) != RTVFSOBJ_F_OPEN_ANY
2263 && !RTFS_IS_DIRECTORY(pObjInfo->Attr.fMode))
2264 rc = VERR_NOT_A_DIRECTORY;
2265 break;
2266 }
2267 }
2268
2269 RTVFSOBJ hVfsObj;
2270 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
2271 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName,
2272 RTFILE_O_ACCESS_ATTR_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN,
2273 fObjFlags, &hVfsObj);
2274 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
2275 if (RT_FAILURE(rc))
2276 break;
2277
2278 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
2279 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
2280 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
2281 {
2282 rc = RTVfsObjQueryInfo(hVfsObj, pObjInfo, enmAddAttr);
2283 RTVfsObjRelease(hVfsObj);
2284 break;
2285 }
2286
2287 /* Follow symbolic link. */
2288 if (cLoops < RTVFS_MAX_LINKS)
2289 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fObjFlags & RTPATH_F_MASK);
2290 else
2291 rc = VERR_TOO_MANY_SYMLINKS;
2292 RTVfsObjRelease(hVfsObj);
2293 if (RT_FAILURE(rc))
2294 break;
2295 }
2296 RTVfsDirRelease(pVfsParentDir);
2297 }
2298 RTVfsParsePathFree(pPath);
2299 }
2300 return rc;
2301}
2302
2303
2304
2305RTDECL(int) RTVfsQueryRangeState(RTVFS hVfs, uint64_t off, size_t cb, bool *pfUsed)
2306{
2307 RTVFSINTERNAL *pThis = hVfs;
2308 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2309 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
2310
2311 if (!pThis->pOps->pfnQueryRangeState)
2312 return VERR_NOT_SUPPORTED;
2313 RTVfsLockAcquireRead(pThis->Base.hLock);
2314 int rc = pThis->pOps->pfnQueryRangeState(pThis->Base.pvThis, off, cb, pfUsed);
2315 RTVfsLockReleaseRead(pThis->Base.hLock);
2316
2317 return rc;
2318}
2319
2320
2321RTDECL(int) RTVfsQueryLabel(RTVFS hVfs, bool fAlternative, char *pszLabel, size_t cbLabel, size_t *pcbActual)
2322{
2323 RTVFSINTERNAL *pThis = hVfs;
2324 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2325 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
2326
2327 if (cbLabel > 0)
2328 AssertPtrReturn(pszLabel, VERR_INVALID_POINTER);
2329
2330 int rc;
2331 if (pThis->pOps->Obj.pfnQueryInfoEx)
2332 {
2333 size_t cbActualIgn;
2334 if (!pcbActual)
2335 pcbActual = &cbActualIgn;
2336
2337 RTVfsLockAcquireRead(pThis->Base.hLock);
2338 rc = pThis->pOps->Obj.pfnQueryInfoEx(pThis->Base.pvThis, !fAlternative ? RTVFSQIEX_VOL_LABEL : RTVFSQIEX_VOL_LABEL_ALT,
2339 pszLabel, cbLabel, pcbActual);
2340 RTVfsLockReleaseRead(pThis->Base.hLock);
2341 }
2342 else
2343 rc = VERR_NOT_SUPPORTED;
2344 return rc;
2345}
2346
2347
2348
2349/*
2350 *
2351 * F I L E S Y S T E M S T R E A M
2352 * F I L E S Y S T E M S T R E A M
2353 * F I L E S Y S T E M S T R E A M
2354 *
2355 */
2356
2357
2358RTDECL(int) RTVfsNewFsStream(PCRTVFSFSSTREAMOPS pFsStreamOps, size_t cbInstance, RTVFS hVfs, RTVFSLOCK hLock, uint32_t fAccess,
2359 PRTVFSFSSTREAM phVfsFss, void **ppvInstance)
2360{
2361 /*
2362 * Validate the input, be extra strict in strict builds.
2363 */
2364 AssertPtr(pFsStreamOps);
2365 AssertReturn(pFsStreamOps->uVersion == RTVFSFSSTREAMOPS_VERSION, VERR_VERSION_MISMATCH);
2366 AssertReturn(pFsStreamOps->uEndMarker == RTVFSFSSTREAMOPS_VERSION, VERR_VERSION_MISMATCH);
2367 Assert(!pFsStreamOps->fReserved);
2368 RTVFSOBJ_ASSERT_OPS(&pFsStreamOps->Obj, RTVFSOBJTYPE_FS_STREAM);
2369 Assert((fAccess & (RTFILE_O_READ | RTFILE_O_WRITE)) == fAccess);
2370 Assert(fAccess);
2371 if (fAccess & RTFILE_O_READ)
2372 AssertPtr(pFsStreamOps->pfnNext);
2373 if (fAccess & RTFILE_O_WRITE)
2374 {
2375 AssertPtr(pFsStreamOps->pfnAdd);
2376 AssertPtr(pFsStreamOps->pfnEnd);
2377 }
2378 Assert(cbInstance > 0);
2379 AssertPtr(ppvInstance);
2380 AssertPtr(phVfsFss);
2381 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
2382
2383 /*
2384 * Allocate the handle + instance data.
2385 */
2386 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSFSSTREAMINTERNAL), RTVFS_INST_ALIGNMENT)
2387 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
2388 RTVFSFSSTREAMINTERNAL *pThis = (RTVFSFSSTREAMINTERNAL *)RTMemAllocZ(cbThis);
2389 if (!pThis)
2390 return VERR_NO_MEMORY;
2391
2392 int rc = rtVfsObjInitNewObject(&pThis->Base, &pFsStreamOps->Obj, hVfs, false /*fNoVfsRef*/, hLock,
2393 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
2394
2395 if (RT_FAILURE(rc))
2396 {
2397 RTMemFree(pThis);
2398 return rc;
2399 }
2400
2401 pThis->uMagic = RTVFSFSSTREAM_MAGIC;
2402 pThis->pOps = pFsStreamOps;
2403 pThis->fFlags = fAccess;
2404 if (fAccess == RTFILE_O_READ)
2405 pThis->fFlags |= RTFILE_O_OPEN | RTFILE_O_DENY_NONE;
2406 else if (fAccess == RTFILE_O_WRITE)
2407 pThis->fFlags |= RTFILE_O_CREATE | RTFILE_O_DENY_ALL;
2408 else
2409 pThis->fFlags |= RTFILE_O_OPEN | RTFILE_O_DENY_ALL;
2410
2411 *phVfsFss = pThis;
2412 *ppvInstance = pThis->Base.pvThis;
2413 return VINF_SUCCESS;
2414}
2415
2416
2417#ifdef DEBUG
2418# undef RTVfsFsStrmRetain
2419#endif
2420RTDECL(uint32_t) RTVfsFsStrmRetain(RTVFSFSSTREAM hVfsFss)
2421{
2422 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2423 AssertPtrReturn(pThis, UINT32_MAX);
2424 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, UINT32_MAX);
2425 return rtVfsObjRetain(&pThis->Base);
2426}
2427#ifdef DEBUG
2428# define RTVfsFsStrmRetain(hVfsFss) RTVfsFsStrmRetainDebug(hVfsFss, RT_SRC_POS)
2429#endif
2430
2431
2432RTDECL(uint32_t) RTVfsFsStrmRetainDebug(RTVFSFSSTREAM hVfsFss, RT_SRC_POS_DECL)
2433{
2434 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2435 AssertPtrReturn(pThis, UINT32_MAX);
2436 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, UINT32_MAX);
2437 return rtVfsObjRetainDebug(&pThis->Base, "RTVfsFsStrmRetain", RT_SRC_POS_ARGS);
2438}
2439
2440
2441RTDECL(uint32_t) RTVfsFsStrmRelease(RTVFSFSSTREAM hVfsFss)
2442{
2443 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2444 if (pThis == NIL_RTVFSFSSTREAM)
2445 return 0;
2446 AssertPtrReturn(pThis, UINT32_MAX);
2447 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, UINT32_MAX);
2448 return rtVfsObjRelease(&pThis->Base);
2449}
2450
2451
2452RTDECL(int) RTVfsFsStrmQueryInfo(RTVFSFSSTREAM hVfsFss, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
2453{
2454 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2455 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2456 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, VERR_INVALID_HANDLE);
2457 return RTVfsObjQueryInfo(&pThis->Base, pObjInfo, enmAddAttr);
2458}
2459
2460
2461RTDECL(int) RTVfsFsStrmNext(RTVFSFSSTREAM hVfsFss, char **ppszName, RTVFSOBJTYPE *penmType, PRTVFSOBJ phVfsObj)
2462{
2463 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2464 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2465 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, VERR_INVALID_HANDLE);
2466 AssertPtrNullReturn(ppszName, VERR_INVALID_POINTER);
2467 if (ppszName)
2468 *ppszName = NULL;
2469 AssertPtrNullReturn(penmType, VERR_INVALID_POINTER);
2470 if (penmType)
2471 *penmType = RTVFSOBJTYPE_INVALID;
2472 AssertPtrNullReturn(penmType, VERR_INVALID_POINTER);
2473 if (phVfsObj)
2474 *phVfsObj = NIL_RTVFSOBJ;
2475
2476 AssertReturn(pThis->fFlags & RTFILE_O_READ, VERR_INVALID_FUNCTION);
2477
2478 return pThis->pOps->pfnNext(pThis->Base.pvThis, ppszName, penmType, phVfsObj);
2479}
2480
2481
2482RTDECL(int) RTVfsFsStrmAdd(RTVFSFSSTREAM hVfsFss, const char *pszPath, RTVFSOBJ hVfsObj, uint32_t fFlags)
2483{
2484 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2485 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2486 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, VERR_INVALID_HANDLE);
2487 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2488 AssertReturn(*pszPath != '\0', VERR_INVALID_NAME);
2489 AssertPtrReturn(hVfsObj, VERR_INVALID_HANDLE);
2490 AssertReturn(hVfsObj->uMagic == RTVFSOBJ_MAGIC, VERR_INVALID_HANDLE);
2491 AssertReturn(!(fFlags & ~RTVFSFSSTRM_ADD_F_VALID_MASK), VERR_INVALID_FLAGS);
2492 AssertReturn(pThis->fFlags & RTFILE_O_WRITE, VERR_INVALID_FUNCTION);
2493
2494 return pThis->pOps->pfnAdd(pThis->Base.pvThis, pszPath, hVfsObj, fFlags);
2495}
2496
2497
2498RTDECL(int) RTVfsFsStrmPushFile(RTVFSFSSTREAM hVfsFss, const char *pszPath, uint64_t cbFile,
2499 PCRTFSOBJINFO paObjInfo, uint32_t cObjInfo, uint32_t fFlags, PRTVFSIOSTREAM phVfsIos)
2500{
2501 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2502 AssertPtrReturn(phVfsIos, VERR_INVALID_POINTER);
2503 *phVfsIos = NIL_RTVFSIOSTREAM;
2504
2505 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2506 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, VERR_INVALID_HANDLE);
2507
2508 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2509 AssertReturn(*pszPath != '\0', VERR_INVALID_NAME);
2510
2511 AssertReturn(!(fFlags & ~RTVFSFSSTRM_PUSH_F_VALID_MASK), VERR_INVALID_FLAGS);
2512 AssertReturn(RT_BOOL(cbFile == UINT64_MAX) == RT_BOOL(fFlags & RTVFSFSSTRM_PUSH_F_STREAM), VERR_INVALID_FLAGS);
2513
2514 if (cObjInfo)
2515 {
2516 AssertPtrReturn(paObjInfo, VERR_INVALID_POINTER);
2517 AssertReturn(paObjInfo[0].Attr.enmAdditional == RTFSOBJATTRADD_UNIX, VERR_INVALID_PARAMETER);
2518 }
2519
2520 AssertReturn(pThis->fFlags & RTFILE_O_WRITE, VERR_INVALID_FUNCTION);
2521 if (pThis->pOps->pfnPushFile)
2522 return pThis->pOps->pfnPushFile(pThis->Base.pvThis, pszPath, cbFile, paObjInfo, cObjInfo, fFlags, phVfsIos);
2523 return VERR_NOT_SUPPORTED;
2524}
2525
2526
2527RTDECL(int) RTVfsFsStrmEnd(RTVFSFSSTREAM hVfsFss)
2528{
2529 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2530 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2531 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, VERR_INVALID_HANDLE);
2532
2533 return pThis->pOps->pfnEnd(pThis->Base.pvThis);
2534}
2535
2536
2537RTDECL(void *) RTVfsFsStreamToPrivate(RTVFSFSSTREAM hVfsFss, PCRTVFSFSSTREAMOPS pFsStreamOps)
2538{
2539 RTVFSFSSTREAMINTERNAL *pThis = hVfsFss;
2540 AssertPtrReturn(pThis, NULL);
2541 AssertReturn(pThis->uMagic == RTVFSFSSTREAM_MAGIC, NULL);
2542 if (pThis->pOps != pFsStreamOps)
2543 return NULL;
2544 return pThis->Base.pvThis;
2545}
2546
2547
2548/*
2549 *
2550 * D I R D I R D I R
2551 * D I R D I R D I R
2552 * D I R D I R D I R
2553 *
2554 */
2555
2556
2557RTDECL(int) RTVfsNewDir(PCRTVFSDIROPS pDirOps, size_t cbInstance, uint32_t fFlags, RTVFS hVfs, RTVFSLOCK hLock,
2558 PRTVFSDIR phVfsDir, void **ppvInstance)
2559{
2560 /*
2561 * Validate the input, be extra strict in strict builds.
2562 */
2563 AssertPtr(pDirOps);
2564 AssertReturn(pDirOps->uVersion == RTVFSDIROPS_VERSION, VERR_VERSION_MISMATCH);
2565 AssertReturn(pDirOps->uEndMarker == RTVFSDIROPS_VERSION, VERR_VERSION_MISMATCH);
2566 Assert(!pDirOps->fReserved);
2567 RTVFSDIR_ASSERT_OPS(pDirOps, RTVFSOBJTYPE_DIR);
2568 Assert(cbInstance > 0);
2569 AssertReturn(!(fFlags & ~RTVFSDIR_F_NO_VFS_REF), VERR_INVALID_FLAGS);
2570 AssertPtr(ppvInstance);
2571 AssertPtr(phVfsDir);
2572 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
2573
2574 /*
2575 * Allocate the handle + instance data.
2576 */
2577 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSDIRINTERNAL), RTVFS_INST_ALIGNMENT)
2578 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
2579 RTVFSDIRINTERNAL *pThis = (RTVFSDIRINTERNAL *)RTMemAllocZ(cbThis);
2580 if (!pThis)
2581 return VERR_NO_MEMORY;
2582
2583 int rc = rtVfsObjInitNewObject(&pThis->Base, &pDirOps->Obj, hVfs, RT_BOOL(fFlags & RTVFSDIR_F_NO_VFS_REF), hLock,
2584 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
2585 if (RT_FAILURE(rc))
2586 {
2587 RTMemFree(pThis);
2588 return rc;
2589 }
2590
2591 pThis->uMagic = RTVFSDIR_MAGIC;
2592 pThis->fReserved = 0;
2593 pThis->pOps = pDirOps;
2594
2595 *phVfsDir = pThis;
2596 *ppvInstance = pThis->Base.pvThis;
2597 return VINF_SUCCESS;
2598}
2599
2600
2601RTDECL(void *) RTVfsDirToPrivate(RTVFSDIR hVfsDir, PCRTVFSDIROPS pDirOps)
2602{
2603 RTVFSDIRINTERNAL *pThis = hVfsDir;
2604 AssertPtrReturn(pThis, NULL);
2605 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, NULL);
2606 if (pThis->pOps != pDirOps)
2607 return NULL;
2608 return pThis->Base.pvThis;
2609}
2610
2611
2612#ifdef DEBUG
2613# undef RTVfsDirRetain
2614#endif
2615RTDECL(uint32_t) RTVfsDirRetain(RTVFSDIR hVfsDir)
2616{
2617 RTVFSDIRINTERNAL *pThis = hVfsDir;
2618 AssertPtrReturn(pThis, UINT32_MAX);
2619 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, UINT32_MAX);
2620 uint32_t cRefs = rtVfsObjRetain(&pThis->Base);
2621 LogFlow(("RTVfsDirRetain(%p/%p) -> %#x\n", pThis, pThis->Base.pvThis, cRefs));
2622 return cRefs;
2623}
2624#ifdef DEBUG
2625# define RTVfsDirRetain(hVfsDir) RTVfsDirRetainDebug(hVfsDir, RT_SRC_POS)
2626#endif
2627
2628
2629RTDECL(uint32_t) RTVfsDirRetainDebug(RTVFSDIR hVfsDir, RT_SRC_POS_DECL)
2630{
2631 RTVFSDIRINTERNAL *pThis = hVfsDir;
2632 AssertPtrReturn(pThis, UINT32_MAX);
2633 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, UINT32_MAX);
2634 return rtVfsObjRetainDebug(&pThis->Base, "RTVfsDirRetain", RT_SRC_POS_ARGS);
2635}
2636
2637
2638RTDECL(uint32_t) RTVfsDirRelease(RTVFSDIR hVfsDir)
2639{
2640 RTVFSDIRINTERNAL *pThis = hVfsDir;
2641 if (pThis == NIL_RTVFSDIR)
2642 return 0;
2643 AssertPtrReturn(pThis, UINT32_MAX);
2644 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, UINT32_MAX);
2645#ifdef LOG_ENABLED
2646 void *pvThis = pThis->Base.pvThis;
2647#endif
2648 uint32_t cRefs = rtVfsObjRelease(&pThis->Base);
2649 LogFlow(("RTVfsDirRelease(%p/%p) -> %#x\n", pThis, pvThis, cRefs));
2650 return cRefs;
2651}
2652
2653
2654RTDECL(int) RTVfsDirOpen(RTVFS hVfs, const char *pszPath, uint32_t fFlags, PRTVFSDIR phVfsDir)
2655{
2656 /*
2657 * Validate input.
2658 */
2659 RTVFSINTERNAL *pThis = hVfs;
2660 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2661 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
2662 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2663 AssertPtrReturn(phVfsDir, VERR_INVALID_POINTER);
2664 AssertReturn(!fFlags, VERR_INVALID_FLAGS); /** @todo sort out flags! */
2665
2666 /*
2667 * Parse the path, assume current directory is root since we've got no
2668 * caller context here.
2669 */
2670 PRTVFSPARSEDPATH pPath;
2671 int rc = RTVfsParsePathA(pszPath, "/", &pPath);
2672 if (RT_SUCCESS(rc))
2673 {
2674 /*
2675 * Tranverse the path, resolving the parent node.
2676 * We'll do the symbolic link checking here with help of pfnOpen/pfnOpenDir.
2677 */
2678 RTVFSDIRINTERNAL *pVfsParentDir;
2679 rc = rtVfsTraverseToParent(pThis, pPath, (fFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK, &pVfsParentDir);
2680 if (RT_SUCCESS(rc))
2681 {
2682 /*
2683 * Do the opening. Loop if we need to follow symbolic links.
2684 */
2685 uint64_t fOpenFlags = RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN;
2686 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_DIRECTORY | RTVFSOBJ_F_OPEN_SYMLINK | RTVFSOBJ_F_CREATE_NOTHING;
2687 for (uint32_t cLoops = 1; ; cLoops++)
2688 {
2689 /* Do the querying. If pfnOpenDir is available, we use it first, falling
2690 back on pfnOpen in case of symbolic links that needs following. */
2691 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
2692 if (pVfsParentDir->pOps->pfnOpenDir)
2693 {
2694 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
2695 rc = pVfsParentDir->pOps->pfnOpenDir(pVfsParentDir->Base.pvThis, pszEntryName, fFlags, phVfsDir);
2696 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
2697 if ( RT_SUCCESS(rc)
2698 || ( rc != VERR_NOT_A_DIRECTORY
2699 && rc != VERR_IS_A_SYMLINK))
2700 break;
2701 }
2702
2703 RTVFSOBJ hVfsObj;
2704 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
2705 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fOpenFlags, fObjFlags, &hVfsObj);
2706 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
2707 if (RT_FAILURE(rc))
2708 break;
2709
2710 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
2711 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
2712 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
2713 {
2714 *phVfsDir = RTVfsObjToDir(hVfsObj);
2715 AssertStmt(*phVfsDir != NIL_RTVFSDIR, rc = VERR_INTERNAL_ERROR_3);
2716 RTVfsObjRelease(hVfsObj);
2717 break;
2718 }
2719
2720 /* Follow symbolic link. */
2721 if (cLoops < RTVFS_MAX_LINKS)
2722 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fObjFlags & RTPATH_F_MASK);
2723 else
2724 rc = VERR_TOO_MANY_SYMLINKS;
2725 RTVfsObjRelease(hVfsObj);
2726 if (RT_FAILURE(rc))
2727 break;
2728 }
2729 RTVfsDirRelease(pVfsParentDir);
2730 }
2731 RTVfsParsePathFree(pPath);
2732 }
2733 return rc;
2734}
2735
2736
2737RTDECL(int) RTVfsDirOpenDir(RTVFSDIR hVfsDir, const char *pszPath, uint32_t fFlags, PRTVFSDIR phVfsDir)
2738{
2739 /*
2740 * Validate input.
2741 */
2742 RTVFSDIRINTERNAL *pThis = hVfsDir;
2743 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2744 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
2745 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2746 AssertPtrReturn(phVfsDir, VERR_INVALID_POINTER);
2747 AssertReturn(!fFlags, VERR_INVALID_FLAGS); /** @todo sort out flags! */
2748
2749 /*
2750 * Parse the path, it's always relative to the given directory.
2751 */
2752 PRTVFSPARSEDPATH pPath;
2753 int rc = RTVfsParsePathA(pszPath, NULL, &pPath);
2754 if (RT_SUCCESS(rc))
2755 {
2756 /*
2757 * Tranverse the path, resolving the parent node.
2758 * We'll do the symbolic link checking here with help of pfnOpen/pfnOpenDir.
2759 */
2760 RTVFSDIRINTERNAL *pVfsParentDir;
2761 uint32_t const fTraverse = (fFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK;
2762 rc = rtVfsDirTraverseToParent(pThis, pPath, fTraverse, &pVfsParentDir);
2763 if (RT_SUCCESS(rc))
2764 {
2765 /*
2766 * Do the opening. Loop if we need to follow symbolic links.
2767 */
2768 uint64_t fOpenFlags = RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN;
2769 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_DIRECTORY | RTVFSOBJ_F_OPEN_SYMLINK | RTVFSOBJ_F_CREATE_NOTHING | fTraverse;
2770 for (uint32_t cLoops = 1; ; cLoops++)
2771 {
2772 /* Do the querying. If pfnOpenDir is available, we use it first, falling
2773 back on pfnOpen in case of symbolic links that needs following. */
2774 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
2775 if (pVfsParentDir->pOps->pfnOpenDir)
2776 {
2777 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
2778 rc = pVfsParentDir->pOps->pfnOpenDir(pVfsParentDir->Base.pvThis, pszEntryName, fFlags, phVfsDir);
2779 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
2780 if ( RT_SUCCESS(rc)
2781 || ( rc != VERR_NOT_A_DIRECTORY
2782 && rc != VERR_IS_A_SYMLINK))
2783 break;
2784 }
2785
2786 RTVFSOBJ hVfsObj;
2787 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
2788 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fOpenFlags, fObjFlags, &hVfsObj);
2789 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
2790 if (RT_FAILURE(rc))
2791 break;
2792
2793 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
2794 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
2795 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
2796 {
2797 *phVfsDir = RTVfsObjToDir(hVfsObj);
2798 AssertStmt(*phVfsDir != NIL_RTVFSDIR, rc = VERR_INTERNAL_ERROR_3);
2799 RTVfsObjRelease(hVfsObj);
2800 break;
2801 }
2802
2803 /* Follow symbolic link. */
2804 if (cLoops < RTVFS_MAX_LINKS)
2805 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fTraverse);
2806 else
2807 rc = VERR_TOO_MANY_SYMLINKS;
2808 RTVfsObjRelease(hVfsObj);
2809 if (RT_FAILURE(rc))
2810 break;
2811 }
2812 RTVfsDirRelease(pVfsParentDir);
2813 }
2814 RTVfsParsePathFree(pPath);
2815 }
2816 return rc;
2817}
2818
2819
2820RTDECL(int) RTVfsDirCreateDir(RTVFSDIR hVfsDir, const char *pszRelPath, RTFMODE fMode, uint32_t fFlags, PRTVFSDIR phVfsDir)
2821{
2822 /*
2823 * Validate input.
2824 */
2825 RTVFSDIRINTERNAL *pThis = hVfsDir;
2826 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2827 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
2828 AssertPtrReturn(pszRelPath, VERR_INVALID_POINTER);
2829 AssertPtrNullReturn(phVfsDir, VERR_INVALID_POINTER);
2830 AssertReturn(!(fFlags & ~RTDIRCREATE_FLAGS_VALID_MASK), VERR_INVALID_FLAGS);
2831 fMode = rtFsModeNormalize(fMode, pszRelPath, 0, RTFS_TYPE_DIRECTORY);
2832 AssertReturn(rtFsModeIsValidPermissions(fMode), VERR_INVALID_FMODE);
2833 if (!(fFlags & RTDIRCREATE_FLAGS_NOT_CONTENT_INDEXED_DONT_SET))
2834 fMode |= RTFS_DOS_NT_NOT_CONTENT_INDEXED;
2835
2836 /*
2837 * Parse the path, it's always relative to the given directory.
2838 */
2839 PRTVFSPARSEDPATH pPath;
2840 int rc = RTVfsParsePathA(pszRelPath, NULL, &pPath);
2841 if (RT_SUCCESS(rc))
2842 {
2843 /*
2844 * Tranverse the path, resolving the parent node.
2845 * We'll do the symbolic link checking here with help of pfnOpen/pfnOpenDir.
2846 */
2847 RTVFSDIRINTERNAL *pVfsParentDir;
2848 uint32_t fTraverse = (fFlags & RTDIRCREATE_FLAGS_NO_SYMLINKS ? RTPATH_F_NO_SYMLINKS : 0) | RTPATH_F_ON_LINK;
2849 rc = rtVfsDirTraverseToParent(pThis, pPath, fTraverse, &pVfsParentDir);
2850 if (RT_SUCCESS(rc))
2851 {
2852 /*
2853 * Do the opening. Loop if we need to follow symbolic links.
2854 */
2855 uint64_t fOpenFlags = RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_CREATE
2856 | ((fMode << RTFILE_O_CREATE_MODE_SHIFT) & RTFILE_O_CREATE_MODE_MASK);
2857 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_SYMLINK | RTVFSOBJ_F_CREATE_DIRECTORY | fTraverse;
2858 for (uint32_t cLoops = 1; ; cLoops++)
2859 {
2860 /* Do the querying. If pfnOpenDir is available, we use it first, falling
2861 back on pfnOpen in case of symbolic links that needs following. */
2862 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
2863 if (pVfsParentDir->pOps->pfnCreateDir)
2864 {
2865 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
2866 rc = pVfsParentDir->pOps->pfnCreateDir(pVfsParentDir->Base.pvThis, pszEntryName, fMode, phVfsDir);
2867 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
2868 if ( RT_SUCCESS(rc)
2869 || ( rc != VERR_NOT_A_DIRECTORY
2870 && rc != VERR_IS_A_SYMLINK))
2871 break;
2872 }
2873
2874 RTVFSOBJ hVfsObj;
2875 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
2876 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fOpenFlags, fObjFlags, &hVfsObj);
2877 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
2878 if (RT_FAILURE(rc))
2879 break;
2880
2881 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
2882 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
2883 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
2884 {
2885 if (phVfsDir)
2886 {
2887 *phVfsDir = RTVfsObjToDir(hVfsObj);
2888 AssertStmt(*phVfsDir != NIL_RTVFSDIR, rc = VERR_INTERNAL_ERROR_3);
2889 }
2890 RTVfsObjRelease(hVfsObj);
2891 break;
2892 }
2893
2894 /* Follow symbolic link. */
2895 if (cLoops < RTVFS_MAX_LINKS)
2896 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fTraverse);
2897 else
2898 rc = VERR_TOO_MANY_SYMLINKS;
2899 RTVfsObjRelease(hVfsObj);
2900 if (RT_FAILURE(rc))
2901 break;
2902 }
2903 RTVfsDirRelease(pVfsParentDir);
2904 }
2905 RTVfsParsePathFree(pPath);
2906 }
2907 return rc;
2908}
2909
2910
2911RTDECL(int) RTVfsDirOpenFile(RTVFSDIR hVfsDir, const char *pszPath, uint64_t fOpen, PRTVFSFILE phVfsFile)
2912{
2913 /*
2914 * Validate input.
2915 */
2916 RTVFSDIRINTERNAL *pThis = hVfsDir;
2917 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2918 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
2919 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
2920 AssertPtrReturn(phVfsFile, VERR_INVALID_POINTER);
2921
2922 int rc = rtFileRecalcAndValidateFlags(&fOpen);
2923 if (RT_FAILURE(rc))
2924 return rc;
2925
2926 /*
2927 * Parse the path, it's always relative to the given directory.
2928 */
2929 PRTVFSPARSEDPATH pPath;
2930 rc = RTVfsParsePathA(pszPath, NULL, &pPath);
2931 if (RT_SUCCESS(rc))
2932 {
2933 /*
2934 * Tranverse the path, resolving the parent node.
2935 * We'll do the symbolic link checking here with help of pfnOpen/pfnOpenFile.
2936 */
2937 RTVFSDIRINTERNAL *pVfsParentDir;
2938 uint32_t const fTraverse = (fOpen & RTFILE_O_NO_SYMLINKS ? RTPATH_F_NO_SYMLINKS : 0) | RTPATH_F_ON_LINK;
2939 rc = rtVfsDirTraverseToParent(pThis, pPath, fTraverse, &pVfsParentDir);
2940 if (RT_SUCCESS(rc))
2941 {
2942 /** @todo join path with RTVfsFileOpen. */
2943
2944 /*
2945 * Do the opening. Loop if we need to follow symbolic links.
2946 */
2947 bool fDirSlash = pPath->fDirSlash;
2948
2949 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_ANY_FILE | RTVFSOBJ_F_OPEN_SYMLINK;
2950 if ( (fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_CREATE
2951 || (fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_CREATE_REPLACE)
2952 fObjFlags |= RTVFSOBJ_F_CREATE_FILE;
2953 else
2954 fObjFlags |= RTVFSOBJ_F_CREATE_NOTHING;
2955 fObjFlags |= fTraverse & RTPATH_F_MASK;
2956
2957 for (uint32_t cLoops = 1;; cLoops++)
2958 {
2959 /* Do the querying. If pfnOpenFile is available, we use it first, falling
2960 back on pfnOpen in case of symbolic links that needs following or we got
2961 a trailing directory slash (to get file-not-found error). */
2962 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
2963 if ( pVfsParentDir->pOps->pfnOpenFile
2964 && !fDirSlash)
2965 {
2966 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
2967 rc = pVfsParentDir->pOps->pfnOpenFile(pVfsParentDir->Base.pvThis, pszEntryName, fOpen, phVfsFile);
2968 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
2969 if ( RT_SUCCESS(rc)
2970 || ( rc != VERR_NOT_A_FILE
2971 && rc != VERR_IS_A_SYMLINK))
2972 break;
2973 }
2974
2975 RTVFSOBJ hVfsObj;
2976 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
2977 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fOpen, fObjFlags, &hVfsObj);
2978 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
2979 if (RT_FAILURE(rc))
2980 break;
2981
2982 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
2983 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
2984 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
2985 {
2986 *phVfsFile = RTVfsObjToFile(hVfsObj);
2987 AssertStmt(*phVfsFile != NIL_RTVFSFILE, rc = VERR_INTERNAL_ERROR_3);
2988 RTVfsObjRelease(hVfsObj);
2989 break;
2990 }
2991
2992 /* Follow symbolic link. */
2993 if (cLoops < RTVFS_MAX_LINKS)
2994 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fTraverse);
2995 else
2996 rc = VERR_TOO_MANY_SYMLINKS;
2997 RTVfsObjRelease(hVfsObj);
2998 if (RT_FAILURE(rc))
2999 break;
3000 fDirSlash |= pPath->fDirSlash;
3001 }
3002 RTVfsDirRelease(pVfsParentDir);
3003 }
3004 RTVfsParsePathFree(pPath);
3005 }
3006 return rc;
3007}
3008
3009
3010RTDECL(int) RTVfsDirOpenFileAsIoStream(RTVFSDIR hVfsDir, const char *pszPath, uint64_t fOpen, PRTVFSIOSTREAM phVfsIos)
3011{
3012 RTVFSFILE hVfsFile;
3013 int rc = RTVfsDirOpenFile(hVfsDir, pszPath, fOpen, &hVfsFile);
3014 if (RT_SUCCESS(rc))
3015 {
3016 *phVfsIos = RTVfsFileToIoStream(hVfsFile);
3017 AssertStmt(*phVfsIos != NIL_RTVFSIOSTREAM, rc = VERR_INTERNAL_ERROR_2);
3018 RTVfsFileRelease(hVfsFile);
3019 }
3020 return rc;
3021}
3022
3023
3024RTDECL(int) RTVfsDirOpenObj(RTVFSDIR hVfsDir, const char *pszPath, uint64_t fFileOpen, uint32_t fObjFlags, PRTVFSOBJ phVfsObj)
3025{
3026 /*
3027 * Validate input.
3028 */
3029 RTVFSDIRINTERNAL *pThis = hVfsDir;
3030 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3031 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
3032 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
3033 AssertPtrReturn(phVfsObj, VERR_INVALID_POINTER);
3034
3035 int rc = rtFileRecalcAndValidateFlags(&fFileOpen);
3036 if (RT_FAILURE(rc))
3037 return rc;
3038 AssertMsgReturn( RTPATH_F_IS_VALID(fObjFlags, RTVFSOBJ_F_VALID_MASK)
3039 && (fObjFlags & RTVFSOBJ_F_CREATE_MASK) <= RTVFSOBJ_F_CREATE_DIRECTORY,
3040 ("fObjFlags=%#x\n", fObjFlags),
3041 VERR_INVALID_FLAGS);
3042
3043 /*
3044 * Parse the relative path. If it ends with a directory slash or it boils
3045 * down to an empty path (i.e. re-opening hVfsDir), adjust the flags to only
3046 * open/create directories.
3047 */
3048 PRTVFSPARSEDPATH pPath;
3049 rc = RTVfsParsePathA(pszPath, NULL, &pPath);
3050 if (RT_SUCCESS(rc))
3051 {
3052 /*
3053 * Tranverse the path, resolving the parent node.
3054 * We'll do the symbolic link checking here with help of pfnOpen.
3055 */
3056 RTVFSDIRINTERNAL *pVfsParentDir;
3057 rc = rtVfsDirTraverseToParent(pThis, pPath, (fObjFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK, &pVfsParentDir);
3058 if (RT_SUCCESS(rc))
3059 {
3060 /*
3061 * Do the opening. Loop if we need to follow symbolic links.
3062 */
3063 for (uint32_t cLoops = 1;; cLoops++)
3064 {
3065 /* If we end with a directory slash, adjust open flags. */
3066 if (pPath->fDirSlash)
3067 {
3068 fObjFlags &= ~RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_OPEN_DIRECTORY;
3069 if ((fObjFlags & RTVFSOBJ_F_CREATE_MASK) != RTVFSOBJ_F_CREATE_DIRECTORY)
3070 fObjFlags = (fObjFlags & ~RTVFSOBJ_F_CREATE_MASK) | RTVFSOBJ_F_CREATE_NOTHING;
3071 }
3072 if (fObjFlags & RTPATH_F_FOLLOW_LINK)
3073 fObjFlags |= RTVFSOBJ_F_OPEN_SYMLINK;
3074
3075 /* Open it. */
3076 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
3077 RTVFSOBJ hVfsObj;
3078 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
3079 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fFileOpen, fObjFlags, &hVfsObj);
3080 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
3081 if (RT_FAILURE(rc))
3082 break;
3083
3084 /* We're done if we don't follow links or this wasn't a link. */
3085 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
3086 || RTVfsObjGetType(*phVfsObj) != RTVFSOBJTYPE_SYMLINK)
3087 {
3088 *phVfsObj = hVfsObj;
3089 break;
3090 }
3091
3092 /* Follow symbolic link. */
3093 if (cLoops < RTVFS_MAX_LINKS)
3094 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fObjFlags & RTPATH_F_MASK);
3095 else
3096 rc = VERR_TOO_MANY_SYMLINKS;
3097 RTVfsObjRelease(hVfsObj);
3098 if (RT_FAILURE(rc))
3099 break;
3100 }
3101
3102 RTVfsDirRelease(pVfsParentDir);
3103 }
3104 RTVfsParsePathFree(pPath);
3105 }
3106 return rc;
3107}
3108
3109
3110RTDECL(int) RTVfsDirQueryPathInfo(RTVFSDIR hVfsDir, const char *pszPath, PRTFSOBJINFO pObjInfo,
3111 RTFSOBJATTRADD enmAddAttr, uint32_t fFlags)
3112{
3113 /*
3114 * Validate input.
3115 */
3116 RTVFSDIRINTERNAL *pThis = hVfsDir;
3117 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3118 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
3119 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
3120 AssertReturn(*pszPath, VERR_INVALID_PARAMETER);
3121 AssertPtrReturn(pObjInfo, VERR_INVALID_POINTER);
3122 AssertReturn(enmAddAttr >= RTFSOBJATTRADD_NOTHING && enmAddAttr <= RTFSOBJATTRADD_LAST, VERR_INVALID_PARAMETER);
3123 AssertMsgReturn(RTPATH_F_IS_VALID(fFlags, 0), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
3124
3125 /*
3126 * Parse the relative path. Then traverse to the parent directory.
3127 */
3128 PRTVFSPARSEDPATH pPath;
3129 int rc = RTVfsParsePathA(pszPath, NULL, &pPath);
3130 if (RT_SUCCESS(rc))
3131 {
3132 /*
3133 * Tranverse the path, resolving the parent node.
3134 * We'll do the symbolic link checking here with help of pfnOpen.
3135 */
3136 RTVFSDIRINTERNAL *pVfsParentDir;
3137 rc = rtVfsDirTraverseToParent(pThis, pPath, (fFlags & RTPATH_F_NO_SYMLINKS) | RTPATH_F_ON_LINK, &pVfsParentDir);
3138 if (RT_SUCCESS(rc))
3139 {
3140 /*
3141 * Do the opening. Loop if we need to follow symbolic links.
3142 */
3143 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_CREATE_NOTHING;
3144 for (uint32_t cLoops = 1;; cLoops++)
3145 {
3146 /* If we end with a directory slash, adjust open flags. */
3147 if (pPath->fDirSlash)
3148 {
3149 fObjFlags &= ~RTVFSOBJ_F_OPEN_ANY | RTVFSOBJ_F_OPEN_DIRECTORY;
3150 if ((fObjFlags & RTVFSOBJ_F_CREATE_MASK) != RTVFSOBJ_F_CREATE_DIRECTORY)
3151 fObjFlags = (fObjFlags & ~RTVFSOBJ_F_CREATE_MASK) | RTVFSOBJ_F_CREATE_NOTHING;
3152 }
3153 if (fObjFlags & RTPATH_F_FOLLOW_LINK)
3154 fObjFlags |= RTVFSOBJ_F_OPEN_SYMLINK;
3155
3156 /* Do the querying. If pfnQueryEntryInfo is available, we use it first,
3157 falling back on pfnOpen in case of symbolic links that needs following. */
3158 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
3159 if (pVfsParentDir->pOps->pfnQueryEntryInfo)
3160 {
3161 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
3162 rc = pVfsParentDir->pOps->pfnQueryEntryInfo(pVfsParentDir->Base.pvThis, pszEntryName, pObjInfo, enmAddAttr);
3163 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
3164 if (RT_FAILURE(rc))
3165 break;
3166 if ( !RTFS_IS_SYMLINK(pObjInfo->Attr.fMode)
3167 || !(fFlags & RTPATH_F_FOLLOW_LINK))
3168 {
3169 if ( (fObjFlags & RTVFSOBJ_F_OPEN_MASK) != RTVFSOBJ_F_OPEN_ANY
3170 && !RTFS_IS_DIRECTORY(pObjInfo->Attr.fMode))
3171 rc = VERR_NOT_A_DIRECTORY;
3172 break;
3173 }
3174 }
3175
3176 RTVFSOBJ hVfsObj;
3177 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
3178 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName,
3179 RTFILE_O_ACCESS_ATTR_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN,
3180 fObjFlags, &hVfsObj);
3181 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
3182 if (RT_FAILURE(rc))
3183 break;
3184
3185 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
3186 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
3187 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
3188 {
3189 rc = RTVfsObjQueryInfo(hVfsObj, pObjInfo, enmAddAttr);
3190 RTVfsObjRelease(hVfsObj);
3191 break;
3192 }
3193
3194 /* Follow symbolic link. */
3195 if (cLoops < RTVFS_MAX_LINKS)
3196 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fObjFlags & RTPATH_F_MASK);
3197 else
3198 rc = VERR_TOO_MANY_SYMLINKS;
3199 RTVfsObjRelease(hVfsObj);
3200 if (RT_FAILURE(rc))
3201 break;
3202 }
3203
3204 RTVfsDirRelease(pVfsParentDir);
3205 }
3206 RTVfsParsePathFree(pPath);
3207 }
3208 return rc;
3209}
3210
3211
3212RTDECL(int) RTVfsDirRemoveDir(RTVFSDIR hVfsDir, const char *pszRelPath, uint32_t fFlags)
3213{
3214 /*
3215 * Validate input.
3216 */
3217 RTVFSDIRINTERNAL *pThis = hVfsDir;
3218 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3219 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
3220 AssertPtrReturn(pszRelPath, VERR_INVALID_POINTER);
3221 AssertReturn(!fFlags, VERR_INVALID_FLAGS);
3222
3223 /*
3224 * Parse the path, it's always relative to the given directory.
3225 */
3226 PRTVFSPARSEDPATH pPath;
3227 int rc = RTVfsParsePathA(pszRelPath, NULL, &pPath);
3228 if (RT_SUCCESS(rc))
3229 {
3230 if (pPath->cComponents > 0)
3231 {
3232 /*
3233 * Tranverse the path, resolving the parent node, not checking for symbolic
3234 * links in the final element, and ask the directory to remove the subdir.
3235 */
3236 RTVFSDIRINTERNAL *pVfsParentDir;
3237 rc = rtVfsDirTraverseToParent(pThis, pPath, RTPATH_F_ON_LINK, &pVfsParentDir);
3238 if (RT_SUCCESS(rc))
3239 {
3240 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
3241
3242 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
3243 rc = pVfsParentDir->pOps->pfnUnlinkEntry(pVfsParentDir->Base.pvThis, pszEntryName, RTFS_TYPE_DIRECTORY);
3244 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
3245
3246 RTVfsDirRelease(pVfsParentDir);
3247 }
3248 }
3249 else
3250 rc = VERR_PATH_ZERO_LENGTH;
3251 RTVfsParsePathFree(pPath);
3252 }
3253 return rc;
3254}
3255
3256
3257
3258RTDECL(int) RTVfsDirReadEx(RTVFSDIR hVfsDir, PRTDIRENTRYEX pDirEntry, size_t *pcbDirEntry, RTFSOBJATTRADD enmAddAttr)
3259{
3260 /*
3261 * Validate input.
3262 */
3263 RTVFSDIRINTERNAL *pThis = hVfsDir;
3264 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3265 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
3266 AssertPtrReturn(pDirEntry, VERR_INVALID_POINTER);
3267 AssertReturn(enmAddAttr >= RTFSOBJATTRADD_NOTHING && enmAddAttr <= RTFSOBJATTRADD_LAST, VERR_INVALID_PARAMETER);
3268
3269 size_t cbDirEntry = sizeof(*pDirEntry);
3270 if (!pcbDirEntry)
3271 pcbDirEntry = &cbDirEntry;
3272 else
3273 {
3274 cbDirEntry = *pcbDirEntry;
3275 AssertMsgReturn(cbDirEntry >= RT_UOFFSETOF(RTDIRENTRYEX, szName[2]),
3276 ("Invalid *pcbDirEntry=%d (min %zu)\n", *pcbDirEntry, RT_UOFFSETOF(RTDIRENTRYEX, szName[2])),
3277 VERR_INVALID_PARAMETER);
3278 }
3279
3280 /*
3281 * Call the directory method.
3282 */
3283 RTVfsLockAcquireRead(pThis->Base.hLock);
3284 int rc = pThis->pOps->pfnReadDir(pThis->Base.pvThis, pDirEntry, pcbDirEntry, enmAddAttr);
3285 RTVfsLockReleaseRead(pThis->Base.hLock);
3286 return rc;
3287}
3288
3289
3290RTDECL(int) RTVfsDirRewind(RTVFSDIR hVfsDir)
3291{
3292 /*
3293 * Validate input.
3294 */
3295 RTVFSDIRINTERNAL *pThis = hVfsDir;
3296 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3297 AssertReturn(pThis->uMagic == RTVFSDIR_MAGIC, VERR_INVALID_HANDLE);
3298
3299 /*
3300 * Call the directory method.
3301 */
3302 RTVfsLockAcquireRead(pThis->Base.hLock);
3303 int rc = pThis->pOps->pfnRewindDir(pThis->Base.pvThis);
3304 RTVfsLockReleaseRead(pThis->Base.hLock);
3305 return rc;
3306}
3307
3308
3309/*
3310 *
3311 * S Y M B O L I C L I N K
3312 * S Y M B O L I C L I N K
3313 * S Y M B O L I C L I N K
3314 *
3315 */
3316
3317RTDECL(int) RTVfsNewSymlink(PCRTVFSSYMLINKOPS pSymlinkOps, size_t cbInstance, RTVFS hVfs, RTVFSLOCK hLock,
3318 PRTVFSSYMLINK phVfsSym, void **ppvInstance)
3319{
3320 /*
3321 * Validate the input, be extra strict in strict builds.
3322 */
3323 AssertPtr(pSymlinkOps);
3324 AssertReturn(pSymlinkOps->uVersion == RTVFSSYMLINKOPS_VERSION, VERR_VERSION_MISMATCH);
3325 AssertReturn(pSymlinkOps->uEndMarker == RTVFSSYMLINKOPS_VERSION, VERR_VERSION_MISMATCH);
3326 Assert(!pSymlinkOps->fReserved);
3327 RTVFSSYMLINK_ASSERT_OPS(pSymlinkOps, RTVFSOBJTYPE_SYMLINK);
3328 Assert(cbInstance > 0);
3329 AssertPtr(ppvInstance);
3330 AssertPtr(phVfsSym);
3331 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
3332
3333 /*
3334 * Allocate the handle + instance data.
3335 */
3336 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSSYMLINKINTERNAL), RTVFS_INST_ALIGNMENT)
3337 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
3338 RTVFSSYMLINKINTERNAL *pThis = (RTVFSSYMLINKINTERNAL *)RTMemAllocZ(cbThis);
3339 if (!pThis)
3340 return VERR_NO_MEMORY;
3341
3342 int rc = rtVfsObjInitNewObject(&pThis->Base, &pSymlinkOps->Obj, hVfs, false /*fNoVfsRef*/, hLock,
3343 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
3344 if (RT_FAILURE(rc))
3345 {
3346 RTMemFree(pThis);
3347 return rc;
3348 }
3349
3350 pThis->uMagic = RTVFSSYMLINK_MAGIC;
3351 pThis->pOps = pSymlinkOps;
3352
3353 *phVfsSym = pThis;
3354 *ppvInstance = pThis->Base.pvThis;
3355 return VINF_SUCCESS;
3356}
3357
3358
3359RTDECL(void *) RTVfsSymlinkToPrivate(RTVFSSYMLINK hVfsSym, PCRTVFSSYMLINKOPS pSymlinkOps)
3360{
3361 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3362 AssertPtrReturn(pThis, NULL);
3363 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, NULL);
3364 if (pThis->pOps != pSymlinkOps)
3365 return NULL;
3366 return pThis->Base.pvThis;
3367}
3368
3369
3370RTDECL(uint32_t) RTVfsSymlinkRetain(RTVFSSYMLINK hVfsSym)
3371{
3372 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3373 AssertPtrReturn(pThis, UINT32_MAX);
3374 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, UINT32_MAX);
3375 return rtVfsObjRetain(&pThis->Base);
3376}
3377
3378
3379RTDECL(uint32_t) RTVfsSymlinkRetainDebug(RTVFSSYMLINK hVfsSym, RT_SRC_POS_DECL)
3380{
3381 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3382 AssertPtrReturn(pThis, UINT32_MAX);
3383 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, UINT32_MAX);
3384 return rtVfsObjRetainDebug(&pThis->Base, "RTVfsSymlinkRetainDebug", RT_SRC_POS_ARGS);
3385}
3386
3387
3388RTDECL(uint32_t) RTVfsSymlinkRelease(RTVFSSYMLINK hVfsSym)
3389{
3390 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3391 if (pThis == NIL_RTVFSSYMLINK)
3392 return 0;
3393 AssertPtrReturn(pThis, UINT32_MAX);
3394 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, UINT32_MAX);
3395 return rtVfsObjRelease(&pThis->Base);
3396}
3397
3398
3399RTDECL(int) RTVfsSymlinkQueryInfo(RTVFSSYMLINK hVfsSym, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
3400{
3401 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3402 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3403 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, VERR_INVALID_HANDLE);
3404 return RTVfsObjQueryInfo(&pThis->Base, pObjInfo, enmAddAttr);
3405}
3406
3407
3408RTDECL(int) RTVfsSymlinkSetMode(RTVFSSYMLINK hVfsSym, RTFMODE fMode, RTFMODE fMask)
3409{
3410 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3411 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3412 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, VERR_INVALID_HANDLE);
3413
3414 fMode = rtFsModeNormalize(fMode, NULL, 0, RTFS_TYPE_SYMLINK);
3415 if (!rtFsModeIsValid(fMode))
3416 return VERR_INVALID_PARAMETER;
3417
3418 RTVfsLockAcquireWrite(pThis->Base.hLock);
3419 int rc = pThis->pOps->ObjSet.pfnSetMode(pThis->Base.pvThis, fMode, fMask);
3420 RTVfsLockReleaseWrite(pThis->Base.hLock);
3421 return rc;
3422}
3423
3424
3425RTDECL(int) RTVfsSymlinkSetTimes(RTVFSSYMLINK hVfsSym, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
3426 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
3427{
3428 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3429 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3430 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, VERR_INVALID_HANDLE);
3431
3432 AssertPtrNullReturn(pAccessTime, VERR_INVALID_POINTER);
3433 AssertPtrNullReturn(pModificationTime, VERR_INVALID_POINTER);
3434 AssertPtrNullReturn(pChangeTime, VERR_INVALID_POINTER);
3435 AssertPtrNullReturn(pBirthTime, VERR_INVALID_POINTER);
3436
3437 RTVfsLockAcquireWrite(pThis->Base.hLock);
3438 int rc = pThis->pOps->ObjSet.pfnSetTimes(pThis->Base.pvThis, pAccessTime, pModificationTime, pChangeTime, pBirthTime);
3439 RTVfsLockReleaseWrite(pThis->Base.hLock);
3440 return rc;
3441}
3442
3443
3444RTDECL(int) RTVfsSymlinkSetOwner(RTVFSSYMLINK hVfsSym, RTUID uid, RTGID gid)
3445{
3446 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3447 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3448 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, VERR_INVALID_HANDLE);
3449
3450 RTVfsLockAcquireWrite(pThis->Base.hLock);
3451 int rc = pThis->pOps->ObjSet.pfnSetOwner(pThis->Base.pvThis, uid, gid);
3452 RTVfsLockReleaseWrite(pThis->Base.hLock);
3453 return rc;
3454}
3455
3456
3457RTDECL(int) RTVfsSymlinkRead(RTVFSSYMLINK hVfsSym, char *pszTarget, size_t cbTarget)
3458{
3459 RTVFSSYMLINKINTERNAL *pThis = hVfsSym;
3460 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3461 AssertReturn(pThis->uMagic == RTVFSSYMLINK_MAGIC, VERR_INVALID_HANDLE);
3462
3463 RTVfsLockAcquireWrite(pThis->Base.hLock);
3464 int rc = pThis->pOps->pfnRead(pThis->Base.pvThis, pszTarget, cbTarget);
3465 RTVfsLockReleaseWrite(pThis->Base.hLock);
3466
3467 return rc;
3468}
3469
3470
3471
3472/*
3473 *
3474 * I / O S T R E A M I / O S T R E A M I / O S T R E A M
3475 * I / O S T R E A M I / O S T R E A M I / O S T R E A M
3476 * I / O S T R E A M I / O S T R E A M I / O S T R E A M
3477 *
3478 */
3479
3480RTDECL(int) RTVfsNewIoStream(PCRTVFSIOSTREAMOPS pIoStreamOps, size_t cbInstance, uint32_t fOpen, RTVFS hVfs, RTVFSLOCK hLock,
3481 PRTVFSIOSTREAM phVfsIos, void **ppvInstance)
3482{
3483 /*
3484 * Validate the input, be extra strict in strict builds.
3485 */
3486 AssertPtr(pIoStreamOps);
3487 AssertReturn(pIoStreamOps->uVersion == RTVFSIOSTREAMOPS_VERSION, VERR_VERSION_MISMATCH);
3488 AssertReturn(pIoStreamOps->uEndMarker == RTVFSIOSTREAMOPS_VERSION, VERR_VERSION_MISMATCH);
3489 Assert(!(pIoStreamOps->fFeatures & ~RTVFSIOSTREAMOPS_FEAT_VALID_MASK));
3490 RTVFSIOSTREAM_ASSERT_OPS(pIoStreamOps, RTVFSOBJTYPE_IO_STREAM);
3491 Assert(cbInstance > 0);
3492 Assert(fOpen & RTFILE_O_ACCESS_MASK);
3493 AssertPtr(ppvInstance);
3494 AssertPtr(phVfsIos);
3495 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
3496
3497 /*
3498 * Allocate the handle + instance data.
3499 */
3500 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSIOSTREAMINTERNAL), RTVFS_INST_ALIGNMENT)
3501 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
3502 RTVFSIOSTREAMINTERNAL *pThis = (RTVFSIOSTREAMINTERNAL *)RTMemAllocZ(cbThis);
3503 if (!pThis)
3504 return VERR_NO_MEMORY;
3505
3506 int rc = rtVfsObjInitNewObject(&pThis->Base, &pIoStreamOps->Obj, hVfs, false /*fNoVfsRef*/, hLock,
3507 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
3508 if (RT_FAILURE(rc))
3509 {
3510 RTMemFree(pThis);
3511 return rc;
3512 }
3513
3514 pThis->uMagic = RTVFSIOSTREAM_MAGIC;
3515 pThis->fFlags = fOpen;
3516 pThis->pOps = pIoStreamOps;
3517
3518 *phVfsIos = pThis;
3519 *ppvInstance = pThis->Base.pvThis;
3520 return VINF_SUCCESS;
3521}
3522
3523
3524RTDECL(void *) RTVfsIoStreamToPrivate(RTVFSIOSTREAM hVfsIos, PCRTVFSIOSTREAMOPS pIoStreamOps)
3525{
3526 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3527 AssertPtrReturn(pThis, NULL);
3528 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, NULL);
3529 if (pThis->pOps != pIoStreamOps)
3530 return NULL;
3531 return pThis->Base.pvThis;
3532}
3533
3534
3535#ifdef DEBUG
3536# undef RTVfsIoStrmRetain
3537#endif
3538RTDECL(uint32_t) RTVfsIoStrmRetain(RTVFSIOSTREAM hVfsIos)
3539{
3540 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3541 AssertPtrReturn(pThis, UINT32_MAX);
3542 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, UINT32_MAX);
3543 return rtVfsObjRetain(&pThis->Base);
3544}
3545#ifdef DEBUG
3546# define RTVfsIoStrmRetain(hVfsIos) RTVfsIoStrmRetainDebug(hVfsIos, RT_SRC_POS)
3547#endif
3548
3549
3550RTDECL(uint32_t) RTVfsIoStrmRetainDebug(RTVFSIOSTREAM hVfsIos, RT_SRC_POS_DECL)
3551{
3552 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3553 AssertPtrReturn(pThis, UINT32_MAX);
3554 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, UINT32_MAX);
3555 return rtVfsObjRetainDebug(&pThis->Base, "RTVfsIoStrmRetainDebug", RT_SRC_POS_ARGS);
3556}
3557
3558
3559RTDECL(uint32_t) RTVfsIoStrmRelease(RTVFSIOSTREAM hVfsIos)
3560{
3561 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3562 if (pThis == NIL_RTVFSIOSTREAM)
3563 return 0;
3564 AssertPtrReturn(pThis, UINT32_MAX);
3565 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, UINT32_MAX);
3566 return rtVfsObjRelease(&pThis->Base);
3567}
3568
3569
3570RTDECL(RTVFSFILE) RTVfsIoStrmToFile(RTVFSIOSTREAM hVfsIos)
3571{
3572 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3573 AssertPtrReturn(pThis, NIL_RTVFSFILE);
3574 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, NIL_RTVFSFILE);
3575
3576 if (pThis->pOps->Obj.enmType == RTVFSOBJTYPE_FILE)
3577 {
3578 rtVfsObjRetainVoid(&pThis->Base, "RTVfsIoStrmToFile");
3579 return RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream);
3580 }
3581
3582 /* this is no crime, so don't assert. */
3583 return NIL_RTVFSFILE;
3584}
3585
3586
3587RTDECL(int) RTVfsIoStrmQueryInfo(RTVFSIOSTREAM hVfsIos, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
3588{
3589 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3590 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3591 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3592 return RTVfsObjQueryInfo(&pThis->Base, pObjInfo, enmAddAttr);
3593}
3594
3595
3596RTDECL(int) RTVfsIoStrmRead(RTVFSIOSTREAM hVfsIos, void *pvBuf, size_t cbToRead, bool fBlocking, size_t *pcbRead)
3597{
3598 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
3599 if (pcbRead)
3600 *pcbRead = 0;
3601 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3602 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3603 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3604 AssertReturn(fBlocking || pcbRead, VERR_INVALID_PARAMETER);
3605 AssertReturn(pThis->fFlags & RTFILE_O_READ, VERR_ACCESS_DENIED);
3606
3607 RTSGSEG Seg = { pvBuf, cbToRead };
3608 RTSGBUF SgBuf;
3609 RTSgBufInit(&SgBuf, &Seg, 1);
3610
3611 RTVfsLockAcquireWrite(pThis->Base.hLock);
3612 int rc = pThis->pOps->pfnRead(pThis->Base.pvThis, -1 /*off*/, &SgBuf, fBlocking, pcbRead);
3613 RTVfsLockReleaseWrite(pThis->Base.hLock);
3614 return rc;
3615}
3616
3617
3618RTDECL(int) RTVfsIoStrmReadAt(RTVFSIOSTREAM hVfsIos, RTFOFF off, void *pvBuf, size_t cbToRead,
3619 bool fBlocking, size_t *pcbRead)
3620{
3621 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
3622 if (pcbRead)
3623 *pcbRead = 0;
3624 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3625 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3626 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3627 AssertReturn(fBlocking || pcbRead, VERR_INVALID_PARAMETER);
3628 AssertReturn(pThis->fFlags & RTFILE_O_READ, VERR_ACCESS_DENIED);
3629
3630 RTSGSEG Seg = { pvBuf, cbToRead };
3631 RTSGBUF SgBuf;
3632 RTSgBufInit(&SgBuf, &Seg, 1);
3633
3634 RTVfsLockAcquireWrite(pThis->Base.hLock);
3635 int rc = pThis->pOps->pfnRead(pThis->Base.pvThis, off, &SgBuf, fBlocking, pcbRead);
3636 RTVfsLockReleaseWrite(pThis->Base.hLock);
3637 return rc;
3638}
3639
3640
3641RTDECL(int) RTVfsIoStrmWrite(RTVFSIOSTREAM hVfsIos, const void *pvBuf, size_t cbToWrite, bool fBlocking, size_t *pcbWritten)
3642{
3643 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
3644 if (pcbWritten)
3645 *pcbWritten = 0;
3646 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3647 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3648 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3649 AssertReturn(fBlocking || pcbWritten, VERR_INVALID_PARAMETER);
3650 AssertReturn(pThis->fFlags & RTFILE_O_WRITE, VERR_ACCESS_DENIED);
3651
3652 int rc;
3653 if (pThis->pOps->pfnWrite)
3654 {
3655 RTSGSEG Seg = { (void *)pvBuf, cbToWrite };
3656 RTSGBUF SgBuf;
3657 RTSgBufInit(&SgBuf, &Seg, 1);
3658
3659 RTVfsLockAcquireWrite(pThis->Base.hLock);
3660 rc = pThis->pOps->pfnWrite(pThis->Base.pvThis, -1 /*off*/, &SgBuf, fBlocking, pcbWritten);
3661 RTVfsLockReleaseWrite(pThis->Base.hLock);
3662 }
3663 else
3664 rc = VERR_WRITE_PROTECT;
3665 return rc;
3666}
3667
3668
3669RTDECL(int) RTVfsIoStrmWriteAt(RTVFSIOSTREAM hVfsIos, RTFOFF off, const void *pvBuf, size_t cbToWrite,
3670 bool fBlocking, size_t *pcbWritten)
3671{
3672 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
3673 if (pcbWritten)
3674 *pcbWritten = 0;
3675 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3676 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3677 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3678 AssertReturn(fBlocking || pcbWritten, VERR_INVALID_PARAMETER);
3679 AssertReturn(pThis->fFlags & RTFILE_O_WRITE, VERR_ACCESS_DENIED);
3680
3681 int rc;
3682 if (pThis->pOps->pfnWrite)
3683 {
3684 RTSGSEG Seg = { (void *)pvBuf, cbToWrite };
3685 RTSGBUF SgBuf;
3686 RTSgBufInit(&SgBuf, &Seg, 1);
3687
3688 RTVfsLockAcquireWrite(pThis->Base.hLock);
3689 rc = pThis->pOps->pfnWrite(pThis->Base.pvThis, off, &SgBuf, fBlocking, pcbWritten);
3690 RTVfsLockReleaseWrite(pThis->Base.hLock);
3691 }
3692 else
3693 rc = VERR_WRITE_PROTECT;
3694 return rc;
3695}
3696
3697
3698RTDECL(int) RTVfsIoStrmSgRead(RTVFSIOSTREAM hVfsIos, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbRead)
3699{
3700 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
3701 if (pcbRead)
3702 *pcbRead = 0;
3703 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3704 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3705 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3706 AssertPtr(pSgBuf);
3707 AssertReturn(fBlocking || pcbRead, VERR_INVALID_PARAMETER);
3708 AssertReturn(pThis->fFlags & RTFILE_O_READ, VERR_ACCESS_DENIED);
3709
3710 RTVfsLockAcquireWrite(pThis->Base.hLock);
3711 int rc;
3712 if (!(pThis->pOps->fFeatures & RTVFSIOSTREAMOPS_FEAT_NO_SG))
3713 rc = pThis->pOps->pfnRead(pThis->Base.pvThis, off, pSgBuf, fBlocking, pcbRead);
3714 else
3715 {
3716 size_t cbRead = 0;
3717 rc = VINF_SUCCESS;
3718
3719 for (uint32_t iSeg = 0; iSeg < pSgBuf->cSegs; iSeg++)
3720 {
3721 RTSGBUF SgBuf;
3722 RTSgBufInit(&SgBuf, &pSgBuf->paSegs[iSeg], 1);
3723
3724 size_t cbReadSeg = pcbRead ? 0 : pSgBuf->paSegs[iSeg].cbSeg;
3725 rc = pThis->pOps->pfnRead(pThis->Base.pvThis, off, &SgBuf, fBlocking, pcbRead ? &cbReadSeg : NULL);
3726 if (RT_FAILURE(rc))
3727 break;
3728 cbRead += cbReadSeg;
3729 if ((pcbRead && cbReadSeg != SgBuf.paSegs[0].cbSeg) || rc != VINF_SUCCESS)
3730 break;
3731 if (off != -1)
3732 off += cbReadSeg;
3733 }
3734
3735 if (pcbRead)
3736 *pcbRead = cbRead;
3737 }
3738 RTVfsLockReleaseWrite(pThis->Base.hLock);
3739 return rc;
3740}
3741
3742
3743RTDECL(int) RTVfsIoStrmSgWrite(RTVFSIOSTREAM hVfsIos, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbWritten)
3744{
3745 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
3746 if (pcbWritten)
3747 *pcbWritten = 0;
3748 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3749 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3750 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3751 AssertPtr(pSgBuf);
3752 AssertReturn(fBlocking || pcbWritten, VERR_INVALID_PARAMETER);
3753 AssertReturn(pThis->fFlags & RTFILE_O_WRITE, VERR_ACCESS_DENIED);
3754
3755 int rc;
3756 if (pThis->pOps->pfnWrite)
3757 {
3758 RTVfsLockAcquireWrite(pThis->Base.hLock);
3759 if (!(pThis->pOps->fFeatures & RTVFSIOSTREAMOPS_FEAT_NO_SG))
3760 rc = pThis->pOps->pfnWrite(pThis->Base.pvThis, off, pSgBuf, fBlocking, pcbWritten);
3761 else
3762 {
3763 size_t cbWritten = 0;
3764 rc = VINF_SUCCESS;
3765
3766 for (uint32_t iSeg = 0; iSeg < pSgBuf->cSegs; iSeg++)
3767 {
3768 RTSGBUF SgBuf;
3769 RTSgBufInit(&SgBuf, &pSgBuf->paSegs[iSeg], 1);
3770
3771 size_t cbWrittenSeg = 0;
3772 rc = pThis->pOps->pfnWrite(pThis->Base.pvThis, off, &SgBuf, fBlocking, pcbWritten ? &cbWrittenSeg : NULL);
3773 if (RT_FAILURE(rc))
3774 break;
3775 if (pcbWritten)
3776 {
3777 cbWritten += cbWrittenSeg;
3778 if (cbWrittenSeg != SgBuf.paSegs[0].cbSeg)
3779 break;
3780 if (off != -1)
3781 off += cbWrittenSeg;
3782 }
3783 else if (off != -1)
3784 off += pSgBuf->paSegs[iSeg].cbSeg;
3785 }
3786
3787 if (pcbWritten)
3788 *pcbWritten = cbWritten;
3789 }
3790 RTVfsLockReleaseWrite(pThis->Base.hLock);
3791 }
3792 else
3793 rc = VERR_WRITE_PROTECT;
3794 return rc;
3795}
3796
3797
3798RTDECL(int) RTVfsIoStrmFlush(RTVFSIOSTREAM hVfsIos)
3799{
3800 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3801 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3802 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3803
3804 RTVfsLockAcquireWrite(pThis->Base.hLock);
3805 int rc = pThis->pOps->pfnFlush(pThis->Base.pvThis);
3806 RTVfsLockReleaseWrite(pThis->Base.hLock);
3807 return rc;
3808}
3809
3810
3811RTDECL(int) RTVfsIoStrmPoll(RTVFSIOSTREAM hVfsIos, uint32_t fEvents, RTMSINTERVAL cMillies, bool fIntr,
3812 uint32_t *pfRetEvents)
3813{
3814 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3815 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
3816 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, VERR_INVALID_HANDLE);
3817
3818 int rc;
3819 if (pThis->pOps->pfnPollOne)
3820 {
3821 RTVfsLockAcquireWrite(pThis->Base.hLock);
3822 rc = pThis->pOps->pfnPollOne(pThis->Base.pvThis, fEvents, cMillies, fIntr, pfRetEvents);
3823 RTVfsLockReleaseWrite(pThis->Base.hLock);
3824 }
3825 /*
3826 * Default implementation. Polling for non-error events returns
3827 * immediately, waiting for errors will work like sleep.
3828 */
3829 else if (fEvents != RTPOLL_EVT_ERROR)
3830 {
3831 *pfRetEvents = fEvents & ~RTPOLL_EVT_ERROR;
3832 rc = VINF_SUCCESS;
3833 }
3834 else if (fIntr)
3835 rc = RTThreadSleep(cMillies);
3836 else
3837 {
3838 uint64_t uMsStart = RTTimeMilliTS();
3839 do
3840 rc = RTThreadSleep(cMillies);
3841 while ( rc == VERR_INTERRUPTED
3842 && !fIntr
3843 && RTTimeMilliTS() - uMsStart < cMillies);
3844 if (rc == VERR_INTERRUPTED)
3845 rc = VERR_TIMEOUT;
3846 }
3847 return rc;
3848}
3849
3850
3851RTDECL(RTFOFF) RTVfsIoStrmTell(RTVFSIOSTREAM hVfsIos)
3852{
3853 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3854 AssertPtrReturn(pThis, -1);
3855 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, -1);
3856
3857 RTFOFF off;
3858 RTVfsLockAcquireRead(pThis->Base.hLock);
3859 int rc = pThis->pOps->pfnTell(pThis->Base.pvThis, &off);
3860 RTVfsLockReleaseRead(pThis->Base.hLock);
3861 if (RT_FAILURE(rc))
3862 off = rc;
3863 return off;
3864}
3865
3866
3867RTDECL(int) RTVfsIoStrmSkip(RTVFSIOSTREAM hVfsIos, RTFOFF cb)
3868{
3869 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3870 AssertPtrReturn(pThis, -1);
3871 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, -1);
3872 AssertReturn(cb >= 0, VERR_INVALID_PARAMETER);
3873
3874 int rc;
3875 if (pThis->pOps->pfnSkip)
3876 {
3877 RTVfsLockAcquireWrite(pThis->Base.hLock);
3878 rc = pThis->pOps->pfnSkip(pThis->Base.pvThis, cb);
3879 RTVfsLockReleaseWrite(pThis->Base.hLock);
3880 }
3881 else if (pThis->pOps->Obj.enmType == RTVFSOBJTYPE_FILE)
3882 {
3883 RTVFSFILEINTERNAL *pThisFile = RT_FROM_MEMBER(pThis, RTVFSFILEINTERNAL, Stream);
3884 RTFOFF offIgnored;
3885
3886 RTVfsLockAcquireWrite(pThis->Base.hLock);
3887 rc = pThisFile->pOps->pfnSeek(pThis->Base.pvThis, cb, RTFILE_SEEK_CURRENT, &offIgnored);
3888 RTVfsLockReleaseWrite(pThis->Base.hLock);
3889 }
3890 else
3891 {
3892 void *pvBuf = RTMemTmpAlloc(_64K);
3893 if (pvBuf)
3894 {
3895 rc = VINF_SUCCESS;
3896 while (cb > 0)
3897 {
3898 size_t cbToRead = (size_t)RT_MIN(cb, _64K);
3899 RTVfsLockAcquireWrite(pThis->Base.hLock);
3900 rc = RTVfsIoStrmRead(hVfsIos, pvBuf, cbToRead, true /*fBlocking*/, NULL);
3901 RTVfsLockReleaseWrite(pThis->Base.hLock);
3902 if (RT_FAILURE(rc))
3903 break;
3904 cb -= cbToRead;
3905 }
3906
3907 RTMemTmpFree(pvBuf);
3908 }
3909 else
3910 rc = VERR_NO_TMP_MEMORY;
3911 }
3912 return rc;
3913}
3914
3915
3916RTDECL(int) RTVfsIoStrmZeroFill(RTVFSIOSTREAM hVfsIos, RTFOFF cb)
3917{
3918 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3919 AssertPtrReturn(pThis, -1);
3920 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, -1);
3921
3922 int rc;
3923 if (pThis->pOps->pfnZeroFill)
3924 {
3925 RTVfsLockAcquireWrite(pThis->Base.hLock);
3926 rc = pThis->pOps->pfnZeroFill(pThis->Base.pvThis, cb);
3927 RTVfsLockReleaseWrite(pThis->Base.hLock);
3928 }
3929 else
3930 {
3931 rc = VINF_SUCCESS;
3932 while (cb > 0)
3933 {
3934 size_t cbToWrite = (size_t)RT_MIN(cb, (ssize_t)sizeof(g_abRTZero64K));
3935 RTVfsLockAcquireWrite(pThis->Base.hLock);
3936 rc = RTVfsIoStrmWrite(hVfsIos, g_abRTZero64K, cbToWrite, true /*fBlocking*/, NULL);
3937 RTVfsLockReleaseWrite(pThis->Base.hLock);
3938 if (RT_FAILURE(rc))
3939 break;
3940 cb -= cbToWrite;
3941 }
3942 }
3943 return rc;
3944}
3945
3946
3947RTDECL(bool) RTVfsIoStrmIsAtEnd(RTVFSIOSTREAM hVfsIos)
3948{
3949 /*
3950 * There is where the zero read behavior comes in handy.
3951 */
3952 char bDummy;
3953 size_t cbRead;
3954 int rc = RTVfsIoStrmRead(hVfsIos, &bDummy, 0 /*cbToRead*/, false /*fBlocking*/, &cbRead);
3955 return rc == VINF_EOF;
3956}
3957
3958
3959
3960RTDECL(uint64_t) RTVfsIoStrmGetOpenFlags(RTVFSIOSTREAM hVfsIos)
3961{
3962 RTVFSIOSTREAMINTERNAL *pThis = hVfsIos;
3963 AssertPtrReturn(pThis, 0);
3964 AssertReturn(pThis->uMagic == RTVFSIOSTREAM_MAGIC, 0);
3965 return pThis->fFlags;
3966}
3967
3968
3969
3970/*
3971 *
3972 * F I L E F I L E F I L E
3973 * F I L E F I L E F I L E
3974 * F I L E F I L E F I L E
3975 *
3976 */
3977
3978RTDECL(int) RTVfsNewFile(PCRTVFSFILEOPS pFileOps, size_t cbInstance, uint32_t fOpen, RTVFS hVfs, RTVFSLOCK hLock,
3979 PRTVFSFILE phVfsFile, void **ppvInstance)
3980{
3981 /*
3982 * Validate the input, be extra strict in strict builds.
3983 */
3984 RTVFSFILE_ASSERT_OPS(pFileOps, RTVFSOBJTYPE_FILE);
3985 Assert(cbInstance > 0);
3986 Assert(fOpen & (RTFILE_O_ACCESS_MASK | RTFILE_O_ACCESS_ATTR_MASK));
3987 AssertPtr(ppvInstance);
3988 AssertPtr(phVfsFile);
3989 RTVFS_ASSERT_VALID_HANDLE_OR_NIL_RETURN(hVfs, VERR_INVALID_HANDLE);
3990
3991 /*
3992 * Allocate the handle + instance data.
3993 */
3994 size_t const cbThis = RT_ALIGN_Z(sizeof(RTVFSFILEINTERNAL), RTVFS_INST_ALIGNMENT)
3995 + RT_ALIGN_Z(cbInstance, RTVFS_INST_ALIGNMENT);
3996 RTVFSFILEINTERNAL *pThis = (RTVFSFILEINTERNAL *)RTMemAllocZ(cbThis);
3997 if (!pThis)
3998 return VERR_NO_MEMORY;
3999
4000 int rc = rtVfsObjInitNewObject(&pThis->Stream.Base, &pFileOps->Stream.Obj, hVfs, false /*fNoVfsRef*/, hLock,
4001 (char *)pThis + RT_ALIGN_Z(sizeof(*pThis), RTVFS_INST_ALIGNMENT));
4002 if (RT_FAILURE(rc))
4003 {
4004 RTMemFree(pThis);
4005 return rc;
4006 }
4007
4008 pThis->uMagic = RTVFSFILE_MAGIC;
4009 pThis->fReserved = 0;
4010 pThis->pOps = pFileOps;
4011 pThis->Stream.uMagic = RTVFSIOSTREAM_MAGIC;
4012 pThis->Stream.fFlags = fOpen;
4013 pThis->Stream.pOps = &pFileOps->Stream;
4014
4015 *phVfsFile = pThis;
4016 *ppvInstance = pThis->Stream.Base.pvThis;
4017 return VINF_SUCCESS;
4018}
4019
4020
4021RTDECL(int) RTVfsFileOpen(RTVFS hVfs, const char *pszFilename, uint64_t fOpen, PRTVFSFILE phVfsFile)
4022{
4023 /*
4024 * Validate input.
4025 */
4026 RTVFSINTERNAL *pThis = hVfs;
4027 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4028 AssertReturn(pThis->uMagic == RTVFS_MAGIC, VERR_INVALID_HANDLE);
4029 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
4030 AssertPtrReturn(phVfsFile, VERR_INVALID_POINTER);
4031
4032 int rc = rtFileRecalcAndValidateFlags(&fOpen);
4033 if (RT_FAILURE(rc))
4034 return rc;
4035
4036 /*
4037 * Parse the path, assume current directory is root since we've got no
4038 * caller context here.
4039 */
4040 PRTVFSPARSEDPATH pPath;
4041 rc = RTVfsParsePathA(pszFilename, "/", &pPath);
4042 if (RT_SUCCESS(rc))
4043 {
4044 /*
4045 * Tranverse the path, resolving the parent node.
4046 * We'll do the symbolic link checking here with help of pfnOpen/pfnOpenFile.
4047 */
4048 RTVFSDIRINTERNAL *pVfsParentDir;
4049 uint32_t const fTraverse = (fOpen & RTFILE_O_NO_SYMLINKS ? RTPATH_F_NO_SYMLINKS : 0) | RTPATH_F_ON_LINK;
4050 rc = rtVfsTraverseToParent(pThis, pPath, fTraverse, &pVfsParentDir);
4051 if (RT_SUCCESS(rc))
4052 {
4053 /** @todo join path with RTVfsDirOpenFile. */
4054 /*
4055 * Do the opening. Loop if we need to follow symbolic links.
4056 */
4057 bool fDirSlash = pPath->fDirSlash;
4058
4059 uint32_t fObjFlags = RTVFSOBJ_F_OPEN_ANY_FILE | RTVFSOBJ_F_OPEN_SYMLINK;
4060 if ( (fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_CREATE
4061 || (fOpen & RTFILE_O_ACTION_MASK) == RTFILE_O_CREATE_REPLACE)
4062 fObjFlags |= RTVFSOBJ_F_CREATE_FILE;
4063 else
4064 fObjFlags |= RTVFSOBJ_F_CREATE_NOTHING;
4065 fObjFlags |= fTraverse & RTPATH_F_MASK;
4066
4067 for (uint32_t cLoops = 1;; cLoops++)
4068 {
4069 /* Do the querying. If pfnOpenFile is available, we use it first, falling
4070 back on pfnOpen in case of symbolic links that needs following or we got
4071 a trailing directory slash (to get file-not-found error). */
4072 const char *pszEntryName = &pPath->szPath[pPath->aoffComponents[pPath->cComponents - 1]];
4073 if ( pVfsParentDir->pOps->pfnOpenFile
4074 && !fDirSlash)
4075 {
4076 RTVfsLockAcquireRead(pVfsParentDir->Base.hLock);
4077 rc = pVfsParentDir->pOps->pfnOpenFile(pVfsParentDir->Base.pvThis, pszEntryName, fOpen, phVfsFile);
4078 RTVfsLockReleaseRead(pVfsParentDir->Base.hLock);
4079 if ( RT_SUCCESS(rc)
4080 || ( rc != VERR_NOT_A_FILE
4081 && rc != VERR_IS_A_SYMLINK))
4082 break;
4083 }
4084
4085 RTVFSOBJ hVfsObj;
4086 RTVfsLockAcquireWrite(pVfsParentDir->Base.hLock);
4087 rc = pVfsParentDir->pOps->pfnOpen(pVfsParentDir->Base.pvThis, pszEntryName, fOpen, fObjFlags, &hVfsObj);
4088 RTVfsLockReleaseWrite(pVfsParentDir->Base.hLock);
4089 if (RT_FAILURE(rc))
4090 break;
4091
4092 /* If we don't follow links or this wasn't a link we just have to do the query and we're done. */
4093 if ( !(fObjFlags & RTPATH_F_FOLLOW_LINK)
4094 || RTVfsObjGetType(hVfsObj) != RTVFSOBJTYPE_SYMLINK)
4095 {
4096 *phVfsFile = RTVfsObjToFile(hVfsObj);
4097 AssertStmt(*phVfsFile != NIL_RTVFSFILE, rc = VERR_INTERNAL_ERROR_3);
4098 RTVfsObjRelease(hVfsObj);
4099 break;
4100 }
4101
4102 /* Follow symbolic link. */
4103 if (cLoops < RTVFS_MAX_LINKS)
4104 rc = rtVfsDirFollowSymlinkObjToParent(&pVfsParentDir, hVfsObj, pPath, fTraverse);
4105 else
4106 rc = VERR_TOO_MANY_SYMLINKS;
4107 RTVfsObjRelease(hVfsObj);
4108 if (RT_FAILURE(rc))
4109 break;
4110 fDirSlash |= pPath->fDirSlash;
4111 }
4112 RTVfsDirRelease(pVfsParentDir);
4113 }
4114 RTVfsParsePathFree(pPath);
4115 }
4116 return rc;
4117
4118}
4119
4120
4121#ifdef DEBUG
4122# undef RTVfsFileRetain
4123#endif
4124RTDECL(uint32_t) RTVfsFileRetain(RTVFSFILE hVfsFile)
4125{
4126 RTVFSFILEINTERNAL *pThis = hVfsFile;
4127 AssertPtrReturn(pThis, UINT32_MAX);
4128 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, UINT32_MAX);
4129 return rtVfsObjRetain(&pThis->Stream.Base);
4130}
4131#ifdef DEBUG
4132# define RTVfsFileRetain(hVfsFile) RTVfsFileRetainDebug(hVfsFile, RT_SRC_POS)
4133#endif
4134
4135
4136RTDECL(uint32_t) RTVfsFileRetainDebug(RTVFSFILE hVfsFile, RT_SRC_POS_DECL)
4137{
4138 RTVFSFILEINTERNAL *pThis = hVfsFile;
4139 AssertPtrReturn(pThis, UINT32_MAX);
4140 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, UINT32_MAX);
4141 return rtVfsObjRetainDebug(&pThis->Stream.Base, "RTVFsFileRetainDebug", RT_SRC_POS_ARGS);
4142}
4143
4144
4145RTDECL(uint32_t) RTVfsFileRelease(RTVFSFILE hVfsFile)
4146{
4147 RTVFSFILEINTERNAL *pThis = hVfsFile;
4148 if (pThis == NIL_RTVFSFILE)
4149 return 0;
4150 AssertPtrReturn(pThis, UINT32_MAX);
4151 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, UINT32_MAX);
4152 return rtVfsObjRelease(&pThis->Stream.Base);
4153}
4154
4155
4156RTDECL(RTVFSIOSTREAM) RTVfsFileToIoStream(RTVFSFILE hVfsFile)
4157{
4158 RTVFSFILEINTERNAL *pThis = hVfsFile;
4159 AssertPtrReturn(pThis, NIL_RTVFSIOSTREAM);
4160 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, NIL_RTVFSIOSTREAM);
4161
4162 rtVfsObjRetainVoid(&pThis->Stream.Base, "RTVfsFileToIoStream");
4163 return &pThis->Stream;
4164}
4165
4166
4167RTDECL(int) RTVfsFileQueryInfo(RTVFSFILE hVfsFile, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
4168{
4169 RTVFSFILEINTERNAL *pThis = hVfsFile;
4170 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4171 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4172 return RTVfsObjQueryInfo(&pThis->Stream.Base, pObjInfo, enmAddAttr);
4173}
4174
4175
4176RTDECL(int) RTVfsFileRead(RTVFSFILE hVfsFile, void *pvBuf, size_t cbToRead, size_t *pcbRead)
4177{
4178 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
4179 if (pcbRead)
4180 *pcbRead = 0;
4181 RTVFSFILEINTERNAL *pThis = hVfsFile;
4182 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4183 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4184 return RTVfsIoStrmRead(&pThis->Stream, pvBuf, cbToRead, true /*fBlocking*/, pcbRead);
4185}
4186
4187
4188RTDECL(int) RTVfsFileWrite(RTVFSFILE hVfsFile, const void *pvBuf, size_t cbToWrite, size_t *pcbWritten)
4189{
4190 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
4191 if (pcbWritten)
4192 *pcbWritten = 0;
4193 RTVFSFILEINTERNAL *pThis = hVfsFile;
4194 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4195 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4196 return RTVfsIoStrmWrite(&pThis->Stream, pvBuf, cbToWrite, true /*fBlocking*/, pcbWritten);
4197}
4198
4199
4200RTDECL(int) RTVfsFileWriteAt(RTVFSFILE hVfsFile, RTFOFF off, const void *pvBuf, size_t cbToWrite, size_t *pcbWritten)
4201{
4202 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
4203 if (pcbWritten)
4204 *pcbWritten = 0;
4205 RTVFSFILEINTERNAL *pThis = hVfsFile;
4206 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4207 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4208
4209 int rc = RTVfsFileSeek(hVfsFile, off, RTFILE_SEEK_BEGIN, NULL);
4210 if (RT_SUCCESS(rc))
4211 rc = RTVfsIoStrmWriteAt(&pThis->Stream, off, pvBuf, cbToWrite, true /*fBlocking*/, pcbWritten);
4212
4213 return rc;
4214}
4215
4216
4217RTDECL(int) RTVfsFileReadAt(RTVFSFILE hVfsFile, RTFOFF off, void *pvBuf, size_t cbToRead, size_t *pcbRead)
4218{
4219 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
4220 if (pcbRead)
4221 *pcbRead = 0;
4222 RTVFSFILEINTERNAL *pThis = hVfsFile;
4223 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4224 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4225
4226 int rc = RTVfsFileSeek(hVfsFile, off, RTFILE_SEEK_BEGIN, NULL);
4227 if (RT_SUCCESS(rc))
4228 rc = RTVfsIoStrmReadAt(&pThis->Stream, off, pvBuf, cbToRead, true /*fBlocking*/, pcbRead);
4229
4230 return rc;
4231}
4232
4233
4234RTDECL(int) RTVfsFileSgRead(RTVFSFILE hVfsFile, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbRead)
4235{
4236 AssertPtrNullReturn(pcbRead, VERR_INVALID_POINTER);
4237 if (pcbRead)
4238 *pcbRead = 0;
4239 RTVFSFILEINTERNAL *pThis = hVfsFile;
4240 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4241 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4242
4243 return RTVfsIoStrmSgRead(&pThis->Stream, off, pSgBuf, fBlocking, pcbRead);
4244}
4245
4246
4247RTDECL(int) RTVfsFileSgWrite(RTVFSFILE hVfsFile, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbWritten)
4248{
4249 AssertPtrNullReturn(pcbWritten, VERR_INVALID_POINTER);
4250 if (pcbWritten)
4251 *pcbWritten = 0;
4252 RTVFSFILEINTERNAL *pThis = hVfsFile;
4253 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4254 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4255
4256 return RTVfsIoStrmSgWrite(&pThis->Stream, off, pSgBuf, fBlocking, pcbWritten);
4257}
4258
4259
4260RTDECL(int) RTVfsFileFlush(RTVFSFILE hVfsFile)
4261{
4262 RTVFSFILEINTERNAL *pThis = hVfsFile;
4263 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4264 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4265 return RTVfsIoStrmFlush(&pThis->Stream);
4266}
4267
4268
4269RTDECL(RTFOFF) RTVfsFilePoll(RTVFSFILE hVfsFile, uint32_t fEvents, RTMSINTERVAL cMillies, bool fIntr,
4270 uint32_t *pfRetEvents)
4271{
4272 RTVFSFILEINTERNAL *pThis = hVfsFile;
4273 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4274 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4275 return RTVfsIoStrmPoll(&pThis->Stream, fEvents, cMillies, fIntr, pfRetEvents);
4276}
4277
4278
4279RTDECL(RTFOFF) RTVfsFileTell(RTVFSFILE hVfsFile)
4280{
4281 RTVFSFILEINTERNAL *pThis = hVfsFile;
4282 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4283 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4284 return RTVfsIoStrmTell(&pThis->Stream);
4285}
4286
4287
4288RTDECL(int) RTVfsFileSeek(RTVFSFILE hVfsFile, RTFOFF offSeek, uint32_t uMethod, uint64_t *poffActual)
4289{
4290 RTVFSFILEINTERNAL *pThis = hVfsFile;
4291 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4292 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4293
4294 AssertReturn( uMethod == RTFILE_SEEK_BEGIN
4295 || uMethod == RTFILE_SEEK_CURRENT
4296 || uMethod == RTFILE_SEEK_END, VERR_INVALID_PARAMETER);
4297 AssertPtrNullReturn(poffActual, VERR_INVALID_POINTER);
4298
4299 RTFOFF offActual = 0;
4300 RTVfsLockAcquireWrite(pThis->Stream.Base.hLock);
4301 int rc = pThis->pOps->pfnSeek(pThis->Stream.Base.pvThis, offSeek, uMethod, &offActual);
4302 RTVfsLockReleaseWrite(pThis->Stream.Base.hLock);
4303 if (RT_SUCCESS(rc) && poffActual)
4304 {
4305 Assert(offActual >= 0);
4306 *poffActual = offActual;
4307 }
4308
4309 return rc;
4310}
4311
4312
4313RTDECL(int) RTVfsFileQuerySize(RTVFSFILE hVfsFile, uint64_t *pcbSize)
4314{
4315 RTVFSFILEINTERNAL *pThis = hVfsFile;
4316 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4317 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4318 AssertPtrReturn(pcbSize, VERR_INVALID_POINTER);
4319
4320 RTVfsLockAcquireWrite(pThis->Stream.Base.hLock);
4321 int rc = pThis->pOps->pfnQuerySize(pThis->Stream.Base.pvThis, pcbSize);
4322 RTVfsLockReleaseWrite(pThis->Stream.Base.hLock);
4323
4324 return rc;
4325}
4326
4327
4328RTDECL(int) RTVfsFileSetSize(RTVFSFILE hVfsFile, uint64_t cbSize, uint32_t fFlags)
4329{
4330 RTVFSFILEINTERNAL *pThis = hVfsFile;
4331 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4332 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4333 AssertReturn(RTVFSFILE_SIZE_F_IS_VALID(fFlags), VERR_INVALID_FLAGS);
4334 AssertReturn(pThis->Stream.fFlags & RTFILE_O_WRITE, VERR_ACCESS_DENIED);
4335
4336 int rc;
4337 if (pThis->pOps->pfnSetSize)
4338 {
4339 RTVfsLockAcquireWrite(pThis->Stream.Base.hLock);
4340 rc = pThis->pOps->pfnSetSize(pThis->Stream.Base.pvThis, cbSize, fFlags);
4341 RTVfsLockReleaseWrite(pThis->Stream.Base.hLock);
4342 }
4343 else
4344 rc = VERR_WRITE_PROTECT;
4345 return rc;
4346}
4347
4348
4349RTDECL(RTFOFF) RTVfsFileGetMaxSize(RTVFSFILE hVfsFile)
4350{
4351 uint64_t cbMax;
4352 int rc = RTVfsFileQueryMaxSize(hVfsFile, &cbMax);
4353 return RT_SUCCESS(rc) ? (RTFOFF)RT_MIN(cbMax, (uint64_t)RTFOFF_MAX) : -1;
4354}
4355
4356
4357RTDECL(int) RTVfsFileQueryMaxSize(RTVFSFILE hVfsFile, uint64_t *pcbMax)
4358{
4359 RTVFSFILEINTERNAL *pThis = hVfsFile;
4360 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
4361 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, VERR_INVALID_HANDLE);
4362 AssertPtrReturn(pcbMax, VERR_INVALID_POINTER);
4363 *pcbMax = RTFOFF_MAX;
4364
4365 int rc;
4366 if (pThis->pOps->pfnQueryMaxSize)
4367 {
4368 RTVfsLockAcquireWrite(pThis->Stream.Base.hLock);
4369 rc = pThis->pOps->pfnQueryMaxSize(pThis->Stream.Base.pvThis, pcbMax);
4370 RTVfsLockReleaseWrite(pThis->Stream.Base.hLock);
4371 }
4372 else
4373 rc = VERR_WRITE_PROTECT;
4374 return rc;
4375}
4376
4377
4378RTDECL(uint64_t) RTVfsFileGetOpenFlags(RTVFSFILE hVfsFile)
4379{
4380 RTVFSFILEINTERNAL *pThis = hVfsFile;
4381 AssertPtrReturn(pThis, 0);
4382 AssertReturn(pThis->uMagic == RTVFSFILE_MAGIC, 0);
4383 return pThis->Stream.fFlags;
4384}
4385
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