reactos/drivers/filesystems/udfs/udf_info/dirtree.cpp
2019-04-02 10:53:23 +02:00

1481 lines
47 KiB
C++

////////////////////////////////////////////////////////////////////
// Copyright (C) Alexander Telyatnikov, Ivan Keliukh, Yegor Anchishkin, SKIF Software, 1999-2013. Kiev, Ukraine
// All rights reserved
// This file was released under the GPLv2 on June 2015.
////////////////////////////////////////////////////////////////////
/*
Module name:
udf_info.cpp
Abstract:
This file contains filesystem-specific routines
for Directory tree & related structures support
*/
#include "udf.h"
#ifdef UDF_CHECK_UTIL
#include "..\namesup.h"
#else
#ifdef UDF_BUG_CHECK_ID
#undef UDF_BUG_CHECK_ID
#endif //UDF_BUG_CHECK_ID
#endif //UDF_CHECK_UTIL
#define UDF_BUG_CHECK_ID UDF_FILE_UDF_INFO_DIR
#define MEM_USDIRHASH_TAG "USDirHash"
#define UDF_DUMP_DIRTREE
#ifdef UDF_DUMP_DIRTREE
#define DirPrint(x) UDFPrint(x)
#else
#define DirPrint(x) {;}
#endif
/*
This routine initializes DirIndex array
*/
PDIR_INDEX_HDR
UDFDirIndexAlloc(
IN uint_di i
)
{
uint_di j,k;
PDIR_INDEX_HDR hDirNdx;
PDIR_INDEX_ITEM* FrameList;
if(!i)
return NULL;
#ifdef UDF_LIMIT_DIR_SIZE
if(i>UDF_DIR_INDEX_FRAME)
return NULL;
#endif //UDF_LIMIT_DIR_SIZE
j = i >> UDF_DIR_INDEX_FRAME_SH;
i &= (UDF_DIR_INDEX_FRAME-1);
hDirNdx = (PDIR_INDEX_HDR)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, sizeof(DIR_INDEX_HDR)+(j+(i!=0))*sizeof(PDIR_INDEX_ITEM), MEM_DIR_HDR_TAG);
if(!hDirNdx) return NULL;
RtlZeroMemory(hDirNdx, sizeof(DIR_INDEX_HDR));
FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
for(k=0; k<j; k++, FrameList++) {
(*FrameList) = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG);
if(!(*FrameList)) {
free_hdi:
// item pointet by FrameList is NULL, it could not be allocated
while(k) {
k--;
FrameList--;
MyFreePool__(*FrameList);
}
MyFreePool__(hDirNdx);
return NULL;
}
RtlZeroMemory((*FrameList), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM));
}
if(i) {
(*FrameList) = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, AlignDirIndex(i)*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG);
if(!(*FrameList))
goto free_hdi;
RtlZeroMemory((*FrameList), i*sizeof(DIR_INDEX_ITEM));
}
hDirNdx->FrameCount = j+(i!=0);
hDirNdx->LastFrameCount = i ? i : UDF_DIR_INDEX_FRAME;
return hDirNdx;
} // UDFDirIndexAlloc()
/*
This routine releases DirIndex array
*/
void
UDFDirIndexFree(
PDIR_INDEX_HDR hDirNdx
)
{
uint32 k;
PDIR_INDEX_ITEM* FrameList;
FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
if(!hDirNdx) return;
for(k=0; k<hDirNdx->FrameCount; k++, FrameList++) {
if(*FrameList) MyFreePool__(*FrameList);
}
MyFreePool__(hDirNdx);
} // UDFDirIndexFree();
/*
This routine grows DirIndex array
*/
OSSTATUS
UDFDirIndexGrow(
IN PDIR_INDEX_HDR* _hDirNdx,
IN uint_di d // increment
)
{
uint_di j,k;
PDIR_INDEX_HDR hDirNdx = *_hDirNdx;
PDIR_INDEX_ITEM* FrameList;
if(d > UDF_DIR_INDEX_FRAME)
return STATUS_INVALID_PARAMETER;
j = hDirNdx->LastFrameCount+d;
if(j > UDF_DIR_INDEX_FRAME) {
#ifndef UDF_LIMIT_DIR_SIZE // release
// Grow header
k = hDirNdx->FrameCount;
if(!MyReallocPool__((int8*)hDirNdx, sizeof(DIR_INDEX_HDR) + k*sizeof(PDIR_INDEX_ITEM),
(int8**)(&hDirNdx), sizeof(DIR_INDEX_HDR) + (k+1)*sizeof(PDIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
// Grow last frame
if(!MyReallocPool__((int8*)(FrameList[k-1]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
(int8**)(&(FrameList[k-1])), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
RtlZeroMemory(&(FrameList[k-1][hDirNdx->LastFrameCount]),
(UDF_DIR_INDEX_FRAME-hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM));
hDirNdx->LastFrameCount = UDF_DIR_INDEX_FRAME;
// Allocate new frame
FrameList[k] = (PDIR_INDEX_ITEM)MyAllocatePoolTag__(UDF_DIR_INDEX_MT, AlignDirIndex(j-UDF_DIR_INDEX_FRAME)*sizeof(DIR_INDEX_ITEM), MEM_DIR_NDX_TAG );
if(!FrameList[k])
return STATUS_INSUFFICIENT_RESOURCES;
hDirNdx->FrameCount++;
RtlZeroMemory(FrameList[k], (j-UDF_DIR_INDEX_FRAME)*sizeof(DIR_INDEX_ITEM));
hDirNdx->LastFrameCount = j-UDF_DIR_INDEX_FRAME;
(*_hDirNdx) = hDirNdx;
#else // UDF_LIMIT_DIR_SIZE
return STATUS_INSUFFICIENT_RESOURCES;
#endif // UDF_LIMIT_DIR_SIZE
} else {
k = hDirNdx->FrameCount;
FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
if(!MyReallocPool__((int8*)(FrameList[k-1]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
(int8**)(&(FrameList[k-1])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
RtlZeroMemory(&(FrameList[k-1][hDirNdx->LastFrameCount]),
(j-hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM));
hDirNdx->LastFrameCount = j;
}
return STATUS_SUCCESS;
} // end UDFDirIndexGrow()
/*
Thisd routine truncates DirIndex array
*/
OSSTATUS
UDFDirIndexTrunc(
IN PDIR_INDEX_HDR* _hDirNdx,
IN uint_di d // decrement
)
{
uint_di j,k;
if(d > UDF_DIR_INDEX_FRAME) {
OSSTATUS status;
while(d) {
k = (d > UDF_DIR_INDEX_FRAME) ? UDF_DIR_INDEX_FRAME : d;
if(!OS_SUCCESS(status = UDFDirIndexTrunc(_hDirNdx, k))) {
return status;
}
d -= k;
}
return STATUS_SUCCESS;
}
PDIR_INDEX_HDR hDirNdx = *_hDirNdx;
PDIR_INDEX_ITEM* FrameList;
j = UDF_DIR_INDEX_FRAME+hDirNdx->LastFrameCount-d;
FrameList = (PDIR_INDEX_ITEM*)(hDirNdx+1);
k = hDirNdx->FrameCount-1;
if(j <= UDF_DIR_INDEX_FRAME) {
// free last frame
if(!k && (j < 2)) {
// someone tries to trunc. residual entries...
return STATUS_INVALID_PARAMETER;
}
MyFreePool__(FrameList[k]);
FrameList[k] = NULL;
hDirNdx->LastFrameCount = UDF_DIR_INDEX_FRAME;
hDirNdx->FrameCount--;
// Truncate new last frame
if(!MyReallocPool__((int8*)(FrameList[k-1]), UDF_DIR_INDEX_FRAME*sizeof(DIR_INDEX_ITEM),
(int8**)(&(FrameList[k-1])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
hDirNdx->LastFrameCount = j;
// Truncate header
if(!MyReallocPool__((int8*)hDirNdx, sizeof(DIR_INDEX_HDR) + (k+1)*sizeof(PDIR_INDEX_ITEM),
(int8**)(&hDirNdx), sizeof(DIR_INDEX_HDR) + k*sizeof(PDIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
(*_hDirNdx) = hDirNdx;
} else {
j -= UDF_DIR_INDEX_FRAME;
if(!k && (j < 2)) {
// someone tries to trunc. residual entries...
return STATUS_INVALID_PARAMETER;
}
if(!MyReallocPool__((int8*)(FrameList[k]), AlignDirIndex(hDirNdx->LastFrameCount)*sizeof(DIR_INDEX_ITEM),
(int8**)(&(FrameList[k])), AlignDirIndex(j)*sizeof(DIR_INDEX_ITEM) ) )
return STATUS_INSUFFICIENT_RESOURCES;
hDirNdx->LastFrameCount = j;
}
return STATUS_SUCCESS;
} // end UDFDirIndexTrunc()
#if defined _X86_ && !defined UDF_LIMIT_DIR_SIZE
#ifdef _MSC_VER
#pragma warning(disable:4035) // re-enable below
#endif
/*
This routine returns pointer to DirIndex item with index i.
*/
#if defined(_MSC_VER) && !defined(__clang__)
__declspec (naked)
#endif
PDIR_INDEX_ITEM
__fastcall
UDFDirIndex(
IN PDIR_INDEX_HDR hDirNdx, // ECX
IN uint32 i // EDX
)
{
#if defined(_MSC_VER) && !defined(__clang__)
__asm {
push ebx
push ecx
push edx
// mov ebx,hDirNdx
mov ebx,ecx
mov ecx,edx
or ebx,ebx
jz EO_udi_err
mov eax,ecx
shr ecx,UDF_DIR_INDEX_FRAME_SH ; ecx = j
mov edx,[ebx]hDirNdx.FrameCount ; edx = k
cmp ecx,edx
jae EO_udi_err
and eax,(1 shl UDF_DIR_INDEX_FRAME_SH)-1 ; eax = i
dec edx
cmp ecx,edx
jb No_check
cmp eax,[ebx].LastFrameCount
jae EO_udi_err
No_check:
add ebx,size DIR_INDEX_HDR ; ((PDIR_INDEX_ITEM*)(hDirNdx+1))...
mov ebx,[ebx+ecx*4] ; ...[j]...
mov edx,size DIR_INDEX_ITEM
mul edx ; ...[i]...
add eax,ebx ; &(...)
jmp udi_OK
EO_udi_err:
xor eax,eax
udi_OK:
pop edx
pop ecx
pop ebx
ret
}
#else
/* FIXME ReactOS */
uint_di j, k;
if( hDirNdx &&
((j = (i >> UDF_DIR_INDEX_FRAME_SH)) < (k = hDirNdx->FrameCount) ) &&
((i = (i & (UDF_DIR_INDEX_FRAME-1))) < ((j < (k-1)) ? UDF_DIR_INDEX_FRAME : hDirNdx->LastFrameCount)) )
return &( (((PDIR_INDEX_ITEM*)(hDirNdx+1))[j])[i] );
return NULL;
#endif
}
#ifdef _MSC_VER
#pragma warning(default:4035)
#endif
#endif // _X86_
/*
This routine returns pointer to DirIndex'es frame & index inside it
according to start Index parameter. It also initializes scan parameters
*/
PDIR_INDEX_ITEM
UDFDirIndexGetFrame(
IN PDIR_INDEX_HDR hDirNdx,
IN uint32 Frame,
OUT uint32* FrameLen,
OUT uint_di* Index,
IN uint_di Rel
)
{
if(Frame >= hDirNdx->FrameCount)
return NULL;
if(Index) {
#ifdef UDF_LIMIT_DIR_SIZE
(*Index) = Rel;
// if(FrameLen)
(*FrameLen) = hDirNdx->LastFrameCount;
#else //UDF_LIMIT_DIR_SIZE
(*Index) = Frame*UDF_DIR_INDEX_FRAME+Rel;
// if(FrameLen)
(*FrameLen) = (Frame < (hDirNdx->FrameCount-1)) ? UDF_DIR_INDEX_FRAME :
hDirNdx->LastFrameCount;
#endif //UDF_LIMIT_DIR_SIZE
}
return ((PDIR_INDEX_ITEM*)(hDirNdx+1))[Frame]+Rel;
} // end UDFDirIndexGetFrame()
/*
This routine initializes indexes for optimized DirIndex scan
according to start Index parameter
*/
BOOLEAN
UDFDirIndexInitScan(
IN PUDF_FILE_INFO DirInfo, //
OUT PUDF_DIR_SCAN_CONTEXT Context,
IN uint_di Index
)
{
Context->DirInfo = DirInfo;
Context->hDirNdx = DirInfo->Dloc->DirIndex;
if( (Context->frame = (Index >> UDF_DIR_INDEX_FRAME_SH)) >=
Context->hDirNdx->FrameCount) {
return FALSE;
}
if( (Context->j = Index & (UDF_DIR_INDEX_FRAME-1)) >=
((Context->frame < (Context->hDirNdx->FrameCount-1))
?
UDF_DIR_INDEX_FRAME : Context->hDirNdx->LastFrameCount) ) {
return FALSE;
}
Context->DirNdx = UDFDirIndexGetFrame(Context->hDirNdx,
Context->frame,
&(Context->d),
&(Context->i),
Context->j);
Context->i--;
Context->j--;
Context->DirNdx--;
return TRUE;
} // end UDFDirIndexInitScan()
PDIR_INDEX_ITEM
UDFDirIndexScan(
PUDF_DIR_SCAN_CONTEXT Context,
PUDF_FILE_INFO* _FileInfo
)
{
PUDF_FILE_INFO FileInfo;
PUDF_FILE_INFO ParFileInfo;
Context->i++;
Context->j++;
Context->DirNdx++;
if(Context->j >= Context->d) {
Context->j=0;
Context->frame++;
Context->DirNdx = UDFDirIndexGetFrame(Context->hDirNdx,
Context->frame,
&(Context->d),
&(Context->i),
Context->j);
}
if(!Context->DirNdx) {
if(_FileInfo)
(*_FileInfo) = NULL;
return NULL;
}
if(_FileInfo) {
if((FileInfo = Context->DirNdx->FileInfo)) {
if(FileInfo->ParentFile != Context->DirInfo) {
ParFileInfo = UDFLocateParallelFI(Context->DirInfo,
Context->i,
FileInfo);
#ifdef UDF_DBG
if(ParFileInfo->ParentFile != Context->DirInfo) {
BrutePoint();
}
#endif // UDF_DBG
FileInfo = ParFileInfo;
}
}
(*_FileInfo) = FileInfo;
}
return (Context->DirNdx);
} // end UDFDirIndexScan()
/*
This routine calculates hashes for directory search
*/
uint8
UDFBuildHashEntry(
IN PVCB Vcb,
IN PUNICODE_STRING Name,
OUT PHASH_ENTRY hashes,
IN uint8 Mask
)
{
UNICODE_STRING UName;
WCHAR ShortNameBuffer[13];
uint8 RetFlags = 0;
if(!Name->Buffer) return 0;
if(Mask & HASH_POSIX)
hashes->hPosix = crc32((uint8*)(Name->Buffer), Name->Length);
if(Mask & HASH_ULFN) {
/* if(OS_SUCCESS(MyInitUnicodeString(&UName, L"")) &&
OS_SUCCESS(MyAppendUnicodeStringToStringTag(&UName, Name, MEM_USDIRHASH_TAG))) {*/
if(OS_SUCCESS(MyCloneUnicodeString(&UName, Name))) {
RtlUpcaseUnicodeString(&UName, &UName, FALSE);
/* if(!RtlCompareUnicodeString(Name, &UName, FALSE)) {
RetFlags |= UDF_FI_FLAG_LFN;
}*/
hashes->hLfn = crc32((uint8*)(UName.Buffer), UName.Length);
} else {
BrutePoint();
}
MyFreePool__(UName.Buffer);
}
if(Mask & HASH_DOS) {
UName.Buffer = (PWCHAR)(&ShortNameBuffer);
UName.MaximumLength = 13*sizeof(WCHAR);
UDFDOSName(Vcb, &UName, Name, (Mask & HASH_KEEP_NAME) ? TRUE : FALSE);
if(!RtlCompareUnicodeString(Name, &UName, TRUE)) {
RetFlags |= UDF_FI_FLAG_DOS;
}
hashes->hDos = crc32((uint8*)(UName.Buffer), UName.Length);
}
return RetFlags;
} // UDFBuildHashEntry()
#ifdef UDF_CHECK_UTIL
uint32
UDFFindNextFI(
IN int8* buff,
IN uint32 prevOffset,
IN uint32 Length
)
{
PFILE_IDENT_DESC FileId;
while(prevOffset+sizeof(FILE_IDENT_DESC) < Length) {
prevOffset++;
FileId = (PFILE_IDENT_DESC)(buff+prevOffset);
if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC)
continue;
if(FileId->descTag.descVersion != 2 && FileId->descTag.descVersion != 3)
continue;
if(FileId->fileVersionNum != 1)
continue;
if(FileId->fileCharacteristics & (~0x1f))
continue;
if(prevOffset + ((FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3))) <= Length) {
UDFPrint(("UDFFindNextFI OK: %x\n", prevOffset));
return prevOffset;
}
}
return 0;
} // end UDFFindNextFI()
#else //UDF_CHECK_UTIL
#define UDFFindNextFI(a,b,c) 0
#endif //UDF_CHECK_UTIL
/*
This routine scans directory extent & builds index table for FileIdents
*/
OSSTATUS
UDFIndexDirectory(
IN PVCB Vcb,
IN OUT PUDF_FILE_INFO FileInfo
)
{
PDIR_INDEX_HDR hDirNdx;
PDIR_INDEX_ITEM DirNdx;
PFILE_IDENT_DESC FileId;
uint32 Offset = 0;
// uint32 prevOffset = 0;
uint_di Count = 0;
OSSTATUS status;
int8* buff;
PEXTENT_INFO ExtInfo; // Extent array for directory
uint16 PartNum;
SIZE_T ReadBytes;
uint16 valueCRC;
if(!FileInfo) return STATUS_INVALID_PARAMETER;
ValidateFileInfo(FileInfo);
ExtInfo = &(FileInfo->Dloc->DataLoc);
FileInfo->Dloc->DirIndex = NULL;
UDFPrint(("UDF: scaning directory\n"));
// allocate buffer for the whole directory
ASSERT((uint32)(ExtInfo->Length));
if(!ExtInfo->Length)
return STATUS_FILE_CORRUPT_ERROR;
buff = (int8*)DbgAllocatePool(PagedPool, (uint32)(ExtInfo->Length));
if(!buff)
return STATUS_INSUFFICIENT_RESOURCES;
ExtInfo->Flags |= EXTENT_FLAG_ALLOC_SEQUENTIAL;
// read FileIdents
status = UDFReadExtent(Vcb, ExtInfo, 0, (uint32)(ExtInfo->Length), FALSE, buff, &ReadBytes);
if(!OS_SUCCESS(status)) {
DbgFreePool(buff);
return status;
}
// scan Dir to get entry counter
FileId = (PFILE_IDENT_DESC)buff;
DirPrint((" ExtInfo->Length %x\n", ExtInfo->Length));
// prevOffset = 0;
while(Offset<ExtInfo->Length) {
DirPrint((" Offset %x\n", Offset));
if(!FileId->descTag.tagIdent) {
DirPrint((" term item\n"));
break;
}
if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC) {
DirPrint((" Inv. tag %x\n", FileId->descTag.tagIdent));
Offset = UDFFindNextFI(buff, prevOffset, (ULONG)(ExtInfo->Length));
if(!Offset) {
DirPrint((" can't find next\n"));
break;
} else {
DirPrint((" found next offs %x\n", Offset));
FileId = (PFILE_IDENT_DESC)((buff)+Offset);
}
}
if(((ULONG)Offset & (Vcb->LBlockSize-1)) > (Vcb->LBlockSize-sizeof(FILE_IDENT_DESC))) {
DirPrint((" badly aligned\n", Offset));
if(Vcb->Modified) {
DirPrint((" queue repack request\n"));
FileInfo->Dloc->DirIndex->DelCount = Vcb->PackDirThreshold+1;
}
}
// prevOffset = Offset;
Offset += (FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3));
FileId = (PFILE_IDENT_DESC)((buff)+Offset);
Count++;
if(Offset+sizeof(FILE_IDENT_DESC) > ExtInfo->Length) {
if(Offset != ExtInfo->Length) {
UDFPrint((" Trash at the end of Dir\n"));
}
// BrutePoint();
break;
}
}
DirPrint((" final Offset %x\n", Offset));
if(Offset > ExtInfo->Length) {
BrutePoint();
UDFPrint((" Unexpected end of Dir\n"));
DbgFreePool(buff);
return STATUS_FILE_CORRUPT_ERROR;
}
// allocate buffer for directory index & zero it
DirPrint((" Count %x\n", Count));
hDirNdx = UDFDirIndexAlloc(Count+1);
if(!hDirNdx) {
DbgFreePool(buff);
return STATUS_INSUFFICIENT_RESOURCES;
}
Offset = Count = 0;
hDirNdx->DIFlags |= (ExtInfo->Offset ? UDF_DI_FLAG_INIT_IN_ICB : 0);
// add entry pointing to the directory itself
DirNdx = UDFDirIndex(hDirNdx,0);
ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
DirNdx->FileEntryLoc.partitionReferenceNum = PartNum =
(uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
ASSERT(PartNum != -1);
DirNdx->FileEntryLoc.logicalBlockNum =
UDFPhysLbaToPart(Vcb, PartNum, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
if(DirNdx->FileEntryLoc.logicalBlockNum == (ULONG)-1) {
DirPrint((" err: FileEntryLoc=-1\n"));
DbgFreePool(buff);
UDFDirIndexFree(hDirNdx);
return STATUS_FILE_CORRUPT_ERROR;
}
DirNdx->FileCharacteristics = (FileInfo->FileIdent) ?
FileInfo->FileIdent->fileCharacteristics :
FILE_DIRECTORY;
// DirNdx->Offset = 0;
// DirNdx->Length = 0;
RtlInitUnicodeString(&DirNdx->FName, L".");
DirNdx->FileInfo = FileInfo;
DirNdx->FI_Flags |= UDF_FI_FLAG_KEEP_NAME;
DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes),
HASH_ALL | HASH_KEEP_NAME);
Count++;
FileId = (PFILE_IDENT_DESC)buff;
status = STATUS_SUCCESS;
// prevOffset = 0;
while((Offset<ExtInfo->Length) && FileId->descTag.tagIdent) {
// add new entry to index list
if(FileId->descTag.tagIdent != TID_FILE_IDENT_DESC) {
UDFPrint((" Invalid tagIdent %x (expected %x) offst %x\n", FileId->descTag.tagIdent, TID_FILE_IDENT_DESC, Offset));
DirPrint((" FileId: filen %x, iulen %x, charact %x\n",
FileId->lengthFileIdent, FileId->lengthOfImpUse, FileId->fileCharacteristics));
DirPrint((" loc: @%x\n", UDFExtentOffsetToLba(Vcb, ExtInfo->Mapping, Offset, NULL, NULL, NULL, NULL)));
KdDump(FileId, sizeof(FileId->descTag));
Offset = UDFFindNextFI(buff, prevOffset, (ULONG)(ExtInfo->Length));
if(!Offset) {
DbgFreePool(buff);
UDFDirIndexFree(hDirNdx);
return STATUS_FILE_CORRUPT_ERROR;
} else {
DirPrint((" found next offs %x\n", Offset));
FileId = (PFILE_IDENT_DESC)((buff)+Offset);
}
}
DirNdx = UDFDirIndex(hDirNdx,Count);
// allocate buffer & fill it with decompressed unicode filename
if(FileId->fileCharacteristics & FILE_DELETED) {
DirPrint((" FILE_DELETED\n"));
hDirNdx->DelCount++;
}
DirPrint((" FileId: offs %x, filen %x, iulen %x\n", Offset, FileId->lengthFileIdent, FileId->lengthOfImpUse));
DirNdx->Length = (FileId->lengthFileIdent + FileId->lengthOfImpUse + sizeof(FILE_IDENT_DESC) + 3) & (~((uint32)3));
DirPrint((" DirNdx: Length %x, Charact %x\n", DirNdx->Length, FileId->fileCharacteristics));
if(FileId->fileCharacteristics & FILE_PARENT) {
DirPrint((" parent\n"));
// init 'parent' entry
// '..' points to Parent Object (if any),
// otherwise it points to the Dir itself
RtlInitUnicodeString(&DirNdx->FName, L"..");
DirNdx->FileInfo = (FileInfo->ParentFile) ?
FileInfo->ParentFile : FileInfo;
DirNdx->FI_Flags |= UDF_FI_FLAG_KEEP_NAME;
DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes), HASH_ALL | HASH_KEEP_NAME);
} else {
// init plain file/dir entry
ASSERT( (Offset+sizeof(FILE_IDENT_DESC)+FileId->lengthOfImpUse+FileId->lengthFileIdent) <=
ExtInfo->Length );
UDFDecompressUnicode(&(DirNdx->FName),
((uint8*)(FileId+1)) + (FileId->lengthOfImpUse),
FileId->lengthFileIdent,
&valueCRC);
UDFNormalizeFileName(&(DirNdx->FName), valueCRC);
DirNdx->FI_Flags |= UDFBuildHashEntry(Vcb, &(DirNdx->FName), &(DirNdx->hashes), HASH_ALL);
}
if((FileId->fileCharacteristics & FILE_METADATA)
||
!DirNdx->FName.Buffer
||
((DirNdx->FName.Length >= sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) &&
(RtlCompareMemory(DirNdx->FName.Buffer, UDF_RESERVED_NAME_HDR, sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) == sizeof(UDF_RESERVED_NAME_HDR)-sizeof(WCHAR)) )) {
DirPrint((" metadata\n"));
DirNdx->FI_Flags |= UDF_FI_FLAG_FI_INTERNAL;
}
#if 0
UDFPrint(("%ws\n", DirNdx->FName.Buffer));
#endif
DirPrint(("%ws\n", DirNdx->FName.Buffer));
// remember FileEntry location...
DirNdx->FileEntryLoc = FileId->icb.extLocation;
// ... and some file characteristics
DirNdx->FileCharacteristics = FileId->fileCharacteristics;
DirNdx->Offset = Offset;
#ifdef UDF_CHECK_DISK_ALLOCATION
if(!(FileId->fileCharacteristics & FILE_DELETED) &&
(UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) != LBA_OUT_OF_EXTENT) &&
UDFGetFreeBit(((uint32*)(Vcb->FSBM_Bitmap)), UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) )) {
AdPrint(("Ref to Discarded block %x\n",UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) ));
BrutePoint();
FileId->fileCharacteristics |= FILE_DELETED;
} else
if(UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) == LBA_OUT_OF_EXTENT) {
AdPrint(("Ref to Invalid block %x\n", UDFPartLbaToPhys(Vcb, &(DirNdx->FileEntryLoc)) ));
BrutePoint();
FileId->fileCharacteristics |= FILE_DELETED;
}
#endif // UDF_CHECK_DISK_ALLOCATION
// prevOffset = Offset;
Offset += DirNdx->Length;
FileId = (PFILE_IDENT_DESC)(((int8*)FileId)+DirNdx->Length);
Count++;
if(Offset+sizeof(FILE_IDENT_DESC) > ExtInfo->Length) {
if(Offset != ExtInfo->Length) {
UDFPrint((" Trash at the end of Dir (2)\n"));
}
// BrutePoint();
break;
}
} // while()
// we needn't writing terminator 'cause the buffer is already zero-filled
DbgFreePool(buff);
if(Count < 2) {
UDFDirIndexFree(hDirNdx);
UDFPrint((" Directory too short\n"));
return STATUS_FILE_CORRUPT_ERROR;
}
// store index
FileInfo->Dloc->DirIndex = hDirNdx;
return status;
} // end UDFIndexDirectory()
#ifndef UDF_READ_ONLY_BUILD
/*
This routine removes all DELETED entries from Dir & resizes it.
It must be called before closing, no files sould be opened.
*/
OSSTATUS
UDFPackDirectory__(
IN PVCB Vcb,
IN OUT PUDF_FILE_INFO FileInfo // source (opened)
)
{
#ifdef UDF_PACK_DIRS
uint32 d, LBS;
uint_di i, j;
uint32 IUl, FIl, l;
uint32 DataLocOffset;
uint32 Offset, curOffset;
int8* Buf;
OSSTATUS status;
SIZE_T ReadBytes;
int8* storedFI;
PUDF_FILE_INFO curFileInfo;
PDIR_INDEX_ITEM DirNdx, DirNdx2;
UDF_DIR_SCAN_CONTEXT ScanContext;
uint_di dc=0;
uint16 PartNum;
#endif //UDF_PACK_DIRS
ValidateFileInfo(FileInfo);
PDIR_INDEX_HDR hDirNdx = FileInfo->Dloc->DirIndex;
if(!hDirNdx) return STATUS_NOT_A_DIRECTORY;
#ifndef UDF_PACK_DIRS
return STATUS_SUCCESS;
#else // UDF_PACK_DIRS
// do not pack dirs on unchanged disks
if(!Vcb->Modified)
return STATUS_SUCCESS;
// start packing
LBS = Vcb->LBlockSize;
Buf = (int8*)DbgAllocatePool(PagedPool, LBS*2);
if(!Buf) return STATUS_INSUFFICIENT_RESOURCES;
// we shall never touch 1st entry 'cause it can't be deleted
Offset = UDFDirIndex(hDirNdx,2)->Offset;
DataLocOffset = FileInfo->Dloc->DataLoc.Offset;
i=j=2;
if(!UDFDirIndexInitScan(FileInfo, &ScanContext, i)) {
DbgFreePool(Buf);
return STATUS_SUCCESS;
}
ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
PartNum = (uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
ASSERT(PartNum != -1);
while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
if(UDFIsDeleted(DirNdx))
dc++;
if(!UDFIsDeleted(DirNdx) ||
DirNdx->FileInfo) {
// move down valid entry
status = UDFReadFile__(Vcb, FileInfo, curOffset = DirNdx->Offset,
l = DirNdx->Length, FALSE, Buf, &ReadBytes);
if(!OS_SUCCESS(status)) {
DbgFreePool(Buf);
return status;
}
// remove ImpUse field
IUl = ((PFILE_IDENT_DESC)Buf)->lengthOfImpUse;
curFileInfo = DirNdx->FileInfo;
// align next entry
if((d = LBS - ((curOffset + (l - IUl) + DataLocOffset) & (LBS-1)) ) < sizeof(FILE_IDENT_DESC)) {
// insufficient space at the end of last sector for
// next FileIdent's tag. fill it with ImpUse data
// generally, all data should be DWORD-aligned, but if it is not so
// this opearation will help us to avoid glitches
d = (d+3) & ~(3);
if(d != IUl) {
l = l + d - IUl;
FIl = ((PFILE_IDENT_DESC)Buf)->lengthFileIdent;
// copy filename to upper addr
RtlMoveMemory(Buf+sizeof(FILE_IDENT_DESC)+d,
Buf+sizeof(FILE_IDENT_DESC)+IUl, FIl);
RtlZeroMemory(Buf+sizeof(FILE_IDENT_DESC), d);
((PFILE_IDENT_DESC)Buf)->lengthOfImpUse = (uint16)d;
if(curFileInfo && curFileInfo->FileIdent) {
// update stored FI if any
if(!MyReallocPool__((int8*)(curFileInfo->FileIdent), l,
(int8**)&(curFileInfo->FileIdent), (l+IUl-d) )) {
DbgFreePool(Buf);
return STATUS_INSUFFICIENT_RESOURCES;
}
storedFI = (int8*)(curFileInfo->FileIdent);
RtlMoveMemory(storedFI+sizeof(FILE_IDENT_DESC)+d,
storedFI+sizeof(FILE_IDENT_DESC)+IUl, FIl);
RtlZeroMemory(storedFI+sizeof(FILE_IDENT_DESC), d);
((PFILE_IDENT_DESC)storedFI)->lengthOfImpUse = (uint16)d;
FileInfo->Dloc->FELoc.Modified = TRUE;
FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED;
}
}
} else {
d = 0;
}
// write modified to new addr
if((d != IUl) ||
(curOffset != Offset)) {
UDFSetUpTag(Vcb, (tag*)Buf, (uint16)l,
UDFPhysLbaToPart(Vcb, PartNum,
UDFExtentOffsetToLba(Vcb, FileInfo->Dloc->DataLoc.Mapping,
Offset, NULL, NULL, NULL, NULL)));
status = UDFWriteFile__(Vcb, FileInfo, Offset, l, FALSE, Buf, &ReadBytes);
if(!OS_SUCCESS(status)) {
DbgFreePool(Buf);
return status;
}
}
DirNdx2 = UDFDirIndex(hDirNdx, j);
*DirNdx2 = *DirNdx;
DirNdx2->Offset = Offset;
DirNdx2->Length = l;
if(curFileInfo) {
curFileInfo->Index = j;
DirNdx2->FI_Flags |= UDF_FI_FLAG_FI_MODIFIED;
}
Offset += l;
j++;
}
}
// resize DirIndex
DbgFreePool(Buf);
if(dc) {
if(!OS_SUCCESS(status = UDFDirIndexTrunc(&(FileInfo->Dloc->DirIndex), dc))) {
return status;
}
}
// terminator is set by UDFDirIndexTrunc()
FileInfo->Dloc->DirIndex->DelCount = 0;
ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
// now Offset points to EOF. Let's truncate directory
return UDFResizeFile__(Vcb, FileInfo, Offset);
#endif // UDF_PACK_DIRS
} // end UDFPackDirectory__()
/*
This routine rebuilds tags for all entries from Dir.
*/
OSSTATUS
UDFReTagDirectory(
IN PVCB Vcb,
IN OUT PUDF_FILE_INFO FileInfo // source (opened)
)
{
uint32 l;
uint32 Offset;
int8* Buf;
OSSTATUS status;
SIZE_T ReadBytes;
PUDF_FILE_INFO curFileInfo;
PDIR_INDEX_ITEM DirNdx;
UDF_DIR_SCAN_CONTEXT ScanContext;
uint16 PartNum;
ValidateFileInfo(FileInfo);
PDIR_INDEX_HDR hDirNdx = FileInfo->Dloc->DirIndex;
if(!hDirNdx) return STATUS_NOT_A_DIRECTORY;
// do not pack dirs on unchanged disks
if(!Vcb->Modified)
return STATUS_SUCCESS;
if( ((hDirNdx->DIFlags & UDF_DI_FLAG_INIT_IN_ICB) ? TRUE : FALSE) ==
((FileInfo->Dloc->DataLoc.Offset) ? TRUE : FALSE) ) {
return STATUS_SUCCESS;
}
// start packing
Buf = (int8*)DbgAllocatePool(PagedPool, Vcb->LBlockSize*2);
if(!Buf) return STATUS_INSUFFICIENT_RESOURCES;
Offset = UDFDirIndex(hDirNdx,1)->Offset;
if(!UDFDirIndexInitScan(FileInfo, &ScanContext, 1)) {
DbgFreePool(Buf);
return STATUS_SUCCESS;
}
ASSERT(FileInfo->Dloc->FELoc.Mapping[0].extLocation);
PartNum = (uint16)UDFGetPartNumByPhysLba(Vcb, FileInfo->Dloc->FELoc.Mapping[0].extLocation);
ASSERT(PartNum != -1);
while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
status = UDFReadFile__(Vcb, FileInfo, Offset = DirNdx->Offset,
l = DirNdx->Length, FALSE, Buf, &ReadBytes);
if(!OS_SUCCESS(status)) {
DbgFreePool(Buf);
return status;
}
curFileInfo = DirNdx->FileInfo;
// write modified
UDFSetUpTag(Vcb, (tag*)Buf, (uint16)l,
UDFPhysLbaToPart(Vcb, PartNum,
UDFExtentOffsetToLba(Vcb, FileInfo->Dloc->DataLoc.Mapping,
Offset, NULL, NULL, NULL, NULL)));
if(curFileInfo && curFileInfo->FileIdent) {
FileInfo->Dloc->FELoc.Modified = TRUE;
FileInfo->Dloc->FE_Flags |= UDF_FE_FLAG_FE_MODIFIED;
}
status = UDFWriteFile__(Vcb, FileInfo, Offset, l, FALSE, Buf, &ReadBytes);
if(!OS_SUCCESS(status)) {
DbgFreePool(Buf);
return status;
}
if(curFileInfo) {
DirNdx->FI_Flags |= UDF_FI_FLAG_FI_MODIFIED;
}
}
// resize DirIndex
DbgFreePool(Buf);
hDirNdx->DIFlags &= ~UDF_DI_FLAG_INIT_IN_ICB;
hDirNdx->DIFlags |= (FileInfo->Dloc->DataLoc.Offset ? UDF_DI_FLAG_INIT_IN_ICB : 0);
return status;
} // end UDFReTagDirectory()
#endif //UDF_READ_ONLY_BUILD
/*
This routine performs search for specified file in specified directory &
returns corresponding offset in extent if found.
*/
OSSTATUS
UDFFindFile(
IN PVCB Vcb,
IN BOOLEAN IgnoreCase,
IN BOOLEAN NotDeleted,
IN PUNICODE_STRING Name,
IN PUDF_FILE_INFO DirInfo,
IN OUT uint_di* Index // IN:start index OUT:found file index
)
{
// PDIR_INDEX_HDR hDirIndex = DirInfo->Dloc->DirIndex;
UNICODE_STRING ShortName;
WCHAR ShortNameBuffer[13];
PDIR_INDEX_ITEM DirNdx;
UDF_DIR_SCAN_CONTEXT ScanContext;
uint_di j=(-1), k=(-1);
HASH_ENTRY hashes;
BOOLEAN CanBe8d3;
UDFBuildHashEntry(Vcb, Name, &hashes, HASH_POSIX | HASH_ULFN);
if((CanBe8d3 = UDFCanNameBeA8dot3(Name))) {
ShortName.MaximumLength = 13 * sizeof(WCHAR);
ShortName.Buffer = (PWCHAR)&ShortNameBuffer;
}
if(!UDFDirIndexInitScan(DirInfo, &ScanContext, (*Index)))
return STATUS_OBJECT_NAME_NOT_FOUND;
if(!IgnoreCase && !CanBe8d3) {
// perform case sensetive sequential directory scan
while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
if( (DirNdx->hashes.hPosix == hashes.hPosix) &&
DirNdx->FName.Buffer &&
(!RtlCompareUnicodeString(&(DirNdx->FName), Name, FALSE)) &&
( (!UDFIsDeleted(DirNdx)) || (!NotDeleted) ) ) {
(*Index) = ScanContext.i;
return STATUS_SUCCESS;
}
}
return STATUS_OBJECT_NAME_NOT_FOUND;
}
if(hashes.hPosix == hashes.hLfn) {
while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
if(!DirNdx->FName.Buffer ||
(NotDeleted && UDFIsDeleted(DirNdx)) )
continue;
if( (DirNdx->hashes.hLfn == hashes.hLfn) &&
(!RtlCompareUnicodeString(&(DirNdx->FName), Name, IgnoreCase)) ) {
(*Index) = ScanContext.i;
return STATUS_SUCCESS;
} else
if( CanBe8d3 &&
!(DirNdx->FI_Flags & UDF_FI_FLAG_DOS) &&
(DirNdx->hashes.hDos == hashes.hLfn) &&
(k == (uint_di)(-1))) {
UDFDOSName(Vcb, &ShortName, &(DirNdx->FName), ScanContext.i < 2) ;
if(!RtlCompareUnicodeString(&ShortName, Name, IgnoreCase))
k = ScanContext.i;
}
}
} else {
while((DirNdx = UDFDirIndexScan(&ScanContext, NULL))) {
// perform sequential directory scan
if(!DirNdx->FName.Buffer ||
(NotDeleted && UDFIsDeleted(DirNdx)) )
continue;
if( (DirNdx->hashes.hPosix == hashes.hPosix) &&
(!RtlCompareUnicodeString(&(DirNdx->FName), Name, FALSE)) ) {
(*Index) = ScanContext.i;
return STATUS_SUCCESS;
} else
if( (DirNdx->hashes.hLfn == hashes.hLfn) &&
(j == (uint_di)(-1)) &&
(!RtlCompareUnicodeString(&(DirNdx->FName), Name, IgnoreCase)) ) {
j = ScanContext.i;
} else
if( CanBe8d3 &&
!(DirNdx->FI_Flags & UDF_FI_FLAG_DOS) &&
(DirNdx->hashes.hDos == hashes.hLfn) &&
(k == (uint_di)(-1))) {
UDFDOSName(Vcb, &ShortName, &(DirNdx->FName), ScanContext.i < 2 );
if(!RtlCompareUnicodeString(&ShortName, Name, IgnoreCase)) {
k = ScanContext.i;
}
}
}
}
if(j != (uint_di)(-1)) {
(*Index) = j;
return STATUS_SUCCESS;
} else
if(k != (uint_di)(-1)) {
(*Index) = k;
return STATUS_SUCCESS;
}
return STATUS_OBJECT_NAME_NOT_FOUND;
} // end UDFFindFile()
/*
This routine returns pointer to parent DirIndex
*/
PDIR_INDEX_HDR
UDFGetDirIndexByFileInfo(
IN PUDF_FILE_INFO FileInfo
)
{
ValidateFileInfo(FileInfo);
if(!FileInfo) {
BrutePoint();
return NULL;
}
if (FileInfo->ParentFile) {
ValidateFileInfo(FileInfo->ParentFile);
if(UDFIsAStreamDir(FileInfo))
return NULL;
if(FileInfo->ParentFile->Dloc)
return FileInfo->ParentFile->Dloc->DirIndex;
return NULL;
}
if(FileInfo->Dloc)
return FileInfo->Dloc->DirIndex;
return NULL;
}
/*
File Data Location support routines (UDFXxxDloc)
This group is responsible for caching FE locations
If requested FE referenced by another FI the file is assumed to be linked
All linked files reference to common Data Location (& attr) structure
*/
/*
Check if given FE is already in use
*/
LONG
UDFFindDloc(
IN PVCB Vcb,
IN uint32 Lba
)
{
PUDF_DATALOC_INDEX DlocList;
uint32 l;
if(!(DlocList = Vcb->DlocList) || !Lba) return (-1);
// scan FE location cache
l = Vcb->DlocCount;
for(uint32 i=0; i<l; i++, DlocList++) {
if(DlocList->Lba == Lba)
return i;
}
return (-1);
} // end UDFFindDloc()
/*
Check if given FE is already stored in memory
*/
LONG
UDFFindDlocInMem(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
PUDF_DATALOC_INDEX DlocList;
uint32 l;
if(!(DlocList = Vcb->DlocList) || !Dloc) return (-1);
// scan FE location cache
l = Vcb->DlocCount;
for(uint32 i=0; i<l; i++, DlocList++) {
if(DlocList->Dloc == Dloc)
return i;
}
return (-1);
} // end UDFFindDlocInMem()
/*
Find free cache entry
*/
LONG
UDFFindFreeDloc(
IN PVCB Vcb,
IN uint32 Lba
)
{
PUDF_DATALOC_INDEX DlocList;
uint32 l;
if(!Vcb->DlocList) {
// init FE location cache
if(!(Vcb->DlocList = (PUDF_DATALOC_INDEX)MyAllocatePoolTag__(NonPagedPool, sizeof(UDF_DATALOC_INDEX)*DLOC_LIST_GRANULARITY, MEM_DLOC_NDX_TAG)))
return (-1);
RtlZeroMemory(Vcb->DlocList, DLOC_LIST_GRANULARITY*sizeof(UDF_DATALOC_INDEX));
Vcb->DlocCount = DLOC_LIST_GRANULARITY;
}
// scan for free entry
DlocList = Vcb->DlocList;
l = Vcb->DlocCount;
for(uint32 i=0; i<l; i++, DlocList++) {
if(!DlocList->Dloc)
return i;
}
// alloc some free entries
if(!MyReallocPool__((int8*)(Vcb->DlocList), Vcb->DlocCount*sizeof(UDF_DATALOC_INDEX),
(int8**)&(Vcb->DlocList), (Vcb->DlocCount+DLOC_LIST_GRANULARITY)*sizeof(UDF_DATALOC_INDEX))) {
return (-1);
}
RtlZeroMemory(&(Vcb->DlocList[Vcb->DlocCount]), DLOC_LIST_GRANULARITY*sizeof(UDF_DATALOC_INDEX));
Vcb->DlocCount += DLOC_LIST_GRANULARITY;
return (Vcb->DlocCount - DLOC_LIST_GRANULARITY);
} // end UDFFindFreeDloc()
/*
*/
OSSTATUS
UDFAcquireDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
UDFAcquireResourceExclusive(&(Vcb->DlocResource2),TRUE);
if(Dloc->FE_Flags & UDF_FE_FLAG_UNDER_INIT) {
UDFReleaseResource(&(Vcb->DlocResource2));
return STATUS_SHARING_PAUSED;
}
Dloc->FE_Flags |= UDF_FE_FLAG_UNDER_INIT;
UDFReleaseResource(&(Vcb->DlocResource2));
return STATUS_SUCCESS;
} // end UDFAcquireDloc()
/*
*/
OSSTATUS
UDFReleaseDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
UDFAcquireResourceExclusive(&(Vcb->DlocResource2),TRUE);
Dloc->FE_Flags &= ~UDF_FE_FLAG_UNDER_INIT;
UDFReleaseResource(&(Vcb->DlocResource2));
return STATUS_SUCCESS;
} // end UDFReleaseDloc()
/*
Try to store FE location in cache
If it is already in use, caller will be informed about it
*/
OSSTATUS
UDFStoreDloc(
IN PVCB Vcb,
IN PUDF_FILE_INFO fi,
IN uint32 Lba
)
{
LONG i;
PUDF_DATALOC_INFO Dloc;
if(!Lba) return STATUS_INVALID_PARAMETER;
if(Lba == (ULONG)-1) return STATUS_INVALID_PARAMETER;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
// check if FE specified is already in use
if((i = UDFFindDloc(Vcb, Lba)) == (-1)) {
// not used
if((i = UDFFindFreeDloc(Vcb, Lba)) == (-1)) {
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_INSUFFICIENT_RESOURCES;
}
} else {
if(!OS_SUCCESS(UDFAcquireDloc(Vcb, Dloc = Vcb->DlocList[i].Dloc))) {
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_SHARING_PAUSED;
}
// update caller's structures & exit
fi->Dloc = Dloc;
UDFReleaseDloc(Vcb, Dloc);
#if defined UDF_DBG && !defined _CONSOLE
if(fi->Dloc->CommonFcb) {
ASSERT((uint32)(fi->Dloc->CommonFcb) != 0xDEADDA7A);
ASSERT(fi->Dloc->CommonFcb->CommonFCBHeader.NodeTypeCode == UDF_NODE_TYPE_NT_REQ_FCB);
}
#endif // UDF_DBG
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_SUCCESS;
}
// allocate common DataLocation (Dloc) descriptor
Dloc = fi->Dloc = (PUDF_DATALOC_INFO)MyAllocatePoolTag__(UDF_DATALOC_INFO_MT, sizeof(UDF_DATALOC_INFO), MEM_DLOC_INF_TAG);
if(!Dloc) {
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_INSUFFICIENT_RESOURCES;
}
Vcb->DlocList[i].Lba = Lba;
Vcb->DlocList[i].Dloc = Dloc;
RtlZeroMemory(Dloc, sizeof(UDF_DATALOC_INFO));
Dloc->LinkedFileInfo = fi;
UDFAcquireDloc(Vcb, Dloc);
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_SUCCESS;
} // end UDFStoreDloc()
/*
Remove unreferenced FE location from cache & free allocated memory
This routine must be invoked when there are no more opened files
associated with given FE
*/
OSSTATUS
UDFRemoveDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
LONG i;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
if((i = UDFFindDlocInMem(Vcb, Dloc)) == (-1)) {
// FE specified is not in cache. exit
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_INVALID_PARAMETER;
}
// remove from cache
ASSERT(Vcb->DlocList);
RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
UDFReleaseResource(&(Vcb->DlocResource));
MyFreePool__(Dloc);
return STATUS_SUCCESS;
} // end UDFRemoveDloc()
/*
Remove unlinked FE location from cache & keep allocated memory
This routine must be invoked when there are no more opened files
associated with given FE
*/
OSSTATUS
UDFUnlinkDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
LONG i;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
if((i = UDFFindDlocInMem(Vcb, Dloc)) == (-1)) {
// FE specified is not in cache. exit
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_INVALID_PARAMETER;
}
// remove from cache
ASSERT(Vcb->DlocList);
RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
UDFReleaseResource(&(Vcb->DlocResource));
return STATUS_SUCCESS;
} // end UDFUnlinkDloc()
/*
This routine releases memory allocated for Dloc & removes it from
cache (if it is still there)
*/
void
UDFFreeDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc
)
{
LONG i;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
if((i = UDFFindDlocInMem(Vcb, Dloc)) != (-1)) {
ASSERT(Vcb->DlocList);
RtlZeroMemory(&(Vcb->DlocList[i]), sizeof(UDF_DATALOC_INDEX));
}
UDFReleaseResource(&(Vcb->DlocResource));
MyFreePool__(Dloc);
} // end UDFFreeDloc()
/*
This routine updates Dloc LBA after relocation
*/
void
UDFRelocateDloc(
IN PVCB Vcb,
IN PUDF_DATALOC_INFO Dloc,
IN uint32 NewLba
)
{
LONG i;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
if((i = UDFFindDlocInMem(Vcb, Dloc)) != (-1)) {
ASSERT(Vcb->DlocList);
Vcb->DlocList[i].Lba = NewLba;
}
UDFReleaseResource(&(Vcb->DlocResource));
} // end UDFRelocateDloc()
/*
Release FE cache
*/
void
UDFReleaseDlocList(
IN PVCB Vcb
)
{
if(!Vcb->DlocList) return;
UDFAcquireResourceExclusive(&(Vcb->DlocResource),TRUE);
for(uint32 i=0; i<Vcb->DlocCount; i++) {
if(Vcb->DlocList[i].Dloc)
MyFreePool__(Vcb->DlocList[i].Dloc);
}
MyFreePool__(Vcb->DlocList);
Vcb->DlocList = NULL;
Vcb->DlocCount = 0;
UDFReleaseResource(&(Vcb->DlocResource));
} // end UDFReleaseDlocList()
/*
This routine walks through Linked/Parallel FI chain and looks for
FE with same Index & Parent File
*/
PUDF_FILE_INFO
UDFLocateParallelFI(
PUDF_FILE_INFO di, // parent FileInfo
uint_di i, // Index
PUDF_FILE_INFO fi // FileInfo to start search from
)
{
PUDF_FILE_INFO ParFileInfo = fi->NextLinkedFile;
// PUDF_DATALOC_INFO Dloc = di->Dloc;
while((ParFileInfo != fi) &&
((ParFileInfo->ParentFile != di) ||
(ParFileInfo->Index != i)) ) {
ParFileInfo = ParFileInfo->NextLinkedFile;
}
return ParFileInfo;
// BrutePoint();
} // end UDFLocateParallelFI()
/*
This routine walks through Linked/Parallel FI chain and looks for
FE with same Index & Parent Dloc
*/
PUDF_FILE_INFO
UDFLocateAnyParallelFI(
PUDF_FILE_INFO fi // FileInfo to start search from
)
{
if(!fi->ParentFile) {
if(fi->NextLinkedFile == fi)
return NULL;
return fi->NextLinkedFile;
}
PUDF_FILE_INFO ParFileInfo = fi->NextLinkedFile;
PUDF_DATALOC_INFO Dloc = fi->ParentFile->Dloc;
uint_di i = fi->Index;
BOOLEAN NotFound = TRUE;
while((ParFileInfo != fi) &&
(NotFound =
((ParFileInfo->Index != i) ||
(ParFileInfo->ParentFile->Dloc != Dloc))) ) {
ParFileInfo = ParFileInfo->NextLinkedFile;
}
/* if(NotFound) {
if((ParFileInfo->Index == i) &&
(ParFileInfo->ParentFile->Dloc == Dloc))
return ParFileInfo;
return NULL;
}
return ParFileInfo;*/
return NotFound ? NULL : ParFileInfo;
// BrutePoint();
} // end UDFLocateAnyParallelFI()
void
UDFInsertLinkedFile(
PUDF_FILE_INFO fi, // FileInfo to be added to chain
PUDF_FILE_INFO fi2 // any FileInfo fro the chain
)
{
fi->NextLinkedFile = fi2->NextLinkedFile;
fi->PrevLinkedFile = fi2;
fi->NextLinkedFile->PrevLinkedFile =
fi->PrevLinkedFile->NextLinkedFile = fi;
return;
} // end UDFInsertLinkedFile()