windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/misc.cxx

3714 lines
88 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) 1998-2001 Microsoft Corporation
Module Name:
misc.cxx
Abstract:
This module contains the miscellaneous UL routines.
Author:
Keith Moore (keithmo) 10-Jun-1998
Revision History:
--*/
#include "precomp.h"
ULONG HttpChars[256];
USHORT FastPopChars[256];
USHORT DummyPopChars[256];
//
// Private prototypes.
//
NTSTATUS
UlpRestartDeviceControl(
IN PDEVICE_OBJECT pDeviceObject,
IN PIRP pIrp,
IN PVOID pContext
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, UlOpenRegistry )
#pragma alloc_text( PAGE, UlReadLongParameter )
#pragma alloc_text( PAGE, UlReadLongLongParameter )
#pragma alloc_text( PAGE, UlReadGenericParameter )
#pragma alloc_text( PAGE, UlIssueDeviceControl )
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlBuildDeviceControlIrp
NOT PAGEABLE -- UlULongLongToAscii
NOT PAGEABLE -- UlpRestartDeviceControl
NOT PAGEABLE -- UlAllocateReceiveBufferPool
NOT PAGEABLE -- UlFreeReceiveBufferPool
NOT PAGEABLE -- UlAllocateIrpContextPool
NOT PAGEABLE -- UlFreeIrpContextPool
NOT PAGEABLE -- UlAllocateRequestBufferPool
NOT PAGEABLE -- UlFreeRequestBufferPool
NOT PAGEABLE -- UlAllocateInternalRequestPool
NOT PAGEABLE -- UlFreeInternalRequestPool
NOT PAGEABLE -- UlAllocateChunkTrackerPool
NOT PAGEABLE -- UlFreeChunkTrackerPool
NOT PAGEABLE -- UlAllocateFullTrackerPool
NOT PAGEABLE -- UlFreeFullTrackerPool
NOT PAGEABLE -- UlAllocateResponseBufferPool
NOT PAGEABLE -- UlFreeResponseBufferPool
NOT PAGEABLE -- UlAllocateLogBufferPool
NOT PAGEABLE -- UlFreeLogBufferPool
NOT PAGEABLE -- UlInvokeCompletionRoutine
NOT PAGEABLE -- UlUlInterlockedIncrement64
NOT PAGEABLE -- UlUlInterlockedDecrement64
NOT PAGEABLE -- UlUlInterlockedAdd64
NOT PAGEABLE -- UlUlInterlockedExchange64
NOT PAGEABLE -- TwoDigitsToUnicode
NOT PAGEABLE -- TimeFieldsToHttpDate
NOT PAGEABLE -- AsciiToShort
NOT PAGEABLE -- TwoAsciisToShort
NOT PAGEABLE -- NumericToAsciiMonth
NOT PAGEABLE -- StringTimeToSystemTime
#endif
//
// Public functions.
//
/***************************************************************************++
Routine Description:
Opens a handle to the UL's Parameters registry key.
Arguments:
BaseName - Supplies the name of the parent registry key containing
the Parameters key.
ParametersHandle - Returns a handle to the Parameters key.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlOpenRegistry(
IN PUNICODE_STRING BaseName,
OUT PHANDLE ParametersHandle
)
{
HANDLE configHandle;
NTSTATUS status;
PWSTR parametersString = REGISTRY_PARAMETERS;
UNICODE_STRING parametersKeyName;
OBJECT_ATTRIBUTES objectAttributes;
ULONG disposition;
//
// Sanity check.
//
PAGED_CODE();
//
// Open the registry for the initial string.
//
InitializeObjectAttributes(
&objectAttributes, // ObjectAttributes
BaseName, // ObjectName
OBJ_CASE_INSENSITIVE | // Attributes
UL_KERNEL_HANDLE,
NULL, // RootDirectory
NULL // SecurityDescriptor
);
UlAttachToSystemProcess();
status = ZwOpenKey( &configHandle, KEY_READ, &objectAttributes );
if (!NT_SUCCESS(status))
{
UlDetachFromSystemProcess();
return STATUS_UNSUCCESSFUL;
}
//
// Now open the parameters key.
//
RtlInitUnicodeString( &parametersKeyName, parametersString );
InitializeObjectAttributes(
&objectAttributes, // ObjectAttributes
&parametersKeyName, // ObjectName
OBJ_CASE_INSENSITIVE, // Attributes
configHandle, // RootDirectory
NULL // SecurityDescriptor
);
status = ZwOpenKey( ParametersHandle, KEY_READ, &objectAttributes );
if (!NT_SUCCESS(status))
{
ZwClose( configHandle );
UlDetachFromSystemProcess();
return status;
}
//
// All keys successfully opened or created.
//
ZwClose( configHandle );
UlDetachFromSystemProcess();
return STATUS_SUCCESS;
} // UlOpenRegistry
/***************************************************************************++
Routine Description:
Reads a single (LONG/ULONG) value from the registry.
Arguments:
ParametersHandle - Supplies an open registry handle.
ValueName - Supplies the name of the value to read.
DefaultValue - Supplies the default value.
Return Value:
LONG - The value read from the registry or the default if the
registry data was unavailable or incorrect.
--***************************************************************************/
LONG
UlReadLongParameter(
IN HANDLE ParametersHandle,
IN PWCHAR ValueName,
IN LONG DefaultValue
)
{
PKEY_VALUE_PARTIAL_INFORMATION information;
UNICODE_STRING valueKeyName;
ULONG informationLength;
LONG returnValue;
NTSTATUS status;
UCHAR buffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) + sizeof(LONG)];
//
// Sanity check.
//
PAGED_CODE();
//
// Build the value name, read it from the registry.
//
RtlInitUnicodeString(
&valueKeyName,
ValueName
);
information = (PKEY_VALUE_PARTIAL_INFORMATION)buffer;
status = ZwQueryValueKey(
ParametersHandle,
&valueKeyName,
KeyValuePartialInformation,
(PVOID)information,
sizeof(buffer),
&informationLength
);
//
// If the read succeeded, the type is DWORD and the length is
// sane, use it. Otherwise, use the default.
//
if (status == STATUS_SUCCESS &&
information->Type == REG_DWORD &&
information->DataLength == sizeof(returnValue))
{
RtlMoveMemory( &returnValue, information->Data, sizeof(returnValue) );
} else {
returnValue = DefaultValue;
}
return returnValue;
} // UlReadLongParameter
/***************************************************************************++
Routine Description:
Reads a single (LONGLONG/ULONGLONG) value from the registry.
Arguments:
ParametersHandle - Supplies an open registry handle.
ValueName - Supplies the name of the value to read.
DefaultValue - Supplies the default value.
Return Value:
LONGLONG - The value read from the registry or the default if the
registry data was unavailable or incorrect.
--***************************************************************************/
LONGLONG
UlReadLongLongParameter(
IN HANDLE ParametersHandle,
IN PWCHAR ValueName,
IN LONGLONG DefaultValue
)
{
PKEY_VALUE_PARTIAL_INFORMATION information;
UNICODE_STRING valueKeyName;
ULONG informationLength;
LONGLONG returnValue;
NTSTATUS status;
UCHAR buffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) + sizeof(LONGLONG)];
//
// Sanity check.
//
PAGED_CODE();
//
// Build the value name, read it from the registry.
//
RtlInitUnicodeString(
&valueKeyName,
ValueName
);
information = (PKEY_VALUE_PARTIAL_INFORMATION)buffer;
status = ZwQueryValueKey(
ParametersHandle,
&valueKeyName,
KeyValuePartialInformation,
(PVOID)information,
sizeof(buffer),
&informationLength
);
//
// If the read succeeded, the type is DWORD and the length is
// sane, use it. Otherwise, use the default.
//
if (status == STATUS_SUCCESS &&
information->Type == REG_QWORD &&
information->DataLength == sizeof(returnValue))
{
RtlMoveMemory( &returnValue, information->Data, sizeof(returnValue) );
} else {
returnValue = DefaultValue;
}
return returnValue;
} // UlReadLongLongParameter
/***************************************************************************++
Routine Description:
Reads a single free-form value from the registry.
Arguments:
ParametersHandle - Supplies an open registry handle.
ValueName - Supplies the name of the value to read.
Value - Receives the value read from the registry.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlReadGenericParameter(
IN HANDLE ParametersHandle,
IN PWCHAR ValueName,
OUT PKEY_VALUE_PARTIAL_INFORMATION * Value
)
{
KEY_VALUE_PARTIAL_INFORMATION partialInfo;
UNICODE_STRING valueKeyName;
ULONG informationLength;
NTSTATUS status;
PKEY_VALUE_PARTIAL_INFORMATION newValue;
ULONG dataLength;
//
// Sanity check.
//
PAGED_CODE();
//
// Build the value name, then perform an initial read. The read
// should fail with buffer overflow, but that's OK. We just want
// to get the length of the data.
//
RtlInitUnicodeString( &valueKeyName, ValueName );
status = ZwQueryValueKey(
ParametersHandle,
&valueKeyName,
KeyValuePartialInformation,
(PVOID)&partialInfo,
sizeof(partialInfo),
&informationLength
);
if (NT_ERROR(status))
{
return status;
}
//
// Determine the data length. Ensure that strings and multi-sz get
// properly terminated.
//
dataLength = partialInfo.DataLength - 1;
if (partialInfo.Type == REG_SZ || partialInfo.Type == REG_EXPAND_SZ)
{
dataLength += 1;
}
if (partialInfo.Type == REG_MULTI_SZ)
{
dataLength += 2;
}
//
// Allocate the buffer.
//
newValue = UL_ALLOCATE_STRUCT_WITH_SPACE(
PagedPool,
KEY_VALUE_PARTIAL_INFORMATION,
dataLength,
UL_REGISTRY_DATA_POOL_TAG
);
if (newValue == NULL)
{
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// update the actually allocated length for later use
//
dataLength += sizeof(KEY_VALUE_PARTIAL_INFORMATION);
RtlZeroMemory( newValue, dataLength );
//
// Perform the actual read.
//
status = ZwQueryValueKey(
ParametersHandle,
&valueKeyName,
KeyValuePartialInformation,
(PVOID)(newValue),
dataLength,
&informationLength
);
if (NT_SUCCESS(status))
{
*Value = newValue;
}
else
{
UL_FREE_POOL( newValue, UL_REGISTRY_DATA_POOL_TAG );
}
return status;
} // UlReadGenericParameter
/***************************************************************************++
Routine Description:
Builds a properly formatted device control IRP.
Arguments:
Irp - Supplies the IRP to format.
IoControlCode - Supplies the device IO control code.
InputBuffer - Supplies the input buffer.
InputBufferLength - Supplies the length of InputBuffer.
OutputBuffer - Supplies the output buffer.
OutputBufferLength - Supplies the length of OutputBuffer.
MdlAddress - Supplies a MDL to attach to the IRP. This is assumed to
be a non-paged MDL.
FileObject - Supplies the file object for the target driver.
DeviceObject - Supplies the correct device object for the target
driver.
IoStatusBlock - Receives the final completion status of the request.
CompletionRoutine - Supplies a pointer to a completion routine to
call after the request completes. This will only be called if
this routine returns STATUS_PENDING.
CompletionContext - Supplies an uninterpreted context value passed
to the completion routine.
TargetThread - Optionally supplies a target thread for the IRP. If
this value is NULL, then the current thread is used.
--***************************************************************************/
VOID
UlBuildDeviceControlIrp(
IN OUT PIRP Irp,
IN ULONG IoControlCode,
IN PVOID InputBuffer OPTIONAL,
IN ULONG InputBufferLength,
IN PVOID OutputBuffer OPTIONAL,
IN ULONG OutputBufferLength,
IN PMDL MdlAddress,
IN PFILE_OBJECT FileObject,
IN PDEVICE_OBJECT DeviceObject,
IN PIO_STATUS_BLOCK IoStatusBlock,
IN PIO_COMPLETION_ROUTINE CompletionRoutine,
IN PVOID CompletionContext,
IN PETHREAD TargetThread OPTIONAL
)
{
PIO_STACK_LOCATION irpSp;
//
// Sanity check.
//
ASSERT( Irp != NULL );
ASSERT( FileObject != NULL );
ASSERT( DeviceObject != NULL );
//
// Fill in the service independent parameters in the IRP.
//
Irp->Flags = 0;
Irp->RequestorMode = KernelMode;
Irp->PendingReturned = FALSE;
Irp->UserIosb = IoStatusBlock;
Irp->UserEvent = NULL;
Irp->AssociatedIrp.SystemBuffer = InputBuffer ? InputBuffer : OutputBuffer;
Irp->UserBuffer = OutputBuffer;
Irp->MdlAddress = MdlAddress;
Irp->Overlay.AsynchronousParameters.UserApcRoutine = NULL;
Irp->Tail.Overlay.Thread = TargetThread ? TargetThread : PsGetCurrentThread();
Irp->Tail.Overlay.OriginalFileObject = FileObject;
Irp->Tail.Overlay.AuxiliaryBuffer = NULL;
//
// Put the file object pointer in the stack location.
//
irpSp = IoGetNextIrpStackLocation( Irp );
irpSp->FileObject = FileObject;
irpSp->DeviceObject = DeviceObject;
//
// Fill in the service dependent parameters in the IRP stack.
//
irpSp->Parameters.DeviceIoControl.IoControlCode = IoControlCode;
irpSp->Parameters.DeviceIoControl.InputBufferLength = InputBufferLength;
irpSp->Parameters.DeviceIoControl.OutputBufferLength = OutputBufferLength;
irpSp->Parameters.DeviceIoControl.Type3InputBuffer = InputBuffer;
irpSp->MajorFunction = IRP_MJ_DEVICE_CONTROL;
irpSp->MinorFunction = 0;
//
// Set the completion routine appropriately.
//
if (CompletionRoutine == NULL)
{
IoSetCompletionRoutine(
Irp,
NULL,
NULL,
FALSE,
FALSE,
FALSE
);
}
else
{
IoSetCompletionRoutine(
Irp,
CompletionRoutine,
CompletionContext,
TRUE,
TRUE,
TRUE
);
}
} // UlBuildDeviceControlIrp
/***************************************************************************++
Routine Description:
Converts the given ULONGLLONG to an ASCII representation and stores it
in the given string.
Arguments:
String - Receives the ASCII representation of the ULONGLONG.
Value - Supplies the ULONGLONG to convert.
Return Value:
PSTR - Pointer to the next character in String *after* the converted
ULONGLONG.
--***************************************************************************/
PSTR
UlULongLongToAscii(
IN PSTR String,
IN ULONGLONG Value
)
{
PSTR p1;
PSTR p2;
CHAR ch;
ULONG digit;
//
// Special case 0 to make the rest of the routine simpler.
//
if (Value == 0)
{
*String++ = '0';
}
else
{
//
// Convert the ULONG. Note that this will result in the string
// being backwards in memory.
//
p1 = String;
p2 = String;
while (Value != 0)
{
digit = (ULONG)( Value % 10 );
Value = Value / 10;
*p1++ = '0' + (CHAR)digit;
}
//
// Reverse the string.
//
String = p1;
p1--;
while (p1 > p2)
{
ch = *p1;
*p1 = *p2;
*p2 = ch;
p2++;
p1--;
}
}
*String = '\0';
return String;
} // UlULongLongToAscii
NTSTATUS
_RtlIntegerToUnicode(
IN ULONG Value,
IN ULONG Base OPTIONAL,
IN LONG BufferLength,
OUT PWSTR String
)
{
PWSTR p1;
PWSTR p2;
WCHAR ch;
ULONG digit;
//
// Special case 0 to make the rest of the routine simpler.
//
if (Value == 0)
{
*String++ = L'0';
}
else
{
//
// Convert the ULONG. Note that this will result in the string
// being backwards in memory.
//
p1 = String;
p2 = String;
while (Value != 0)
{
digit = (ULONG)( Value % 10 );
Value = Value / 10;
*p1++ = L'0' + (WCHAR)digit;
}
//
// Reverse the string.
//
String = p1;
p1--;
while (p1 > p2)
{
ch = *p1;
*p1 = *p2;
*p2 = ch;
p2++;
p1--;
}
}
*String = L'\0';
return STATUS_SUCCESS;
} // _RtlIntegerToUnicode
/***************************************************************************++
Routine Description:
Converts an ansi string to an integer. fails if any non-digit characters
appears in the string. fails on negative numbers, and assumes no preceding
spaces.
Arguments:
PUCHAR pString the string to convert
ULONG Base the base, must be 10 or 16
PULONG pValue the return value of the converted integer
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlAnsiToULongLong(
PUCHAR pString,
ULONG Base,
PULONGLONG pValue
)
{
ULONGLONG Value;
ULONGLONG NewValue;
if (Base != 10 && Base != 16)
RETURN(STATUS_INVALID_PARAMETER);
//
// No preceding space, we already skipped it
//
ASSERT(IS_HTTP_LWS(pString[0]) == FALSE);
Value = 0;
while (pString[0] != ANSI_NULL)
{
if (
(Base == 10 && IS_HTTP_DIGIT(pString[0]) == FALSE) ||
(Base == 16 && IS_HTTP_HEX(pString[0]) == FALSE)
)
{
//
// Not valid , bad!
//
RETURN(STATUS_INVALID_PARAMETER);
}
if (Base == 16)
{
if (IS_HTTP_ALPHA(pString[0]))
{
NewValue = 16 * Value + (UPCASE_CHAR(pString[0]) - 'A' + 10);
}
else
{
NewValue = 16 * Value + (pString[0] - '0');
}
}
else
{
NewValue = 10 * Value + (pString[0] - '0');
}
if (NewValue < Value)
{
//
// Very bad... we overflew
//
RETURN(STATUS_SECTION_TOO_BIG);
}
Value = NewValue;
pString += 1;
}
*pValue = Value;
return STATUS_SUCCESS;
} // UlAnsiToULongLong
/***************************************************************************++
Routine Description:
Converts a unicode string to an integer. fails if any non-digit characters
appear in the string. fails on negative numbers, and assumes no preceding
spaces.
Arguments:
PWCHAR pString the string to convert
ULONG Base the base, must be 10 or 16
PULONG pValue the return value of the converted integer
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlUnicodeToULongLong(
PWCHAR pString,
ULONG Base,
PULONGLONG pValue
)
{
ULONGLONG Value;
ULONGLONG NewValue;
if (Base != 10 && Base != 16)
RETURN(STATUS_INVALID_PARAMETER);
//
// No preceding space, we already skipped it
//
ASSERT(pString[0] < 128 && IS_HTTP_LWS(pString[0]) == FALSE);
Value = 0;
while (pString[0] != UNICODE_NULL)
{
if ((Base == 10 &&
(pString[0] >= 128 || IS_HTTP_DIGIT(pString[0]) == FALSE)) ||
(Base == 16 &&
(pString[0] >= 128 || IS_HTTP_HEX(pString[0]) == FALSE)))
{
//
// Not valid , bad!
//
RETURN(STATUS_INVALID_PARAMETER);
}
if (Base == 16)
{
if (IS_HTTP_ALPHA(pString[0]))
{
NewValue = 16 * Value + (pString[0] - L'A' + 10);
}
else
{
NewValue = 16 * Value + (pString[0] - L'0');
}
}
else
{
NewValue = 10 * Value + (pString[0] - L'0');
}
if (NewValue < Value)
{
//
// Very bad... we overflew
//
RETURN(STATUS_INVALID_PARAMETER);
}
Value = NewValue;
pString += 1;
}
*pValue = Value;
return STATUS_SUCCESS;
} // UlUnicodeToULongLong
/***************************************************************************++
Routine Description:
Synchronously issues a device control request to the TDI provider.
Arguments:
pTdiObject - Supplies a pointer to the TDI object.
pIrpParameters - Supplies a pointer to the IRP parameters.
IrpParametersLength - Supplies the length of pIrpParameters.
pMdlBuffer - Optionally supplies a pointer to a buffer to be mapped
into a MDL and placed in the MdlAddress field of the IRP.
MdlBufferLength - Optionally supplies the length of pMdlBuffer.
MinorFunction - Supplies the minor function code of the request.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlIssueDeviceControl(
IN PUX_TDI_OBJECT pTdiObject,
IN PVOID pIrpParameters,
IN ULONG IrpParametersLength,
IN PVOID pMdlBuffer OPTIONAL,
IN ULONG MdlBufferLength OPTIONAL,
IN UCHAR MinorFunction
)
{
NTSTATUS status;
PIRP pIrp;
PIO_STACK_LOCATION pIrpSp;
UL_STATUS_BLOCK ulStatus;
PMDL pMdl;
//
// Sanity check.
//
PAGED_CODE();
//
// Initialize the event that will signal I/O completion.
//
UlInitializeStatusBlock( &ulStatus );
//
// Set the file object event to the non-signaled state.
//
KeResetEvent( &pTdiObject->pFileObject->Event );
//
// Allocate an IRP for the request.
//
pIrp = UlAllocateIrp(
pTdiObject->pDeviceObject->StackSize, // StackSize
FALSE // ChargeQuota
);
if (pIrp == NULL)
{
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Establish the service independent parameters.
//
pIrp->Flags = IRP_SYNCHRONOUS_API;
pIrp->RequestorMode = KernelMode;
pIrp->PendingReturned = FALSE;
pIrp->Tail.Overlay.Thread = PsGetCurrentThread();
pIrp->Tail.Overlay.OriginalFileObject = pTdiObject->pFileObject;
//
// If we have a MDL buffer, allocate a new MDL and map the
// buffer into it.
//
if (pMdlBuffer != NULL)
{
pMdl = UlAllocateMdl(
pMdlBuffer, // VirtualAddress
MdlBufferLength, // Length
FALSE, // SecondaryBuffer
FALSE, // ChargeQuota
pIrp // Irp
);
if (pMdl == NULL)
{
UlFreeIrp( pIrp );
return STATUS_INSUFFICIENT_RESOURCES;
}
MmBuildMdlForNonPagedPool( pMdl );
}
else
{
pIrp->MdlAddress = NULL;
}
//
// Initialize the IRP stack location.
//
pIrpSp = IoGetNextIrpStackLocation( pIrp );
pIrpSp->FileObject = pTdiObject->pFileObject;
pIrpSp->DeviceObject = pTdiObject->pDeviceObject;
ASSERT( IrpParametersLength <= sizeof(pIrpSp->Parameters) );
RtlCopyMemory(
&pIrpSp->Parameters,
pIrpParameters,
IrpParametersLength
);
pIrpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
pIrpSp->MinorFunction = MinorFunction;
//
// Reference the file object.
//
ObReferenceObject( pTdiObject->pFileObject );
//
// Establish a completion routine to free the MDL and dereference
// the FILE_OBJECT.
//
IoSetCompletionRoutine(
pIrp, // Irp
&UlpRestartDeviceControl, // CompletionRoutine
&ulStatus, // Context
TRUE, // InvokeOnSuccess
TRUE, // InvokeOnError
TRUE // InvokeOnCancel
);
//
// Issue the request.
//
status = UlCallDriver( pTdiObject->pDeviceObject, pIrp );
//
// If necessary, wait for the request to complete and snag the
// final completion status.
//
if (status == STATUS_PENDING)
{
UlWaitForStatusBlockEvent( &ulStatus );
status = ulStatus.IoStatus.Status;
}
return status;
} // UlIssueDeviceControl
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_RECEIVE_BUFFER structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be sizeof(UL_RECEIVE_BUFFER), but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_RCV_BUFFER_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateReceiveBufferPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_RECEIVE_BUFFER pBuffer;
SIZE_T irpLength;
SIZE_T mdlLength;
SIZE_T ExtraLength;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == sizeof(UL_RECEIVE_BUFFER) );
ASSERT( Tag == UL_RCV_BUFFER_POOL_TAG );
//
// Calculate the required length of the buffer & allocate it.
//
irpLength = IoSizeOfIrp( g_UlIrpStackSize );
irpLength = ALIGN_UP( irpLength, PVOID );
mdlLength = MmSizeOfMdl( (PVOID)(PAGE_SIZE - 1), g_UlReceiveBufferSize );
mdlLength = ALIGN_UP( mdlLength, PVOID );
ExtraLength = irpLength + (mdlLength*2) + g_UlReceiveBufferSize;
ASSERT( ( ExtraLength & (sizeof(PVOID) - 1) ) == 0 );
pBuffer = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_RECEIVE_BUFFER,
ExtraLength,
UL_RCV_BUFFER_POOL_TAG
);
if (pBuffer != NULL)
{
PUCHAR pRawBuffer = (PUCHAR)(pBuffer);
//
// Initialize the IRP, MDL, and data pointers within the buffer.
//
// CODEWORK: the signature should be set in invalid here, but
// there's no wrapper around the PplAllocate/Free functions
// for this structure, so set it to the valid sig for now.
//
pBuffer->Signature = UL_RECEIVE_BUFFER_SIGNATURE;
pRawBuffer += ALIGN_UP( sizeof(UL_RECEIVE_BUFFER), PVOID );
pBuffer->pIrp = (PIRP)pRawBuffer;
pRawBuffer += irpLength;
pBuffer->pMdl = (PMDL)pRawBuffer;
pRawBuffer += mdlLength;
pBuffer->pPartialMdl = (PMDL)pRawBuffer;
pRawBuffer += mdlLength;
pBuffer->pDataArea = (PVOID)pRawBuffer;
pBuffer->UnreadDataLength = 0;
//
// Initialize the IRP.
//
IoInitializeIrp(
pBuffer->pIrp, // Irp
(USHORT)irpLength, // PacketSize
g_UlIrpStackSize // StackSize
);
//
// Initialize the primary MDL.
//
MmInitializeMdl(
pBuffer->pMdl, // MemoryDescriptorList
pBuffer->pDataArea, // BaseVa
g_UlReceiveBufferSize // Length
);
MmBuildMdlForNonPagedPool( pBuffer->pMdl );
}
return (PVOID)pBuffer;
} // UlAllocateReceiveBufferPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_RECEIVE_BUFFER structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeReceiveBufferPool(
IN PVOID pBuffer
)
{
PUL_RECEIVE_BUFFER pReceiveBuffer;
//
// Sanity check.
//
pReceiveBuffer = (PUL_RECEIVE_BUFFER)pBuffer;
//
// Kill the signature, then free it.
//
pReceiveBuffer->Signature = UL_RECEIVE_BUFFER_SIGNATURE_X;
UL_FREE_POOL( pReceiveBuffer, UL_RCV_BUFFER_POOL_TAG );
} // UlFreeReceiveBufferPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_IRP_CONTEXT structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be sizeof(UL_IRP_CONTEXT), but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_IRP_CONTEXT_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateIrpContextPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_IRP_CONTEXT pIrpContext;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == sizeof(UL_IRP_CONTEXT) );
ASSERT( Tag == UL_IRP_CONTEXT_POOL_TAG );
//
// Allocate the IRP context.
//
pIrpContext = UL_ALLOCATE_STRUCT(
NonPagedPool,
UL_IRP_CONTEXT,
UL_IRP_CONTEXT_POOL_TAG
);
if (pIrpContext != NULL)
{
//
// Initialize it.
//
//
// CODEWORK: It's bogus for us to set the valid signature
// here. It should only be valid once the object is really
// in use.
//
pIrpContext->Signature = UL_IRP_CONTEXT_SIGNATURE;
#if DBG
pIrpContext->pCompletionRoutine = &UlDbgInvalidCompletionRoutine;
#endif
ASSERT( IS_VALID_IRP_CONTEXT( pIrpContext ) );
}
return (PVOID)pIrpContext;
} // UlAllocateIrpContextPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_IRP_CONTEXT structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeIrpContextPool(
IN PVOID pBuffer
)
{
PUL_IRP_CONTEXT pIrpContext;
//
// Sanity check.
//
pIrpContext = (PUL_IRP_CONTEXT)pBuffer;
ASSERT( IS_VALID_IRP_CONTEXT( pIrpContext ) );
//
// Kill the signature, then free it.
//
pIrpContext->Signature = UL_IRP_CONTEXT_SIGNATURE_X;
UL_FREE_POOL( pIrpContext, UL_IRP_CONTEXT_POOL_TAG );
} // UlFreeIrpContextPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_REQUEST_BUFFER structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be DEFAULT_MAX_REQUEST_BUFFER_SIZE but is basically
ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_REQUEST_BUFFER_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateRequestBufferPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_REQUEST_BUFFER pRequestBuffer;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == DEFAULT_MAX_REQUEST_BUFFER_SIZE );
ASSERT( Tag == UL_REQUEST_BUFFER_POOL_TAG );
//
// Allocate the request buffer.
//
pRequestBuffer = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_REQUEST_BUFFER,
DEFAULT_MAX_REQUEST_BUFFER_SIZE,
UL_REQUEST_BUFFER_POOL_TAG
);
if (pRequestBuffer != NULL)
{
//
// Initialize it.
//
pRequestBuffer->Signature = MAKE_FREE_TAG(UL_REQUEST_BUFFER_POOL_TAG);
}
return (PVOID)pRequestBuffer;
} // UlAllocateRequestBufferPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_REQUEST_BUFFER structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeRequestBufferPool(
IN PVOID pBuffer
)
{
PUL_REQUEST_BUFFER pRequestBuffer;
//
// Sanity check.
//
pRequestBuffer = (PUL_REQUEST_BUFFER)pBuffer;
//
// Kill the signature, then free it.
//
UL_FREE_POOL_WITH_SIG(pRequestBuffer, UL_REQUEST_BUFFER_POOL_TAG);
} // UlFreeRequestBufferPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_INTERNAL_REQUEST structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be sizeof(UL_INTERNAL_REQUEST) but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_INTERNAL_REQUEST_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateInternalRequestPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_INTERNAL_REQUEST pRequest;
PUL_FULL_TRACKER pTracker;
ULONG SpaceLength;
NTSTATUS Status;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == sizeof(UL_INTERNAL_REQUEST) );
ASSERT( Tag == UL_INTERNAL_REQUEST_POOL_TAG );
//
// Allocate the request buffer plus the default cooked URL buffer and
// the full tracker plus the auxiliary buffer.
//
SpaceLength =
g_UlFullTrackerSize +
(g_UlMaxInternalUrlLength/sizeof(WCHAR) + 1) * sizeof(WCHAR);
pRequest = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_INTERNAL_REQUEST,
SpaceLength,
UL_INTERNAL_REQUEST_POOL_TAG
);
if (pRequest != NULL)
{
//
// Initialize it.
//
pRequest->Signature = MAKE_FREE_TAG(UL_INTERNAL_REQUEST_POOL_TAG);
pRequest->pTracker =
(PUL_FULL_TRACKER)((PCHAR)pRequest +
ALIGN_UP(sizeof(UL_INTERNAL_REQUEST), PVOID));
pRequest->pUrlBuffer =
(PWSTR)((PCHAR)pRequest->pTracker + g_UlFullTrackerSize);
//
// Initialize the fast/cache tracker.
//
pTracker = pRequest->pTracker;
pTracker->Signature = UL_FULL_TRACKER_POOL_TAG;
pTracker->IrpContext.Signature = UL_IRP_CONTEXT_SIGNATURE;
pTracker->IsFromLookaside = FALSE;
pTracker->IsFromRequest = TRUE;
pTracker->AuxilaryBufferLength =
g_UlMaxFixedHeaderSize +
g_UlMaxVariableHeaderSize +
g_UlMaxCopyThreshold;
UlInitializeFullTrackerPool( pTracker, DEFAULT_MAX_IRP_STACK_SIZE );
}
return (PVOID)pRequest;
} // UlAllocateInternalRequestPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_INTERNAL_REQUEST structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeInternalRequestPool(
IN PVOID pBuffer
)
{
PUL_INTERNAL_REQUEST pRequest;
//
// Sanity check.
//
pRequest = (PUL_INTERNAL_REQUEST)pBuffer;
//
// Free the resource.
//
ASSERT( pRequest->Signature == MAKE_FREE_TAG(UL_INTERNAL_REQUEST_POOL_TAG));
//
// Kill the signature, then free it.
//
UL_FREE_POOL_WITH_SIG( pRequest, UL_INTERNAL_REQUEST_POOL_TAG );
} // UlFreeInternalRequestPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_CHUNK_TRACKER structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be g_UlChunkTrackerSize but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_CHUNK_TRACKER_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateChunkTrackerPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_CHUNK_TRACKER pTracker;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == g_UlChunkTrackerSize );
ASSERT( Tag == UL_CHUNK_TRACKER_POOL_TAG );
//
// Allocate the tracker buffer.
//
pTracker = (PUL_CHUNK_TRACKER)UL_ALLOCATE_POOL(
NonPagedPool,
g_UlChunkTrackerSize,
UL_CHUNK_TRACKER_POOL_TAG
);
if (pTracker != NULL)
{
pTracker->Signature = MAKE_FREE_TAG(UL_CHUNK_TRACKER_POOL_TAG);
pTracker->IrpContext.Signature = UL_IRP_CONTEXT_SIGNATURE;
pTracker->IsFromLookaside = TRUE;
//
// Set up the IRP.
//
pTracker->pReadIrp =
(PIRP)((PCHAR)pTracker +
ALIGN_UP(sizeof(UL_CHUNK_TRACKER), PVOID));
IoInitializeIrp(
pTracker->pReadIrp,
IoSizeOfIrp(DEFAULT_MAX_IRP_STACK_SIZE),
DEFAULT_MAX_IRP_STACK_SIZE
);
pTracker->pSendIrp =
(PIRP)((PCHAR)pTracker->pReadIrp +
ALIGN_UP(IoSizeOfIrp(DEFAULT_MAX_IRP_STACK_SIZE), PVOID));
IoInitializeIrp(
pTracker->pSendIrp,
IoSizeOfIrp(DEFAULT_MAX_IRP_STACK_SIZE),
DEFAULT_MAX_IRP_STACK_SIZE
);
//
// Set up the variable header pointer.
//
pTracker->pVariableHeader =
(PUCHAR)((PCHAR)pTracker->pSendIrp +
ALIGN_UP(IoSizeOfIrp(DEFAULT_MAX_IRP_STACK_SIZE), PVOID));
}
return pTracker;
} // UlAllocateChunkTrackerPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_CHUNK_TRACKER structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeChunkTrackerPool(
IN PVOID pBuffer
)
{
PUL_CHUNK_TRACKER pTracker = (PUL_CHUNK_TRACKER)pBuffer;
UL_FREE_POOL_WITH_SIG( pTracker, UL_CHUNK_TRACKER_POOL_TAG );
} // UlFreeChunkTrackerPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_FULL_TRACKER structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be g_UlFullTrackerSize but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_FULL_TRACKER_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateFullTrackerPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_FULL_TRACKER pTracker;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == g_UlFullTrackerSize );
ASSERT( Tag == UL_FULL_TRACKER_POOL_TAG );
//
// Allocate the tracker buffer.
//
pTracker = (PUL_FULL_TRACKER)UL_ALLOCATE_POOL(
NonPagedPool,
g_UlFullTrackerSize,
UL_FULL_TRACKER_POOL_TAG
);
if (pTracker != NULL)
{
pTracker->Signature = MAKE_FREE_TAG(UL_FULL_TRACKER_POOL_TAG);
pTracker->IrpContext.Signature = UL_IRP_CONTEXT_SIGNATURE;
pTracker->IsFromLookaside = TRUE;
pTracker->IsFromRequest = FALSE;
pTracker->AuxilaryBufferLength =
g_UlMaxFixedHeaderSize +
g_UlMaxVariableHeaderSize +
g_UlMaxCopyThreshold;
UlInitializeFullTrackerPool( pTracker, DEFAULT_MAX_IRP_STACK_SIZE );
}
return pTracker;
} // UlAllocateFullTrackerPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_FULL_TRACKER structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeFullTrackerPool(
IN PVOID pBuffer
)
{
PUL_FULL_TRACKER pTracker = (PUL_FULL_TRACKER)pBuffer;
UL_FREE_POOL_WITH_SIG( pTracker, UL_FULL_TRACKER_POOL_TAG );
} // UlFreeFullTrackerPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_INTERNAL_RESPONSE structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be NonPagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be g_UlResponseBufferSize but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_INTERNAL_RESPONSE_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateResponseBufferPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == g_UlResponseBufferSize );
ASSERT( Tag == UL_INTERNAL_RESPONSE_POOL_TAG );
//
// Allocate the default internal response buffer.
//
return UL_ALLOCATE_POOL(
NonPagedPool,
g_UlResponseBufferSize,
UL_INTERNAL_RESPONSE_POOL_TAG
);
} // UlAllocateResponseBufferPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_INTERNAL_RESPONSE structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeResponseBufferPool(
IN PVOID pBuffer
)
{
PUL_INTERNAL_RESPONSE pResponseBuffer;
pResponseBuffer = (PUL_INTERNAL_RESPONSE)pBuffer;
UL_FREE_POOL_WITH_SIG( pResponseBuffer, UL_INTERNAL_RESPONSE_POOL_TAG );
} // UlFreeResponseBufferPool
/***************************************************************************++
Routine Description:
Allocates the pool necessary for a new UL_FILE_LOG_BUFFER structure and
initializes the structure.
Arguments:
PoolType - Supplies the type of pool to allocate. This must always
be PagedPool.
ByteLength - Supplies the byte length for the allocation request.
This should be sizeof(UL_LOG_FILE_BUFFER) but is basically ignored.
Tag - Supplies the tag to use for the pool. This should be
UL_LOG_FILE_BUFFER_POOL_TAG, but is basically ignored.
Note: These parameters are required so that this function has a
signature identical to ExAllocatePoolWithTag.
Return Value:
PVOID - Pointer to the newly allocated block if successful, FALSE
otherwise.
--***************************************************************************/
PVOID
UlAllocateLogBufferPool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
PUL_LOG_FILE_BUFFER pLogBuffer;
//
// Sanity check.
//
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == sizeof(UL_LOG_FILE_BUFFER) );
ASSERT( Tag == UL_LOG_FILE_BUFFER_POOL_TAG );
//
// Allocate the default log buffer.
//
pLogBuffer = UL_ALLOCATE_STRUCT_WITH_SPACE(
PagedPool,
UL_LOG_FILE_BUFFER,
g_UlLogBufferSize,
UL_LOG_FILE_BUFFER_POOL_TAG
);
if ( pLogBuffer != NULL )
{
pLogBuffer->Signature = MAKE_FREE_TAG(UL_LOG_FILE_BUFFER_POOL_TAG);
pLogBuffer->BufferUsed = 0;
pLogBuffer->Buffer = (PUCHAR) (pLogBuffer + 1);
}
return pLogBuffer;
} // UlAllocateLogBufferPool
/***************************************************************************++
Routine Description:
Frees the pool allocated for a UL_LOG_FILE_BUFFER structure.
Arguments:
pBuffer - Supplies the buffer to free.
--***************************************************************************/
VOID
UlFreeLogBufferPool(
IN PVOID pBuffer
)
{
PUL_LOG_FILE_BUFFER pLogBuffer;
pLogBuffer = (PUL_LOG_FILE_BUFFER) pBuffer;
UL_FREE_POOL_WITH_SIG( pLogBuffer, UL_LOG_FILE_BUFFER_POOL_TAG );
} // UlFreeLogBufferPool
//
// Private routines.
//
/***************************************************************************++
Routine Description:
Completion handler for device control IRPs.
Arguments:
pDeviceObject - Supplies the device object for the IRP being
completed.
pIrp - Supplies the IRP being completed.
pContext - Supplies the context associated with this request. In
this case, it's a pointer to a UL_STATUS_BLOCK structure.
Return Value:
NTSTATUS - STATUS_SUCCESS if IO should continue processing this
IRP, STATUS_MORE_PROCESSING_REQUIRED if IO should stop processing
this IRP.
--***************************************************************************/
NTSTATUS
UlpRestartDeviceControl(
IN PDEVICE_OBJECT pDeviceObject,
IN PIRP pIrp,
IN PVOID pContext
)
{
PUL_STATUS_BLOCK pStatus;
//
// If we attached an MDL to the IRP, then free it here and reset
// the MDL pointer to NULL. IO can't handle a nonpaged MDL in an
// IRP, so we do it here.
//
if (pIrp->MdlAddress != NULL)
{
UlFreeMdl( pIrp->MdlAddress );
pIrp->MdlAddress = NULL;
}
//
// Complete the request.
//
pStatus = (PUL_STATUS_BLOCK)pContext;
UlSignalStatusBlock(
pStatus,
pIrp->IoStatus.Status,
pIrp->IoStatus.Information
);
//
// Tell IO to continue processing this IRP.
//
return STATUS_SUCCESS;
} // UlpRestartDeviceControl
/*++
Routine Description:
Routine to initialize the utilitu code.
Arguments:
Return Value:
--*/
NTSTATUS
InitializeHttpUtil(
VOID
)
{
ULONG i;
UCHAR c;
// Initialize the HttpChars array appropriately.
for (i = 0; i < 128; i++)
{
HttpChars[i] = HTTP_CHAR;
}
for (i = 'A'; i <= 'Z'; i++)
{
HttpChars[i] |= HTTP_UPCASE;
}
for (i = 'a'; i <= 'z'; i++)
{
HttpChars[i] |= HTTP_LOCASE;
}
for (i = '0'; i <= '9'; i++)
{
HttpChars[i] |= (HTTP_DIGIT | HTTP_HEX);
}
for (i = 0; i <= 31; i++)
{
HttpChars[i] |= HTTP_CTL;
}
HttpChars[127] |= HTTP_CTL;
HttpChars[SP] |= HTTP_LWS;
HttpChars[HT] |= HTTP_LWS;
for (i = 'A'; i <= 'F'; i++)
{
HttpChars[i] |= HTTP_HEX;
}
for (i = 'a'; i <= 'f'; i++)
{
HttpChars[i] |= HTTP_HEX;
}
HttpChars['('] |= HTTP_SEPERATOR;
HttpChars[')'] |= HTTP_SEPERATOR;
HttpChars['<'] |= HTTP_SEPERATOR;
HttpChars['>'] |= HTTP_SEPERATOR;
HttpChars['@'] |= HTTP_SEPERATOR;
HttpChars[','] |= HTTP_SEPERATOR;
HttpChars[';'] |= HTTP_SEPERATOR;
HttpChars[':'] |= HTTP_SEPERATOR;
HttpChars['\\'] |= HTTP_SEPERATOR;
HttpChars['"'] |= HTTP_SEPERATOR;
HttpChars['/'] |= HTTP_SEPERATOR;
HttpChars['['] |= HTTP_SEPERATOR;
HttpChars[']'] |= HTTP_SEPERATOR;
HttpChars['?'] |= HTTP_SEPERATOR;
HttpChars['='] |= HTTP_SEPERATOR;
HttpChars['{'] |= HTTP_SEPERATOR;
HttpChars['}'] |= HTTP_SEPERATOR;
HttpChars[SP] |= HTTP_SEPERATOR;
HttpChars[HT] |= HTTP_SEPERATOR;
//
// URL "reserved" characters (rfc2396)
//
HttpChars[';'] |= URL_LEGAL;
HttpChars['/'] |= URL_LEGAL;
HttpChars['\\'] |= URL_LEGAL;
HttpChars['?'] |= URL_LEGAL;
HttpChars[':'] |= URL_LEGAL;
HttpChars['@'] |= URL_LEGAL;
HttpChars['&'] |= URL_LEGAL;
HttpChars['='] |= URL_LEGAL;
HttpChars['+'] |= URL_LEGAL;
HttpChars['$'] |= URL_LEGAL;
HttpChars[','] |= URL_LEGAL;
//
// URL escape character
//
HttpChars['%'] |= URL_LEGAL;
//
// URL "mark" characters (rfc2396)
//
HttpChars['-'] |= URL_LEGAL;
HttpChars['_'] |= URL_LEGAL;
HttpChars['.'] |= URL_LEGAL;
HttpChars['!'] |= URL_LEGAL;
HttpChars['~'] |= URL_LEGAL;
HttpChars['*'] |= URL_LEGAL;
HttpChars['\''] |= URL_LEGAL;
HttpChars['('] |= URL_LEGAL;
HttpChars[')'] |= URL_LEGAL;
//
// RFC2396 describes these characters as `unwise' "because gateways and
// other transport agents are known to sometimes modify such characters,
// or they are used as delimiters". However, for compatibility with IIS
// 5.0 and DAV, we must allow these unwise characters in URLs.
//
HttpChars['{'] |= URL_LEGAL;
HttpChars['}'] |= URL_LEGAL;
HttpChars['|'] |= URL_LEGAL;
HttpChars['^'] |= URL_LEGAL;
HttpChars['['] |= URL_LEGAL;
HttpChars[']'] |= URL_LEGAL;
HttpChars['`'] |= URL_LEGAL;
//
// These US-ASCII characters are "excluded"; i.e. not URL_LEGAL (see RFC):
// '<' | '>' | '#' | '%' | '"' (0x22) | ' ' (0x20)
// In addition, control characters (0x00-0x1F and 0x7F) and
// non US-ASCII characters (0x80-0xFF) are not URL_LEGAL.
//
for (i = 0; i < 128; i++)
{
if (!IS_HTTP_SEPERATOR(i) && !IS_HTTP_CTL(i))
{
HttpChars[i] |= HTTP_TOKEN;
}
}
//
// Fast path for PopChar
//
RtlZeroMemory(FastPopChars, 256 * sizeof(USHORT));
RtlZeroMemory(DummyPopChars, 256 * sizeof(USHORT));
for (i = 0; i < 256; i++)
{
c = (UCHAR)i;
if (IS_URL_TOKEN(c) && c != '%' && (c & 0x80) != 0x80)
{
FastPopChars[i] = (USHORT)c;
}
}
//
// Turn backslashes into forward slashes
//
FastPopChars['\\'] = L'/';
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
Allocates brand new UL_NONPAGED_RESOURCE structures from the
lookaside lists
Arguments:
Return Value:
the resource. NULL = out of memory.
--***************************************************************************/
PUL_NONPAGED_RESOURCE
UlResourceNew(
ULONG OwnerTag
)
{
PUL_NONPAGED_RESOURCE pResource;
pResource = (PUL_NONPAGED_RESOURCE)(
PplAllocate(
g_pUlNonpagedData->ResourceLookaside
)
);
if (pResource != NULL)
{
pResource->Signature = UL_NONPAGED_RESOURCE_SIGNATURE;
#if DBG
pResource->Resource.OwnerTag = OwnerTag;
#endif
pResource->RefCount = 1;
UlTrace(
REFCOUNT,
("ul!UlResourceNew res=%p tag=0x%08x refcount=%d\n",
pResource, OwnerTag,
pResource->RefCount)
);
}
return pResource;
}
/***************************************************************************++
Routine Description:
Reference a resource.
Arguments:
pResource - the resource
Return Value:
VOID
--***************************************************************************/
VOID
UlReferenceResource(
PUL_NONPAGED_RESOURCE pResource
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(IS_VALID_UL_NONPAGED_RESOURCE(pResource));
refCount = InterlockedIncrement( &pResource->RefCount );
UlTrace(
REFCOUNT,
("ul!UlReferenceResource res=%p refcount=%d\n",
pResource,
refCount)
);
} // UlReferenceResource
/***************************************************************************++
Routine Description:
Dereference a resource.
Arguments:
pResource - the resource
Return Value:
VOID
--***************************************************************************/
VOID
UlDereferenceResource(
PUL_NONPAGED_RESOURCE pResource
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(IS_VALID_UL_NONPAGED_RESOURCE(pResource));
refCount = InterlockedDecrement( &pResource->RefCount );
UlTrace(
REFCOUNT,
("ul!UlDereferenceResource res=%p refcount=%d\n",
pResource,
refCount)
);
if (refCount == 0)
{
//
// free it
//
PplFree(
g_pUlNonpagedData->ResourceLookaside,
pResource
);
}
} // UlDereferenceResource
/***************************************************************************++
Routine Description:
support function for lookasides to allocate the memory for
UL_NONPAGED_RESOURCEs
Arguments:
PoolType - the pool type
ByteLength - how much to alloc
Tag - the tag to use
Return Value:
nt status code
--***************************************************************************/
PVOID
UlResourceAllocatePool(
IN POOL_TYPE PoolType,
IN SIZE_T ByteLength,
IN ULONG Tag
)
{
NTSTATUS Status;
PUL_NONPAGED_RESOURCE pResource;
ASSERT( PoolType == NonPagedPool );
ASSERT( ByteLength == sizeof(UL_NONPAGED_RESOURCE) );
ASSERT( Tag == UL_NONPAGED_RESOURCE_POOL_TAG );
pResource = UL_ALLOCATE_STRUCT(
NonPagedPool,
UL_NONPAGED_RESOURCE,
UL_NONPAGED_RESOURCE_POOL_TAG
);
if (pResource != NULL)
{
pResource->Signature = UL_NONPAGED_RESOURCE_SIGNATURE;
pResource->RefCount = 1;
Status = UlInitializeResource(
&pResource->Resource,
"UL_NONPAGED_RESOURCE[%p].Resource",
pResource,
UL_NONPAGED_RESOURCE_POOL_TAG
);
if (NT_SUCCESS(Status) == FALSE)
{
UL_FREE_POOL_WITH_SIG(pResource, UL_NONPAGED_RESOURCE_POOL_TAG);
return NULL;
}
}
return (PVOID)(pResource);
}
/***************************************************************************++
Routine Description:
support function for lookasides to free the memory for
UL_NONPAGED_RESOURCE's
Arguments:
pBuffer - the UL_NONPAGED_RESOURCE buffer to free
Return Value:
VOID
--***************************************************************************/
VOID
UlResourceFreePool(
IN PVOID pBuffer
)
{
PUL_NONPAGED_RESOURCE pResource;
NTSTATUS Status;
pResource = (PUL_NONPAGED_RESOURCE)(pBuffer);
ASSERT(pResource != NULL);
ASSERT(pResource->Signature == UL_NONPAGED_RESOURCE_SIGNATURE);
pResource->Signature = UL_NONPAGED_RESOURCE_SIGNATURE_X;
Status = UlDeleteResource(&pResource->Resource);
ASSERT(NT_SUCCESS(Status));
UL_FREE_POOL_WITH_SIG(
pResource,
UL_NONPAGED_RESOURCE_POOL_TAG
);
} // UlResourceFreePool
/***************************************************************************++
Routine Description:
Invokes the completion routine (if specified) and determines the
appropriate return code. This routine ensures that, if the completion
routine is invoked, the caller always returns STATUS_PENDING.
Arguments:
Status - Supplies the completion status.
Information - Optionally supplies additional information about
the completed operation, such as the number of bytes
transferred.
pCompletionRoutine - Supplies a pointer to a completion routine to
invoke after the listening endpoint is fully closed.
pCompletionContext - Supplies an uninterpreted context value for the
completion routine.
Return Value:
NTSTATUS - Completion status. Will always be STATUS_PENDING if the
completion routine is invoked.
--***************************************************************************/
NTSTATUS
UlInvokeCompletionRoutine(
IN NTSTATUS Status,
IN ULONG_PTR Information,
IN PUL_COMPLETION_ROUTINE pCompletionRoutine,
IN PVOID pCompletionContext
)
{
if (pCompletionRoutine != NULL)
{
(pCompletionRoutine)(
pCompletionContext,
Status,
Information
);
Status = STATUS_PENDING;
}
return Status;
} // UlInvokeCompletionRoutine
//
// constants used by the date formatter
//
const PWSTR pDays[] =
{
L"Sun", L"Mon", L"Tue", L"Wed", L"Thu", L"Fri", L"Sat"
};
const PWSTR pMonths[] =
{
L"Jan", L"Feb", L"Mar", L"Apr", L"May", L"Jun", L"Jul",
L"Aug", L"Sep", L"Oct", L"Nov", L"Dec"
};
__inline
VOID
TwoDigitsToUnicode(
PWSTR pBuffer,
ULONG Number
)
{
pBuffer[0] = L'0' + (WCHAR)(Number / 10);
pBuffer[1] = L'0' + (WCHAR)(Number % 10);
}
/***************************************************************************++
Routine Description:
Converts the given system time to string representation containing
GMT Formatted String.
Arguments:
pTime - System time that needs to be converted.
pBuffer - pointer to string which will contain the GMT time on
successful return.
BufferLength - size of pszBuff in bytes
Return Value:
NTSTATUS
History:
MuraliK 3-Jan-1995
paulmcd 4-Mar-1999 copied to ul
--***************************************************************************/
NTSTATUS
TimeFieldsToHttpDate(
IN PTIME_FIELDS pTime,
OUT PWSTR pBuffer,
IN ULONG BufferLength
)
{
NTSTATUS Status;
ASSERT(pBuffer != NULL);
if (BufferLength < (HTTP_DATE_COUNT + 1)*sizeof(WCHAR))
{
return STATUS_BUFFER_TOO_SMALL;
}
// 0 1 2
// 01234567890123456789012345678
// Formats a string like: "Thu, 14 Jul 1994 15:26:05 GMT"
//
//
// write the constants
//
pBuffer[3] = L',';
pBuffer[4] = pBuffer[7] = pBuffer[11] = L' ';
pBuffer[19] = pBuffer[22] = L':';
//
// now the variants
//
//
// 0-based Weekday
//
RtlCopyMemory(&(pBuffer[0]), pDays[pTime->Weekday], 3*sizeof(WCHAR));
TwoDigitsToUnicode(&(pBuffer[5]), pTime->Day);
//
// 1-based Month
//
RtlCopyMemory(&(pBuffer[8]), pMonths[pTime->Month - 1], 3*sizeof(WCHAR)); // 1-based
Status = _RtlIntegerToUnicode(pTime->Year, 10, 5, &(pBuffer[12]));
ASSERT(NT_SUCCESS(Status));
pBuffer[16] = L' ';
TwoDigitsToUnicode(&(pBuffer[17]), pTime->Hour);
TwoDigitsToUnicode(&(pBuffer[20]), pTime->Minute);
TwoDigitsToUnicode(&(pBuffer[23]), pTime->Second);
RtlCopyMemory(&(pBuffer[25]), L" GMT", sizeof(L" GMT"));
return STATUS_SUCCESS;
} // TimeFieldsToHttpDate
__inline
SHORT
FASTCALL
AsciiToShort(
PCHAR pString
)
{
return (SHORT)atoi(pString);
}
__inline
SHORT
FASTCALL
TwoAsciisToShort(
PCHAR pString
)
{
SHORT Value;
SHORT Number;
Number = pString[1] - '0';
if (Number <= 9)
{
Value = Number;
Number = pString[0] - '0';
if (Number <= 9)
{
Value += Number * 10;
return Value;
}
}
return 0;
}
/***************************************************************************++
DateTime function ported from user mode W3SVC
--***************************************************************************/
/************************************************************
* Data
************************************************************/
static const PSTR s_rgchMonths[] = {
"Jan", "Feb", "Mar", "Apr",
"May", "Jun", "Jul", "Aug",
"Sep", "Oct", "Nov", "Dec"
};
// Custom hash table for NumericToAsciiMonth() for mapping "Apr" to 4
static const CHAR MonthIndexTable[64] = {
-1,'A', 2, 12, -1, -1, -1, 8, // A to G
-1, -1, -1, -1, 7, -1,'N', -1, // F to O
9, -1,'R', -1, 10, -1, 11, -1, // P to W
-1, 5, -1, -1, -1, -1, -1, -1, // X to Z
-1,'A', 2, 12, -1, -1, -1, 8, // a to g
-1, -1, -1, -1, 7, -1,'N', -1, // f to o
9, -1,'R', -1, 10, -1, 11, -1, // p to w
-1, 5, -1, -1, -1, -1, -1, -1 // x to z
};
/************************************************************
* Functions
************************************************************/
/***************************************************************************++
Converts three letters of a month to numeric month
Arguments:
s String to convert
Returns:
numeric equivalent, 0 on failure.
--***************************************************************************/
__inline
SHORT
FASTCALL
NumericToAsciiMonth(
PCHAR s
)
{
UCHAR monthIndex;
UCHAR c;
PSTR monthString;
//
// use the third character as the index
//
c = (s[2] - 0x40) & 0x3F;
monthIndex = MonthIndexTable[c];
if ( monthIndex < 13 ) {
goto verify;
}
//
// ok, we need to look at the second character
//
if ( monthIndex == 'N' ) {
//
// we got an N which we need to resolve further
//
//
// if s[1] is 'u' then Jun, if 'a' then Jan
//
if ( MonthIndexTable[(s[1]-0x40) & 0x3f] == 'A' ) {
monthIndex = 1;
} else {
monthIndex = 6;
}
} else if ( monthIndex == 'R' ) {
//
// if s[1] is 'a' then March, if 'p' then April
//
if ( MonthIndexTable[(s[1]-0x40) & 0x3f] == 'A' ) {
monthIndex = 3;
} else {
monthIndex = 4;
}
} else {
goto error_exit;
}
verify:
monthString = (PSTR) s_rgchMonths[monthIndex-1];
if ( (s[0] == monthString[0]) &&
(s[1] == monthString[1]) &&
(s[2] == monthString[2]) ) {
return(monthIndex);
} else if ( (toupper(s[0]) == monthString[0]) &&
(tolower(s[1]) == monthString[1]) &&
(tolower(s[2]) == monthString[2]) ) {
return monthIndex;
}
error_exit:
return(0);
} // NumericToAsciiMonth
/***************************************************************************++
Converts a string representation of a GMT time (three different
varieties) to an NT representation of a file time.
We handle the following variations:
Sun, 06 Nov 1994 08:49:37 GMT (RFC 822 updated by RFC 1123)
Sunday, 06-Nov-94 08:49:37 GMT (RFC 850)
Sun Nov 6 08:49:37 1994 (ANSI C's asctime() format
Arguments:
pszTime String representation of time field
pliTime large integer containing the time in NT format.
Returns:
TRUE on success and FALSE on failure.
History:
Johnl 24-Jan-1995 Modified from WWW library
ericsten 30-Nov-2000 Ported from user-mode W3SVC
--***************************************************************************/
BOOLEAN
StringTimeToSystemTime(
IN const PSTR pszTime,
OUT LARGE_INTEGER * pliTime
)
{
PCHAR s;
TIME_FIELDS st;
if (pszTime == NULL) {
return FALSE;
}
st.Milliseconds = 0;
if ((s = strchr(pszTime, ','))) {
ULONG len;
//
// Thursday, 10-Jun-93 01:29:59 GMT
// or: Thu, 10 Jan 1993 01:29:59 GMT */
//
s++;
while (*s && *s==' ') s++;
len = strlen(s);
if (len < 18) {
return FALSE;
}
if ( *(s+2) == '-' ) { /* First format */
st.Day = AsciiToShort(s);
st.Month = NumericToAsciiMonth(s+3);
st.Year = AsciiToShort(s+7);
st.Hour = AsciiToShort(s+10);
st.Minute = AsciiToShort(s+13);
st.Second = AsciiToShort(s+16);
} else { /* Second format */
if (len < 20) {
return FALSE;
}
st.Day = TwoAsciisToShort(s);
st.Month = NumericToAsciiMonth(s+3);
st.Year = TwoAsciisToShort(s+7) * 100 + TwoAsciisToShort(s+9);
st.Hour = TwoAsciisToShort(s+12);
st.Minute = TwoAsciisToShort(s+15);
st.Second = TwoAsciisToShort(s+18);
}
} else { /* Try the other format: Wed Jun 9 01:29:59 1993 GMT */
s = (PCHAR) pszTime;
while (*s && *s==' ') s++;
if ((int)strlen(s) < 24) {
return FALSE;
}
if (isdigit(*(s+8))) {
st.Day = AsciiToShort(s+8);
} else {
if ( ' ' != *(s+8) ) {
return FALSE;
}
st.Day = AsciiToShort(s+9);
}
st.Month = NumericToAsciiMonth(s+4);
st.Year = AsciiToShort(s+20);
st.Hour = AsciiToShort(s+11);
st.Minute = AsciiToShort(s+14);
st.Second = AsciiToShort(s+17);
}
//
// Adjust for dates with only two digits
//
if ( st.Year < 1000 ) {
if ( st.Year < 50 ) {
st.Year += 2000;
} else {
st.Year += 1900;
}
}
if ( !RtlTimeFieldsToTime( &st, pliTime )) {
return FALSE;
}
return(TRUE);
}
/***************************************************************************++
End of DateTime function ported from user mode W3SVC
--***************************************************************************/
//
// Some Unicode to Utf8 conversion utilities taken and modified frm
// base\win32\winnls\utf.c. Use this until they expose the same functionality
// in kernel.
//
/***************************************************************************++
Routine Description:
Maps a Unicode character string to its UTF-8 string counterpart
Conversion continues until the source is finished or an error happens in
either case it returns the number of UTF-8 characters written.
If the supllied buffer is not big enough it returns 0.
--***************************************************************************/
ULONG
HttpUnicodeToUTF8(
IN PCWSTR lpSrcStr,
IN LONG cchSrc,
OUT LPSTR lpDestStr,
IN LONG cchDest
)
{
LPCWSTR lpWC = lpSrcStr;
LONG cchU8 = 0; // # of UTF8 chars generated
DWORD dwSurrogateChar;
WCHAR wchHighSurrogate = 0;
BOOLEAN bHandled;
while ((cchSrc--) && ((cchDest == 0) || (cchU8 < cchDest)))
{
bHandled = FALSE;
//
// Check if high surrogate is available
//
if ((*lpWC >= HIGH_SURROGATE_START) && (*lpWC <= HIGH_SURROGATE_END))
{
if (cchDest)
{
// Another high surrogate, then treat the 1st as normal
// Unicode character.
if (wchHighSurrogate)
{
if ((cchU8 + 2) < cchDest)
{
lpDestStr[cchU8++] = UTF8_1ST_OF_3 | HIGHER_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | MIDDLE_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | LOWER_6_BIT(wchHighSurrogate);
}
else
{
// not enough buffer
cchSrc++;
break;
}
}
}
else
{
cchU8 += 3;
}
wchHighSurrogate = *lpWC;
bHandled = TRUE;
}
if (!bHandled && wchHighSurrogate)
{
if ((*lpWC >= LOW_SURROGATE_START) && (*lpWC <= LOW_SURROGATE_END))
{
// wheee, valid surrogate pairs
if (cchDest)
{
if ((cchU8 + 3) < cchDest)
{
dwSurrogateChar = (((wchHighSurrogate-0xD800) << 10) + (*lpWC - 0xDC00) + 0x10000);
lpDestStr[cchU8++] = (UTF8_1ST_OF_4 | (unsigned char)(dwSurrogateChar >> 18)); // 3 bits from 1st byte
lpDestStr[cchU8++] = (UTF8_TRAIL | (unsigned char)((dwSurrogateChar >> 12) & 0x3f)); // 6 bits from 2nd byte
lpDestStr[cchU8++] = (UTF8_TRAIL | (unsigned char)((dwSurrogateChar >> 6) & 0x3f)); // 6 bits from 3rd byte
lpDestStr[cchU8++] = (UTF8_TRAIL | (unsigned char)(0x3f &dwSurrogateChar)); // 6 bits from 4th byte
}
else
{
// not enough buffer
cchSrc++;
break;
}
}
else
{
// we already counted 3 previously (in high surrogate)
cchU8 += 1;
}
bHandled = TRUE;
}
else
{
// Bad Surrogate pair : ERROR
// Just process wchHighSurrogate , and the code below will
// process the current code point
if (cchDest)
{
if ((cchU8 + 2) < cchDest)
{
lpDestStr[cchU8++] = UTF8_1ST_OF_3 | HIGHER_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | MIDDLE_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | LOWER_6_BIT(wchHighSurrogate);
}
else
{
// not enough buffer
cchSrc++;
break;
}
}
}
wchHighSurrogate = 0;
}
if (!bHandled)
{
if (*lpWC <= ASCII)
{
//
// Found ASCII.
//
if (cchDest)
{
lpDestStr[cchU8] = (char)*lpWC;
}
cchU8++;
}
else if (*lpWC <= UTF8_2_MAX)
{
//
// Found 2 byte sequence if < 0x07ff (11 bits).
//
if (cchDest)
{
if ((cchU8 + 1) < cchDest)
{
//
// Use upper 5 bits in first byte.
// Use lower 6 bits in second byte.
//
lpDestStr[cchU8++] = UTF8_1ST_OF_2 | (*lpWC >> 6);
lpDestStr[cchU8++] = UTF8_TRAIL | LOWER_6_BIT(*lpWC);
}
else
{
//
// Error - buffer too small.
//
cchSrc++;
break;
}
}
else
{
cchU8 += 2;
}
}
else
{
//
// Found 3 byte sequence.
//
if (cchDest)
{
if ((cchU8 + 2) < cchDest)
{
//
// Use upper 4 bits in first byte.
// Use middle 6 bits in second byte.
// Use lower 6 bits in third byte.
//
lpDestStr[cchU8++] = UTF8_1ST_OF_3 | HIGHER_6_BIT(*lpWC);
lpDestStr[cchU8++] = UTF8_TRAIL | MIDDLE_6_BIT(*lpWC);
lpDestStr[cchU8++] = UTF8_TRAIL | LOWER_6_BIT(*lpWC);
}
else
{
//
// Error - buffer too small.
//
cchSrc++;
break;
}
}
else
{
cchU8 += 3;
}
}
}
lpWC++;
}
//
// If the last character was a high surrogate, then handle it as a normal
// unicode character.
//
if ((cchSrc < 0) && (wchHighSurrogate != 0))
{
if (cchDest)
{
if ((cchU8 + 2) < cchDest)
{
lpDestStr[cchU8++] = UTF8_1ST_OF_3 | HIGHER_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | MIDDLE_6_BIT(wchHighSurrogate);
lpDestStr[cchU8++] = UTF8_TRAIL | LOWER_6_BIT(wchHighSurrogate);
}
else
{
cchSrc++;
}
}
}
//
// Make sure the destination buffer was large enough.
//
if (cchDest && (cchSrc >= 0))
{
return 0;
}
//
// Return the number of UTF-8 characters written.
//
return cchU8;
}
/*++
Routine Description:
Search input list of ETags for one that matches our local ETag.
Arguments:
pLocalETag - The local ETag we're using.
pETagList - The ETag list we've received from the client.
bWeakCompare - Whether using Weak Comparison is ok
Returns:
TRUE if we found a matching ETag, FALSE otherwise.
Author:
Anil Ruia (AnilR) 3-Apr-2000
History:
Eric Stenson (EricSten) 6-Dec-2000 ported from user-mode
--*/
BOOLEAN FindInETagList(
IN PUCHAR pLocalETag,
IN PUCHAR pETagList,
IN BOOLEAN fWeakCompare
)
{
ULONG QuoteCount;
PUCHAR pFileETag;
BOOLEAN Matched;
// We'll loop through the ETag string, looking for ETag to
// compare, as long as we have an ETag to look at.
do
{
while (isspace(*pETagList))
{
pETagList++;
}
if (!*pETagList)
{
// Ran out of ETag.
return FALSE;
}
// If this ETag is *, it's a match.
if (*pETagList == '*')
{
return TRUE;
}
// See if this ETag is weak.
if (pETagList[0] == 'W' && pETagList[1] == '/')
{
// This is a weak validator. If we're not doing the weak
// comparison, fail.
if (!fWeakCompare)
{
return FALSE;
}
// Skip over the 'W/', and any intervening whitespace.
pETagList += 2;
while (isspace(*pETagList))
{
pETagList++;
}
if (!*pETagList)
{
// Ran out of ETag.
return FALSE;
}
}
if (*pETagList != '"')
{
// This isn't a quoted string, so fail.
return FALSE;
}
// OK, right now we should be at the start of a quoted string that
// we can compare against our current ETag.
QuoteCount = 0;
Matched = TRUE;
pFileETag = pLocalETag;
// Do the actual compare. We do this by scanning the current ETag,
// which is a quoted string. We look for two quotation marks, the
// the delimiters if the quoted string. If after we find two quotes
// in the ETag everything has matched, then we've matched this ETag.
// Otherwise we'll try the next one.
do
{
CHAR Temp;
Temp = *pETagList;
if (Temp == '"')
{
QuoteCount++;
}
if (*pFileETag != Temp)
{
Matched = FALSE;
}
if (!Temp)
{
return FALSE;
}
pETagList++;
if (*pFileETag == '\0')
{
break;
}
pFileETag++;
}
while (QuoteCount != 2);
if (Matched)
{
return TRUE;
}
// Otherwise, at this point we need to look at the next ETag.
while (QuoteCount != 2)
{
if (*pETagList == '"')
{
QuoteCount++;
}
else
{
if (*pETagList == '\0')
{
return FALSE;
}
}
pETagList++;
}
while (isspace(*pETagList))
{
pETagList++;
}
if (*pETagList == ',')
{
pETagList++;
}
else
{
return FALSE;
}
}
while ( *pETagList );
return FALSE;
}
/*++
Routine Description:
Build a NULL terminated UNICODE string from the IP and Port address
Arguments:
IpAddressStringW - String buffer to place the UNICODE string
(caller allocated)
IpAddress - 32-bit version of the IP address
IpPortNum - 16-bit version of the TCP Port
Returns:
Count of bytes written into IpAddressStringW.
Author:
Eric Stenson (EricSten) 29-Jan-2001
--*/
ULONG
HostAddressAndPortToStringW(
IN OUT PWCHAR IpAddressStringW,
IN ULONG IpAddress,
IN USHORT IpPortNum
)
{
PWCHAR pszW = IpAddressStringW;
pszW = UlStrPrintUlongW(pszW, (IpAddress >> 24) & 0xFF, 0, '.');
pszW = UlStrPrintUlongW(pszW, (IpAddress >> 16) & 0xFF, 0, '.');
pszW = UlStrPrintUlongW(pszW, (IpAddress >> 8) & 0xFF, 0, '.');
pszW = UlStrPrintUlongW(pszW, (IpAddress >> 0) & 0xFF, 0, ':');
pszW = UlStrPrintUlongW(pszW, IpPortNum, 0, '\0');
return DIFF(pszW - IpAddressStringW) * sizeof(WCHAR);
}
/*++
Routine Description:
Calculates current bias (daylight time aware) and time zone ID.
Captured from base\client\datetime.c
Until this two functions are exposed in the kernel we have to
keep them here.
Arguments:
IN CONST TIME_ZONE_INFORMATION *ptzi - time zone for which to calculate bias
OUT KSYSTEM_TIME *pBias - current bias
Return Value:
TIME_ZONE_ID_UNKNOWN - daylight saving time is not used in the
current time zone.
TIME_ZONE_ID_STANDARD - The system is operating in the range covered
by StandardDate.
TIME_ZONE_ID_DAYLIGHT - The system is operating in the range covered
by DaylightDate.
TIME_ZONE_ID_INVALID - The operation failed.
--*/
ULONG
UlCalcTimeZoneIdAndBias(
IN RTL_TIME_ZONE_INFORMATION *ptzi,
OUT PLONG pBias
)
{
LARGE_INTEGER TimeZoneBias;
LARGE_INTEGER NewTimeZoneBias;
LARGE_INTEGER LocalCustomBias;
LARGE_INTEGER UtcStandardTime;
LARGE_INTEGER UtcDaylightTime;
LARGE_INTEGER StandardTime;
LARGE_INTEGER DaylightTime;
LARGE_INTEGER CurrentUniversalTime;
ULONG CurrentTimeZoneId = UL_TIME_ZONE_ID_INVALID;
NewTimeZoneBias.QuadPart = Int32x32To64(ptzi->Bias*60, 10000000);
//
// Now see if we have stored cutover times
//
if (ptzi->StandardStart.Month && ptzi->DaylightStart.Month)
{
KeQuerySystemTime(&CurrentUniversalTime);
//
// We have timezone cutover information. Compute the
// cutover dates and compute what our current bias
// is
//
if((!UlCutoverTimeToSystemTime(
&ptzi->StandardStart,
&StandardTime,
&CurrentUniversalTime)
) ||
(!UlCutoverTimeToSystemTime(
&ptzi->DaylightStart,
&DaylightTime,
&CurrentUniversalTime)
)
)
{
return UL_TIME_ZONE_ID_INVALID;
}
//
// Convert standard time and daylight time to utc
//
LocalCustomBias.QuadPart = Int32x32To64(ptzi->StandardBias*60, 10000000);
TimeZoneBias.QuadPart = NewTimeZoneBias.QuadPart + LocalCustomBias.QuadPart;
UtcDaylightTime.QuadPart = DaylightTime.QuadPart + TimeZoneBias.QuadPart;
LocalCustomBias.QuadPart = Int32x32To64(ptzi->DaylightBias*60, 10000000);
TimeZoneBias.QuadPart = NewTimeZoneBias.QuadPart + LocalCustomBias.QuadPart;
UtcStandardTime.QuadPart = StandardTime.QuadPart + TimeZoneBias.QuadPart;
//
// If daylight < standard, then time >= daylight and
// less than standard is daylight
//
if (UtcDaylightTime.QuadPart < UtcStandardTime.QuadPart)
{
//
// If today is >= DaylightTime and < StandardTime, then
// We are in daylight savings time
//
if ((CurrentUniversalTime.QuadPart >= UtcDaylightTime.QuadPart) &&
(CurrentUniversalTime.QuadPart < UtcStandardTime.QuadPart))
{
CurrentTimeZoneId = UL_TIME_ZONE_ID_DAYLIGHT;
}
else
{
CurrentTimeZoneId = UL_TIME_ZONE_ID_STANDARD;
}
}
else
{
//
// If today is >= StandardTime and < DaylightTime, then
// We are in standard time
//
if ((CurrentUniversalTime.QuadPart >= UtcStandardTime.QuadPart) &&
(CurrentUniversalTime.QuadPart < UtcDaylightTime.QuadPart))
{
CurrentTimeZoneId = UL_TIME_ZONE_ID_STANDARD;
}
else
{
CurrentTimeZoneId = UL_TIME_ZONE_ID_DAYLIGHT;
}
}
// Bias in minutes
*pBias = ptzi->Bias + (CurrentTimeZoneId == UL_TIME_ZONE_ID_DAYLIGHT ?
ptzi->DaylightBias : ptzi->StandardBias
);
}
else
{
*pBias = ptzi->Bias;
CurrentTimeZoneId = UL_TIME_ZONE_ID_UNKNOWN;
}
return CurrentTimeZoneId;
}
BOOLEAN
UlCutoverTimeToSystemTime(
PTIME_FIELDS CutoverTime,
PLARGE_INTEGER SystemTime,
PLARGE_INTEGER CurrentSystemTime
)
{
TIME_FIELDS CurrentTimeFields;
//
// Get the current system time
//
RtlTimeToTimeFields(CurrentSystemTime,&CurrentTimeFields);
//
// check for absolute time field. If the year is specified,
// the the time is an abosulte time
//
if ( CutoverTime->Year )
{
return FALSE;
}
else
{
TIME_FIELDS WorkingTimeField;
TIME_FIELDS ScratchTimeField;
LARGE_INTEGER ScratchTime;
CSHORT BestWeekdayDate;
CSHORT WorkingWeekdayNumber;
CSHORT TargetWeekdayNumber;
CSHORT TargetYear;
CSHORT TargetMonth;
CSHORT TargetWeekday; // range [0..6] == [Sunday..Saturday]
BOOLEAN MonthMatches;
//
// The time is an day in the month style time
//
// the convention is the Day is 1-5 specifying 1st, 2nd... Last
// day within the month. The day is WeekDay.
//
//
// Compute the target month and year
//
TargetWeekdayNumber = CutoverTime->Day;
if ( TargetWeekdayNumber > 5 || TargetWeekdayNumber == 0 ) {
return FALSE;
}
TargetWeekday = CutoverTime->Weekday;
TargetMonth = CutoverTime->Month;
MonthMatches = FALSE;
TargetYear = CurrentTimeFields.Year;
try_next_year:
BestWeekdayDate = 0;
WorkingTimeField.Year = TargetYear;
WorkingTimeField.Month = TargetMonth;
WorkingTimeField.Day = 1;
WorkingTimeField.Hour = CutoverTime->Hour;
WorkingTimeField.Minute = CutoverTime->Minute;
WorkingTimeField.Second = CutoverTime->Second;
WorkingTimeField.Milliseconds = CutoverTime->Milliseconds;
WorkingTimeField.Weekday = 0;
//
// Convert to time and then back to time fields so we can determine
// the weekday of day 1 on the month
//
if ( !RtlTimeFieldsToTime(&WorkingTimeField,&ScratchTime) ) {
return FALSE;
}
RtlTimeToTimeFields(&ScratchTime,&ScratchTimeField);
//
// Compute bias to target weekday
//
if ( ScratchTimeField.Weekday > TargetWeekday ) {
WorkingTimeField.Day += (7-(ScratchTimeField.Weekday - TargetWeekday));
}
else if ( ScratchTimeField.Weekday < TargetWeekday ) {
WorkingTimeField.Day += (TargetWeekday - ScratchTimeField.Weekday);
}
//
// We are now at the first weekday that matches our target weekday
//
BestWeekdayDate = WorkingTimeField.Day;
WorkingWeekdayNumber = 1;
//
// Keep going one week at a time until we either pass the
// target weekday, or we match exactly
//
while ( WorkingWeekdayNumber < TargetWeekdayNumber ) {
WorkingTimeField.Day += 7;
if ( !RtlTimeFieldsToTime(&WorkingTimeField,&ScratchTime) ) {
break;
}
RtlTimeToTimeFields(&ScratchTime,&ScratchTimeField);
WorkingWeekdayNumber++;
BestWeekdayDate = ScratchTimeField.Day;
}
WorkingTimeField.Day = BestWeekdayDate;
//
// If the months match, and the date is less than the current
// date, then be have to go to next year.
//
if ( !RtlTimeFieldsToTime(&WorkingTimeField,&ScratchTime) ) {
return FALSE;
}
if ( MonthMatches ) {
if ( WorkingTimeField.Day < CurrentTimeFields.Day ) {
MonthMatches = FALSE;
TargetYear++;
goto try_next_year;
}
if ( WorkingTimeField.Day == CurrentTimeFields.Day ) {
if (ScratchTime.QuadPart < CurrentSystemTime->QuadPart) {
MonthMatches = FALSE;
TargetYear++;
goto try_next_year;
}
}
}
*SystemTime = ScratchTime;
return TRUE;
}
}