NT4/private/ntos/fastfat/dirsup.c
2020-09-30 17:12:29 +02:00

3231 lines
86 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
DirSup.c
Abstract:
This module implements the dirent support routines for Fat.
Author:
DavidGoebel [DavidGoe] 08-Nov-90
--*/
#include "FatProcs.h"
//
// The Bug check file id for this module
//
#define BugCheckFileId (FAT_BUG_CHECK_DIRSUP)
//
// Local debug trace level
//
#define Dbg (DEBUG_TRACE_DIRSUP)
//
// The following three macro all assume the input dirent has been zeroed.
//
//
// VOID
// FatConstructDot (
// IN PIRP_CONTEXT IrpContext,
// IN PDCB Directory,
// IN PDIRENT ParentDirent,
// IN OUT PDIRENT Dirent
// );
//
// The following macro is called to initalize the "." dirent.
//
#define FatConstructDot(IRPCONTEXT,DCB,PARENT,DIRENT) { \
\
RtlCopyMemory( (PUCHAR)(DIRENT), ". ", 11 ); \
(DIRENT)->Attributes = FAT_DIRENT_ATTR_DIRECTORY; \
(DIRENT)->LastWriteTime = (PARENT)->LastWriteTime; \
if (FatData.ChicagoMode) { \
(DIRENT)->CreationTime = (PARENT)->CreationTime; \
(DIRENT)->CreationMSec = (PARENT)->CreationMSec; \
(DIRENT)->LastAccessDate = (PARENT)->LastAccessDate; \
} \
(DIRENT)->FirstClusterOfFile = (FAT_ENTRY)(DCB)->FirstClusterOfFile; \
}
//
// VOID
// FatConstructDotDot (
// IN PIRP_CONTEXT IrpContext,
// IN PDCB Directory,
// IN PDIRENT ParentDirent,
// IN OUT PDIRENT Dirent
// );
//
// The following macro is called to initalize the ".." dirent.
//
#define FatConstructDotDot(IRPCONTEXT,DCB,PARENT,DIRENT) { \
\
RtlCopyMemory( (PUCHAR)(DIRENT), ".. ", 11 ); \
(DIRENT)->Attributes = FAT_DIRENT_ATTR_DIRECTORY; \
(DIRENT)->LastWriteTime = (PARENT)->LastWriteTime; \
if (FatData.ChicagoMode) { \
(DIRENT)->CreationTime = (PARENT)->CreationTime; \
(DIRENT)->CreationMSec = (PARENT)->CreationMSec; \
(DIRENT)->LastAccessDate = (PARENT)->LastAccessDate; \
} \
(DIRENT)->FirstClusterOfFile = (FAT_ENTRY) ( \
NodeType((DCB)->ParentDcb) == FAT_NTC_ROOT_DCB ? \
0 : (DCB)->ParentDcb->FirstClusterOfFile); \
}
//
// VOID
// FatConstructEndDirent (
// IN PIRP_CONTEXT IrpContext,
// IN OUT PDIRENT Dirent
// );
//
// The following macro created the end dirent. Note that since the
// dirent was zeroed, the first byte of the name already contains 0x0,
// so there is nothing to do.
//
#define FatConstructEndDirent(IRPCONTEXT,DIRENT) NOTHING
//
// VOID
// FatReadDirent (
// IN PIRP_CONTEXT IrpContext,
// IN PDCB Dcb,
// IN VBO Vbo,
// OUT PBCB *Bcb,
// OUT PVOID *Dirent,
// OUT PNTSTATUS Status
// );
//
//
// This macro reads in a page of dirents when we step onto a new page,
// or this is the first itteration of a loop and Bcb is NULL.
//
#define FatReadDirent(IRPCONTEXT,DCB,VBO,BCB,DIRENT,STATUS) \
if ((VBO) >= (DCB)->Header.AllocationSize.LowPart) { \
*(STATUS) = STATUS_END_OF_FILE; \
FatUnpinBcb( (IRPCONTEXT), *(BCB) ); \
} else if ( ((VBO) % PAGE_SIZE == 0) || (*(BCB) == NULL) ) { \
FatUnpinBcb( (IRPCONTEXT), *(BCB) ); \
FatReadDirectoryFile( (IRPCONTEXT), \
(DCB), \
(VBO) & ~(PAGE_SIZE - 1), \
PAGE_SIZE, \
FALSE, \
(BCB), \
(PVOID *)(DIRENT), \
(STATUS) ); \
*(DIRENT) = (PVOID)((PUCHAR)*(DIRENT) + ((VBO) % PAGE_SIZE)); \
}
//
// Internal support routines
//
UCHAR
FatComputeLfnChecksum (
PDIRENT Dirent
);
VOID
FatRescanDirectory (
PIRP_CONTEXT IrpContext,
PDCB Dcb
);
ULONG
FatDefragDirectory (
IN PIRP_CONTEXT IrpContext,
IN PDCB Dcb,
IN ULONG DirentsNeeded
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, FatComputeLfnChecksum)
#pragma alloc_text(PAGE, FatConstructDirent)
#pragma alloc_text(PAGE, FatConstructLabelDirent)
#pragma alloc_text(PAGE, FatCreateNewDirent)
#pragma alloc_text(PAGE, FatDefragDirectory)
#pragma alloc_text(PAGE, FatDeleteDirent)
#pragma alloc_text(PAGE, FatGetDirentFromFcbOrDcb)
#pragma alloc_text(PAGE, FatInitializeDirectoryDirent)
#pragma alloc_text(PAGE, FatIsDirectoryEmpty)
#pragma alloc_text(PAGE, FatLfnDirentExists)
#pragma alloc_text(PAGE, FatLocateDirent)
#pragma alloc_text(PAGE, FatLocateSimpleOemDirent)
#pragma alloc_text(PAGE, FatLocateVolumeLabel)
#pragma alloc_text(PAGE, FatRescanDirectory)
#pragma alloc_text(PAGE, FatSetFileSizeInDirent)
#pragma alloc_text(PAGE, FatTunnelFcbOrDcb)
#endif
ULONG
FatCreateNewDirent (
IN PIRP_CONTEXT IrpContext,
IN PDCB ParentDirectory,
IN ULONG DirentsNeeded
)
/*++
Routine Description:
This routine allocates on the disk a new dirent inside of the
parent directory. If a new dirent cannot be allocated (i.e.,
because the disk is full or the root directory is full) then
it raises the appropriate status. The dirent itself is
neither initialized nor pinned by this procedure.
Arguments:
ParentDirectory - Supplies the DCB for the directory in which
to create the new dirent
DirentsNeeded - This is the number of continginous dirents required
Return Value:
ByteOffset - Returns the VBO within the Parent directory where
the dirent has been allocated
--*/
{
VBO UnusedVbo;
VBO DeletedHint;
ULONG ByteOffset;
PBCB Bcb = NULL;
PDIRENT Dirent;
NTSTATUS Status;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatCreateNewDirent\n", 0);
DebugTrace( 0, Dbg, " ParentDirectory = %08lx\n", ParentDirectory);
//
// If UnusedDirentVbo is within our current file allocation then we
// don't have to search through the directory at all; we know just
// where to put it.
//
// If UnusedDirentVbo is beyond the current file allocation then
// there are no more unused dirents in the current allocation, though
// upon adding another cluster of allocation UnusedDirentVbo
// will point to an unused dirent. Haveing found no unused dirents
// we use the DeletedDirentHint to try and find a deleted dirent in
// the current allocation. In this also runs off the end of the file,
// we finally have to break down and allocate another sector. Note
// that simply writing beyond the current allocation will automatically
// do just this.
//
// We also must deal with the special case where UnusedDirentVbo and
// DeletedDirentHint have yet to be initialized. In this case we must
// first walk through the directory looking for the first deleted entry
// first unused dirent. After this point we continue as before.
// This virgin state is denoted by the special value of 0xffffffff.
//
UnusedVbo = ParentDirectory->Specific.Dcb.UnusedDirentVbo;
DeletedHint = ParentDirectory->Specific.Dcb.DeletedDirentHint;
//
// Check for our first call to this routine with this Dcb. If so
// we have to correctly set the two hints in the Dcb.
//
if (UnusedVbo == 0xffffffff) {
FatRescanDirectory( IrpContext, ParentDirectory );
UnusedVbo = ParentDirectory->Specific.Dcb.UnusedDirentVbo;
DeletedHint = ParentDirectory->Specific.Dcb.DeletedDirentHint;
}
//
// Now we know that UnusedDirentVbo and DeletedDirentHint are correctly
// set so we check if there is already an unused dirent in the the
// current allocation. This is the easy case.
//
DebugTrace( 0, Dbg, " UnusedVbo = %08lx\n", UnusedVbo);
DebugTrace( 0, Dbg, " DeletedHint = %08lx\n", DeletedHint);
if ( UnusedVbo + (DirentsNeeded * sizeof(DIRENT)) <=
ParentDirectory->Header.AllocationSize.LowPart ) {
//
// Get this unused dirent for the caller. We have a
// sporting chance that we won't have to wait.
//
DebugTrace( 0, Dbg, "There is a never used entry.\n", 0);
ByteOffset = UnusedVbo;
UnusedVbo += DirentsNeeded * sizeof(DIRENT);
} else {
//
// Life is tough. We have to march from the DeletedDirentHint
// looking for a deleted dirent. If we get to EOF without finding
// one, we will have to allocate a new cluster.
//
ByteOffset =
RtlFindClearBits( &ParentDirectory->Specific.Dcb.FreeDirentBitmap,
DirentsNeeded,
DeletedHint / sizeof(DIRENT) );
//
// Do a quick check for a root directory allocation that failed
// simply because of fragmentation. Also, only attempt to defrag
// if the length is less that 0x40000. This is to avoid
// complications arrising from crossing a cache manager VMCB block
// (by default on DOS the root directory is only 0x2000 long).
//
if ((ByteOffset == -1) &&
(NodeType(ParentDirectory) == FAT_NTC_ROOT_DCB) &&
(ParentDirectory->Header.AllocationSize.LowPart <= 0x40000)) {
ByteOffset = FatDefragDirectory( IrpContext, ParentDirectory, DirentsNeeded );
}
if (ByteOffset != -1) {
//
// If we consuemed deleted dirents at Deleted Hint, update.
// We also may have consumed some un-used dirents as well,
// so be sure to check for that as well.
//
ByteOffset *= sizeof(DIRENT);
if (ByteOffset == DeletedHint) {
DeletedHint += DirentsNeeded * sizeof(DIRENT);
}
if (ByteOffset + DirentsNeeded * sizeof(DIRENT) > UnusedVbo) {
UnusedVbo = ByteOffset + DirentsNeeded * sizeof(DIRENT);
}
} else {
//
// We are going to have to allocate another cluster. Do
// so, update both the UnusedVbo and the DeletedHint and bail.
//
DebugTrace( 0, Dbg, "We have to allocate another cluster.\n", 0);
//
// Make sure we are not trying to expand the root directory.
//
if ( NodeType(ParentDirectory) == FAT_NTC_ROOT_DCB ) {
DebugTrace(0, Dbg, "Disk Full. Raise Status.\n", 0);
FatRaiseStatus( IrpContext, STATUS_DISK_FULL );
}
//
// Take the last dirent(s) in this cluster. We will allocate
// more clusters below.
//
ByteOffset = UnusedVbo;
UnusedVbo += DirentsNeeded * sizeof(DIRENT);
//
// Touch the directory file to cause space for the new dirents
// to be allocated.
//
Bcb = NULL;
try {
ULONG ClusterSize;
PVOID Buffer;
ClusterSize =
1 << ParentDirectory->Vcb->AllocationSupport.LogOfBytesPerCluster;
FatPrepareWriteDirectoryFile( IrpContext,
ParentDirectory,
UnusedVbo,
1,
&Bcb,
&Buffer,
FALSE,
&Status );
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
}
}
//
// If we are only requesting a single dirent, and we did not get the
// first dirent in a directory, then check that the preceding dirent
// is not an orphaned LFN. If it is, then mark it deleted. Thus
// reducing the possibility of an accidental pairing.
//
// Only do this when we are in Chicago Mode.
//
Bcb = NULL;
if (FatData.ChicagoMode &&
(DirentsNeeded == 1) &&
(ByteOffset > (NodeType(ParentDirectory) == FAT_NTC_ROOT_DCB ?
0 : 2 * sizeof(DIRENT)))) {
try {
FatReadDirent( IrpContext,
ParentDirectory,
ByteOffset - sizeof(DIRENT),
&Bcb,
&Dirent,
&Status );
if ((Status != STATUS_SUCCESS) ||
(Dirent->FileName[0] == FAT_DIRENT_NEVER_USED)) {
FatPopUpFileCorrupt( IrpContext, ParentDirectory );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
if ((Dirent->Attributes == FAT_DIRENT_ATTR_LFN) &&
(Dirent->FileName[0] != FAT_DIRENT_DELETED)) {
//
// Pin it, mark it, and set it dirty.
//
FatPinMappedData( IrpContext,
ParentDirectory,
ByteOffset - sizeof(DIRENT),
sizeof(DIRENT),
&Bcb );
Dirent->FileName[0] = FAT_DIRENT_DELETED;
FatSetDirtyBcb( IrpContext, Bcb, ParentDirectory->Vcb );
ASSERT( RtlAreBitsSet( &ParentDirectory->Specific.Dcb.FreeDirentBitmap,
(ByteOffset - sizeof(DIRENT))/ sizeof(DIRENT),
DirentsNeeded ) );
RtlClearBits( &ParentDirectory->Specific.Dcb.FreeDirentBitmap,
(ByteOffset - sizeof(DIRENT))/ sizeof(DIRENT),
DirentsNeeded );
}
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
}
//
// Assert that the dirents are in fact unused
//
try {
ULONG i;
Bcb = NULL;
for (i = 0; i < DirentsNeeded; i++) {
FatReadDirent( IrpContext,
ParentDirectory,
ByteOffset + i*sizeof(DIRENT),
&Bcb,
&Dirent,
&Status );
if ((Status != STATUS_SUCCESS) ||
((Dirent->FileName[0] != FAT_DIRENT_NEVER_USED) &&
(Dirent->FileName[0] != FAT_DIRENT_DELETED))) {
FatPopUpFileCorrupt( IrpContext, ParentDirectory );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
}
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
//
// Set the Bits in the bitmap and move the Unused Dirent Vbo.
//
ASSERT( RtlAreBitsClear( &ParentDirectory->Specific.Dcb.FreeDirentBitmap,
ByteOffset / sizeof(DIRENT),
DirentsNeeded ) );
RtlSetBits( &ParentDirectory->Specific.Dcb.FreeDirentBitmap,
ByteOffset / sizeof(DIRENT),
DirentsNeeded );
//
// Save the newly computed values in the Parent Directory Fcb
//
ParentDirectory->Specific.Dcb.UnusedDirentVbo = UnusedVbo;
ParentDirectory->Specific.Dcb.DeletedDirentHint = DeletedHint;
DebugTrace(-1, Dbg, "FatCreateNewDirent -> (VOID)\n", 0);
return ByteOffset;
}
VOID
FatInitializeDirectoryDirent (
IN PIRP_CONTEXT IrpContext,
IN PDCB Dcb,
IN PDIRENT ParentDirent
)
/*++
Routine Description:
This routine converts a dirent into a directory on the disk. It does this
setting the directory flag in the dirent, and by allocating the necessary
space for the "." and ".." dirents and initializing them.
If a new dirent cannot be allocated (i.e., because the disk is full) then
it raises the appropriate status.
Arguments:
Dcb - Supplies the Dcb denoting the file that is to be made into a
directory. This must be input a completely empty file with
an allocation size of zero.
ParentDirent - Provides the parent Dirent for a time-stamp model.
Return Value:
None.
--*/
{
PBCB Bcb;
PVOID Buffer;
NTSTATUS DontCare;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatInitializeDirectoryDirent\n", 0);
DebugTrace( 0, Dbg, " Dcb = %08lx\n", Dcb);
//
// Assert that we are not attempting this on the root directory.
//
ASSERT( NodeType(Dcb) != FAT_NTC_ROOT_DCB );
//
// Assert that this is only attempted on newly created directories.
//
ASSERT( Dcb->Header.AllocationSize.LowPart == 0 );
//
// Prepare the directory file for writing. Note that we can use a single
// Bcb for these two entries because we know they are the first two in
// the directory, and thus together do not span a page boundry. Also
// note that we prepare write 2 entries: one for "." and one for "..".
// The end of directory marker is automatically set since the whole
// directory is initially zero (DIRENT_NEVER_USED).
//
FatPrepareWriteDirectoryFile( IrpContext,
Dcb,
0,
2 * sizeof(DIRENT),
&Bcb,
&Buffer,
FALSE,
&DontCare );
ASSERT( NT_SUCCESS( DontCare ));
//
// Add the . and .. entries
//
try {
FatConstructDot( IrpContext, Dcb, ParentDirent, (PDIRENT)Buffer + 0);
FatConstructDotDot( IrpContext, Dcb, ParentDirent, (PDIRENT)Buffer + 1);
//
// Unpin the buffer and return to the caller.
//
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
DebugTrace(-1, Dbg, "FatInitializeDirectoryDirent -> (VOID)\n", 0);
return;
}
VOID
FatTunnelFcbOrDcb (
IN PFCB FcbOrDcb,
IN PCCB Ccb OPTIONAL
)
/*++
Routine Description:
This routine handles tunneling of an Fcb or Dcb associated with
an object whose name is disappearing from a directory.
Arguments:
FcbOrDcb - Supplies the Fcb/Dcb whose name will be going away
Ccb - Supplies the Ccb for the Fcb (not reqired for a Dcb) so
that we know which name the Fcb was opened by
Return Value:
None.
--*/
{
UNICODE_STRING ShortNameWithCase;
UNICODE_STRING DownCaseSeg;
WCHAR ShortNameBuffer[8+1+3];
NTSTATUS Status;
USHORT i;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatTunnelFcbOrDcb\n", 0);
if (FcbOrDcb->Header.NodeTypeCode == FAT_NTC_DCB) {
//
// Directory deletion. Flush all entries from this directory in
// the cache for this volume
//
FsRtlDeleteKeyFromTunnelCache( &FcbOrDcb->Vcb->Tunnel,
FatDirectoryKey(FcbOrDcb) );
} else {
//
// Was a file, so throw it into the tunnel cache
//
//
// Get the short name into UNICODE
//
ShortNameWithCase.Length = 0;
ShortNameWithCase.MaximumLength = sizeof(ShortNameBuffer);
ShortNameWithCase.Buffer = ShortNameBuffer;
Status = RtlOemStringToCountedUnicodeString( &ShortNameWithCase,
&FcbOrDcb->ShortName.Name.Oem,
FALSE);
ASSERT(ShortNameWithCase.Length != 0);
ASSERT(NT_SUCCESS(Status));
if (FlagOn(FcbOrDcb->FcbState, FCB_STATE_8_LOWER_CASE | FCB_STATE_3_LOWER_CASE)) {
//
// Have to repair the case of the short name
//
for (i = 0; i < (ShortNameWithCase.Length/sizeof(WCHAR)) &&
ShortNameWithCase.Buffer[i] != L'.'; i++);
//
// Now pointing at the '.', or otherwise the end of name component
//
if (FlagOn(FcbOrDcb->FcbState, FCB_STATE_8_LOWER_CASE)) {
DownCaseSeg.Buffer = ShortNameWithCase.Buffer;
DownCaseSeg.MaximumLength = DownCaseSeg.Length = i*sizeof(WCHAR);
FatDowncaseUnicodeString(&DownCaseSeg, &DownCaseSeg, FALSE);
}
i++;
//
// Now pointing at first wchar of the extension.
//
if (FlagOn(FcbOrDcb->FcbState, FCB_STATE_3_LOWER_CASE)) {
//
// It is not neccesarily the case that we can rely on the flag
// indicating that we really have an extension.
//
if ((i*sizeof(WCHAR)) < ShortNameWithCase.Length) {
DownCaseSeg.Buffer = &ShortNameWithCase.Buffer[i];
DownCaseSeg.MaximumLength = DownCaseSeg.Length = ShortNameWithCase.Length - i*sizeof(WCHAR);
FatDowncaseUnicodeString(&DownCaseSeg, &DownCaseSeg, FALSE);
}
}
}
//
// ... and add it in
//
FsRtlAddToTunnelCache( &FcbOrDcb->Vcb->Tunnel,
FatDirectoryKey(FcbOrDcb->ParentDcb),
&ShortNameWithCase,
&FcbOrDcb->ExactCaseLongName,
BooleanFlagOn(Ccb->Flags, CCB_FLAG_OPENED_BY_SHORTNAME),
sizeof(LARGE_INTEGER),
&FcbOrDcb->CreationTime );
}
DebugTrace(-1, Dbg, "FatTunnelFcbOrDcb -> (VOID)\n", 0);
return;
}
VOID
FatDeleteDirent (
IN PIRP_CONTEXT IrpContext,
IN PFCB FcbOrDcb,
IN PDELETE_CONTEXT DeleteContext OPTIONAL,
IN BOOLEAN DeleteEa
)
/*++
Routine Description:
This routine Deletes on the disk the indicated dirent. It does
this by marking the dirent as deleted.
Arguments:
FcbOrDcb - Supplies the FCB/DCB for the file/directory being
deleted. For a file the file size and allocation must be zero.
(Zero allocation is implied by a zero cluster index).
For a directory the allocation must be zero.
DeleteContext - This variable, if speicified, may be used to preserve
the file size and first cluster of file information in the dirent
fot the benefit of unerase utilities.
DeleteEa - Tells us whether to delete the EA and whether to check
for no allocation/ Mainly TRUE. FALSE passed in from rename.
Return Value:
None.
--*/
{
PBCB Bcb = NULL;
PDIRENT Dirent;
NTSTATUS DontCare;
ULONG Offset;
ULONG DirentsToDelete;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatDeleteDirent\n", 0);
DebugTrace( 0, Dbg, " FcbOrDcb = %08lx\n", FcbOrDcb);
//
// Assert that we are not attempting this on the root directory.
//
ASSERT( NodeType(FcbOrDcb) != FAT_NTC_ROOT_DCB );
//
// Make sure all requests have zero allocation/file size
//
if (DeleteEa &&
((FcbOrDcb->Header.AllocationSize.LowPart != 0) ||
((NodeType(FcbOrDcb) == FAT_NTC_FCB) &&
(FcbOrDcb->Header.FileSize.LowPart != 0)))) {
DebugTrace( 0, Dbg, "Called with non zero allocation/file size.\n", 0);
FatBugCheck( 0, 0, 0 );
}
//
// Now, mark the dirents deleted, unpin the Bcb, and return to the caller.
// Assert that there isn't any allocation associated with this dirent.
//
// Note that this loop will end with Dirent pointing to the short name.
//
try {
for ( Offset = FcbOrDcb->LfnOffsetWithinDirectory;
Offset <= FcbOrDcb->DirentOffsetWithinDirectory;
Offset += sizeof(DIRENT), Dirent += 1 ) {
//
// If we stepped onto a new page, or this is the first itteration,
// unpin the old page, and pin the new one.
//
if ((Offset == FcbOrDcb->LfnOffsetWithinDirectory) ||
((Offset & (PAGE_SIZE - 1)) == 0)) {
FatUnpinBcb( IrpContext, Bcb );
FatPrepareWriteDirectoryFile( IrpContext,
FcbOrDcb->ParentDcb,
Offset,
sizeof(DIRENT),
&Bcb,
(PVOID *)&Dirent,
FALSE,
&DontCare );
}
ASSERT( (Dirent->FirstClusterOfFile == 0) || !DeleteEa );
Dirent->FileName[0] = FAT_DIRENT_DELETED;
}
//
// Back Dirent off by one to point back to the short dirent.
//
Dirent -= 1;
//
// If there are extended attributes for this dirent, we will attempt
// to remove them. We ignore any errors in removing Eas.
//
if (DeleteEa && (Dirent->ExtendedAttributes != 0)) {
try {
FatDeleteEa( IrpContext,
FcbOrDcb->Vcb,
Dirent->ExtendedAttributes,
&FcbOrDcb->ShortName.Name.Oem );
} except(FatExceptionFilter( IrpContext, GetExceptionInformation() )) {
//
// We catch all exceptions that Fat catches, but don't do
// anything with them.
//
}
}
//
// Now clear the bits in the free dirent mask.
//
DirentsToDelete = (FcbOrDcb->DirentOffsetWithinDirectory -
FcbOrDcb->LfnOffsetWithinDirectory) / sizeof(DIRENT) + 1;
ASSERT( (FcbOrDcb->ParentDcb->Specific.Dcb.UnusedDirentVbo == 0xffffffff) ||
RtlAreBitsSet( &FcbOrDcb->ParentDcb->Specific.Dcb.FreeDirentBitmap,
FcbOrDcb->LfnOffsetWithinDirectory / sizeof(DIRENT),
DirentsToDelete ) );
RtlClearBits( &FcbOrDcb->ParentDcb->Specific.Dcb.FreeDirentBitmap,
FcbOrDcb->LfnOffsetWithinDirectory / sizeof(DIRENT),
DirentsToDelete );
//
// Now, if the caller specified a DeleteContext, use it.
//
if ( ARGUMENT_PRESENT( DeleteContext ) ) {
Dirent->FileSize = DeleteContext->FileSize;
Dirent->FirstClusterOfFile = (FAT_ENTRY)DeleteContext->FirstClusterOfFile;
}
//
// If this newly deleted dirent is before the DeletedDirentHint, change
// the DeletedDirentHint to point here.
//
if (FcbOrDcb->DirentOffsetWithinDirectory <
FcbOrDcb->ParentDcb->Specific.Dcb.DeletedDirentHint) {
FcbOrDcb->ParentDcb->Specific.Dcb.DeletedDirentHint =
FcbOrDcb->LfnOffsetWithinDirectory;
}
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
DebugTrace(-1, Dbg, "FatDeleteDirent -> (VOID)\n", 0);
return;
}
BOOLEAN
FatLfnDirentExists (
IN PIRP_CONTEXT IrpContext,
IN PDCB Dcb,
IN PUNICODE_STRING Lfn,
IN PUNICODE_STRING LfnTmp
)
/*++
Routine Description:
This routine looks for a given Lfn in a directory
Arguments:
Dcb - The directory to search
Lfn - The Lfn to look for
Lfn - Temporary buffer to use to search for Lfn with (must be MAX_LFN)
Retrn Value:
BOOLEAN TRUE if it exists, FALSE if not
--*/
{
CCB Ccb;
PDIRENT Dirent;
PBCB DirentBcb;
VBO DirentByteOffset;
//
// Pay performance penalty by forcing the compares to be case insensitive as
// opposed to grabbing more pool for a monocased copy of the Lfn. This is slight.
//
Ccb.UnicodeQueryTemplate = *Lfn;
Ccb.ContainsWildCards = FALSE;
Ccb.Flags = CCB_FLAG_SKIP_SHORT_NAME_COMPARE | CCB_FLAG_QUERY_TEMPLATE_MIXED;
DirentBcb = NULL;
FatLocateDirent( IrpContext,
Dcb,
&Ccb,
0,
&Dirent,
&DirentBcb,
&DirentByteOffset,
NULL,
LfnTmp );
if (DirentBcb) {
FatUnpinBcb(IrpContext, DirentBcb);
return TRUE;
}
return FALSE;
}
VOID
FatLocateDirent (
IN PIRP_CONTEXT IrpContext,
IN PDCB ParentDirectory,
IN PCCB Ccb,
IN VBO OffsetToStartSearchFrom,
OUT PDIRENT *Dirent,
OUT PBCB *Bcb,
OUT PVBO ByteOffset,
OUT PBOOLEAN FileNameDos OPTIONAL,
IN OUT PUNICODE_STRING LongFileName OPTIONAL
)
/*++
Routine Description:
This routine locates on the disk an undeleted dirent matching a given name.
Arguments:
ParentDirectory - Supplies the DCB for the directory to search
Ccb - Contains a context control block with all matching information.
OffsetToStartSearchFrom - Supplies the VBO within the parent directory
from which to start looking for another real dirent.
Dirent - Receives a pointer to the located dirent if one was found
or NULL otherwise.
Bcb - Receives the Bcb for the located dirent if one was found or
NULL otherwise.
ByteOffset - Receives the VBO within the Parent directory for
the located dirent if one was found, or 0 otherwise.
FileNameDos - Receives TRUE if the element of the dirent we hit on
was the short (non LFN) side
LongFileName - If specified, this parameter returns the long file name
associated with the returned dirent. Note that it is the caller's
responsibility to provide the buffer (and set MaximumLength
accordingly) for this unicode string. The Length field is reset
to 0 by this routine on invocation.
Return Value:
None.
--*/
{
NTSTATUS Status;
OEM_STRING Name;
UCHAR NameBuffer[12];
UNICODE_STRING LocalUpcasedLfn;
UNICODE_STRING PoolUpcasedLfn;
PUNICODE_STRING UpcasedLfn;
WCHAR LocalLfnBuffer[32];
BOOLEAN LfnInProgress = FALSE;
UCHAR LfnChecksum;
ULONG LfnSize;
ULONG LfnIndex;
UCHAR Ordinal;
VBO LfnByteOffset;
TimerStart(Dbg);
PAGED_CODE();
DebugTrace(+1, Dbg, "FatLocateDirent\n", 0);
DebugTrace( 0, Dbg, " ParentDirectory = %08lx\n", ParentDirectory);
DebugTrace( 0, Dbg, " OffsetToStartSearchFrom = %08lx\n", OffsetToStartSearchFrom);
DebugTrace( 0, Dbg, " Dirent = %08lx\n", Dirent);
DebugTrace( 0, Dbg, " Bcb = %08lx\n", Bcb);
DebugTrace( 0, Dbg, " ByteOffset = %08lx\n", ByteOffset);
//
// The algorithm here is pretty simple. We just walk through the
// parent directory until we:
//
// A) Find a matching entry.
// B) Can't Wait
// C) Hit the End of Directory
// D) Hit Eof
//
// In the first case we found it, in the latter three cases we did not.
//
//
// Set up the strings that receives file names from our search
//
Name.MaximumLength = 12;
Name.Buffer = NameBuffer;
PoolUpcasedLfn.Length =
PoolUpcasedLfn.MaximumLength = 0;
PoolUpcasedLfn.Buffer = NULL;
LocalUpcasedLfn.Length = 0;
LocalUpcasedLfn.MaximumLength = 32*sizeof(WCHAR);
LocalUpcasedLfn.Buffer = LocalLfnBuffer;
//
// If we were given a non-NULL Bcb, compute the new Dirent address
// from the prior one, or unpin the Bcb if the new Dirent is not pinned.
//
if (*Bcb != NULL) {
if ((OffsetToStartSearchFrom / PAGE_SIZE) == (*ByteOffset / PAGE_SIZE)) {
*Dirent += (OffsetToStartSearchFrom - *ByteOffset) / sizeof(DIRENT);
} else {
FatUnpinBcb( IrpContext, *Bcb );
}
}
//
// Init the Lfn if we were given one.
//
if (ARGUMENT_PRESENT(LongFileName)) {
LongFileName->Length = 0;
}
//
// Init the FileNameDos flag
//
if (FileNameDos) {
*FileNameDos = FALSE;
}
//
// Round up OffsetToStartSearchFrom to the nearest Dirent, and store
// in ByteOffset. Note that this wipes out the prior value.
//
*ByteOffset = (OffsetToStartSearchFrom + (sizeof(DIRENT) - 1))
& ~(sizeof(DIRENT) - 1);
try {
while ( TRUE ) {
BOOLEAN FoundValidLfn;
//
// Try to read in the dirent
//
FatReadDirent( IrpContext,
ParentDirectory,
*ByteOffset,
Bcb,
Dirent,
&Status );
//
// If End Directory dirent or EOF, set all out parameters to
// indicate entry not found and, like, bail.
//
// Note that the order of evaluation here is important since we
// cannot check the first character of the dirent until after we
// know we are not beyond EOF
//
if ((Status == STATUS_END_OF_FILE) ||
((*Dirent)->FileName[0] == FAT_DIRENT_NEVER_USED)) {
DebugTrace( 0, Dbg, "End of directory: entry not found.\n", 0);
//
// If there is a Bcb, unpin it and set it to null
//
FatUnpinBcb( IrpContext, *Bcb );
*Dirent = NULL;
*ByteOffset = 0;
break;
}
//
// If we have wandered onto an LFN entry, try to interpret it.
//
if (FatData.ChicagoMode &&
ARGUMENT_PRESENT(LongFileName) &&
((*Dirent)->Attributes == FAT_DIRENT_ATTR_LFN)) {
PLFN_DIRENT Lfn;
Lfn = (PLFN_DIRENT)*Dirent;
if (LfnInProgress) {
//
// Check for a proper continuation of the Lfn in progress.
//
if ((Lfn->Ordinal & FAT_LAST_LONG_ENTRY) ||
(Lfn->Ordinal == 0) ||
(Lfn->Ordinal != Ordinal - 1) ||
(Lfn->Checksum != LfnChecksum) ||
(Lfn->MustBeZero != 0)) {
//
// The Lfn is not proper, stop constructing it.
//
LfnInProgress = FALSE;
} else {
ASSERT( ((LfnIndex % 13) == 0) && LfnIndex );
LfnIndex -= 13;
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+0],
&Lfn->Name1[0],
5*sizeof(WCHAR) );
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+5],
&Lfn->Name2[0],
6 * sizeof(WCHAR) );
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+11],
&Lfn->Name3[0],
2 * sizeof(WCHAR) );
Ordinal = Lfn->Ordinal;
LfnByteOffset = *ByteOffset;
}
}
//
// Now check (maybe again) if we should analyze this entry
// for a possible last entry.
//
if ((!LfnInProgress) &&
(Lfn->Ordinal & FAT_LAST_LONG_ENTRY) &&
((Lfn->Ordinal & ~FAT_LAST_LONG_ENTRY) <= MAX_LFN_DIRENTS) &&
(Lfn->MustBeZero == 0)) {
BOOLEAN CheckTail = FALSE;
Ordinal = Lfn->Ordinal & ~FAT_LAST_LONG_ENTRY;
LfnIndex = (Ordinal - 1) * 13;
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+0],
&Lfn->Name1[0],
5*sizeof(WCHAR));
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+5],
&Lfn->Name2[0],
6 * sizeof(WCHAR) );
RtlCopyMemory( &LongFileName->Buffer[LfnIndex+11],
&Lfn->Name3[0],
2 * sizeof(WCHAR) );
//
// Now compute the Lfn size and make sure that the tail
// bytes are correct.
//
while (LfnIndex != (ULONG)Ordinal * 13) {
if (!CheckTail) {
if (LongFileName->Buffer[LfnIndex] == 0x0000) {
LfnSize = LfnIndex;
CheckTail = TRUE;
}
} else {
if (LongFileName->Buffer[LfnIndex] != 0xffff) {
break;
}
}
LfnIndex += 1;
}
//
// If we exited this loop prematurely, the LFN is not valid.
//
if (LfnIndex == (ULONG)Ordinal * 13) {
//
// If we didn't find the NULL terminator, then the size
// is LfnIndex.
//
if (!CheckTail) {
LfnSize = LfnIndex;
}
LfnIndex -= 13;
LfnInProgress = TRUE;
LfnChecksum = Lfn->Checksum;
LfnByteOffset = *ByteOffset;
}
}
//
// Move on to the next dirent.
//
goto GetNextDirent;
}
//
// If the file is not deleted and is not the volume label, check
// for a match.
//
if ( ((*Dirent)->FileName[0] == FAT_DIRENT_DELETED) ||
FlagOn((*Dirent)->Attributes, FAT_DIRENT_ATTR_VOLUME_ID)) {
//
// Move on to the next dirent.
//
goto GetNextDirent;
}
//
// We may have just stepped off a valid Lfn run. Check to see if
// it is indeed valid for the following dirent.
//
if (LfnInProgress &&
(*ByteOffset == LfnByteOffset + sizeof(DIRENT)) &&
(LfnIndex == 0) &&
(FatComputeLfnChecksum(*Dirent) == LfnChecksum)) {
ASSERT( Ordinal == 1);
FoundValidLfn = TRUE;
LongFileName->Length = (USHORT)(LfnSize * sizeof(WCHAR));
} else {
FoundValidLfn = FALSE;
}
//
// If we are supposed to match all entries, then match this entry.
//
if (FlagOn(Ccb->Flags, CCB_FLAG_MATCH_ALL)) {
break;
}
//
// Check against the short name given if one was.
//
if (!FlagOn( Ccb->Flags, CCB_FLAG_SKIP_SHORT_NAME_COMPARE )) {
if (Ccb->ContainsWildCards) {
//
// If we get one, note that all out parameters are already set.
//
(VOID)Fat8dot3ToString( IrpContext, (*Dirent), FALSE, &Name );
//
// For fat we special case the ".." dirent because we want it to
// match ????????.??? and to do that we change ".." to "." before
// calling the Fsrtl routine. But only do this if the expression
// is greater than one character long.
//
if ((Name.Length == 2) &&
(Name.Buffer[0] == '.') &&
(Name.Buffer[1] == '.') &&
(Ccb->OemQueryTemplate.Wild.Length > 1)) {
Name.Length = 1;
}
if (FatIsNameInExpression( IrpContext,
Ccb->OemQueryTemplate.Wild,
Name)) {
DebugTrace( 0, Dbg, "Entry found: Name = \"%Z\"\n", &Name);
DebugTrace( 0, Dbg, " VBO = %08lx\n", *ByteOffset);
if (FileNameDos) {
*FileNameDos = TRUE;
}
break;
}
} else {
//
// Do the quickest 8.3 equivalency check possible
//
if (!FlagOn((*Dirent)->Attributes, FAT_DIRENT_ATTR_VOLUME_ID) &&
(*(PULONG)&(Ccb->OemQueryTemplate.Constant[0]) == *(PULONG)&((*Dirent)->FileName[0])) &&
(*(PULONG)&(Ccb->OemQueryTemplate.Constant[4]) == *(PULONG)&((*Dirent)->FileName[4])) &&
(*(PUSHORT)&(Ccb->OemQueryTemplate.Constant[8]) == *(PUSHORT)&((*Dirent)->FileName[8])) &&
(*(PUCHAR)&(Ccb->OemQueryTemplate.Constant[10]) == *(PUCHAR)&((*Dirent)->FileName[10]))) {
DebugTrace( 0, Dbg, "Entry found.\n", 0);
if (FileNameDos) {
*FileNameDos = TRUE;
}
break;
}
}
}
//
// No matches were found with the short name. If an LFN exists,
// use it for the search.
//
if (FoundValidLfn) {
//
// First do a quick check here for different sized constant
// name and expression before upcasing.
//
if (!Ccb->ContainsWildCards &&
Ccb->UnicodeQueryTemplate.Length != (USHORT)(LfnSize * sizeof(WCHAR))) {
//
// Move on to the next dirent.
//
FoundValidLfn = FALSE;
LongFileName->Length = 0;
goto GetNextDirent;
}
//
// We need to upcase the name we found.
// We need a buffer. Try to avoid doing an allocation.
//
if (LongFileName->Length <= 32*sizeof(WCHAR)) {
UpcasedLfn = &LocalUpcasedLfn;
} else if (LongFileName->Length <= PoolUpcasedLfn.MaximumLength) {
UpcasedLfn = &PoolUpcasedLfn;
} else {
//
// Free the old buffer now, and get a new one.
//
if (PoolUpcasedLfn.Buffer) {
ExFreePool( PoolUpcasedLfn.Buffer );
PoolUpcasedLfn.Buffer = NULL;
}
PoolUpcasedLfn.Buffer =
FsRtlAllocatePool( PagedPool,
LongFileName->Length );
PoolUpcasedLfn.MaximumLength = LongFileName->Length;
UpcasedLfn = &PoolUpcasedLfn;
}
Status = FatUpcaseUnicodeString( UpcasedLfn,
LongFileName,
FALSE );
if (!NT_SUCCESS(Status)) {
FatNormalizeAndRaiseStatus( IrpContext, Status );
}
//
// Do the compare
//
if (Ccb->ContainsWildCards) {
if (FsRtlIsNameInExpression( &Ccb->UnicodeQueryTemplate,
UpcasedLfn,
TRUE,
NULL )) {
break;
}
} else {
if (FsRtlAreNamesEqual( &Ccb->UnicodeQueryTemplate,
UpcasedLfn,
BooleanFlagOn( Ccb->Flags, CCB_FLAG_QUERY_TEMPLATE_MIXED ),
NULL )) {
break;
}
}
}
//
// This long name was not a match. Zero out the Length field.
//
if (FoundValidLfn) {
FoundValidLfn = FALSE;
LongFileName->Length = 0;
}
GetNextDirent:
//
// Move on to the next dirent.
//
*ByteOffset += sizeof(DIRENT);
*Dirent += 1;
}
} finally {
if (PoolUpcasedLfn.Buffer != NULL) {
ExFreePool( PoolUpcasedLfn.Buffer );
}
}
DebugTrace(-1, Dbg, "FatLocateDirent -> (VOID)\n", 0);
TimerStop(Dbg,"FatLocateDirent");
return;
}
VOID
FatLocateSimpleOemDirent (
IN PIRP_CONTEXT IrpContext,
IN PDCB ParentDirectory,
IN POEM_STRING FileName,
OUT PDIRENT *Dirent,
OUT PBCB *Bcb,
OUT PVBO ByteOffset
)
/*++
Routine Description:
This routine locates on the disk an undelted simple Oem dirent. By simple
I mean that FileName cannot contain any extended characters, and we do
not search LFNs or return them.
Arguments:
ParentDirectory - Supplies the DCB for the directory in which
to search
FileName - Supplies the filename to search for. The name may contain
wild cards
OffsetToStartSearchFrom - Supplies the VBO within the parent directory
from which to start looking for another real dirent.
Dirent - Receives a pointer to the located dirent if one was found
or NULL otherwise.
Bcb - Receives the Bcb for the located dirent if one was found or
NULL otherwise.
ByteOffset - Receives the VBO within the Parent directory for
the located dirent if one was found, or 0 otherwise.
Return Value:
None.
--*/
{
CCB LocalCcb;
PAGED_CODE();
//
// Note, this routine is called rarely, so performance is not critical.
// Just fill in a Ccb structure on my stack with the values that are
// required.
//
FatStringTo8dot3( IrpContext,
*FileName,
&LocalCcb.OemQueryTemplate.Constant );
LocalCcb.ContainsWildCards = FALSE;
LocalCcb.Flags = 0;
FatLocateDirent( IrpContext,
ParentDirectory,
&LocalCcb,
0,
Dirent,
Bcb,
ByteOffset,
NULL,
NULL );
return;
}
VOID
FatLocateVolumeLabel (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
OUT PDIRENT *Dirent,
OUT PBCB *Bcb,
OUT PVBO ByteOffset
)
/*++
Routine Description:
This routine locates on the disk a dirent representing the volume
label. It does this by searching the root directory for a special
volume label dirent.
Arguments:
Vcb - Supplies the VCB for the volume to search
Dirent - Receives a pointer to the located dirent if one was found
or NULL otherwise.
Bcb - Receives the Bcb for the located dirent if one was found or
NULL otherwise.
ByteOffset - Receives the VBO within the Parent directory for
the located dirent if one was found, or 0 otherwise.
Return Value:
None.
--*/
{
NTSTATUS Status;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatLocateVolumeLabel\n", 0);
DebugTrace( 0, Dbg, " Vcb = %08lx\n", Vcb);
DebugTrace( 0, Dbg, " Dirent = %08lx\n", Dirent);
DebugTrace( 0, Dbg, " Bcb = %08lx\n", Bcb);
DebugTrace( 0, Dbg, " ByteOffset = %08lx\n", ByteOffset);
//
// The algorithm here is really simple. We just walk through the
// root directory until we:
//
// A) Find the non-deleted volume label
// B) Can't Wait
// C) Hit the End of Directory
// D) Hit Eof
//
// In the first case we found it, in the latter three cases we did not.
//
*Bcb = NULL;
*ByteOffset = 0;
while ( TRUE ) {
//
// Try to read in the dirent
//
FatReadDirent( IrpContext,
Vcb->RootDcb,
*ByteOffset,
Bcb,
Dirent,
&Status );
//
// If End Directory dirent or EOF, set all out parameters to
// indicate volume label not found and, like, bail.
//
// Note that the order of evaluation here is important since we cannot
// check the first character of the dirent until after we know we
// are not beyond EOF
//
if ((Status == STATUS_END_OF_FILE) || ((*Dirent)->FileName[0] ==
FAT_DIRENT_NEVER_USED)) {
DebugTrace( 0, Dbg, "Volume label not found.\n", 0);
//
// If there is a Bcb, unpin it and set it to null
//
FatUnpinBcb( IrpContext, *Bcb );
*Dirent = NULL;
*ByteOffset = 0;
break;
}
//
// If the entry is the non-deleted volume label break from the loop.
//
// Note that all out parameters are already correctly set.
//
if ((((*Dirent)->Attributes & ~FAT_DIRENT_ATTR_ARCHIVE) == FAT_DIRENT_ATTR_VOLUME_ID) &&
((*Dirent)->FileName[0] != FAT_DIRENT_DELETED)) {
DebugTrace( 0, Dbg, "Volume label found at VBO = %08lx\n", *ByteOffset);
//
// We may set this dirty, so pin it.
//
FatPinMappedData( IrpContext,
Vcb->RootDcb,
*ByteOffset,
sizeof(DIRENT),
Bcb );
break;
}
//
// Move on to the next dirent.
//
*ByteOffset += sizeof(DIRENT);
*Dirent += 1;
}
DebugTrace(-1, Dbg, "FatLocateVolumeLabel -> (VOID)\n", 0);
return;
}
VOID
FatGetDirentFromFcbOrDcb (
IN PIRP_CONTEXT IrpContext,
IN PFCB FcbOrDcb,
OUT PDIRENT *Dirent,
OUT PBCB *Bcb
)
/*++
Routine Description:
This routine reads locates on the disk the dirent denoted by the
specified Fcb/Dcb.
Arguments:
FcbOrDcb - Supplies the FCB/DCB for the file/directory whose dirent
we are trying to read in. This must not be the root dcb.
Dirent - Receives a pointer to the dirent
Bcb - Receives the Bcb for the dirent
Return Value:
None.
--*/
{
NTSTATUS DontCare;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatGetDirentFromFcbOrDcb\n", 0);
DebugTrace( 0, Dbg, " FcbOrDcb = %08lx\n", FcbOrDcb);
DebugTrace( 0, Dbg, " Dirent = %08lx\n", Dirent);
DebugTrace( 0, Dbg, " Bcb = %08lx\n", Bcb);
//
// Assert that we are not attempting this on the root directory.
//
ASSERT( NodeType(FcbOrDcb) != FAT_NTC_ROOT_DCB );
//
// We know the offset of the dirent within the directory file,
// so we just read it (with pinning).
//
FatReadDirectoryFile( IrpContext,
FcbOrDcb->ParentDcb,
FcbOrDcb->DirentOffsetWithinDirectory,
sizeof(DIRENT),
TRUE,
Bcb,
(PVOID *)Dirent,
&DontCare );
ASSERT( NT_SUCCESS( DontCare ));
//
// We should never be accessing a deleted entry (except on removeable
// media where the dirent may have changed out from underneath us).
//
ASSERT( FlagOn( FcbOrDcb->Vcb->VcbState, VCB_STATE_FLAG_REMOVABLE_MEDIA) ?
TRUE : (BOOLEAN)((*Dirent)->FileName[0] != FAT_DIRENT_DELETED ));
DebugTrace(-1, Dbg, "FatGetDirentFromFcbOrDcb -> (VOID)\n", 0);
return;
}
BOOLEAN
FatIsDirectoryEmpty (
IN PIRP_CONTEXT IrpContext,
IN PDCB Dcb
)
/*++
Routine Description:
This routine indicates to the caller if the specified directory
is empty. (i.e., it is not the root dcb and it only contains
the "." and ".." entries, or deleted files).
Arguments:
Dcb - Supplies the DCB for the directory being queried.
Return Value:
BOOLEAN - Returns TRUE if the directory is empty and
FALSE if the directory and is not empty.
--*/
{
PBCB Bcb;
ULONG ByteOffset;
PDIRENT Dirent;
BOOLEAN IsDirectoryEmpty;
NTSTATUS Status;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatIsDirectoryEmpty\n", 0);
DebugTrace( 0, Dbg, " Dcb = %08lx\n", Dcb);
DebugTrace( 0, Dbg, " IsDirectoryEmpty = %08lx\n", IsDirectoryEmpty);
//
// Check to see if the first entry is an and of directory marker.
// For the root directory we check at Vbo = 0, for normal directories
// we check after the "." and ".." entries.
//
ByteOffset = (NodeType(Dcb) == FAT_NTC_ROOT_DCB) ? 0 : 2*sizeof(DIRENT);
//
// We just march through the directory looking for anything other
// than deleted files, LFNs, an EOF, or end of directory marker.
//
Bcb = NULL;
try {
while ( TRUE ) {
//
// Try to read in the dirent
//
FatReadDirent( IrpContext,
Dcb,
ByteOffset,
&Bcb,
&Dirent,
&Status );
//
// If End Directory dirent or EOF, set IsDirectoryEmpty to TRUE and,
// like, bail.
//
// Note that the order of evaluation here is important since we cannot
// check the first character of the dirent until after we know we
// are not beyond EOF
//
if ((Status == STATUS_END_OF_FILE) ||
(Dirent->FileName[0] == FAT_DIRENT_NEVER_USED)) {
DebugTrace( 0, Dbg, "Empty. Last exempt entry at VBO = %08lx\n", ByteOffset);
IsDirectoryEmpty = TRUE;
break;
}
//
// If this dirent is NOT deleted or an LFN set IsDirectoryEmpty to
// FALSE and, like, bail.
//
if ((Dirent->FileName[0] != FAT_DIRENT_DELETED) &&
(Dirent->Attributes != FAT_DIRENT_ATTR_LFN)) {
DebugTrace( 0, Dbg, "Not Empty. First entry at VBO = %08lx\n", ByteOffset);
IsDirectoryEmpty = FALSE;
break;
}
//
// Move on to the next dirent.
//
ByteOffset += sizeof(DIRENT);
Dirent += 1;
}
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
DebugTrace(-1, Dbg, "FatIsDirectoryEmpty -> %ld\n", IsDirectoryEmpty);
return IsDirectoryEmpty;
}
VOID
FatConstructDirent (
IN PIRP_CONTEXT IrpContext,
IN OUT PDIRENT Dirent,
IN POEM_STRING FileName,
IN BOOLEAN ComponentReallyLowercase,
IN BOOLEAN ExtensionReallyLowercase,
IN PUNICODE_STRING Lfn OPTIONAL,
IN UCHAR Attributes,
IN BOOLEAN ZeroAndSetTimeFields,
IN PLARGE_INTEGER SetCreationTime OPTIONAL
)
/*++
Routine Description:
This routine modifies the fields of a dirent.
Arguments:
Dirent - Supplies the dirent being modified.
FileName - Supplies the name to store in the Dirent. This
name must not contain wildcards.
ComponentReallyLowercase - This boolean indicates that the User Specified
compoent name was really all a-z and < 0x80 characters. We set the
magic bit in this case.
ExtensionReallyLowercase - Same as above, but for the extension.
Lfn - May supply a long file name.
Attributes - Supplies the attributes to store in the dirent
ZeroAndSetTimeFields - Tells whether or not to initially zero the dirent
and update the time fields.
SetCreationTime - If specified, contains a timestamp to use as the creation
time of this dirent
Return Value:
None.
--*/
{
PAGED_CODE();
DebugTrace(+1, Dbg, "FatConstructDirent\n", 0);
DebugTrace( 0, Dbg, " Dirent = %08lx\n", Dirent);
DebugTrace( 0, Dbg, " FileName = %Z\n", FileName);
DebugTrace( 0, Dbg, " Attributes = %08lx\n", Attributes);
if (ZeroAndSetTimeFields) {
RtlZeroMemory( Dirent, sizeof(DIRENT) );
}
//
// We just merrily go and fill up the dirent with the fields given.
//
FatStringTo8dot3( IrpContext, *FileName, (PFAT8DOT3)&Dirent->FileName[0] );
if (ZeroAndSetTimeFields || SetCreationTime) {
LARGE_INTEGER Time, SaveTime;
KeQuerySystemTime( &Time );
if (FatData.ChicagoMode) {
if (!SetCreationTime || !FatNtTimeToFatTime( IrpContext,
SetCreationTime,
FALSE,
&Dirent->CreationTime,
&Dirent->CreationMSec )) {
//
// No tunneled time or the tunneled time was bogus. Since we aren't
// responsible for initializing the to-be-created Fcb with creation
// time, we can't do the usual thing and let NtTimeToFatTime perform
// rounding on the timestamp - this would mess up converting to the
// LastWriteTime below.
//
SaveTime = Time;
if (!FatNtTimeToFatTime( IrpContext,
&SaveTime,
FALSE,
&Dirent->CreationTime,
&Dirent->CreationMSec )) {
//
// Failed again. Wow.
//
RtlZeroMemory( &Dirent->CreationTime, sizeof(FAT_TIME_STAMP));
Dirent->CreationMSec = 0;
}
}
}
if (ZeroAndSetTimeFields) {
//
// We only touch the other timestamps if we are initializing the dirent
//
if (!FatNtTimeToFatTime( IrpContext,
&Time,
TRUE,
&Dirent->LastWriteTime,
NULL )) {
DebugTrace( 0, Dbg, "Current time invalid.\n", 0);
RtlZeroMemory( &Dirent->LastWriteTime, sizeof(FAT_TIME_STAMP) );
}
if (FatData.ChicagoMode) {
Dirent->LastAccessDate = Dirent->LastWriteTime.Date;
}
}
}
//
// Copy the attributes
//
Dirent->Attributes = Attributes;
//
// Set the magic bit here, to tell dirctrl.c that this name is really
// lowercase.
//
Dirent->NtByte = 0;
if (ComponentReallyLowercase) {
SetFlag( Dirent->NtByte, FAT_DIRENT_NT_BYTE_8_LOWER_CASE );
}
if (ExtensionReallyLowercase) {
SetFlag( Dirent->NtByte, FAT_DIRENT_NT_BYTE_3_LOWER_CASE );
}
//
// See if we have to create an Lfn entry
//
if (ARGUMENT_PRESENT(Lfn)) {
UCHAR DirentChecksum;
UCHAR DirentsInLfn;
UCHAR LfnOrdinal;
PWCHAR LfnBuffer;
PLFN_DIRENT LfnDirent;
ASSERT( FatData.ChicagoMode );
DirentChecksum = FatComputeLfnChecksum( Dirent );
LfnOrdinal =
DirentsInLfn = FAT_LFN_DIRENTS_NEEDED(Lfn);
LfnBuffer = &Lfn->Buffer[(DirentsInLfn - 1) * 13];
ASSERT( DirentsInLfn <= MAX_LFN_DIRENTS );
for (LfnDirent = (PLFN_DIRENT)Dirent - DirentsInLfn;
LfnDirent < (PLFN_DIRENT)Dirent;
LfnDirent += 1, LfnOrdinal -= 1, LfnBuffer -= 13) {
WCHAR FinalLfnBuffer[13];
PWCHAR Buffer;
//
// We need to special case the "final" dirent.
//
if (LfnOrdinal == DirentsInLfn) {
ULONG i;
ULONG RemainderChars;
RemainderChars = (Lfn->Length / sizeof(WCHAR)) % 13;
LfnDirent->Ordinal = LfnOrdinal | FAT_LAST_LONG_ENTRY;
if (RemainderChars != 0) {
RtlCopyMemory( &FinalLfnBuffer,
LfnBuffer,
RemainderChars * sizeof(WCHAR) );
for (i = RemainderChars; i < 13; i++) {
//
// Figure out which character to use.
//
if (i == RemainderChars) {
FinalLfnBuffer[i] = 0x0000;
} else {
FinalLfnBuffer[i] = 0xffff;
}
}
Buffer = FinalLfnBuffer;
} else {
Buffer = LfnBuffer;
}
} else {
LfnDirent->Ordinal = LfnOrdinal;
Buffer = LfnBuffer;
}
//
// Now fill in the name.
//
RtlCopyMemory( &LfnDirent->Name1[0],
&Buffer[0],
5 * sizeof(WCHAR) );
RtlCopyMemory( &LfnDirent->Name2[0],
&Buffer[5],
6 * sizeof(WCHAR) );
RtlCopyMemory( &LfnDirent->Name3[0],
&Buffer[11],
2 * sizeof(WCHAR) );
//
// And the other fields
//
LfnDirent->Attributes = FAT_DIRENT_ATTR_LFN;
LfnDirent->Type = 0;
LfnDirent->Checksum = DirentChecksum;
LfnDirent->MustBeZero = 0;
}
}
DebugTrace(-1, Dbg, "FatConstructDirent -> (VOID)\n", 0);
return;
}
VOID
FatConstructLabelDirent (
IN PIRP_CONTEXT IrpContext,
IN OUT PDIRENT Dirent,
IN POEM_STRING Label
)
/*++
Routine Description:
This routine modifies the fields of a dirent to be used for a label.
Arguments:
Dirent - Supplies the dirent being modified.
Label - Supplies the name to store in the Dirent. This
name must not contain wildcards.
Return Value:
None.
--*/
{
PAGED_CODE();
DebugTrace(+1, Dbg, "FatConstructLabelDirent\n", 0);
DebugTrace( 0, Dbg, " Dirent = %08lx\n", Dirent);
DebugTrace( 0, Dbg, " Label = %Z\n", Label);
RtlZeroMemory( Dirent, sizeof(DIRENT) );
//
// We just merrily go and fill up the dirent with the fields given.
//
RtlCopyMemory( Dirent->FileName, Label->Buffer, Label->Length );
//
// Pad the label with spaces, not nulls.
//
RtlFillMemory( &Dirent->FileName[Label->Length], 11 - Label->Length, ' ');
Dirent->LastWriteTime = FatGetCurrentFatTime( IrpContext );
Dirent->Attributes = FAT_DIRENT_ATTR_VOLUME_ID;
Dirent->ExtendedAttributes = 0;
Dirent->FileSize = 0;
DebugTrace(-1, Dbg, "FatConstructLabelDirent -> (VOID)\n", 0);
return;
}
VOID
FatSetFileSizeInDirent (
IN PIRP_CONTEXT IrpContext,
IN PFCB Fcb,
IN PULONG AlternativeFileSize OPTIONAL
)
/*++
Routine Description:
This routine saves the file size in an fcb into its dirent.
Arguments:
Fcb - Supplies the Fcb being referenced
AlternativeFileSize - If non-null we use the ULONG it points to as
the new file size. Otherwise we use the one in the Fcb.
Return Value:
None.
--*/
{
PDIRENT Dirent;
PBCB DirentBcb;
PAGED_CODE();
//****ASSERT( !FlagOn( Fcb->DirentFatFlags, FAT_DIRENT_ATTR_READ_ONLY ));
FatGetDirentFromFcbOrDcb( IrpContext,
Fcb,
&Dirent,
&DirentBcb );
try {
Dirent->FileSize = ARGUMENT_PRESENT( AlternativeFileSize ) ?
*AlternativeFileSize : Fcb->Header.FileSize.LowPart;
FatSetDirtyBcb( IrpContext, DirentBcb, Fcb->Vcb );
} finally {
FatUnpinBcb( IrpContext, DirentBcb );
}
}
//
// Internal support routine
//
UCHAR
FatComputeLfnChecksum (
PDIRENT Dirent
)
/*++
Routine Description:
This routine computes the Chicago long file name checksum.
Arguments:
Dirent - Specifies the dirent that we are to compute a checksum for.
Return Value:
The checksum.
--*/
{
ULONG i;
UCHAR Checksum;
PAGED_CODE();
Checksum = Dirent->FileName[0];
for (i=1; i < 11; i++) {
Checksum = ((Checksum & 1) ? 0x80 : 0) +
(Checksum >> 1) +
Dirent->FileName[i];
}
return Checksum;
}
#if 0 // It turns out Win95 is still creating short names without a ~
//
// Internal support routine
//
BOOLEAN
FatIsLfnPairValid (
PWCHAR Lfn,
ULONG LfnSize,
PDIRENT Dirent
)
/*++
Routine Description:
This routine does a few more checks to make sure that a LFN/short
name pairing is legitimate. Basically this is the test:
Pairing is valid if:
DIRENT has a ~ character ||
(LFN is 8.3 compliant &&
(LFN has extended character(s) ? TRUE :
LFN upcases to DIRENT))
When checking for the presence of a tilda character in the short
name, note that we purposely do a single byte search instead of
converting the name to UNICODE and looking there for the tilda.
This protects us from accidently missing the tilda if the
preceding byte is a lead byte in the current Oem code page,
but wasn't in the Oem code page that created the file.
Also note that if the LFN is longer than 12 characters, then the
second clause of the OR must be false.
Arguments:
Lfn - Points to a buffer of UNICODE chars.
LfnSize - This is the size of the LFN in characters.
Dirent - Specifies the dirent we are to consider.
Return Value:
TRUE if the Lfn/DIRENT form a legitimate pair, FALSE otherwise.
--*/
{
ULONG i;
BOOLEAN ExtendedChars;
ULONG DirentBuffer[3];
PUCHAR DirentName;
ULONG DirentIndex;
BOOLEAN DotEncountered;
//
// First, look for a tilda
//
for (i=0; i<11; i++) {
if (Dirent->FileName[i] == '~') {
return TRUE;
}
}
//
// No tilda. If the LFN is longer than 12 characters, then it can
// neither upcase to the DIRENT nor be 8.3 complient.
//
if (LfnSize > 12) {
return FALSE;
}
//
// Now see if the name is 8.3, and build an upcased DIRENT as well.
//
DirentBuffer[0] = 0x20202020;
DirentBuffer[1] = 0x20202020;
DirentBuffer[2] = 0x20202020;
DirentName = (PUCHAR)DirentBuffer;
ExtendedChars = FALSE;
DirentIndex = 0;
DotEncountered = FALSE;
for (i=0; i < LfnSize; i++) {
//
// Do dot transition work
//
if (Lfn[i] == L'.') {
if (DotEncountered ||
(i > 8) ||
((LfnSize - i) > 4) ||
(i && Lfn[i-1] == L' ')) {
return FALSE;
}
DotEncountered = TRUE;
DirentIndex = 8;
continue;
}
//
// The character must be legal in order to be 8.3
//
if ((Lfn[i] < 0x80) &&
!FsRtlIsAnsiCharacterLegalFat((UCHAR)Lfn[i], FALSE)) {
return FALSE;
}
//
// If the name contains no extended chars, continue building DIRENT
//
if (!ExtendedChars) {
if (Lfn[i] > 0x7f) {
ExtendedChars = TRUE;
} else {
DirentName[DirentIndex++] = (UCHAR) (
Lfn[i] < 'a' ? Lfn[i] : Lfn[i] <= 'z' ? Lfn[i] - ('a' - 'A') : Lfn[i]);
}
}
}
//
// If the LFN ended in a space, or there was no dot and the name
// has more than 8 characters, then it is not 8.3 compliant.
//
if ((Lfn[LfnSize - 1] == L' ') ||
(!DotEncountered && (LfnSize > 8))) {
return FALSE;
}
//
// OK, now if we got this far then the LFN is 8dot3. If there are
// no extended characters, then we can also check to make sure that
// the LFN is only a case varient of the DIRENT.
//
if (!ExtendedChars &&
!RtlEqualMemory(Dirent->FileName, DirentName, 11)) {
return FALSE;
}
//
// We have now verified this pairing the very best we can without
// knowledge of the code page that the file was created under.
//
return TRUE;
}
#endif //0
//
// Internal support routine
//
VOID
FatRescanDirectory (
PIRP_CONTEXT IrpContext,
PDCB Dcb
)
/*++
Routine Description:
This routine rescans the given directory, finding the first unused
dirent, first deleted dirent, and setting the free dirent bitmap
appropriately.
Arguments:
Dcb - Supplies the directory to rescan.
Return Value:
None.
--*/
{
PBCB Bcb = NULL;
PDIRENT Dirent;
NTSTATUS Status;
ULONG UnusedVbo;
ULONG DeletedHint;
ULONG DirentIndex;
ULONG DirentsThisRun;
ULONG StartIndexOfThisRun;
enum RunType {
InitialRun,
FreeDirents,
AllocatedDirents,
} CurrentRun;
PAGED_CODE();
DebugTrace( 0, Dbg, "We must scan the whole directory.\n", 0);
UnusedVbo = 0;
DeletedHint = 0xffffffff;
//
// To start with, we have to find out if the first dirent is free.
//
CurrentRun = InitialRun;
DirentIndex =
StartIndexOfThisRun = 0;
try {
while ( TRUE ) {
BOOLEAN DirentDeleted;
//
// Read a dirent
//
FatReadDirent( IrpContext,
Dcb,
UnusedVbo,
&Bcb,
&Dirent,
&Status );
//
// If EOF, or we found a NEVER_USED entry, we exit the loop
//
if ( (Status == STATUS_END_OF_FILE ) ||
(Dirent->FileName[0] == FAT_DIRENT_NEVER_USED)) {
break;
}
//
// If the dirent is DELETED, and it is the first one we found, set
// it in the deleted hint.
//
if (Dirent->FileName[0] == FAT_DIRENT_DELETED) {
DirentDeleted = TRUE;
if (DeletedHint == 0xffffffff) {
DeletedHint = UnusedVbo;
}
} else {
DirentDeleted = FALSE;
}
//
// Check for the first time through the loop, and determine
// the current run type.
//
if (CurrentRun == InitialRun) {
CurrentRun = DirentDeleted ?
FreeDirents : AllocatedDirents;
} else {
//
// Are we switching from a free run to an allocated run?
//
if ((CurrentRun == FreeDirents) && !DirentDeleted) {
DirentsThisRun = DirentIndex - StartIndexOfThisRun;
RtlClearBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
StartIndexOfThisRun,
DirentsThisRun );
CurrentRun = AllocatedDirents;
StartIndexOfThisRun = DirentIndex;
}
//
// Are we switching from an allocated run to a free run?
//
if ((CurrentRun == AllocatedDirents) && DirentDeleted) {
DirentsThisRun = DirentIndex - StartIndexOfThisRun;
RtlSetBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
StartIndexOfThisRun,
DirentsThisRun );
CurrentRun = FreeDirents;
StartIndexOfThisRun = DirentIndex;
}
}
//
// Move on to the next dirent.
//
UnusedVbo += sizeof(DIRENT);
Dirent += 1;
DirentIndex += 1;
}
//
// Now we have to record the final run we encoutered
//
DirentsThisRun = DirentIndex - StartIndexOfThisRun;
if ((CurrentRun == FreeDirents) || (CurrentRun == InitialRun)) {
RtlClearBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
StartIndexOfThisRun,
DirentsThisRun );
} else {
RtlSetBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
StartIndexOfThisRun,
DirentsThisRun );
}
//
// Now if there we bailed prematurely out of the loop because
// we hit an unused entry, set all the rest as free.
//
if (UnusedVbo < Dcb->Header.AllocationSize.LowPart) {
StartIndexOfThisRun = UnusedVbo / sizeof(DIRENT);
DirentsThisRun = (Dcb->Header.AllocationSize.LowPart -
UnusedVbo) / sizeof(DIRENT);
RtlClearBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
StartIndexOfThisRun,
DirentsThisRun);
}
} finally {
FatUnpinBcb( IrpContext, Bcb );
}
//
// If there weren't any DELETED entries, set the index to our current
// position.
//
if (DeletedHint == 0xffffffff) { DeletedHint = UnusedVbo; }
Dcb->Specific.Dcb.UnusedDirentVbo = UnusedVbo;
Dcb->Specific.Dcb.DeletedDirentHint = DeletedHint;
return;
}
//
// Internal support routine
//
ULONG
FatDefragDirectory (
IN PIRP_CONTEXT IrpContext,
IN PDCB Dcb,
IN ULONG DirentsNeeded
)
/*++
Routine Description:
This routine determines if the requested number of dirents can be found
in the directory, looking for deleted dirents and orphaned LFNs. If the
request can be satisifed, orphaned LFNs are marked as deleted, and deleted
dirents are all grouped together at the end of the directory.
Note that this routine is currently used only on the root directory, but
it is completely general and could be used on any directory.
Arguments:
Dcb - Supplies the directory to defrag.
Return Value:
The Index of the first dirent available for use, or -1 if the
request cannot be satisfied.
--*/
{
ULONG SavedIrpContextFlag;
PLIST_ENTRY Links;
ULONG ReturnValue;
PFCB Fcb;
PBCB Bcb = NULL;
PDIRENT Dirent = NULL;
UNICODE_STRING Lfn = {0,0,NULL};
MCB Mcb;
BOOLEAN McbInitialized = FALSE;
PUCHAR Directory;
PUCHAR UnusedDirents;
PUCHAR UnusedDirentBuffer = NULL;
PUCHAR UsedDirents;
PUCHAR UsedDirentBuffer = NULL;
PBCB *Bcbs = NULL;
ULONG Page;
ULONG PagesPinned;
ULONG DcbSize;
ULONG TotalBytesAllocated = 0;
PAGED_CODE();
//
// We assume we own the Vcb.
//
ASSERT( FatVcbAcquiredExclusive(IrpContext, Dcb->Vcb) );
//
// We will only attempt this on directories less than 0x40000 bytes
// long (by default on DOS the root directory is only 0x2000 long).
// This is to avoid a cache manager complication.
//
DcbSize = Dcb->Header.AllocationSize.LowPart;
if (DcbSize > 0x40000) {
return (ULONG)-1;
}
//
// Force wait to TRUE
//
SavedIrpContextFlag = IrpContext->Flags;
SetFlag( IrpContext->Flags,
IRP_CONTEXT_FLAG_WAIT | IRP_CONTEXT_FLAG_WRITE_THROUGH );
//
// Now acquire all open Fcbs in the Dcb exclusive.
//
for (Links = Dcb->Specific.Dcb.ParentDcbQueue.Flink;
Links != &Dcb->Specific.Dcb.ParentDcbQueue;
Links = Links->Flink) {
Fcb = CONTAINING_RECORD( Links, FCB, ParentDcbLinks );
(VOID)ExAcquireResourceExclusive( Fcb->Header.Resource, TRUE );
}
try {
CCB Ccb;
ULONG BytesUsed = 0;
ULONG QueryOffset = 0;
ULONG FoundOffset = 0;
NTSTATUS DontCare;
ULONG Run;
ULONG TotalRuns;
BOOLEAN Result;
PUCHAR Char;
//
// We are going to build a new, bitmap that will show all orphaned
// LFNs as well as deleted dirents as available.
//
// Initialize our local CCB that will match all files.
//
RtlZeroMemory( &Ccb, sizeof(CCB) );
Ccb.Flags = CCB_FLAG_MATCH_ALL;
//
// Init the Long File Name string.
//
Lfn.MaximumLength = 260 * sizeof(WCHAR);
Lfn.Buffer = FsRtlAllocatePool( PagedPool, 260*sizeof(WCHAR) );
//
// Initalize the Mcb. We use this structure to keep track of runs
// of free and allocated dirents. Runs are identity allocations, and
// holes are free dirents.
//
FsRtlInitializeMcb( &Mcb, PagedPool );
McbInitialized = TRUE;
do {
FatLocateDirent( IrpContext,
Dcb,
&Ccb,
QueryOffset,
&Dirent,
&Bcb,
&FoundOffset,
NULL,
&Lfn );
if (Dirent != NULL) {
ULONG BytesUsed;
ULONG LfnByteOffset;
//
// Compute the LfnByteOffset.
//
LfnByteOffset = FoundOffset -
FAT_LFN_DIRENTS_NEEDED(&Lfn) * sizeof(LFN_DIRENT);
BytesUsed = FoundOffset - LfnByteOffset + sizeof(DIRENT);
//
// Set a run to represent all the dirent used for this
// file in the Dcb dir. We add 0x40000 to the LBN so that
// it will always to non-zero and thus not confused with a
// hole.
//
Result = FsRtlAddMcbEntry( &Mcb,
LfnByteOffset,
LfnByteOffset + 0x40000,
BytesUsed );
ASSERT( Result );
//
// Move on to the next dirent.
//
TotalBytesAllocated += BytesUsed;
QueryOffset = FoundOffset + sizeof(DIRENT);
}
} while ((Dirent != NULL) && (QueryOffset < DcbSize));
if (Bcb != NULL) {
FatUnpinBcb( IrpContext, Bcb );
}
//
// If we need more dirents than are available, bail.
//
if (DirentsNeeded > (DcbSize - TotalBytesAllocated)/sizeof(DIRENT)) {
try_return(ReturnValue = (ULONG)-1);
}
//
// Now we are going to copy all the used and un-used parts of the
// directory to separate pool.
//
// Allocate these buffers and pin the entire directory.
//
UnusedDirents =
UnusedDirentBuffer = FsRtlAllocatePool( PagedPool,
DcbSize - TotalBytesAllocated );
UsedDirents =
UsedDirentBuffer = FsRtlAllocatePool( PagedPool,
TotalBytesAllocated );
PagesPinned = (DcbSize + (PAGE_SIZE - 1 )) / PAGE_SIZE;
Bcbs = FsRtlAllocatePool( PagedPool, PagesPinned * sizeof(PBCB) );
RtlZeroMemory( Bcbs, PagesPinned * sizeof(PBCB) );
for (Page = 0; Page < PagesPinned; Page += 1) {
ULONG PinSize;
//
// Don't try to pin beyond the Dcb size.
//
if ((Page + 1) * PAGE_SIZE > DcbSize) {
PinSize = DcbSize - (Page * PAGE_SIZE);
} else {
PinSize = PAGE_SIZE;
}
FatPrepareWriteDirectoryFile( IrpContext,
Dcb,
Page * PAGE_SIZE,
PinSize,
&Bcbs[Page],
&Dirent,
FALSE,
&DontCare );
if (Page == 0) {
Directory = (PUCHAR)Dirent;
}
}
TotalRuns = FsRtlNumberOfRunsInMcb( &Mcb );
for (Run = 0; Run < TotalRuns; Run++) {
VBO Vbo;
LBO Lbo;
Result = FsRtlGetNextMcbEntry( &Mcb,
Run,
&Vbo,
&Lbo,
&BytesUsed );
ASSERT(Result);
//
// Copy each run to their specific pool.
//
if (Lbo != 0) {
RtlCopyMemory( UsedDirents,
Directory + Vbo,
BytesUsed );
UsedDirents += BytesUsed;
} else {
RtlCopyMemory( UnusedDirents,
Directory + Vbo,
BytesUsed );
UnusedDirents += BytesUsed;
}
}
//
// Marking all the un-used dirents as "deleted". This will reclaim
// storage used by orphaned LFNs.
//
for (Char = UnusedDirentBuffer; Char < UnusedDirents; Char += sizeof(DIRENT)) {
*Char = FAT_DIRENT_DELETED;
}
//
// Now, for the permanent step. Copy the two pool buffer back to the
// real Dcb directory, and flush the Dcb directory
//
ASSERT( TotalBytesAllocated == (ULONG)(UsedDirents - UsedDirentBuffer) );
RtlCopyMemory( Directory, UsedDirentBuffer, TotalBytesAllocated );
RtlCopyMemory( Directory + TotalBytesAllocated,
UnusedDirentBuffer,
UnusedDirents - UnusedDirentBuffer );
//
// We need to unpin here so that the UnpinRepinned won't deadlock.
//
if (Bcbs) {
for (Page = 0; Page < PagesPinned; Page += 1) {
FatUnpinBcb( IrpContext, Bcbs[Page] );
}
ExFreePool(Bcbs);
Bcbs = NULL;
}
//
// Flush the directory to disk.
//
FatUnpinRepinnedBcbs( IrpContext );
//
// OK, now nothing can go wrong. We have two more things to do.
// First, we have to fix up all the dirent offsets in any open Fcbs.
// If we cannot now find the Fcb, the file is marked invalid. Also,
// we skip deleted files.
//
for (Links = Dcb->Specific.Dcb.ParentDcbQueue.Flink;
Links != &Dcb->Specific.Dcb.ParentDcbQueue;
Links = Links->Flink) {
PBCB TmpBcb = NULL;
ULONG TmpOffset;
PDIRENT TmpDirent = NULL;
ULONG PreviousLfnSpread;
Fcb = CONTAINING_RECORD( Links, FCB, ParentDcbLinks );
if (IsFileDeleted( IrpContext, Fcb )) {
continue;
}
FatLocateSimpleOemDirent( IrpContext,
Dcb,
&Fcb->ShortName.Name.Oem,
&TmpDirent,
&TmpBcb,
&TmpOffset );
if (TmpBcb == NULL) {
FatMarkFcbCondition( IrpContext, Fcb, FcbBad );
} else {
FatUnpinBcb( IrpContext, TmpBcb );
PreviousLfnSpread = Fcb->DirentOffsetWithinDirectory -
Fcb->LfnOffsetWithinDirectory;
Fcb->DirentOffsetWithinDirectory = TmpOffset;
Fcb->LfnOffsetWithinDirectory = TmpOffset - PreviousLfnSpread;
}
}
//
// Now, finally, make the free dirent bitmap reflect the new
// state of the Dcb directory.
//
RtlSetBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
0,
TotalBytesAllocated / sizeof(DIRENT) );
RtlClearBits( &Dcb->Specific.Dcb.FreeDirentBitmap,
TotalBytesAllocated / sizeof(DIRENT),
(DcbSize - TotalBytesAllocated) / sizeof(DIRENT) );
ReturnValue = TotalBytesAllocated / sizeof(DIRENT);
try_exit: NOTHING;
} finally {
//
// Free all our resources and stuff.
//
if (McbInitialized) {
FsRtlUninitializeMcb( &Mcb );
}
if (Lfn.Buffer) {
ExFreePool( Lfn.Buffer );
}
if (UnusedDirentBuffer) {
ExFreePool( UnusedDirentBuffer );
}
if (UsedDirentBuffer) {
ExFreePool( UsedDirentBuffer );
}
if (Bcbs) {
for (Page = 0; Page < PagesPinned; Page += 1) {
FatUnpinBcb( IrpContext, Bcbs[Page] );
}
ExFreePool(Bcbs);
}
if (Bcb) {
FatUnpinBcb( IrpContext, Bcb );
}
for (Links = Dcb->Specific.Dcb.ParentDcbQueue.Flink;
Links != &Dcb->Specific.Dcb.ParentDcbQueue;
Links = Links->Flink) {
Fcb = CONTAINING_RECORD( Links, FCB, ParentDcbLinks );
ExReleaseResource( Fcb->Header.Resource );
}
IrpContext->Flags = SavedIrpContextFlag;
}
//
// Now return the offset of the first free dirent to the caller.
//
return ReturnValue;
}