- Timestamp:
- Dec 13, 2018 7:26:25 PM (6 years ago)
- svn:sync-xref-src-repo-rev:
- 127483
- Location:
- trunk/src/VBox/Runtime
- Files:
-
- 2 edited
- 1 moved
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Runtime/Makefile.kmk
r76115 r76216 438 438 common/err/RTErrConvertFromErrno.cpp \ 439 439 common/err/RTErrConvertToErrno.cpp \ 440 common/fs/ext 2vfs.cpp \440 common/fs/extvfs.cpp \ 441 441 common/fs/fatvfs.cpp \ 442 442 common/fs/isovfs.cpp \ -
trunk/src/VBox/Runtime/common/fs/extvfs.cpp
r76215 r76216 5 5 6 6 /* 7 * Copyright (C) 2012-201 7Oracle Corporation7 * Copyright (C) 2012-2018 Oracle Corporation 8 8 * 9 9 * This file is part of VirtualBox Open Source Edition (OSE), as … … 33 33 34 34 #include <iprt/assert.h> 35 #include <iprt/file.h> 35 36 #include <iprt/log.h> 36 37 #include <iprt/mem.h> … … 38 39 #include <iprt/vfs.h> 39 40 #include <iprt/vfslowlevel.h> 40 #include <iprt/formats/ext 2.h>41 #include <iprt/formats/ext.h> 41 42 42 43 … … 47 48 * Cached block group descriptor data. 48 49 */ 49 typedef struct RTF ILESYSTEMEXTBLKGRP50 typedef struct RTFSEXTBLKGRP 50 51 { 51 52 /** Start offset (in bytes and from the start of the disk). */ … … 56 57 * and number of blocks per group). */ 57 58 uint8_t abBlockBitmap[1]; 58 } RTF ILESYSTEMEXTBLKGRP;59 } RTFSEXTBLKGRP; 59 60 /** Pointer to block group descriptor data. */ 60 typedef RTFILESYSTEMEXTBLKGRP *PRTFILESYSTEMEXTBLKGRP; 61 62 /** 63 * Ext2/3 filesystem data. 64 */ 65 typedef struct RTFILESYSTEMEXT 66 { 67 /** VFS file handle. */ 68 RTVFSFILE hVfsFile; 61 typedef RTFSEXTBLKGRP *PRTFSEXTBLKGRP; 62 63 /** 64 * Ext2/3/4 filesystem volume. 65 */ 66 typedef struct RTFSEXTVOL 67 { 68 /** Handle to itself. */ 69 RTVFS hVfsSelf; 70 /** The file, partition, or whatever backing the ext volume. */ 71 RTVFSFILE hVfsBacking; 72 /** The size of the backing thingy. */ 73 uint64_t cbBacking; 74 /** The formatted size of the volume. */ 75 uint64_t cbVolume; 76 /** cbVolume expressed as a cluster count. */ 77 uint64_t cClusters; 78 79 /** RTVFSMNT_F_XXX. */ 80 uint32_t fMntFlags; 81 /** RTFSEXTVFS_F_XXX (currently none defined). */ 82 uint32_t fExtFlags; 83 84 /** The (logical) sector size. */ 85 uint32_t cbSector; 86 87 /** The (logical) cluster size. */ 88 uint32_t cbCluster; 89 69 90 /** Block number of the superblock. */ 70 uint32_t 91 uint32_t iSbBlock; 71 92 /** Size of one block. */ 72 size_t 93 size_t cbBlock; 73 94 /** Number of blocks in one group. */ 74 unsigned 95 unsigned cBlocksPerGroup; 75 96 /** Number of blocks groups in the volume. */ 76 unsigned 97 unsigned cBlockGroups; 77 98 /** Cached block group descriptor data. */ 78 PRTF ILESYSTEMEXTBLKGRPpBlkGrpDesc;79 } RTF ILESYSTEMEXT;99 PRTFSEXTBLKGRP pBlkGrpDesc; 100 } RTFSEXTVOL; 80 101 /** Pointer to the ext filesystem data. */ 81 typedef RTF ILESYSTEMEXT *PRTFILESYSTEMEXT;102 typedef RTFSEXTVOL *PRTFSEXTVOL; 82 103 83 104 … … 90 111 * @param iBlkGrp Block group number to load. 91 112 */ 92 static int rtFsExtLoadBlkGrpDesc(PRTF ILESYSTEMEXTpThis, uint32_t iBlkGrp)113 static int rtFsExtLoadBlkGrpDesc(PRTFSEXTVOL pThis, uint32_t iBlkGrp) 93 114 { 94 115 size_t cbBlockBitmap = pThis->cBlocksPerGroup / 8; … … 96 117 cbBlockBitmap++; 97 118 98 PRTF ILESYSTEMEXTBLKGRP pBlkGrpDesc = pThis->pBlkGrpDesc;119 PRTFSEXTBLKGRP pBlkGrpDesc = pThis->pBlkGrpDesc; 99 120 if (!pBlkGrpDesc) 100 121 { 101 size_t cbBlkDesc = RT_UOFFSETOF_DYN(RTF ILESYSTEMEXTBLKGRP, abBlockBitmap[cbBlockBitmap]);102 pBlkGrpDesc = (PRTF ILESYSTEMEXTBLKGRP)RTMemAllocZ(cbBlkDesc);122 size_t cbBlkDesc = RT_UOFFSETOF_DYN(RTFSEXTBLKGRP, abBlockBitmap[cbBlockBitmap]); 123 pBlkGrpDesc = (PRTFSEXTBLKGRP)RTMemAllocZ(cbBlkDesc); 103 124 if (!pBlkGrpDesc) 104 125 return VERR_NO_MEMORY; 105 126 } 106 127 107 uint64_t 108 EXT 2BLOCKGROUPDESC BlkDesc;109 int rc = RTVfsFileReadAt(pThis->hVfs File, offRead, &BlkDesc, sizeof(BlkDesc), NULL);128 uint64_t offRead = (pThis->iSbBlock + 1) * pThis->cbBlock; 129 EXTBLOCKGROUPDESC BlkDesc; 130 int rc = RTVfsFileReadAt(pThis->hVfsBacking, offRead, &BlkDesc, sizeof(BlkDesc), NULL); 110 131 if (RT_SUCCESS(rc)) 111 132 { 112 133 pBlkGrpDesc->offStart = pThis->iSbBlock + (uint64_t)iBlkGrp * pThis->cBlocksPerGroup * pThis->cbBlock; 113 134 pBlkGrpDesc->offLast = pBlkGrpDesc->offStart + pThis->cBlocksPerGroup * pThis->cbBlock; 114 rc = RTVfsFileReadAt(pThis->hVfs File, BlkDesc.offBlockBitmap * pThis->cbBlock,135 rc = RTVfsFileReadAt(pThis->hVfsBacking, BlkDesc.offBlockBitmap * pThis->cbBlock, 115 136 &pBlkGrpDesc->abBlockBitmap[0], cbBlockBitmap, NULL); 116 137 } … … 121 142 122 143 123 static bool rtFsExtIsBlockRangeInUse(PRTF ILESYSTEMEXTBLKGRP pBlkGrpDesc, uint32_t offBlockStart, size_t cBlocks)144 static bool rtFsExtIsBlockRangeInUse(PRTFSEXTBLKGRP pBlkGrpDesc, uint32_t offBlockStart, size_t cBlocks) 124 145 { 125 146 while (cBlocks) … … 142 163 * @interface_method_impl{RTVFSOBJOPS::Obj,pfnClose} 143 164 */ 144 static DECLCALLBACK(int) rtFsExt 2Vol_Close(void *pvThis)145 { 146 PRTF ILESYSTEMEXT pThis = (PRTFILESYSTEMEXT)pvThis;165 static DECLCALLBACK(int) rtFsExtVol_Close(void *pvThis) 166 { 167 PRTFSEXTVOL pThis = (PRTFSEXTVOL)pvThis; 147 168 148 169 if (pThis->pBlkGrpDesc) … … 156 177 * @interface_method_impl{RTVFSOBJOPS::Obj,pfnQueryInfo} 157 178 */ 158 static DECLCALLBACK(int) rtFsExt 2Vol_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)179 static DECLCALLBACK(int) rtFsExtVol_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr) 159 180 { 160 181 RT_NOREF(pvThis, pObjInfo, enmAddAttr); … … 166 187 * @interface_method_impl{RTVFSOBJOPS::Obj,pfnOpenRoot} 167 188 */ 168 static DECLCALLBACK(int) rtFsExt 2_OpenRoot(void *pvThis, PRTVFSDIR phVfsDir)189 static DECLCALLBACK(int) rtFsExtVol_OpenRoot(void *pvThis, PRTVFSDIR phVfsDir) 169 190 { 170 191 NOREF(pvThis); … … 177 198 * @interface_method_impl{RTVFSOBJOPS::Obj,pfnQueryRangeState} 178 199 */ 179 static DECLCALLBACK(int) rtFsExt 2_QueryRangeState(void *pvThis, uint64_t off, size_t cb, bool *pfUsed)180 { 181 int 182 PRTF ILESYSTEMEXT pThis = (PRTFILESYSTEMEXT)pvThis;200 static DECLCALLBACK(int) rtFsExtVol_QueryRangeState(void *pvThis, uint64_t off, size_t cb, bool *pfUsed) 201 { 202 int rc = VINF_SUCCESS; 203 PRTFSEXTVOL pThis = (PRTFSEXTVOL)pvThis; 183 204 184 205 *pfUsed = false; … … 216 237 217 238 218 DECL_HIDDEN_CONST(const RTVFSOPS) g_rtFsExt 2VolOps =239 DECL_HIDDEN_CONST(const RTVFSOPS) g_rtFsExtVolOps = 219 240 { 220 241 /* .Obj = */ … … 222 243 /* .uVersion = */ RTVFSOBJOPS_VERSION, 223 244 /* .enmType = */ RTVFSOBJTYPE_VFS, 224 /* .pszName = */ "Ext 2Vol",225 /* .pfnClose = */ rtFsExt 2Vol_Close,226 /* .pfnQueryInfo = */ rtFsExt 2Vol_QueryInfo,245 /* .pszName = */ "ExtVol", 246 /* .pfnClose = */ rtFsExtVol_Close, 247 /* .pfnQueryInfo = */ rtFsExtVol_QueryInfo, 227 248 /* .uEndMarker = */ RTVFSOBJOPS_VERSION 228 249 }, 229 250 /* .uVersion = */ RTVFSOPS_VERSION, 230 251 /* .fFeatures = */ 0, 231 /* .pfnOpenRoot = */ rtFsExt 2_OpenRoot,232 /* .pfnQueryRangeState = */ rtFsExt 2_QueryRangeState,252 /* .pfnOpenRoot = */ rtFsExtVol_OpenRoot, 253 /* .pfnQueryRangeState = */ rtFsExtVol_QueryRangeState, 233 254 /* .uEndMarker = */ RTVFSOPS_VERSION 234 255 }; 235 256 236 257 237 RTDECL(int) RTFsExt 2VolOpen(RTVFSFILE hVfsFileIn, uint32_t fMntFlags, uint32_t fExtFlags, PRTVFS phVfs, PRTERRINFO pErrInfo)258 RTDECL(int) RTFsExtVolOpen(RTVFSFILE hVfsFileIn, uint32_t fMntFlags, uint32_t fExtFlags, PRTVFS phVfs, PRTERRINFO pErrInfo) 238 259 { 239 260 AssertPtrReturn(phVfs, VERR_INVALID_POINTER); … … 244 265 AssertReturn(cRefs != UINT32_MAX, VERR_INVALID_HANDLE); 245 266 246 PRTF ILESYSTEMEXTpThis;247 int rc = RTVfsNew(&g_rtFsExt 2VolOps, sizeof(*pThis), NIL_RTVFS, RTVFSLOCK_CREATE_RW, phVfs, (void **)&pThis);267 PRTFSEXTVOL pThis; 268 int rc = RTVfsNew(&g_rtFsExtVolOps, sizeof(*pThis), NIL_RTVFS, RTVFSLOCK_CREATE_RW, phVfs, (void **)&pThis); 248 269 if (RT_SUCCESS(rc)) 249 270 { 250 pThis->hVfs File= hVfsFileIn;271 pThis->hVfsBacking = hVfsFileIn; 251 272 pThis->pBlkGrpDesc = NULL; 252 273 253 EXT 2SUPERBLOCK SuperBlock;254 rc = RTVfsFileReadAt(hVfsFileIn, 1024, &SuperBlock, sizeof(EXT 2SUPERBLOCK), NULL);274 EXTSUPERBLOCK SuperBlock; 275 rc = RTVfsFileReadAt(hVfsFileIn, 1024, &SuperBlock, sizeof(EXTSUPERBLOCK), NULL); 255 276 if (RT_SUCCESS(rc)) 256 277 { … … 258 279 /** @todo Convert to host endianess. */ 259 280 #endif 260 if (SuperBlock.u16FilesystemState == EXT 2_STATE_ERRORS)261 rc = RTERRINFO_LOG_SET(pErrInfo, VERR_FILESYSTEM_CORRUPT, "EXT 2_STATE_ERRORS");281 if (SuperBlock.u16FilesystemState == EXT_STATE_ERRORS) 282 rc = RTERRINFO_LOG_SET(pErrInfo, VERR_FILESYSTEM_CORRUPT, "EXT_STATE_ERRORS"); 262 283 else 263 284 { … … 287 308 } 288 309 310 311 /** 312 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnValidate} 313 */ 314 static DECLCALLBACK(int) rtVfsChainExtVol_Validate(PCRTVFSCHAINELEMENTREG pProviderReg, PRTVFSCHAINSPEC pSpec, 315 PRTVFSCHAINELEMSPEC pElement, uint32_t *poffError, PRTERRINFO pErrInfo) 316 { 317 RT_NOREF(pProviderReg); 318 319 /* 320 * Basic checks. 321 */ 322 if (pElement->enmTypeIn != RTVFSOBJTYPE_FILE) 323 return pElement->enmTypeIn == RTVFSOBJTYPE_INVALID ? VERR_VFS_CHAIN_CANNOT_BE_FIRST_ELEMENT : VERR_VFS_CHAIN_TAKES_FILE; 324 if ( pElement->enmType != RTVFSOBJTYPE_VFS 325 && pElement->enmType != RTVFSOBJTYPE_DIR) 326 return VERR_VFS_CHAIN_ONLY_DIR_OR_VFS; 327 if (pElement->cArgs > 1) 328 return VERR_VFS_CHAIN_AT_MOST_ONE_ARG; 329 330 /* 331 * Parse the flag if present, save in pElement->uProvider. 332 */ 333 bool fReadOnly = (pSpec->fOpenFile & RTFILE_O_ACCESS_MASK) == RTFILE_O_READ; 334 if (pElement->cArgs > 0) 335 { 336 const char *psz = pElement->paArgs[0].psz; 337 if (*psz) 338 { 339 if (!strcmp(psz, "ro")) 340 fReadOnly = true; 341 else if (!strcmp(psz, "rw")) 342 fReadOnly = false; 343 else 344 { 345 *poffError = pElement->paArgs[0].offSpec; 346 return RTErrInfoSet(pErrInfo, VERR_VFS_CHAIN_INVALID_ARGUMENT, "Expected 'ro' or 'rw' as argument"); 347 } 348 } 349 } 350 351 pElement->uProvider = fReadOnly ? RTVFSMNT_F_READ_ONLY : 0; 352 return VINF_SUCCESS; 353 } 354 355 356 /** 357 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnInstantiate} 358 */ 359 static DECLCALLBACK(int) rtVfsChainExtVol_Instantiate(PCRTVFSCHAINELEMENTREG pProviderReg, PCRTVFSCHAINSPEC pSpec, 360 PCRTVFSCHAINELEMSPEC pElement, RTVFSOBJ hPrevVfsObj, 361 PRTVFSOBJ phVfsObj, uint32_t *poffError, PRTERRINFO pErrInfo) 362 { 363 RT_NOREF(pProviderReg, pSpec, poffError); 364 365 int rc; 366 RTVFSFILE hVfsFileIn = RTVfsObjToFile(hPrevVfsObj); 367 if (hVfsFileIn != NIL_RTVFSFILE) 368 { 369 RTVFS hVfs; 370 rc = RTFsFatVolOpen(hVfsFileIn, (uint32_t)pElement->uProvider, (uint32_t)(pElement->uProvider >> 32), &hVfs, pErrInfo); 371 RTVfsFileRelease(hVfsFileIn); 372 if (RT_SUCCESS(rc)) 373 { 374 *phVfsObj = RTVfsObjFromVfs(hVfs); 375 RTVfsRelease(hVfs); 376 if (*phVfsObj != NIL_RTVFSOBJ) 377 return VINF_SUCCESS; 378 rc = VERR_VFS_CHAIN_CAST_FAILED; 379 } 380 } 381 else 382 rc = VERR_VFS_CHAIN_CAST_FAILED; 383 return rc; 384 } 385 386 387 /** 388 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnCanReuseElement} 389 */ 390 static DECLCALLBACK(bool) rtVfsChainExtVol_CanReuseElement(PCRTVFSCHAINELEMENTREG pProviderReg, 391 PCRTVFSCHAINSPEC pSpec, PCRTVFSCHAINELEMSPEC pElement, 392 PCRTVFSCHAINSPEC pReuseSpec, PCRTVFSCHAINELEMSPEC pReuseElement) 393 { 394 RT_NOREF(pProviderReg, pSpec, pReuseSpec); 395 if ( pElement->paArgs[0].uProvider == pReuseElement->paArgs[0].uProvider 396 || !pReuseElement->paArgs[0].uProvider) 397 return true; 398 return false; 399 } 400 401 402 /** VFS chain element 'ext'. */ 403 static RTVFSCHAINELEMENTREG g_rtVfsChainExtVolReg = 404 { 405 /* uVersion = */ RTVFSCHAINELEMENTREG_VERSION, 406 /* fReserved = */ 0, 407 /* pszName = */ "ext", 408 /* ListEntry = */ { NULL, NULL }, 409 /* pszHelp = */ "Open a EXT file system, requires a file object on the left side.\n" 410 "First argument is an optional 'ro' (read-only) or 'rw' (read-write) flag.\n", 411 /* pfnValidate = */ rtVfsChainExtVol_Validate, 412 /* pfnInstantiate = */ rtVfsChainExtVol_Instantiate, 413 /* pfnCanReuseElement = */ rtVfsChainExtVol_CanReuseElement, 414 /* uEndMarker = */ RTVFSCHAINELEMENTREG_VERSION 415 }; 416 417 RTVFSCHAIN_AUTO_REGISTER_ELEMENT_PROVIDER(&g_rtVfsChainExtVolReg, rtVfsChainExtVolReg); 418 -
trunk/src/VBox/Runtime/common/vfs/vfsmount.cpp
r73097 r76216 45 45 #include <iprt/formats/iso9660.h> 46 46 #include <iprt/formats/udf.h> 47 #include <iprt/formats/ext 2.h>47 #include <iprt/formats/ext.h> 48 48 49 49 … … 352 352 353 353 /** 354 * Check if the given bootsector is an HPFS boot sector.354 * Check if the given bootsector is an ext2/3/4 super block. 355 355 * 356 356 * @returns true if NTFS, false if not. 357 357 * @param pSuperBlock The ext2 superblock. 358 358 */ 359 static bool rtVfsMountIsExt 2(PCEXT2SUPERBLOCK pSuperBlock)360 { 361 if (RT_LE2H_U16(pSuperBlock->u16Signature) != EXT 2_SIGNATURE)359 static bool rtVfsMountIsExt(PCEXTSUPERBLOCK pSuperBlock) 360 { 361 if (RT_LE2H_U16(pSuperBlock->u16Signature) != EXT_SIGNATURE) 362 362 return false; 363 363 … … 365 365 if (cShift > 54) 366 366 { 367 Log2(("rtVfsMountIsExt 2: cBitsShiftLeftBlockSize=%#x: out of range\n", cShift));367 Log2(("rtVfsMountIsExt: cBitsShiftLeftBlockSize=%#x: out of range\n", cShift)); 368 368 return false; 369 369 } … … 372 372 if (cShift > 54) 373 373 { 374 Log2(("rtVfsMountIsExt 2: cBitsShiftLeftFragmentSize=%#x: out of range\n", cShift));374 Log2(("rtVfsMountIsExt: cBitsShiftLeftFragmentSize=%#x: out of range\n", cShift)); 375 375 return false; 376 376 } … … 435 435 } 436 436 437 AssertCompile(sizeof(*pBuf) >= 1024 + sizeof(EXT 2SUPERBLOCK));438 if (rtVfsMountIsExt 2((PCEXT2SUPERBLOCK)&pBuf->ab[1024]))439 { 440 Log(("RTVfsMount: Detected ISO-9660 or UDF.\n"));441 return RTFsExt 2VolOpen(hVfsFileIn, fFlags, 0 /*fExt2Flags*/, phVfs, pErrInfo);437 AssertCompile(sizeof(*pBuf) >= 1024 + sizeof(EXTSUPERBLOCK)); 438 if (rtVfsMountIsExt((PCEXTSUPERBLOCK)&pBuf->ab[1024])) 439 { 440 Log(("RTVfsMount: Detected EXT2/3/4.\n")); 441 return RTFsExtVolOpen(hVfsFileIn, fFlags, 0 /*fExt2Flags*/, phVfs, pErrInfo); 442 442 } 443 443
Note:
See TracChangeset
for help on using the changeset viewer.