mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
1141 lines
26 KiB
C
1141 lines
26 KiB
C
/*++
|
|
|
|
Copyright (c) 1991-2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
NameSup.c
|
|
|
|
Abstract:
|
|
|
|
This module implements the Cdfs Name support routines
|
|
|
|
|
|
--*/
|
|
|
|
#include "cdprocs.h"
|
|
|
|
//
|
|
// The Bug check file id for this module
|
|
//
|
|
|
|
#define BugCheckFileId (CDFS_BUG_CHECK_NAMESUP)
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, CdConvertBigToLittleEndian)
|
|
#pragma alloc_text(PAGE, CdConvertNameToCdName)
|
|
#pragma alloc_text(PAGE, CdDissectName)
|
|
#pragma alloc_text(PAGE, CdGenerate8dot3Name)
|
|
#pragma alloc_text(PAGE, CdFullCompareNames)
|
|
#pragma alloc_text(PAGE, CdIsLegalName)
|
|
#pragma alloc_text(PAGE, CdIs8dot3Name)
|
|
#pragma alloc_text(PAGE, CdIsNameInExpression)
|
|
#pragma alloc_text(PAGE, CdShortNameDirentOffset)
|
|
#pragma alloc_text(PAGE, CdUpcaseName)
|
|
#endif
|
|
|
|
|
|
_Post_satisfies_(_Old_(CdName->FileName.Length) >=
|
|
CdName->FileName.Length + CdName->VersionString.Length)
|
|
VOID
|
|
CdConvertNameToCdName (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_Inout_ PCD_NAME CdName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to convert a string of bytes into a CdName.
|
|
|
|
The full name is already in the CdName structure in the FileName field.
|
|
We split this into the filename and version strings.
|
|
|
|
Arguments:
|
|
|
|
CdName - Pointer to CdName structure to update.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG NameLength = 0;
|
|
PWCHAR CurrentCharacter = CdName->FileName.Buffer;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// Look for a separator character.
|
|
//
|
|
|
|
while ((NameLength < CdName->FileName.Length) &&
|
|
(*CurrentCharacter != L';')) {
|
|
|
|
CurrentCharacter += 1;
|
|
NameLength += 2;
|
|
}
|
|
|
|
//
|
|
// If there is at least one more character after a possible separator then it
|
|
// and all following characters are part of the version string.
|
|
//
|
|
|
|
CdName->VersionString.Length = 0;
|
|
if (NameLength + sizeof( WCHAR ) < CdName->FileName.Length) {
|
|
|
|
CdName->VersionString.MaximumLength =
|
|
CdName->VersionString.Length = (USHORT) (CdName->FileName.Length - NameLength - sizeof( WCHAR ));
|
|
CdName->VersionString.Buffer = Add2Ptr( CdName->FileName.Buffer,
|
|
NameLength + sizeof( WCHAR ),
|
|
PWCHAR );
|
|
}
|
|
|
|
//
|
|
// Now update the filename portion of the name.
|
|
//
|
|
|
|
CdName->FileName.Length = (USHORT) NameLength;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
VOID
|
|
CdConvertBigToLittleEndian (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_reads_bytes_(ByteCount) PCHAR BigEndian,
|
|
_In_ ULONG ByteCount,
|
|
_Out_writes_bytes_(ByteCount) PCHAR LittleEndian
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to convert a unicode string in big endian to
|
|
little endian. We start by copying all of the source bytes except
|
|
the first. This will put the low order bytes in the correct position.
|
|
We then copy each high order byte in its correct position.
|
|
|
|
Arguments:
|
|
|
|
BigEndian - Pointer to the string of big endian characters.
|
|
|
|
ByteCount - Number of unicode characters in this string.
|
|
|
|
LittleEndian - Pointer to array to store the little endian characters.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG RemainingByteCount = ByteCount;
|
|
|
|
PCHAR Source = BigEndian;
|
|
PCHAR Destination = LittleEndian;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// If the byte count isn't an even number then the disk is corrupt.
|
|
//
|
|
|
|
if (FlagOn( ByteCount, 1 )) {
|
|
|
|
CdRaiseStatus( IrpContext, STATUS_DISK_CORRUPT_ERROR );
|
|
}
|
|
|
|
//
|
|
// Start by copy the low-order bytes into the correct position. Do
|
|
// this by skipping the first byte in the BigEndian string.
|
|
//
|
|
|
|
RtlCopyMemory( Destination,
|
|
Source + 1,
|
|
RemainingByteCount - 1 );
|
|
|
|
//
|
|
// Now move the high-order bytes into position.
|
|
//
|
|
|
|
Destination += 1;
|
|
|
|
while (RemainingByteCount != 0) {
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(push)
|
|
#pragma prefast(suppress:26014, "RemainingByteCount is even")
|
|
#endif
|
|
*Destination = *Source;
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(pop)
|
|
#endif
|
|
|
|
Source += 2;
|
|
Destination += 2;
|
|
|
|
RemainingByteCount -= 2;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
VOID
|
|
CdUpcaseName (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PCD_NAME Name,
|
|
_Inout_ PCD_NAME UpcaseName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to upcase a CdName structure. We will do both
|
|
the filename and version strings.
|
|
|
|
Arguments:
|
|
|
|
Name - This is the mixed case version of the name.
|
|
|
|
UpcaseName - This is the destination for the upcase operation.
|
|
|
|
Return Value:
|
|
|
|
None. This routine will raise all errors.
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// If the name structures are different then initialize the different components.
|
|
//
|
|
|
|
if (Name != UpcaseName) {
|
|
|
|
//
|
|
// Initialize the version string portion of the name.
|
|
//
|
|
|
|
UpcaseName->VersionString.Length = 0;
|
|
|
|
if (Name->VersionString.Length != 0) {
|
|
|
|
UpcaseName->VersionString.MaximumLength =
|
|
UpcaseName->VersionString.Length = Name->VersionString.Length;
|
|
|
|
//
|
|
// Initially set the buffer to point to where we need to insert
|
|
// the separator.
|
|
//
|
|
|
|
UpcaseName->VersionString.Buffer = Add2Ptr( UpcaseName->FileName.Buffer,
|
|
Name->FileName.Length,
|
|
PWCHAR );
|
|
|
|
//
|
|
// We are currently pointing to the location to store the separator.
|
|
// Store the ';' and then move to the next character to
|
|
// copy the data.
|
|
//
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast( suppress:26015, "CD_NAME structures have two UNICODE_STRING structures pointing to the same allocation. there is no way to tell prefast this is the case and that the allocation is always big enough.");
|
|
#endif
|
|
*(UpcaseName->VersionString.Buffer) = L';';
|
|
|
|
UpcaseName->VersionString.Buffer += 1;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Upcase the string using the correct upcase routine.
|
|
//
|
|
|
|
Status = RtlUpcaseUnicodeString( &UpcaseName->FileName,
|
|
&Name->FileName,
|
|
FALSE );
|
|
|
|
//
|
|
// This should never fail.
|
|
//
|
|
|
|
NT_ASSERT( Status == STATUS_SUCCESS );
|
|
__analysis_assert( Status == STATUS_SUCCESS );
|
|
|
|
if (Name->VersionString.Length != 0) {
|
|
|
|
Status = RtlUpcaseUnicodeString( &UpcaseName->VersionString,
|
|
&Name->VersionString,
|
|
FALSE );
|
|
|
|
//
|
|
// This should never fail.
|
|
//
|
|
|
|
NT_ASSERT( Status == STATUS_SUCCESS );
|
|
__analysis_assert( Status == STATUS_SUCCESS );
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
VOID
|
|
CdDissectName (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_Inout_ PUNICODE_STRING RemainingName,
|
|
_Out_ PUNICODE_STRING FinalName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to strip off leading components of the name strings. We search
|
|
for either the end of the string or separating characters. The input remaining
|
|
name strings should have neither a trailing or leading backslash.
|
|
|
|
Arguments:
|
|
|
|
RemainingName - Remaining name.
|
|
|
|
FinalName - Location to store next component of name.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG NameLength;
|
|
PWCHAR NextWchar;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// Find the offset of the next component separators.
|
|
//
|
|
|
|
for (NameLength = 0, NextWchar = RemainingName->Buffer;
|
|
(NameLength < RemainingName->Length) && (*NextWchar != L'\\');
|
|
NameLength += sizeof( WCHAR) , NextWchar += 1);
|
|
|
|
//
|
|
// Adjust all the strings by this amount.
|
|
//
|
|
|
|
FinalName->Buffer = RemainingName->Buffer;
|
|
|
|
FinalName->MaximumLength = FinalName->Length = (USHORT) NameLength;
|
|
|
|
//
|
|
// If this is the last component then set the RemainingName lengths to zero.
|
|
//
|
|
|
|
if (NameLength == RemainingName->Length) {
|
|
|
|
RemainingName->Length = 0;
|
|
|
|
//
|
|
// Otherwise we adjust the string by this amount plus the separating character.
|
|
//
|
|
|
|
} else {
|
|
|
|
RemainingName->MaximumLength -= (USHORT) (NameLength + sizeof( WCHAR ));
|
|
RemainingName->Length -= (USHORT) (NameLength + sizeof( WCHAR ));
|
|
RemainingName->Buffer = Add2Ptr( RemainingName->Buffer,
|
|
NameLength + sizeof( WCHAR ),
|
|
PWCHAR );
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
BOOLEAN
|
|
CdIsLegalName (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PUNICODE_STRING FileName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine checks if the name is a legal ISO 9660 name.
|
|
|
|
Arguments:
|
|
|
|
FileName - String of bytes containing the name.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN - TRUE if this name is a legal, FALSE otherwise.
|
|
|
|
--*/
|
|
|
|
{
|
|
PWCHAR Wchar;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// Check if name corresponds to a legal file name.
|
|
//
|
|
|
|
for (Wchar = FileName->Buffer;
|
|
Wchar < Add2Ptr( FileName->Buffer, FileName->Length, PWCHAR );
|
|
Wchar++) {
|
|
|
|
if ((*Wchar < 0xff) &&
|
|
!FsRtlIsAnsiCharacterLegalHpfs( *Wchar, FALSE ) &&
|
|
(*Wchar != L'"') &&
|
|
(*Wchar != L'<') &&
|
|
(*Wchar != L'>') &&
|
|
(*Wchar != L'|')) {
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOLEAN
|
|
CdIs8dot3Name (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ UNICODE_STRING FileName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine checks if the name follows the 8.3 name conventions. We check for
|
|
the name length and whether the characters are valid.
|
|
|
|
Arguments:
|
|
|
|
FileName - String of bytes containing the name.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN - TRUE if this name is a legal 8.3 name, FALSE otherwise.
|
|
|
|
--*/
|
|
|
|
{
|
|
CHAR DbcsNameBuffer[ BYTE_COUNT_8_DOT_3 ];
|
|
STRING DbcsName = {0};
|
|
|
|
PWCHAR NextWchar;
|
|
ULONG Count;
|
|
|
|
ULONG DotCount = 0;
|
|
BOOLEAN LastCharDot = FALSE;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// The length must be less than 24 bytes.
|
|
//
|
|
|
|
NT_ASSERT( FileName.Length != 0 );
|
|
if (FileName.Length > BYTE_COUNT_8_DOT_3) {
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Walk though and check for a space character.
|
|
//
|
|
|
|
NextWchar = FileName.Buffer;
|
|
Count = 0;
|
|
|
|
do {
|
|
|
|
//
|
|
// No spaces allowed.
|
|
//
|
|
|
|
if (*NextWchar == L' ') { return FALSE; }
|
|
|
|
if (*NextWchar == L'.') {
|
|
|
|
//
|
|
// Not an 8.3 name if more than 1 dot or more than 8 characters
|
|
// remaining. (It is legal for the dot to be in the ninth
|
|
// position)
|
|
//
|
|
|
|
if ((DotCount > 0) ||
|
|
(Count > 8 * sizeof( WCHAR ))) {
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
DotCount += 1;
|
|
LastCharDot = TRUE;
|
|
|
|
} else {
|
|
|
|
LastCharDot = FALSE;
|
|
}
|
|
|
|
Count += 2;
|
|
NextWchar += 1;
|
|
|
|
} while (Count < FileName.Length);
|
|
|
|
//
|
|
// Go ahead and truncate the dot if at the end.
|
|
//
|
|
|
|
if (LastCharDot) {
|
|
|
|
FileName.Length -= sizeof( WCHAR );
|
|
}
|
|
|
|
//
|
|
// Create an Oem name to use to check for a valid short name.
|
|
//
|
|
|
|
DbcsName.MaximumLength = BYTE_COUNT_8_DOT_3;
|
|
DbcsName.Buffer = DbcsNameBuffer;
|
|
|
|
if (!NT_SUCCESS( RtlUnicodeStringToCountedOemString( &DbcsName,
|
|
&FileName,
|
|
FALSE ))) {
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// We have now initialized the Oem string. Call the FsRtl package to check for a
|
|
// valid FAT name.
|
|
//
|
|
|
|
return FsRtlIsFatDbcsLegal( DbcsName, FALSE, FALSE, FALSE );
|
|
}
|
|
|
|
|
|
VOID
|
|
CdGenerate8dot3Name (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PUNICODE_STRING FileName,
|
|
_In_ ULONG DirentOffset,
|
|
_Out_writes_bytes_to_(BYTE_COUNT_8_DOT_3, *ShortByteCount) PWCHAR ShortFileName,
|
|
_Out_ PUSHORT ShortByteCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to generate a short name from the given long name. We will
|
|
generate a short name from the given long name.
|
|
|
|
We go through the following steps to make this conversion.
|
|
|
|
1 - Generate the generic short name. This will also be in unicode format.
|
|
|
|
2 - Build the string representation of the dirent offset.
|
|
|
|
3 - Build the biased short name string by combining the generic short name with
|
|
the dirent offset string.
|
|
|
|
4 - Copy the final unicode string back to our caller's buffer.
|
|
|
|
Arguments:
|
|
|
|
FileName - String of bytes containing the name.
|
|
|
|
DirentOffset - Offset in the directory for this filename. We incorporate the offset into
|
|
the short name by dividing this by 32 and prepending a tilde character to the
|
|
digit character. We then append this to the base of the generated short name.
|
|
|
|
ShortFileName - Pointer to the buffer to store the short name into.
|
|
|
|
ShortByteCount - Address to store the number of bytes in the short name.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
UNICODE_STRING ShortName;
|
|
UNICODE_STRING BiasedShortName;
|
|
WCHAR ShortNameBuffer[ BYTE_COUNT_8_DOT_3 / sizeof( WCHAR ) ] = {0};
|
|
WCHAR BiasedShortNameBuffer[ BYTE_COUNT_8_DOT_3 / sizeof( WCHAR ) ];
|
|
|
|
GENERATE_NAME_CONTEXT NameContext;
|
|
|
|
ULONG BiasedDirentOffset;
|
|
|
|
ULONG MaximumBaseBytes;
|
|
ULONG BaseNameOffset;
|
|
|
|
PWCHAR NextWchar;
|
|
WCHAR ThisWchar;
|
|
USHORT Length;
|
|
|
|
BOOLEAN FoundTilde = FALSE;
|
|
|
|
OEM_STRING OemName = {0};
|
|
USHORT OemNameOffset = 0;
|
|
BOOLEAN OverflowBuffer = FALSE;
|
|
|
|
PAGED_CODE();
|
|
|
|
//
|
|
// Initialize the short string to use the input buffer.
|
|
//
|
|
|
|
ShortName.Buffer = ShortNameBuffer;
|
|
ShortName.MaximumLength = BYTE_COUNT_8_DOT_3;
|
|
|
|
//
|
|
// Initialize the name context.
|
|
//
|
|
|
|
RtlZeroMemory( &NameContext, sizeof( GENERATE_NAME_CONTEXT ));
|
|
|
|
//
|
|
// We now have the unicode name for the input string. Go ahead and generate
|
|
// the short name.
|
|
//
|
|
|
|
RtlGenerate8dot3Name( FileName, TRUE, &NameContext, &ShortName );
|
|
|
|
//
|
|
// We now have the generic short name. We want incorporate the dirent offset
|
|
// into the name in order to reduce the chance of name conflicts. We will use
|
|
// a tilde character followed by a character representation of the dirent offset.
|
|
// This will be the hexadecimal representation of the dirent offset in the directory.
|
|
// It is actuall this offset divided by 32 since we don't need the full
|
|
// granularity.
|
|
//
|
|
|
|
BiasedDirentOffset = DirentOffset >> SHORT_NAME_SHIFT;
|
|
|
|
//
|
|
// Point to a local buffer to store the offset string. We start
|
|
// at the end of the buffer and work backwards.
|
|
//
|
|
|
|
NextWchar = Add2Ptr( BiasedShortNameBuffer,
|
|
BYTE_COUNT_8_DOT_3,
|
|
PWCHAR );
|
|
|
|
BiasedShortName.MaximumLength = BYTE_COUNT_8_DOT_3;
|
|
|
|
//
|
|
// Generate an OEM version of the string so that we can check for double
|
|
// byte characters.
|
|
//
|
|
|
|
Status = RtlUnicodeStringToOemString(&OemName, &ShortName, TRUE);
|
|
|
|
//
|
|
// If this failed, bail out. Don't expect any problems other than no mem.
|
|
//
|
|
|
|
if (!NT_SUCCESS( Status)) {
|
|
|
|
NT_ASSERT( STATUS_INSUFFICIENT_RESOURCES == Status);
|
|
CdRaiseStatus( IrpContext, Status);
|
|
}
|
|
|
|
Length = 0;
|
|
|
|
//
|
|
// Now add the characters for the dirent offset. We need to start
|
|
// from the least significant digit and work backwards.
|
|
//
|
|
|
|
do {
|
|
|
|
NextWchar -= 1;
|
|
|
|
ThisWchar = (WCHAR) (BiasedDirentOffset & 0x0000000f);
|
|
|
|
//
|
|
// Store in the next character. Bias against either '0' or 'A'
|
|
//
|
|
|
|
if (ThisWchar <= 9) {
|
|
|
|
*NextWchar = ThisWchar + L'0';
|
|
|
|
} else {
|
|
|
|
*NextWchar = ThisWchar + L'A' - 0xA;
|
|
}
|
|
|
|
Length += sizeof( WCHAR );
|
|
|
|
//
|
|
// Shift out the low 4 bits of the offset.
|
|
//
|
|
|
|
BiasedDirentOffset >>= 4;
|
|
|
|
} while (BiasedDirentOffset != 0);
|
|
|
|
//
|
|
// Now store in the tilde character.
|
|
//
|
|
|
|
NextWchar -= 1;
|
|
*NextWchar = L'~';
|
|
Length += sizeof( WCHAR );
|
|
|
|
//
|
|
// Set the length of this string.
|
|
//
|
|
|
|
BiasedShortName.Length = Length;
|
|
BiasedShortName.Buffer = NextWchar;
|
|
|
|
//
|
|
// Figure out the maximum number of characters we can copy of the base
|
|
// name. We subract the number of characters in the dirent string from 8.
|
|
// We will copy this many characters or stop when we reach a '.' character
|
|
// or a '~' character in the name.
|
|
//
|
|
|
|
MaximumBaseBytes = 16 - Length;
|
|
|
|
BaseNameOffset = 0;
|
|
|
|
//
|
|
// Keep copying from the base name until we hit a '.', '~' or the end of
|
|
// the short name.
|
|
//
|
|
|
|
NextWchar = ShortFileName;
|
|
Length = 0;
|
|
|
|
while ((BaseNameOffset < ShortName.Length) &&
|
|
(ShortName.Buffer[BaseNameOffset / 2] != L'.')) {
|
|
|
|
//
|
|
// Remember if we found a tilde character in the short name,
|
|
// so we don't copy it or anything following it.
|
|
//
|
|
|
|
if (ShortName.Buffer[BaseNameOffset / 2] == L'~') {
|
|
|
|
FoundTilde = TRUE;
|
|
}
|
|
|
|
//
|
|
// We need to consider the DBCS code page, because Unicode characters
|
|
// may use 2 bytes as DBCS characters.
|
|
//
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(push)
|
|
#pragma prefast(suppress:26014, "OemNameOffset <= BaseNameOffset throughout this loop; OemName buffer previously allocated based on ShortName's length.")
|
|
#endif
|
|
if (FsRtlIsLeadDbcsCharacter(OemName.Buffer[OemNameOffset])) {
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(pop)
|
|
#endif
|
|
|
|
OemNameOffset += 2;
|
|
|
|
if ((OemNameOffset + (BiasedShortName.Length / sizeof(WCHAR))) > 8) {
|
|
|
|
OverflowBuffer = TRUE;
|
|
}
|
|
}
|
|
else {
|
|
|
|
OemNameOffset++;
|
|
}
|
|
|
|
//
|
|
// Only copy the bytes if we still have space for the dirent string.
|
|
//
|
|
|
|
if (!FoundTilde && !OverflowBuffer && (BaseNameOffset < MaximumBaseBytes)) {
|
|
|
|
*NextWchar = ShortName.Buffer[BaseNameOffset / 2];
|
|
Length += sizeof( WCHAR );
|
|
NextWchar += 1;
|
|
}
|
|
|
|
BaseNameOffset += 2;
|
|
}
|
|
|
|
RtlFreeOemString(&OemName);
|
|
|
|
//
|
|
// Now copy the dirent string into the biased name buffer.
|
|
//
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(push)
|
|
#endif
|
|
RtlCopyMemory( NextWchar,
|
|
BiasedShortName.Buffer,
|
|
BiasedShortName.Length );
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(pop)
|
|
#endif
|
|
|
|
Length += BiasedShortName.Length;
|
|
NextWchar += (BiasedShortName.Length / sizeof( WCHAR ));
|
|
|
|
//
|
|
// Now copy any remaining bytes over to the biased short name.
|
|
//
|
|
|
|
if (BaseNameOffset != ShortName.Length) {
|
|
|
|
RtlCopyMemory( NextWchar,
|
|
&ShortName.Buffer[BaseNameOffset / 2],
|
|
ShortName.Length - BaseNameOffset );
|
|
|
|
Length += (ShortName.Length - (USHORT) BaseNameOffset);
|
|
}
|
|
|
|
//
|
|
// The final short name is stored in the user's buffer.
|
|
//
|
|
|
|
*ShortByteCount = Length;
|
|
}
|
|
|
|
|
|
BOOLEAN
|
|
CdIsNameInExpression (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PCD_NAME CurrentName,
|
|
_In_ PCD_NAME SearchExpression,
|
|
_In_ ULONG WildcardFlags,
|
|
_In_ BOOLEAN CheckVersion
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will compare two CdName strings. We assume that if this
|
|
is to be a case-insensitive search then they are already upcased.
|
|
|
|
We compare the filename portions of the name and if they match we
|
|
compare the version strings if requested.
|
|
|
|
Arguments:
|
|
|
|
CurrentName - Filename from the disk.
|
|
|
|
SearchExpression - Filename expression to use for match.
|
|
|
|
WildcardFlags - Flags field which indicates which parts of the
|
|
search expression might have wildcards. These flags are the
|
|
same as in the Ccb flags field.
|
|
|
|
CheckVersion - Indicates whether we should check both the name and the
|
|
version strings or just the name.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN - TRUE if the expressions match, FALSE otherwise.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOLEAN Match = TRUE;
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// If there are wildcards in the expression then we call the
|
|
// appropriate FsRtlRoutine.
|
|
//
|
|
|
|
if (FlagOn( WildcardFlags, CCB_FLAG_ENUM_NAME_EXP_HAS_WILD )) {
|
|
|
|
Match = FsRtlIsNameInExpression( &SearchExpression->FileName,
|
|
&CurrentName->FileName,
|
|
FALSE,
|
|
NULL );
|
|
|
|
//
|
|
// Otherwise do a direct memory comparison for the name string.
|
|
//
|
|
|
|
} else {
|
|
|
|
if ((CurrentName->FileName.Length != SearchExpression->FileName.Length) ||
|
|
(!RtlEqualMemory( CurrentName->FileName.Buffer,
|
|
SearchExpression->FileName.Buffer,
|
|
CurrentName->FileName.Length ))) {
|
|
|
|
Match = FALSE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check the version numbers if requested by the user and we have a
|
|
// match on the name and the version number is present.
|
|
//
|
|
|
|
if (Match && CheckVersion && SearchExpression->VersionString.Length &&
|
|
!FlagOn( WildcardFlags, CCB_FLAG_ENUM_VERSION_MATCH_ALL )) {
|
|
|
|
//
|
|
// If there are wildcards in the expression then call the
|
|
// appropriate search expression.
|
|
//
|
|
|
|
if (FlagOn( WildcardFlags, CCB_FLAG_ENUM_VERSION_EXP_HAS_WILD )) {
|
|
|
|
Match = FsRtlIsNameInExpression( &SearchExpression->VersionString,
|
|
&CurrentName->VersionString,
|
|
FALSE,
|
|
NULL );
|
|
|
|
//
|
|
// Otherwise do a direct memory comparison for the name string.
|
|
//
|
|
|
|
} else {
|
|
|
|
if ((CurrentName->VersionString.Length != SearchExpression->VersionString.Length) ||
|
|
(!RtlEqualMemory( CurrentName->VersionString.Buffer,
|
|
SearchExpression->VersionString.Buffer,
|
|
CurrentName->VersionString.Length ))) {
|
|
|
|
Match = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return Match;
|
|
}
|
|
|
|
|
|
ULONG
|
|
CdShortNameDirentOffset (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PUNICODE_STRING Name
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to examine a name to see if the dirent offset string is contained.
|
|
This consists of a tilde character followed by the offset represented as a hexadecimal
|
|
characters. We don't do any other checks to see if this is a short name. We
|
|
catch that later outside this routine.
|
|
|
|
Arguments:
|
|
|
|
Name - This is the CdName to examine.
|
|
|
|
Return Value:
|
|
|
|
ULONG - MAXULONG if there is no valid dirent offset string embedded, otherwise the
|
|
convert the value to numeric form.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG ResultOffset = MAXULONG;
|
|
ULONG RemainingByteCount = Name->Length;
|
|
|
|
BOOLEAN FoundTilde = FALSE;
|
|
|
|
PWCHAR NextWchar;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// Walk through the name until we either reach the end of the name
|
|
// or find a tilde character.
|
|
//
|
|
|
|
for (NextWchar = Name->Buffer;
|
|
RemainingByteCount != 0;
|
|
NextWchar += 1, RemainingByteCount -= sizeof( WCHAR )) {
|
|
|
|
//
|
|
// Check if this is a dot. Stop constructing any string if
|
|
// we found a dot.
|
|
//
|
|
|
|
if (*NextWchar == L'.') {
|
|
|
|
break;
|
|
}
|
|
|
|
//
|
|
// If we already found a tilde then check this character as a
|
|
// valid character. It must be a digit or A to F.
|
|
//
|
|
|
|
if (FoundTilde) {
|
|
|
|
if ((*NextWchar < L'0') ||
|
|
(*NextWchar > L'F') ||
|
|
((*NextWchar > L'9') && (*NextWchar < 'A'))) {
|
|
|
|
ResultOffset = MAXULONG;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Shift the result by 4 bits and add in this new character.
|
|
//
|
|
|
|
ResultOffset <<= 4;
|
|
|
|
if (*NextWchar < L'A') {
|
|
|
|
ResultOffset += *NextWchar - L'0';
|
|
|
|
} else {
|
|
|
|
ResultOffset += (*NextWchar - L'A') + 10;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// If this is a tilde then start building the dirent string.
|
|
//
|
|
|
|
if (*NextWchar == L'~') {
|
|
|
|
FoundTilde = TRUE;
|
|
ResultOffset = 0;
|
|
}
|
|
}
|
|
|
|
return ResultOffset;
|
|
}
|
|
|
|
|
|
//
|
|
// Local support routine
|
|
//
|
|
|
|
FSRTL_COMPARISON_RESULT
|
|
CdFullCompareNames (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PUNICODE_STRING NameA,
|
|
_In_ PUNICODE_STRING NameB
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function compares two names as fast as possible. Note that since
|
|
this comparison is case sensitive we can do a direct memory comparison.
|
|
|
|
Arguments:
|
|
|
|
NameA & NameB - The names to compare.
|
|
|
|
Return Value:
|
|
|
|
COMPARISON - returns
|
|
|
|
LessThan if NameA < NameB lexicalgraphically,
|
|
GreaterThan if NameA > NameB lexicalgraphically,
|
|
EqualTo if NameA is equal to NameB
|
|
|
|
--*/
|
|
|
|
{
|
|
SIZE_T i;
|
|
ULONG MinLength = NameA->Length;
|
|
FSRTL_COMPARISON_RESULT Result = LessThan;
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( IrpContext );
|
|
|
|
//
|
|
// Figure out the minimum of the two lengths
|
|
//
|
|
|
|
if (NameA->Length > NameB->Length) {
|
|
|
|
MinLength = NameB->Length;
|
|
Result = GreaterThan;
|
|
|
|
} else if (NameA->Length == NameB->Length) {
|
|
|
|
Result = EqualTo;
|
|
}
|
|
|
|
//
|
|
// Loop through looking at all of the characters in both strings
|
|
// testing for equalilty, less than, and greater than
|
|
//
|
|
|
|
i = RtlCompareMemory( NameA->Buffer, NameB->Buffer, MinLength );
|
|
|
|
if (i < MinLength) {
|
|
|
|
//
|
|
// We know the offset of the first character which is different.
|
|
//
|
|
|
|
return ((NameA->Buffer[ i / 2 ] < NameB->Buffer[ i / 2 ]) ?
|
|
LessThan :
|
|
GreaterThan);
|
|
}
|
|
|
|
//
|
|
// The names match up to the length of the shorter string.
|
|
// The shorter string lexically appears first.
|
|
//
|
|
|
|
return Result;
|
|
}
|
|
|
|
|
|
|