windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/ullog.cxx
2020-09-26 16:20:57 +08:00

7424 lines
203 KiB
C++

/*++
Copyright (c) 2000-2001 Microsoft Corporation
Module Name:
ullog.cxx (UL IIS6 HIT Logging)
Abstract:
This module implements the logging facilities
for IIS6 including the NCSA, IIS and W3CE types
of logging.
Author:
Ali E. Turkoglu (aliTu) 10-May-2000
Revision History:
--*/
#include "precomp.h"
#include "iiscnfg.h"
#include "ullogp.h"
//
// Generic Private globals.
//
LIST_ENTRY g_LogListHead = {NULL,NULL};
LONG g_LogListEntryCount = 0;
BOOLEAN g_InitLogsCalled = FALSE;
CHAR g_GMTOffset[SIZE_OF_GMT_OFFSET+1];
BOOLEAN g_UTF8Logging = FALSE;
//
// For Log Buffering
//
ULONG g_AllocationGranularity = 0;
KTIMER g_BufferTimer;
BOOLEAN g_BufferTimerInitialized = FALSE;
UL_SPIN_LOCK g_BufferTimerSpinLock;
KDPC g_BufferTimerDpcObject;
//
// For Logging Date & Time caching
//
UL_LOG_DATE_AND_TIME_CACHE
g_UlDateTimeCache[HttpLoggingTypeMaximum];
LARGE_INTEGER g_UlLogSystemTime;
FAST_MUTEX g_LogCacheFastMutex;
//
// For Log Cycling
//
KDPC g_LogTimerDpcObject;
KTIMER g_LogTimer;
BOOLEAN g_LogTimerInitialized = FALSE;
BOOLEAN g_LogTimerStarted = FALSE;
BOOLEAN g_BufferTimerStarted = FALSE;
// This spinlock covers both g_BufferTimerStarted and
// g_LogTimerStarted variables
UL_SPIN_LOCK g_LogTimerSpinLock;
//
// Used to wait for endpoints to close on shutdown
//
UL_SPIN_LOCK g_BufferIoSpinLock;
BOOLEAN g_BufferWaitingForIoComplete = FALSE;
KEVENT g_BufferIoCompleteEvent;
ULONG g_BufferIoCount = 0;
#ifdef ALLOC_PRAGMA
#pragma alloc_text( INIT, UlInitializeLogs )
#pragma alloc_text( PAGE, UlTerminateLogs )
#pragma alloc_text( PAGE, UlpGetGMTOffset )
#pragma alloc_text( PAGE, UlpRecycleLogFile )
#pragma alloc_text( PAGE, UlpGetLogFileLength )
#pragma alloc_text( PAGE, UlCreateLog )
#pragma alloc_text( PAGE, UlRemoveLogFileEntry )
#pragma alloc_text( PAGE, UlpConstructLogFileEntry )
#pragma alloc_text( PAGE, UlProbeLogData )
#pragma alloc_text( PAGE, UlAllocateLogDataBuffer )
#pragma alloc_text( PAGE, UlpConstructFileName )
#pragma alloc_text( PAGE, UlpUpdateLogFlags )
#pragma alloc_text( PAGE, UlpUpdateLogTruncateSize )
#pragma alloc_text( PAGE, UlpUpdatePeriod )
#pragma alloc_text( PAGE, UlpUpdateFormat )
#pragma alloc_text( PAGE, UlReConfigureLogEntry )
#pragma alloc_text( PAGE, UlpGrowLogEntry )
#pragma alloc_text( PAGE, UlLogTimerHandler )
#pragma alloc_text( PAGE, UlBufferTimerHandler )
#pragma alloc_text( PAGE, UlpCalculateTimeToExpire )
#pragma alloc_text( PAGE, UlpCreateSafeDirectory )
#pragma alloc_text( PAGE, UlpAppendW3CLogTitle )
#pragma alloc_text( PAGE, UlpWriteToLogFile )
#pragma alloc_text( PAGE, UlpFlushLogFile )
#pragma alloc_text( PAGE, UlCheckLogDirectory )
#pragma alloc_text( PAGE, UlSetUTF8Logging )
#pragma alloc_text( PAGE, UlCaptureLogFieldsW3C )
#pragma alloc_text( PAGE, UlCaptureLogFieldsNCSA )
#pragma alloc_text( PAGE, UlCaptureLogFieldsIIS )
#pragma alloc_text( PAGE, UlLogHttpCacheHit )
#pragma alloc_text( PAGE, UlLogHttpCacheHitWorker )
#pragma alloc_text( PAGE, UlLogHttpHit )
#pragma alloc_text( PAGE, UlpGenerateDateAndTimeFields )
#pragma alloc_text( PAGE, UlpGetDateTimeFields )
#pragma alloc_text( PAGE, UlpLogCloseHandle )
#pragma alloc_text( PAGE, UlpLogCloseHandleWorker )
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlLogTimerDpcRoutine
NOT PAGEABLE -- UlpTerminateLogTimer
NOT PAGEABLE -- UlpInsertLogFileEntry
NOT PAGEABLE -- UlpSetLogTimer
NOT PAGEABLE -- UlpSetBufferTimer
NOT PAGEABLE -- UlBufferTimerDpcRoutine
NOT PAGEABLE -- UlpTerminateTimers
NOT PAGEABLE -- UlpInitializeTimers
NOT PAGEABLE -- UlWaitForBufferIoToComplete
NOT PAGEABLE -- UlpBufferFlushAPC
NOT PAGEABLE -- UlDestroyLogDataBuffer
NOT PAGEABLE -- UlDestroyLogDataBufferWorker
#endif
//
// Public functions.
//
/***************************************************************************++
Routine Description:
UlInitializeLogs :
Initialize the resource for log list synchronization
--***************************************************************************/
NTSTATUS
UlInitializeLogs (
VOID
)
{
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
ASSERT(!g_InitLogsCalled);
if (!g_InitLogsCalled)
{
InitializeListHead(&g_LogListHead);
Status = UlInitializeResource(
&g_pUlNonpagedData->LogListResource,
"LogListResource",
0,
UL_LOG_LIST_RESOURCE_TAG
);
ASSERT(NT_SUCCESS(Status)); // the call always returns success
if (!NT_SUCCESS(Status))
{
return Status;
}
g_InitLogsCalled = TRUE;
UlpInitializeTimers();
UlpInitializeLogCache();
UlpGetGMTOffset();
UlInitializeSpinLock( &g_BufferIoSpinLock, "g_BufferIoSpinLock" );
//
// Get the allocation granularity from the system
// It will be used as log buffer size if there's no registry
// overwrites.
//
UlpInitializeLogBufferGranularity();
if (g_AllocationGranularity == 0)
{
g_AllocationGranularity = DEFAULT_MAX_LOG_BUFFER_SIZE;
}
//
// Overwrite the log buffer size with the above value
// if registry parameter doesn't exist.
//
if (g_UlLogBufferSize == 0)
{
// No registry parameter available use the system granularity
g_UlLogBufferSize = g_AllocationGranularity;
}
else
{
// Proceed with using the registry provided log buffer size
UlTrace( LOGGING,
("Ul!UlInitializeLogs: Log buffer size %d from registry!\n",
g_UlLogBufferSize
));
}
}
return Status;
}
/***************************************************************************++
Routine Description:
UlTerminateLogs :
Deletes the resource for log list synchronization
--***************************************************************************/
VOID
UlTerminateLogs(
VOID
)
{
NTSTATUS Status;
PAGED_CODE();
if (g_InitLogsCalled)
{
ASSERT( IsListEmpty( &g_LogListHead )) ;
//
// Make sure terminate the log timer before
// deleting the log list resource
//
UlpTerminateTimers();
Status = UlDeleteResource(
&g_pUlNonpagedData->LogListResource
);
ASSERT(NT_SUCCESS(Status));
UlWaitForBufferIoToComplete();
g_InitLogsCalled = FALSE;
}
}
/***************************************************************************++
Routine Description:
UlSetUTF8Logging :
Sets the UTF8Logging on or off. Only once. Initially Utf8Logging is
FALSE and it may only be set during the init once. Following possible
changes won't be taken.
ReConfiguration code is explicitly missing as WAS will anly call this
only once (init) during the lifetime of the control channel.
--***************************************************************************/
NTSTATUS
UlSetUTF8Logging (
IN BOOLEAN UTF8Logging
)
{
PLIST_ENTRY pLink;
PUL_LOG_FILE_ENTRY pEntry;
NTSTATUS Status;
PAGED_CODE();
Status = STATUS_SUCCESS;
//
// Update & Reycle. Need to acquire the logging resource to prevent
// further log hits to be written to file before we finish our
// business. recycle is necessary because files will be renamed to
// have prefix "u_" once we enabled the UTF8.
//
UlTrace(LOGGING,("Ul!UlSetUTF8Logging: UTF8Logging Old %d -> New %d\n",
g_UTF8Logging,UTF8Logging
));
UlAcquireResourceExclusive(&g_pUlNonpagedData->LogListResource, TRUE);
//
// Drop the change if the setting is not changing.
//
if ( g_UTF8Logging == UTF8Logging )
{
goto end;
}
g_UTF8Logging = UTF8Logging;
for (pLink = g_LogListHead.Flink;
pLink != &g_LogListHead;
pLink = pLink->Flink
)
{
pEntry = CONTAINING_RECORD(
pLink,
UL_LOG_FILE_ENTRY,
LogFileListEntry
);
// TODO: Investigate whether we really need to acquire this or not.
UlAcquireResourceExclusive(&pEntry->EntryResource, TRUE);
SET_SEQUNCE_NUMBER_STALE(pEntry);
Status = UlpRecycleLogFile(pEntry);
ASSERT(NT_SUCCESS(Status));
UlReleaseResource(&pEntry->EntryResource);
}
end:
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
return Status;
}
/***************************************************************************++
Routine Description:
UlpGetLogFileLength :
A utility to get the log file length, for a possible size check
Arguments:
hFile - handle to file
Return Value:
ULONG - the length of the file
--***************************************************************************/
ULONG
UlpGetLogFileLength(
IN HANDLE hFile
)
{
NTSTATUS Status;
FILE_STANDARD_INFORMATION StandardInformation;
IO_STATUS_BLOCK IoStatusBlock;
ULONG Length;
PAGED_CODE();
Status = ZwQueryInformationFile(
hFile,
&IoStatusBlock,
&StandardInformation,
sizeof(StandardInformation),
FileStandardInformation
);
ASSERT(NT_SUCCESS(Status));
if (NT_SUCCESS(Status))
{
ASSERT(StandardInformation.EndOfFile.HighPart == 0);
Length = StandardInformation.EndOfFile.LowPart;
}
else
{
Length = 0;
}
return Length;
}
/***************************************************************************++
Routine Description:
Will allocate/fill up a new UNICODE_STRING to hold the directory name info
based on the LocalDrive/UNC.
It's caller's responsibility to cleanup the unicode buffer. If return code
is SUCCESS otherwise no buffer get allocated at all.
Arguments:
PWSTR - the directory name as it's received from the user.
--***************************************************************************/
NTSTATUS
UlpBuildLogDirectory(
IN PUNICODE_STRING pSrcDirName,
IN OUT PUNICODE_STRING pDstDirName
)
{
UNICODE_STRING PathPrefix;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pSrcDirName);
ASSERT(pDstDirName);
UlTrace(LOGGING,(
"Ul!UlpBuildLogDirectory: Directory %S,Length %d,MaxLength %d\n",
pSrcDirName->Buffer,
pSrcDirName->Length,
pSrcDirName->MaximumLength
));
// Allocate a buffer including the terminating NULL and the prefix.
pDstDirName->Length = 0;
pDstDirName->MaximumLength =
pSrcDirName->Length + (UL_MAX_PATH_PREFIX_LENGTH+1) * sizeof(WCHAR);
pDstDirName->Buffer =
(PWSTR) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
pDstDirName->MaximumLength,
UL_CG_LOGDIR_POOL_TAG
);
if (pDstDirName->Buffer == NULL)
{
return STATUS_NO_MEMORY;
}
ASSERT(pSrcDirName->Length > sizeof(WCHAR));
// We store the dir name to cgroup as it is. But when we are constructing
// the filename we skip the second backslash for the UNC shares and for
// local dirs w/o the drive names.
if (pSrcDirName->Buffer[0] == L'\\')
{
if (pSrcDirName->Buffer[1] == L'\\')
{
// UNC share: "\\alitudev\temp"
RtlInitUnicodeString( &PathPrefix, UL_UNC_PATH_PREFIX );
}
else
{
// Local Directory name is missing the device i.e "\temp"
// It should be fully qualified name.
UL_FREE_POOL( pDstDirName->Buffer, UL_CG_LOGDIR_POOL_TAG );
pDstDirName->Buffer = NULL;
return STATUS_NOT_SUPPORTED;
}
RtlCopyUnicodeString( pDstDirName, &PathPrefix );
RtlCopyMemory(
&pDstDirName->Buffer[pDstDirName->Length/sizeof(WCHAR)],
&pSrcDirName->Buffer[1],
pSrcDirName->Length - sizeof(WCHAR)
);
pDstDirName->Length += (pSrcDirName->Length - sizeof(WCHAR));
pDstDirName->Buffer[pDstDirName->Length/sizeof(WCHAR)] = UNICODE_NULL;
}
else
{
RtlInitUnicodeString( &PathPrefix, UL_LOCAL_PATH_PREFIX );
RtlCopyUnicodeString( pDstDirName, &PathPrefix );
RtlAppendUnicodeStringToString( pDstDirName, pSrcDirName );
}
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
A utility to check to see if the directory name is correct or not.
Arguments:
PWSTR - the directory name as it's received from the user.
--***************************************************************************/
NTSTATUS
UlCheckLogDirectory(
IN PUNICODE_STRING pDirName
)
{
NTSTATUS Status;
UNICODE_STRING PathPrefix;
UNICODE_STRING DirectoryName;
PWCHAR pwsz;
//
// Sanity check.
//
PAGED_CODE();
Status = UlpBuildLogDirectory( pDirName, &DirectoryName );
if (!NT_SUCCESS(Status))
{
goto end;
}
//
// Create/Open the director(ies) to see whether it's correct or not.
//
Status = UlpCreateSafeDirectory( &DirectoryName );
end:
UlTrace(LOGGING,("Ul!UlCheckLogDirectory: [%S] -> [%S], Status %08lx\n",
pDirName->Buffer,
DirectoryName.Buffer,
Status
));
if (DirectoryName.Buffer)
{
UL_FREE_POOL( DirectoryName.Buffer, UL_CG_LOGDIR_POOL_TAG );
}
return Status;
}
/***************************************************************************++
Routine Description:
Everytime Aynsc Write Io happens on Log Buffer This APC get called when
completion happens and decrement the global Io Count. If shutting down
we set the event.
This is basically to prevent against shutting down before the Io Complete.
Arguments:
None.
--***************************************************************************/
VOID
UlpBufferFlushAPC(
IN PVOID ApcContext,
IN PIO_STATUS_BLOCK pIoStatusBlock,
IN ULONG Reserved
)
{
PUL_LOG_FILE_BUFFER pLogBuffer;
ULONG IoCount;
KIRQL OldIrql;
UlTrace( LOGGING,("Ul!UlpBufferFlushAPC: entry %p and status %08lx Count %d\n",
ApcContext,
pIoStatusBlock->Status,
(g_BufferIoCount - 1)
));
//
// Free the LogBuffer allocated for this write I/o.
//
pLogBuffer = (PUL_LOG_FILE_BUFFER) ApcContext;
ASSERT(IS_VALID_LOG_FILE_BUFFER(pLogBuffer));
UlPplFreeLogBuffer( pLogBuffer );
//
// Decrement the global outstanding i/o count.
//
IoCount = InterlockedDecrement((PLONG) &g_BufferIoCount);
if ( IoCount == 0 )
{
UlAcquireSpinLock( &g_BufferIoSpinLock, &OldIrql );
//
// Set the event if we hit to zero and waiting for drain.
//
if ( g_BufferWaitingForIoComplete )
{
KeSetEvent( &g_BufferIoCompleteEvent, 0, FALSE );
}
UlReleaseSpinLock( &g_BufferIoSpinLock, OldIrql );
}
}
/***************************************************************************++
Routine Description:
Waits for Io Completions to complete on Log Buffers before shutdown.
Arguments:
None.
--***************************************************************************/
VOID
UlWaitForBufferIoToComplete(
VOID
)
{
KIRQL OldIrql;
BOOLEAN Wait = FALSE;
ASSERT( g_InitLogsCalled );
if ( g_InitLogsCalled )
{
UlAcquireSpinLock( &g_BufferIoSpinLock, &OldIrql );
if ( !g_BufferWaitingForIoComplete )
{
g_BufferWaitingForIoComplete = TRUE;
KeInitializeEvent(
&g_BufferIoCompleteEvent,
NotificationEvent,
FALSE
);
}
//
// If no more i/o operations are happening we are not going to
// wait for them. It is not possible for global i/o counter to
// increment at this time because the log file entry list is empty.
// If there were outstanding i/o then we have to wait them to be
// complete.
//
if ( g_BufferIoCount > 0 )
{
Wait = TRUE;
}
UlReleaseSpinLock( &g_BufferIoSpinLock, OldIrql );
if (Wait)
{
KeWaitForSingleObject(
(PVOID)&g_BufferIoCompleteEvent,
UserRequest,
KernelMode,
FALSE,
NULL
);
}
}
}
/***************************************************************************++
Routine Description:
UlpFlushLogFile :
Assumes you are holding the FastMutex for the buffer & loglist resource
Arguments:
pFile - Handle to a log file entry
--***************************************************************************/
NTSTATUS
UlpFlushLogFile(
IN PUL_LOG_FILE_ENTRY pFile
)
{
NTSTATUS Status;
LARGE_INTEGER EndOfFile;
PUL_LOG_FILE_BUFFER pLogBuffer;
PIO_STATUS_BLOCK pIoStatusBlock;
PAGED_CODE();
ASSERT(pFile!=NULL);
UlTrace( LOGGING, ("Ul!UlpFlushLogFile: pEntry %p\n", pFile ));
pLogBuffer = pFile->LogBuffer;
if ( pLogBuffer==NULL || pLogBuffer->BufferUsed==0 || pFile->hFile==NULL )
{
return STATUS_SUCCESS;
}
pFile->LogBuffer = NULL;
pIoStatusBlock = &pLogBuffer->IoStatusBlock;
EndOfFile.HighPart = -1;
EndOfFile.LowPart = FILE_WRITE_TO_END_OF_FILE;
Status = ZwWriteFile(
pFile->hFile,
NULL,
&UlpBufferFlushAPC,
pLogBuffer,
pIoStatusBlock,
pLogBuffer->Buffer,
pLogBuffer->BufferUsed,
&EndOfFile,
NULL
);
if ( !NT_SUCCESS(Status) )
{
//
// Status maybe STATUS_DISK_FULL,in that case Logging
// will be ceased. Hence log hits stored in this buffer
// are lost.
//
UlTrace( LOGGING,
("Ul!UlpFlushLogFile: ZwWriteFile Failure %08lx \n",
Status
));
UlPplFreeLogBuffer( pLogBuffer );
return Status;
}
//
// Properly keep the number of outstanding Io
//
InterlockedIncrement( (PLONG) &g_BufferIoCount );
return Status;
}
/***************************************************************************++
Routine Description:
UlpWriteToLogFile :
Writes a record to a log file
Arguments:
pFile - Handle to a log file entry
RecordSize - Length of the record to be written.
pRecord - The log record to be written to the log buffer
--***************************************************************************/
NTSTATUS
UlpWriteToLogFile(
IN PUL_LOG_FILE_ENTRY pFile,
IN ULONG RecordSize,
IN PCHAR pRecord,
IN ULONG UsedOffset1,
IN ULONG UsedOffset2
)
{
NTSTATUS Status;
PAGED_CODE();
ASSERT(pRecord!=NULL);
ASSERT(RecordSize!=0);
ASSERT(IS_VALID_LOG_FILE_ENTRY(pFile));
UlTrace(LOGGING, ("Ul!UlpWriteToLogFile: pEntry %p\n", pFile));
if ( pFile==NULL ||
pRecord==NULL ||
RecordSize==0 ||
RecordSize>g_UlLogBufferSize
)
{
return STATUS_INVALID_PARAMETER;
}
//
// We are safe here by dealing only with entry eresource since the
// time based recycling, reconfiguration and periodic buffer flushing
// always acquires the global list eresource exclusively and we are
// already holding it shared. But we should still be carefull about
// file size based recyling and we should only do it while we are
// holding the entries eresource exclusive.I.e. look at the exclusive
// writer down below.
//
if (g_UlDebugLogBufferPeriod)
{
//
// Above global variable is safe to look, it doesn't get changed
// during the life-time of the driver. It's get initialized from
// the registry and disables the log buffering.
//
UlAcquireResourceExclusive(&pFile->EntryResource, TRUE);
Status = UlpWriteToLogFileDebug(
pFile,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
UlReleaseResource(&pFile->EntryResource);
return Status;
}
//
// Try UlpWriteToLogFileShared first which merely moves the
// BufferUsed forward and copy the record to LogBuffer->Buffer.
//
UlAcquireResourceShared(&pFile->EntryResource, TRUE);
Status = UlpWriteToLogFileShared(
pFile,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
UlReleaseResource(&pFile->EntryResource);
if (Status == STATUS_MORE_PROCESSING_REQUIRED)
{
//
// UlpWriteToLogFileShared returns STATUS_MORE_PROCESSING_REQUIRED,
// we need to flush the buffer and try to log again. This time, we
// need to take the entry eresource exclusive.
//
UlAcquireResourceExclusive(&pFile->EntryResource, TRUE);
Status = UlpWriteToLogFileExclusive(
pFile,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
UlReleaseResource(&pFile->EntryResource);
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpAppendToLogBuffer :
Append a record to a log file
REQUIRES you to hold the loglist resource shared and entry mutex
shared or exclusive
Arguments:
pFile - Handle to a log file entry
RecordSize - Length of the record to be written.
pRecord - The log record to be written to the log buffer
--***************************************************************************/
__inline
VOID
FASTCALL
UlpAppendToLogBuffer(
IN PUL_LOG_FILE_ENTRY pFile,
IN ULONG BufferUsed,
IN ULONG RecordSize,
IN PCHAR pRecord,
IN ULONG UsedOffset1,
IN ULONG UsedOffset2
)
{
PUL_LOG_FILE_BUFFER pLogBuffer = pFile->LogBuffer;
UlTrace( LOGGING, ("Ul!UlpAppendToLogBuffer: pEntry %p\n", pFile ));
//
// IIS format log line may be fragmented (identified by looking at the
// UsedOffset2), handle it wisely.
//
if (UsedOffset2)
{
RtlCopyMemory(
pLogBuffer->Buffer + BufferUsed,
&pRecord[0],
UsedOffset1
);
RtlCopyMemory(
pLogBuffer->Buffer + BufferUsed + UsedOffset1,
&pRecord[512],
UsedOffset2
);
RtlCopyMemory(
pLogBuffer->Buffer + BufferUsed + UsedOffset1 + UsedOffset2,
&pRecord[1024],
RecordSize - (UsedOffset1 + UsedOffset2)
);
}
else
{
RtlCopyMemory(
pLogBuffer->Buffer + BufferUsed,
pRecord,
RecordSize
);
}
UlpIncrementBytesWritten(pFile, RecordSize);
}
/***************************************************************************++
Routine Description:
REQUIRES LogListResource Shared & Entry eresource exclusive.
Appends the W3C log file title to the existing buffer.
Arguments:
pFile - Pointer to the logfile entry
pCurrentTimeFields - Current time fields
--***************************************************************************/
NTSTATUS
UlpAppendW3CLogTitle(
IN PUL_LOG_FILE_ENTRY pEntry,
OUT PCHAR pDestBuffer,
IN OUT PULONG pBytesCopied
)
{
NTSTATUS Status = STATUS_SUCCESS;
PCHAR TitleBuffer;
LONG BytesCopied;
ULONG LogExtFileFlags;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
PUL_LOG_FILE_BUFFER pLogBuffer;
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pEntry->Format == HttpLoggingTypeW3C);
pLogBuffer = pEntry->LogBuffer;
LogExtFileFlags = pEntry->LogExtFileFlags;
KeQuerySystemTime(&CurrentTimeStamp);
RtlTimeToTimeFields(&CurrentTimeStamp, &CurrentTimeFields);
if (pDestBuffer)
{
// Append to the provided buffer
ASSERT(pBytesCopied);
ASSERT(*pBytesCopied >= UL_MAX_TITLE_BUFFER_SIZE);
UlTrace(LOGGING,("Ul!UlpAppendW3CLogTitle: Copying to Provided Buffer %p\n",
pDestBuffer));
TitleBuffer = pDestBuffer;
}
else
{
// Append to the entry buffer
ASSERT(pLogBuffer);
ASSERT(pLogBuffer->Buffer);
UlTrace(LOGGING,("Ul!UlpAppendW3CLogTitle: Copying to Entry Buffer %p\n",
pLogBuffer));
TitleBuffer = (PCHAR) pLogBuffer->Buffer + pLogBuffer->BufferUsed;
}
BytesCopied = _snprintf(
TitleBuffer,
UL_MAX_TITLE_BUFFER_SIZE,
// TODO: Make this maintainance friendly
"#Software: Microsoft Internet Information Services 6.0\r\n"
"#Version: 1.0\r\n"
"#Date: %4d-%02d-%02d %02d:%02d:%02d\r\n"
"#Fields:%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls%ls \r\n",
CurrentTimeFields.Year,
CurrentTimeFields.Month,
CurrentTimeFields.Day,
CurrentTimeFields.Hour,
CurrentTimeFields.Minute,
CurrentTimeFields.Second,
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldDate,LogExtFileFlags,MD_EXTLOG_DATE),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldTime,LogExtFileFlags,MD_EXTLOG_TIME),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldSiteName,LogExtFileFlags,MD_EXTLOG_SITE_NAME),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldServerName,LogExtFileFlags,MD_EXTLOG_COMPUTER_NAME),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldServerIp,LogExtFileFlags,MD_EXTLOG_SERVER_IP),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldMethod,LogExtFileFlags,MD_EXTLOG_METHOD),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldUriStem,LogExtFileFlags,MD_EXTLOG_URI_STEM),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldUriQuery,LogExtFileFlags,MD_EXTLOG_URI_QUERY),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldProtocolStatus,LogExtFileFlags,MD_EXTLOG_HTTP_STATUS),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldWin32Status,LogExtFileFlags,MD_EXTLOG_WIN32_STATUS),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldServerPort,LogExtFileFlags,MD_EXTLOG_SERVER_PORT),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldUserName,LogExtFileFlags,MD_EXTLOG_USERNAME),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldClientIp,LogExtFileFlags,MD_EXTLOG_CLIENT_IP),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldProtocolVersion,LogExtFileFlags,MD_EXTLOG_PROTOCOL_VERSION),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldUserAgent,LogExtFileFlags,MD_EXTLOG_USER_AGENT),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldCookie,LogExtFileFlags,MD_EXTLOG_COOKIE),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldReferrer,LogExtFileFlags,MD_EXTLOG_REFERER),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldHost,LogExtFileFlags,MD_EXTLOG_HOST),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldBytesSent,LogExtFileFlags,MD_EXTLOG_BYTES_SENT),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldBytesReceived,LogExtFileFlags,MD_EXTLOG_BYTES_RECV),
UL_GET_LOG_TITLE_IF_PICKED(UlLogFieldTimeTaken,LogExtFileFlags,MD_EXTLOG_TIME_TAKEN)
);
if (BytesCopied < 0)
{
ASSERT(!"Default title buffer size is too small !");
BytesCopied = UL_MAX_TITLE_BUFFER_SIZE;
}
if (pDestBuffer)
{
*pBytesCopied = BytesCopied;
}
else
{
pLogBuffer->BufferUsed += BytesCopied;
UlpIncrementBytesWritten(pEntry, BytesCopied);
}
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
Writes a record to the log buffer and flushes.
This func only get called when debug parameter
g_UlDebugLogBufferPeriod is set.
REQUIRES you to hold the entry eresource EXCLUSIVE.
Arguments:
pFile - Handle to a log file entry
RecordSize - Length of the record to be written.
--***************************************************************************/
NTSTATUS
UlpWriteToLogFileDebug(
IN PUL_LOG_FILE_ENTRY pFile,
IN ULONG RecordSize,
IN PCHAR pRecord,
IN ULONG UsedOffset1,
IN ULONG UsedOffset2
)
{
NTSTATUS Status = STATUS_SUCCESS;
PUL_LOG_FILE_BUFFER pLogBuffer;
ULONG RecordSizePlusTitle = RecordSize;
CHAR TitleBuffer[UL_MAX_TITLE_BUFFER_SIZE];
ULONG TitleBufferSize = UL_MAX_TITLE_BUFFER_SIZE;
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pFile));
ASSERT(UlDbgResourceOwnedExclusive(&pFile->EntryResource));
ASSERT(g_UlDebugLogBufferPeriod!=0);
UlTrace(LOGGING,("Ul!UlpWriteToLogFileDebug: pEntry %p\n", pFile ));
if (!pFile->Flags.LogTitleWritten)
{
// First append to the temp buffer to calculate the size
UlpAppendW3CLogTitle(pFile, TitleBuffer, &TitleBufferSize);
RecordSizePlusTitle += TitleBufferSize;
}
if (UlpIsLogFileOverFlow(pFile,RecordSizePlusTitle))
{
Status = UlpRecycleLogFile(pFile);
}
if (pFile->hFile==NULL || !NT_SUCCESS(Status))
{
//
// If we were unable to acquire a new file handle that means logging
// is temporarly ceased because of either STATUS_DISK_FULL or the
// drive went down for some reason. We just bail out.
//
return Status;
}
if (!pFile->LogBuffer)
{
//
// The buffer will be null for each log hit when log buffering
// is disabled.
//
pFile->LogBuffer = UlPplAllocateLogBuffer();
if (!pFile->LogBuffer)
{
return STATUS_NO_MEMORY;
}
}
pLogBuffer = pFile->LogBuffer;
ASSERT(pLogBuffer->BufferUsed == 0);
if (!pFile->Flags.LogTitleWritten)
{
UlpAppendW3CLogTitle(pFile, NULL, NULL);
}
ASSERT(RecordSize + pLogBuffer->BufferUsed <= g_UlLogBufferSize);
UlpAppendToLogBuffer(
pFile,
pLogBuffer->BufferUsed,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
pLogBuffer->BufferUsed += RecordSize;
Status = UlpFlushLogFile(pFile);
if (!NT_SUCCESS(Status))
{
return Status;
}
pFile->Flags.LogTitleWritten = 1;
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
UlpWriteToLogFileShared :
Writes a record to a log file
REQUIRES you to hold the loglist resource shared
Arguments:
pFile - Handle to a log file entry
RecordSize - Length of the record to be written.
pRecord - The log record to be written to the log buffer
--***************************************************************************/
NTSTATUS
UlpWriteToLogFileShared(
IN PUL_LOG_FILE_ENTRY pFile,
IN ULONG RecordSize,
IN PCHAR pRecord,
IN ULONG UsedOffset1,
IN ULONG UsedOffset2
)
{
PUL_LOG_FILE_BUFFER pLogBuffer;
LONG BufferUsed;
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pFile));
ASSERT(g_UlDebugLogBufferPeriod== 0);
pLogBuffer = pFile->LogBuffer;
UlTrace(LOGGING,("Ul!UlpWriteToLogFileShared: pEntry %p\n", pFile));
//
// Bail out and try the exclusive writer for cases;
//
// 1. No log buffer available.
// 2. Logging ceased. (NULL handle)
// 3. Title needs to be written.
// 4. The actual log file itself has to be recycled.
//
// Otherwise proceed with appending to the current buffer
// if there is enough space avialable for us. If not;
//
// 5. Bail out to get a new buffer
//
if ( pLogBuffer==NULL ||
pFile->hFile==NULL ||
!pFile->Flags.LogTitleWritten ||
UlpIsLogFileOverFlow(pFile,RecordSize)
)
{
return STATUS_MORE_PROCESSING_REQUIRED;
}
//
// Reserve space in pLogBuffer by InterlockedCompareExchange add
// RecordSize. If we exceed the limit, bail out and take the
// exclusive lock to flush the buffer.
//
do
{
BufferUsed = *((volatile LONG *) &pLogBuffer->BufferUsed);
if ( RecordSize + BufferUsed > g_UlLogBufferSize )
{
return STATUS_MORE_PROCESSING_REQUIRED;
}
} while (BufferUsed != InterlockedCompareExchange(
&pLogBuffer->BufferUsed,
RecordSize + BufferUsed,
BufferUsed
));
//
// Keep buffering untill our buffer is full.
//
UlpAppendToLogBuffer(
pFile,
BufferUsed,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
By assuming that it's holding the entrie's eresource exclusively
this function does various functions;
- It Writes a record to a log file
REQUIRES you to hold the loglist resource shared
Arguments:
pFile - Handle to a log file entry
RecordSize - Length of the record to be written.
--***************************************************************************/
NTSTATUS
UlpWriteToLogFileExclusive(
IN PUL_LOG_FILE_ENTRY pFile,
IN ULONG RecordSize,
IN PCHAR pRecord,
IN ULONG UsedOffset1,
IN ULONG UsedOffset2
)
{
PUL_LOG_FILE_BUFFER pLogBuffer;
NTSTATUS Status = STATUS_SUCCESS;
ULONG RecordSizePlusTitle = RecordSize;
CHAR TitleBuffer[UL_MAX_TITLE_BUFFER_SIZE];
ULONG TitleBufferSize = UL_MAX_TITLE_BUFFER_SIZE;
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pFile));
ASSERT(g_UlDebugLogBufferPeriod== 0);
ASSERT(UlDbgResourceOwnedExclusive(&pFile->EntryResource));
UlTrace(LOGGING,("Ul!UlpWriteToLogFileExclusive: pEntry %p\n", pFile));
//
// First append title to the temp buffer to calculate the size of
// the title if we need to write the title as well.
//
if (!pFile->Flags.LogTitleWritten)
{
UlpAppendW3CLogTitle(pFile, TitleBuffer, &TitleBufferSize);
RecordSizePlusTitle += TitleBufferSize;
}
//
// Now check log file overflow.
//
if (UlpIsLogFileOverFlow(pFile,RecordSizePlusTitle))
{
//
// We already acquired the LogListResource Shared and the
// entry eresource exclusive. Therefore ReCycle is fine. Look
// at the comment in UlpWriteToLogFile.
//
Status = UlpRecycleLogFile(pFile);
}
if (pFile->hFile==NULL || !NT_SUCCESS(Status))
{
//
// If somehow the logging ceased and handle released,it happens
// when recycle isn't able to write to the log drive.
//
return Status;
}
pLogBuffer = pFile->LogBuffer;
if (pLogBuffer)
{
//
// There are two conditions we execute the following if block
// 1. We were blocked on eresource exclusive and before us some
// other thread already take care of the buffer flush or recycling.
// 2. Reconfiguration happened and log attempt needs to write the
// title again.
//
if (RecordSizePlusTitle + pLogBuffer->BufferUsed <= g_UlLogBufferSize)
{
//
// If this is the first log attempt after a reconfig, then we have
// to write the title here. Reconfig doesn't immediately write the
// title but rather depend on us by setting the LogTitleWritten flag
// to false.
//
if (!pFile->Flags.LogTitleWritten)
{
ASSERT(RecordSizePlusTitle > RecordSize);
UlpAppendW3CLogTitle(pFile, NULL, NULL);
pFile->Flags.LogTitleWritten = 1;
}
UlpAppendToLogBuffer(
pFile,
pLogBuffer->BufferUsed,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
pLogBuffer->BufferUsed += RecordSize;
return STATUS_SUCCESS;
}
//
// Flush out the buffer first then proceed with allocating a new one.
//
Status = UlpFlushLogFile(pFile);
if (!NT_SUCCESS(Status))
{
return Status;
}
}
ASSERT(pFile->LogBuffer == NULL);
//
// This can be the very first log attempt or the previous allocation
// of LogBuffer failed, or the previous hit flushed and deallocated
// the old buffer. In either case, we allocate a new one,append the
// (title plus) new record and return for more/shared processing.
//
pLogBuffer = pFile->LogBuffer = UlPplAllocateLogBuffer();
if (pLogBuffer == NULL)
{
return STATUS_NO_MEMORY;
}
//
// Very first attempt needs to write the title, as well as the attempt
// which causes the log file recycling. Both cases comes down here
//
if (!pFile->Flags.LogTitleWritten)
{
UlpAppendW3CLogTitle(pFile, NULL, NULL);
pFile->Flags.LogTitleWritten = 1;
}
UlpAppendToLogBuffer(
pFile,
pLogBuffer->BufferUsed,
RecordSize,
pRecord,
UsedOffset1,
UsedOffset2
);
pLogBuffer->BufferUsed += RecordSize;
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
UlpQueryDirectory:
* What file should IIS write to when logging type is daily/weekly/monthly/
hourly if there is already a log file there for that day?
IIS should write to the current day/week/month/hour's log file. For
example, let's say there's an extended log file in my log directory
called ex000727.log. IIS should append new log entries to this log,
as it is for today.
* What file should IIS write to when logging type is MAXSIZE when there are
already log files there for maxsize (like extend0.log, extend1.log, etc.)?
IIS should write to the max extend#.log file, where max(extend#.log)
is has the largest # in the #field for extend#.log. This is provided,
of course, that the MAXSIZE in that file hasn't been exceeded.
* This function quite similar to the implementation of the FindFirstFile
Win32 API. Except that it has been shaped to our purposes.
Arguments:
pEntry - The log file entry which freshly created.
--***************************************************************************/
NTSTATUS
UlpQueryDirectory(
IN OUT PUL_LOG_FILE_ENTRY pEntry
)
{
#define GET_NEXT_FILE(pv, cb) \
(FILE_DIRECTORY_INFORMATION *) ((VOID *) (((UCHAR *) (pv)) + (cb)))
NTSTATUS Status = STATUS_SUCCESS;
OBJECT_ATTRIBUTES ObjectAttributes;
IO_STATUS_BLOCK IoStatusBlock;
LONG WcharsCopied;
HANDLE hDirectory;
ULONG Sequence;
ULONG LastSequence;
UNICODE_STRING Temp;
PWCHAR pTemp;
UNICODE_STRING FileName;
WCHAR _FileName[UL_MAX_FILE_NAME_SUFFIX_LENGTH];
FILE_DIRECTORY_INFORMATION *pFdi;
PUCHAR FileInfoBuffer;
ULARGE_INTEGER FileSize;
WCHAR OriginalWChar;
PAGED_CODE();
Status = STATUS_SUCCESS;
hDirectory = NULL;
FileInfoBuffer = NULL;
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
UlTrace(LOGGING,("Ul!UlpQueryDirectory: %S\n",pEntry->FileName.Buffer));
ASSERT(pEntry->Period == HttpLoggingPeriodMaxSize);
ASSERT(UL_DIRECTORY_SEARCH_BUFFER_SIZE >=
(sizeof(FILE_DIRECTORY_INFORMATION)+UL_MAX_FILE_NAME_SUFFIX_LENGTH));
//
// Open the directory for the list access again. Use the filename in
// pEntry. Where pShortName points to the "\inetsv1.log" portion of
// the whole "\??\c:\whistler\system32\logfiles\w3svc1\inetsv1.log"
// Overwrite the pShortName to get the directory name. Once we are
// done with finding the last sequence we will restore it back later
// on.
//
OriginalWChar = *((PWCHAR)pEntry->pShortName);
*((PWCHAR)pEntry->pShortName) = UNICODE_NULL;
pEntry->FileName.Length =
wcslen(pEntry->FileName.Buffer) * sizeof(WCHAR);
InitializeObjectAttributes(
&ObjectAttributes,
&pEntry->FileName,
OBJ_CASE_INSENSITIVE|UL_KERNEL_HANDLE,
NULL,
NULL
);
Status = ZwCreateFile(
&hDirectory,
SYNCHRONIZE|FILE_LIST_DIRECTORY,
&ObjectAttributes,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
FILE_OPEN,
FILE_SYNCHRONOUS_IO_NONALERT|FILE_DIRECTORY_FILE,
NULL,
0
);
if (!NT_SUCCESS(Status))
{
//
// This call should never fail since CreateLog already created
// the directory for us.
//
ASSERT(!"Directory Invalid!\n");
goto end;
}
//
// Before querrying we need to provide additional DOS-like wildcard
// matching semantics. In our case, only * to DOS_STAR conversion is
// enough though. The following is the pattern we will use for query
// Skipping the first slash character.
//
FileName.Buffer = &_FileName[1];
WcharsCopied = _snwprintf( _FileName,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
L"%s%c.%s",
UL_GET_LOG_FILE_NAME_PREFIX(pEntry->Format),
DOS_STAR,
DEFAULT_LOG_FILE_EXTENSION
);
ASSERT(WcharsCopied > 0);
FileName.Length = wcslen(FileName.Buffer) * sizeof(WCHAR);
FileName.MaximumLength = FileName.Length;
//
// This non-paged buffer should be allocated to be used for storing
// query results.
//
FileInfoBuffer =
UL_ALLOCATE_ARRAY(
NonPagedPool,
UCHAR,
UL_DIRECTORY_SEARCH_BUFFER_SIZE + sizeof(WCHAR),
UL_LOG_GENERIC_POOL_TAG
);
if (FileInfoBuffer == NULL)
{
Status = STATUS_NO_MEMORY;
goto end;
}
//
// The very first call may also fail if there is no log file in the
// current directory.
//
Status = ZwQueryDirectoryFile (
hDirectory,
NULL,
NULL,
NULL,
&IoStatusBlock,
FileInfoBuffer,
UL_DIRECTORY_SEARCH_BUFFER_SIZE,
FileDirectoryInformation,
FALSE,
&FileName,
TRUE
);
if(!NT_SUCCESS(Status))
{
//
// This should never fail with STATUS_BUFFER_OVERFLOW unless the
// buffer size is ridiculously small i.e. 50 bytes or something
//
UlTrace( LOGGING,
("Ul!UlpQueryDirectory: Status %08lx for %S & %S\n",
Status,
pEntry->FileName.Buffer,
FileName.Buffer
));
ASSERT(Status == STATUS_NO_SUCH_FILE);
Status = STATUS_SUCCESS;
goto end;
}
//
// Look into the buffer and get the sequence number from filename.
//
pFdi = (FILE_DIRECTORY_INFORMATION *) FileInfoBuffer;
Sequence = LastSequence = 1;
FileSize.QuadPart = 0;
while (TRUE)
{
//
// Get the latest Sequence Number from the filename
//
if (pTemp = wcsstr(pFdi->FileName,DEFAULT_LOG_FILE_EXTENSION_PLUS_DOT))
{
*pTemp = UNICODE_NULL;
pTemp = pFdi->FileName;
while ( *pTemp != UNICODE_NULL )
{
if ( isdigit((CHAR) (*pTemp)) )
{
Temp.Length = wcslen(pTemp) * sizeof(WCHAR);
Temp.MaximumLength = Temp.Length;
Temp.Buffer = pTemp;
RtlUnicodeStringToInteger( &Temp, 10, &LastSequence );
break;
}
pTemp++;
}
}
else
{
ASSERT(FALSE);
}
if (LastSequence >= Sequence)
{
//
// Bingo ! We have two things to remember though; the file size
// and the sequence number. Cryptic it's that we are getting the
// file size from EOF. Its greater than or equal because we want
// to initialize the FileSize properly even if there's only one
// match.
//
Sequence = LastSequence;
//
// BUGBUG: The HighPart is zero unless the file size is greater
// than the MAXDWORD. If so the file size calculated as in the
// formula; ((HighPart * (MAXDWORD+1)) + LowPart)
//
FileSize.LowPart = pFdi->EndOfFile.LowPart;
}
//
// Keep going until we see no more files
//
if (pFdi->NextEntryOffset != 0)
{
//
// Search through the buffer as long as there is still something
// in there.
//
pFdi = GET_NEXT_FILE(pFdi, pFdi->NextEntryOffset);
}
else
{
//
// Otherwise query again for any other possible log file(s)
//
Status = ZwQueryDirectoryFile (
hDirectory,
NULL,
NULL,
NULL,
&IoStatusBlock,
FileInfoBuffer,
UL_DIRECTORY_SEARCH_BUFFER_SIZE,
FileDirectoryInformation,
FALSE,
NULL,
FALSE
);
if (Status == STATUS_NO_MORE_FILES)
{
Status = STATUS_SUCCESS;
break;
}
if (!NT_SUCCESS(Status))
{
goto end;
}
pFdi = (FILE_DIRECTORY_INFORMATION *) FileInfoBuffer;
}
}
//
// Construct the log file name properly from the sequence number so that
// our caller can create the log file later on.
//
WcharsCopied = _snwprintf( pEntry->pShortName,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
L"%s%d.%s",
UL_GET_LOG_FILE_NAME_PREFIX(pEntry->Format),
Sequence,
DEFAULT_LOG_FILE_EXTENSION
);
ASSERT(WcharsCopied > 0);
pEntry->FileName.Length =
wcslen(pEntry->FileName.Buffer) * sizeof(WCHAR);
//
// Set the next sequence number according to last log file
//
pEntry->SequenceNumber = Sequence + 1;
//
// Update the log file size accordingly in the entry.Otherwise truncation
// will not work properly.
//
pEntry->TotalWritten.QuadPart = FileSize.QuadPart;
UlTrace( LOGGING,
("Ul!UlpQueryDirectory: %S has been found with size %d.\n",
pEntry->FileName.Buffer,
pEntry->TotalWritten.QuadPart
));
end:
if (*((PWCHAR)pEntry->pShortName) == UNICODE_NULL )
{
//
// We have failed for some reason before reconstructing the filename
// Perhaps because the directory was empty. Do not forget to restore
// the pShortName in the pEntry then.
//
*((PWCHAR)pEntry->pShortName) = OriginalWChar;
pEntry->FileName.Length =
wcslen(pEntry->FileName.Buffer) * sizeof(WCHAR);
}
if (FileInfoBuffer)
{
UL_FREE_POOL( FileInfoBuffer, UL_LOG_GENERIC_POOL_TAG );
}
if (!NT_SUCCESS(Status))
{
UlTrace( LOGGING,
("Ul!UlpQueryDirectory: failure %08lx for %S\n",
Status,
pEntry->FileName.Buffer
));
}
if (hDirectory != NULL)
{
ZwClose(hDirectory);
}
return Status;
}
/***************************************************************************++
Routine Description:
UlCreateLog:
Creates a new Logging file and insert a corresponding entry
to the global LoggingList.
Each log file belongs to a single ConfigGroup and can be
created by one. Although we keep a list of already created log files
for convenience, each config group has a pointer to its log file.
If this function fails for any reason, file entry pointer of the
config group will set to NULL.
Arguments:
pConfigGroup - Supplies the necessary information for opening the
log file and gets the result entry pointer.
LogListResource also implicitly protects this pointer
since only possible places which are going to update
it use the resource exclusively.
--***************************************************************************/
NTSTATUS
UlCreateLog(
IN OUT PUL_CONFIG_GROUP_OBJECT pConfigGroup
)
{
NTSTATUS Status;
OBJECT_ATTRIBUTES ObjectAttributes;
IO_STATUS_BLOCK IoStatusBlock;
PUL_LOG_FILE_ENTRY pNewEntry;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
HANDLE hDirectory;
UNICODE_STRING DirectoryName;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pConfigGroup != NULL);
UlTrace(LOGGING, ("Ul!UlCreateLog: pConfigGroup %p Truncate %d\n",
pConfigGroup, pConfigGroup->LoggingConfig.LogFileTruncateSize
));
//
// An important check to ensure that no infinite loop occurs because of
// ridiculusly small truncatesizes. Means smaller than maximum
// allowed log record line (10*1024)
//
if ( pConfigGroup->LoggingConfig.LogPeriod == HttpLoggingPeriodMaxSize &&
pConfigGroup->LoggingConfig.LogFileTruncateSize != HTTP_LIMIT_INFINITE &&
pConfigGroup->LoggingConfig.LogFileTruncateSize < UL_MIN_ALLOWED_TRUNCATESIZE
)
{
UlTrace(LOGGING,
("Ul!UlCreateLog: Truncate size too small pConfigGroup %p : %d\n",
pConfigGroup,
pConfigGroup->LoggingConfig.LogFileTruncateSize
));
return STATUS_INVALID_PARAMETER;
}
//
// We have two criteria for the log file name: its LogFormat and its LogPeriod
//
if ( pConfigGroup->LoggingConfig.LogFormat >= HttpLoggingTypeMaximum ||
pConfigGroup->LoggingConfig.LogPeriod >= HttpLoggingPeriodMaximum )
{
return STATUS_INVALID_PARAMETER;
}
//
// Setup locals so we know how to cleanup on exit.
//
Status = STATUS_SUCCESS;
pNewEntry = NULL;
//
// This value is computed for the GMT time zone.
//
KeQuerySystemTime( &CurrentTimeStamp );
RtlTimeToTimeFields( &CurrentTimeStamp, &CurrentTimeFields );
//
// Allocate a temp buffer to hold the full path name including the
// device prefix and the filename at the end.
//
Status = UlpBuildLogDirectory(
&pConfigGroup->LoggingConfig.LogFileDir,
&DirectoryName
);
if (!NT_SUCCESS(Status))
{
return Status;
}
//
// We have to acquire the LogListresource exclusively, prior to
// the operations Create/Remove/ReConfig. Whenever we acquire the
// LogListResource exclusively we don't need to have the entry mutex
// acquired.
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->LogListResource, TRUE);
pConfigGroup->pLogFileEntry = NULL;
Status = UlpConstructLogFileEntry (
&pConfigGroup->LoggingConfig,
&pNewEntry,
&DirectoryName,
&CurrentTimeFields
);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Create/Open the directory(ies) first.
//
Status = UlpCreateSafeDirectory( &DirectoryName );
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// If the logformat is max_size with truncation we need to scan the
// directory and find the correct last log file to append. Otherwise
// just picking the FILE_OPEN_IF when opening the log file will ensure
// the functionality.
//
if (pNewEntry->Period == HttpLoggingPeriodMaxSize)
{
//
// This call will update the filename and the file size
//
Status = UlpQueryDirectory(pNewEntry);
if (!NT_SUCCESS(Status))
goto end;
}
//
// Create/Open the file.
//
InitializeObjectAttributes(
&ObjectAttributes,
&pNewEntry->FileName, // Full path name
OBJ_CASE_INSENSITIVE | // Attributes
UL_KERNEL_HANDLE,
NULL, // RootDirectory
NULL // SecurityDescriptor
);
//
// Make the created file Aysnc by not picking the sync flag.
//
Status = ZwCreateFile(
&pNewEntry->hFile,
FILE_GENERIC_WRITE,
&ObjectAttributes,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ,
FILE_OPEN_IF,
FILE_NON_DIRECTORY_FILE, // |FILE_SYNCHRONOUS_IO_NONALERT,
NULL,
0);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Get the file size, etc from the file.
//
Status = ZwQueryInformationFile(
pNewEntry->hFile,
&IoStatusBlock,
&pNewEntry->FileInfo,
sizeof(pNewEntry->FileInfo),
FileStandardInformation
);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Add it to our global log list.
//
UlpInsertLogFileEntry( pNewEntry, &CurrentTimeFields );
//
// Success!
//
pConfigGroup->pLogFileEntry = pNewEntry;
UlTrace( LOGGING,
( "Ul!UlCreateLog: entry %p, file %S, handle %lx created\n",
pNewEntry,
pNewEntry->FileName.Buffer,
pNewEntry->hFile )
);
end:
if ( !NT_SUCCESS(Status) )
{
//
// If we made it to this point, then the create/open has failed.
//
UlTrace( LOGGING,
("Ul!UlCreateLog: dirname %S, file %S failure %08lx\n",
pConfigGroup->LoggingConfig.LogFileDir.Buffer,
pNewEntry->FileName.Buffer,
Status
));
if (pNewEntry)
{
NTSTATUS TempStatus;
//
// Now release the entry's resources.
//
if (pNewEntry->hFile != NULL)
{
ZwClose( pNewEntry->hFile );
pNewEntry->hFile = NULL;
}
if ( pNewEntry->LogBuffer )
{
UlPplFreeLogBuffer( pNewEntry->LogBuffer );
}
// Delete the entry eresource
TempStatus = UlDeleteResource( &pNewEntry->EntryResource );
ASSERT(NT_SUCCESS(TempStatus));
UL_FREE_POOL_WITH_SIG(pNewEntry,UL_LOG_FILE_ENTRY_POOL_TAG);
}
pConfigGroup->pLogFileEntry = NULL;
}
else
{
ASSERT(pConfigGroup->pLogFileEntry != NULL);
}
// Cleanup temp dir name buffer
if (DirectoryName.Buffer)
{
UL_FREE_POOL(DirectoryName.Buffer, UL_CG_LOGDIR_POOL_TAG);
}
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
return Status;
}
/***************************************************************************++
Routine Description:
UlpInsertLogFileEntry :
Inserts a log file entry to our global log entry list.
REQUIRES caller to have LogListresource EXCLUSIVELY.
Arguments:
pEntry - The log file entry to be added to the global list
pTimeFields - The current time fields.
--***************************************************************************/
VOID
UlpInsertLogFileEntry(
PUL_LOG_FILE_ENTRY pEntry,
PTIME_FIELDS pTimeFields
)
{
LONG listSize;
HTTP_LOGGING_PERIOD period;
KIRQL oldIrql;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pTimeFields);
//
// add to the list
//
InsertHeadList(&g_LogListHead, &pEntry->LogFileListEntry);
period = pEntry->Period;
listSize = InterlockedIncrement( &g_LogListEntryCount );
ASSERT(listSize >= 1);
//
// Time to start the Log Timer if we haven't done it yet.
// Once we start this timer it keeps working until
// terminaton of the driver. CODEWORK we may start and stop it
// more intelligently, i.e. if no log requires it stop it
//
UlAcquireSpinLock( &g_LogTimerSpinLock, &oldIrql );
if ( g_LogTimerStarted == FALSE )
{
// Only if we are running on time dependent
// format
if ( period != HttpLoggingPeriodMaxSize )
{
UlpSetLogTimer( pTimeFields );
g_LogTimerStarted = TRUE;
}
}
// Go ahead start the buffer timer
// as soon as we have a log file
if ( g_BufferTimerStarted == FALSE )
{
UlpSetBufferTimer();
g_BufferTimerStarted = TRUE;
}
UlReleaseSpinLock( &g_LogTimerSpinLock, oldIrql );
}
/***************************************************************************++
Routine Description:
Simple utility to close the log file handle on a system thread and set the
event to notify the caller that it's done.
Arguments:
pEntry - Acquired from passed-in pWorkItem
--***************************************************************************/
VOID
UlpLogCloseHandleWorker(
IN PUL_WORK_ITEM pWorkItem
)
{
PUL_LOG_FILE_ENTRY pEntry;
PAGED_CODE();
// For this function to not to cause any threats to the safety of the log
// entry. The entry should already been acquired exclusively by our caller.
ASSERT(!UlDbgResourceUnownedExclusive(
&g_pUlNonpagedData->LogListResource));
// Get the log entry
pEntry = CONTAINING_RECORD(
pWorkItem,
UL_LOG_FILE_ENTRY,
WorkItem
);
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pEntry->hFile);
UlTrace(LOGGING,("Ul!UlpLogCloseHandleWorker: pEntry %p hFile %p\n",
pEntry, pEntry->hFile ));
// Close the handle and set the event for the original caller
ZwClose(pEntry->hFile);
pEntry->hFile = NULL;
KeSetEvent(&pEntry->CloseEvent, 0, FALSE);
}
/***************************************************************************++
Routine Description:
Simple utility to close the log file handle on a system thread and set the
event to notify the caller that it's done.
Arguments:
pEntry - Acquired from passed-in pWorkItem
--***************************************************************************/
VOID
UlpLogCloseHandle(
IN PUL_LOG_FILE_ENTRY pEntry
)
{
// Sanity check
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
//ASSERT(UlDbgResourceOwnedExclusive(&pEntry->EntryResource));
ASSERT(g_pUlSystemProcess);
// Close the handle on the system thread and wait until it's done
if (g_pUlSystemProcess == (PKPROCESS)PsGetCurrentProcess())
{
ZwClose(pEntry->hFile);
pEntry->hFile = NULL;
}
else
{
KeAttachProcess(g_pUlSystemProcess);
ZwClose(pEntry->hFile);
pEntry->hFile = NULL;
// Following will bugcheck in the checked kernel if there's
// any user or kernel APCs attached to the thread.
KeDetachProcess();
// TODO: Find a better solution
/*
UL_QUEUE_WORK_ITEM(
&pEntry->WorkItem,
&UlpLogCloseHandleWorker
);
KeWaitForSingleObject(
(PVOID)&pEntry->CloseEvent,
UserRequest,
KernelMode,
FALSE,
NULL
);
KeClearEvent(&pEntry->CloseEvent);
*/
}
}
/***************************************************************************++
Routine Description:
UlRemoveLogFileEntry :
Removes a log file entry from our global log entry list.
Arguments:
pEntry - The log file entry to be removed from the global list
--***************************************************************************/
VOID
UlRemoveLogFileEntry(
PUL_LOG_FILE_ENTRY pEntry
)
{
NTSTATUS Status;
LONG listSize;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
UlAcquireResourceExclusive(&g_pUlNonpagedData->LogListResource, TRUE);
RemoveEntryList(&pEntry->LogFileListEntry);
pEntry->LogFileListEntry.Flink =
pEntry->LogFileListEntry.Blink = NULL;
if (pEntry->hFile != NULL)
{
//
// Make sure that buffer get flushed out before closing the file
// handle. But flush will cause an APC to be queued to the user
// thread, therefore we have to close the handle on one of our
// system threads to avoid the possible bugcheck
// INVALID_PROCESS_DETACH_ATTEMPT condition.
//
UlpFlushLogFile(pEntry);
UlpLogCloseHandle(pEntry);
}
//
// Delete the entry eresource
//
Status = UlDeleteResource( &pEntry->EntryResource );
ASSERT(NT_SUCCESS(Status));
listSize = InterlockedDecrement( &g_LogListEntryCount );
ASSERT(listSize >= 0);
UlTrace( LOGGING,
("Ul!UlRemoveLogFileEntry: entry %p has been removed\n",
pEntry
));
if ( pEntry->LogBuffer )
{
UlPplFreeLogBuffer( pEntry->LogBuffer );
}
UL_FREE_POOL_WITH_SIG(pEntry,UL_LOG_FILE_ENTRY_POOL_TAG);
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
}
/***************************************************************************++
Routine Description:
UlpCreateSafeDirectory :
Creates all of the necessary directories in a given UNICODE directory
pathname.
E.g. For given \??\C:\temp\w3svc1
-> Directories "C:\temp" & "C:\temp\w3svc1" will be created.
This function assumes that directory string starts with "\\??\\"
Arguments:
pDirectoryName - directroy path name string, WARNING this function makes
some inplace modification to the passed directory string
but it restores the original before returning.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpCreateSafeDirectory(
IN PUNICODE_STRING pDirectoryName
)
{
OBJECT_ATTRIBUTES ObjectAttributes;
IO_STATUS_BLOCK IoStatusBlock;
NTSTATUS Status;
HANDLE hDirectory;
PWCHAR pw;
USHORT i;
//
// Sanity check
//
PAGED_CODE();
Status = STATUS_SUCCESS;
ASSERT( pDirectoryName );
ASSERT( pDirectoryName->Buffer );
ASSERT( pDirectoryName->Length );
ASSERT( pDirectoryName->MaximumLength > pDirectoryName->Length );
pw = pDirectoryName->Buffer;
pw[pDirectoryName->Length/sizeof(WCHAR)]=UNICODE_NULL;
// TODO: Handle network mapped drives. Redirector.
if (0 == wcsncmp(pw, UL_UNC_PATH_PREFIX, UL_UNC_PATH_PREFIX_LENGTH))
{
// UNC share
pw += UL_UNC_PATH_PREFIX_LENGTH;
// Bypass "\\machine\share"
i = 0; // Skip two backslashes before reaching to share name
while( *pw != UNICODE_NULL )
{
if ( *pw == L'\\' ) i++;
if ( i == 2 ) break;
pw++;
}
}
else if (0 == wcsncmp(pw, UL_LOCAL_PATH_PREFIX, UL_LOCAL_PATH_PREFIX_LENGTH))
{
// Local Drive
pw += UL_LOCAL_PATH_PREFIX_LENGTH;
// Bypass "C:"
while( *pw != L'\\' && *pw != UNICODE_NULL )
{
pw++;
}
}
else
{
ASSERT(!"Incorrect logging directory name or type !");
return STATUS_INVALID_PARAMETER;
}
if ( *pw == UNICODE_NULL )
{
// Dir. Name cannot be only "\??\C:" or "\dosdevices\UNC\machine
// It should at least be pointing to the root directory.
ASSERT(!"Incomplete logging directory name !");
return STATUS_INVALID_PARAMETER;
}
//
// \??\C:\temp\w3svc1 OR \\dosdevices\UNC\machine\share\w3svc1
// ^ ^
// pw now points to | OR |
//
//
ASSERT( *pw == L'\\' );
do
{
pw++;
if ( *pw == L'\\' || *pw == UNICODE_NULL )
{
//
// Remember the original character
//
WCHAR wcOriginal = *pw;
UNICODE_STRING DirectoryName;
//
// Time to create the directory so far
//
*pw = UNICODE_NULL;
RtlInitUnicodeString( &DirectoryName, pDirectoryName->Buffer );
InitializeObjectAttributes(
&ObjectAttributes,
&DirectoryName,
OBJ_CASE_INSENSITIVE |
UL_KERNEL_HANDLE,
NULL,
NULL
);
Status = ZwCreateFile(
&hDirectory,
FILE_LIST_DIRECTORY|FILE_TRAVERSE,
&ObjectAttributes,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ,
FILE_OPEN_IF,
FILE_DIRECTORY_FILE,
NULL,
0
);
//
// Restore the original character
//
*pw = wcOriginal;
if (NT_SUCCESS(Status) == FALSE)
{
goto end;
}
Status = ZwClose(hDirectory);
if (NT_SUCCESS(Status) == FALSE)
{
goto end;
}
}
}
while( *pw != UNICODE_NULL );
end:
if (!NT_SUCCESS(Status))
{
UlTrace(LOGGING,
("Ul!UlpCreateSafeDirectory: directory %S, failure %08lx\n",
pDirectoryName->Buffer,
Status
));
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpCalculateTimeToExpire :
Shamelessly stolen from IIS 5.1 Logging code and adapted here.
This routine returns the time-to-expire in hours. 1 means the log
will expire in the next timer-fire and so ...
Arguments:
PTIME_FIELDS - Current Time Fields
HTTP_LOGGING_PERIOD - Logging Period
PULONG - Pointer to a buffer to receive result
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpCalculateTimeToExpire(
PTIME_FIELDS pDueTime,
HTTP_LOGGING_PERIOD LogPeriod,
PULONG pTimeRemaining
)
{
NTSTATUS Status;
ULONG NumDays;
PAGED_CODE();
ASSERT(pDueTime!=NULL);
ASSERT(pTimeRemaining!=NULL);
Status = STATUS_SUCCESS;
switch (LogPeriod)
{
case HttpLoggingPeriodMaxSize:
return Status;
case HttpLoggingPeriodHourly:
*pTimeRemaining = 1;
break;
case HttpLoggingPeriodDaily:
*pTimeRemaining = 24 - pDueTime->Hour;
break;
case HttpLoggingPeriodWeekly:
{
ULONG TimeRemainingInTheMonth;
NumDays = UlpGetMonthDays(pDueTime);
TimeRemainingInTheMonth =
NumDays*24 - ((pDueTime->Day-1)*24 + pDueTime->Hour);
// Time Remaining in the week
// Sunday = 0, Monday = 1 ... Saturday = 6
*pTimeRemaining =
7*24 - (pDueTime->Weekday*24 + pDueTime->Hour);
//
// If the time remaining in the month less than time remaining in
// the week then we have to recycle at the end of the month.
// Otherwise we have to recycle at the end of the week. (next sunday)
//
if (TimeRemainingInTheMonth < *pTimeRemaining)
{
*pTimeRemaining = TimeRemainingInTheMonth;
}
}
break;
case HttpLoggingPeriodMonthly:
{
NumDays = UlpGetMonthDays(pDueTime);
//
// Lets not forget that the day starts from 1 .. 31
// Therefore we have to subtract one from the day value.
//
*pTimeRemaining =
NumDays*24 - ((pDueTime->Day-1)*24 + pDueTime->Hour);
}
break;
default:
ASSERT(FALSE);
return STATUS_INVALID_PARAMETER;
}
return Status;
}
/***************************************************************************++
Routine Description:
Initializes the Log & buffering Timers
--***************************************************************************/
VOID
UlpInitializeTimers(
VOID
)
{
/* Log timer */
g_LogTimerInitialized = TRUE;
UlInitializeSpinLock( &g_LogTimerSpinLock, "g_LogTimerSpinLock" );
KeInitializeDpc(
&g_LogTimerDpcObject, // DPC object
&UlLogTimerDpcRoutine, // DPC routine
NULL // context
);
KeInitializeTimer( &g_LogTimer );
/* Buffer timer */
g_BufferTimerInitialized = TRUE;
UlInitializeSpinLock( &g_BufferTimerSpinLock, "g_BufferTimerSpinLock" );
KeInitializeDpc(
&g_BufferTimerDpcObject, // DPC object
&UlBufferTimerDpcRoutine, // DPC routine
NULL // context
);
KeInitializeTimer( &g_BufferTimer );
}
/***************************************************************************++
Routine Description:
Terminates the Log & buffering Timers
--***************************************************************************/
VOID
UlpTerminateTimers(
VOID
)
{
KIRQL oldIrql;
/* Log timer */
if ( g_LogTimerInitialized )
{
UlAcquireSpinLock( &g_LogTimerSpinLock, &oldIrql );
g_LogTimerInitialized = FALSE;
UlReleaseSpinLock( &g_LogTimerSpinLock, oldIrql );
KeCancelTimer( &g_LogTimer );
}
/* Buffer timer */
if ( g_BufferTimerInitialized )
{
UlAcquireSpinLock( &g_BufferTimerSpinLock, &oldIrql );
g_BufferTimerInitialized = FALSE;
UlReleaseSpinLock( &g_BufferTimerSpinLock, oldIrql );
KeCancelTimer( &g_BufferTimer );
}
}
/***************************************************************************++
Routine Description:
UlpSetTimer :
This routine provides the initial due time for the upcoming
periodic hourly timer. We have to align the timer so that it
get signaled at the beginning of each hour. Then it goes with
an hour period until stops.
We keep ONLY one timer for all log periods. A DPC routine will
get called every hour, and it will traverse the log list and
do the cycling properly.
Arguments:
PTIME_FIELDS - Current Time
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
VOID
UlpSetLogTimer(
PTIME_FIELDS pFields
)
{
LONGLONG InitialDueTime100Ns;
LARGE_INTEGER InitialDueTime;
ASSERT(pFields!=NULL);
//
// Remaining time to next hour tick. In seconds
//
InitialDueTime100Ns =
1*60*60 - ( pFields->Minute*60 + pFields->Second );
//
// Also convert it to 100-ns
//
InitialDueTime100Ns =
(InitialDueTime100Ns*1000 - pFields->Milliseconds ) * 1000 * 10;
//
// Negative time for relative value.
//
InitialDueTime.QuadPart = -InitialDueTime100Ns;
KeSetTimerEx(
&g_LogTimer,
InitialDueTime,
1*60*60*1000, // Period of 1 Hours in millisec
&g_LogTimerDpcObject
);
}
/***************************************************************************++
Routine Description:
UlpSetBufferTimer :
We have to introduce a new timer for the log buffering mechanism.
Each log file keeps a system default (64K) buffer do not flush this
out unless it's full or this timer get fired every MINUTE.
The hourly timer get aligned for the beginning of each hour. Therefore
using that existing timer would introduce alot of complexity.
Arguments:
PTIME_FIELDS - Current Time
--***************************************************************************/
VOID
UlpSetBufferTimer(
VOID
)
{
LONGLONG BufferPeriodTime100Ns;
LONG BufferPeriodTimeMs;
LARGE_INTEGER BufferPeriodTime;
//
// Remaining time to next tick. DEFAULT_BUFFER_TIMER_PERIOD is in minutes
//
BufferPeriodTimeMs = DEFAULT_BUFFER_TIMER_PERIOD * 60 * 1000;
BufferPeriodTime100Ns = (LONGLONG) BufferPeriodTimeMs * 10 * 1000;
UlTrace(LOGGING,
("Ul!UlpSetBufferTimer: %d seconds\n",
BufferPeriodTimeMs / 1000
));
//
// Negative time for relative value.
//
BufferPeriodTime.QuadPart = -BufferPeriodTime100Ns;
KeSetTimerEx(
&g_BufferTimer,
BufferPeriodTime, // Must be in nanosec
BufferPeriodTimeMs, // Must be in millisec
&g_BufferTimerDpcObject
);
}
/***************************************************************************++
Routine Description:
UlLogTimerHandler :
Work item for the threadpool that goes thru the log list and
cycle the necessary logs.
Arguments:
PUL_WORK_ITEM - Ignored
Return Value:
None
--***************************************************************************/
VOID
UlLogTimerHandler(
IN PUL_WORK_ITEM pWorkItem
)
{
NTSTATUS Status;
PLIST_ENTRY pLink;
PUL_LOG_FILE_ENTRY pEntry;
PAGED_CODE();
UlTrace(LOGGING,("Ul!UlLogTimerHandler: Scanning the log entries\n"));
UlAcquireResourceExclusive(&g_pUlNonpagedData->LogListResource, TRUE);
// Attempt to reinit the GMT offset every hour, to pickup the changes
// because of the day light changes. Synced by the logging eresource.
UlpGetGMTOffset();
for (pLink = g_LogListHead.Flink;
pLink != &g_LogListHead;
pLink = pLink->Flink
)
{
pEntry = CONTAINING_RECORD(
pLink,
UL_LOG_FILE_ENTRY,
LogFileListEntry
);
//
// We should not recycle this entry if it's period
// is not time based but size based.
//
UlAcquireResourceExclusive(&pEntry->EntryResource, TRUE);
if (pEntry->Period != HttpLoggingPeriodMaxSize)
{
if (pEntry->TimeToExpire==1)
{
// TODO: Don't recycle if the entries logging is disabled.
SET_TIME_TO_EXPIRE_STALE(pEntry);
Status = UlpRecycleLogFile(pEntry);
}
else
{
//
// Just decrement the hourly counter
// for this time.
//
pEntry->TimeToExpire -= 1;
}
}
UlReleaseResource(&pEntry->EntryResource);
}
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
//
// Free the memory allocated (ByDpcRoutine below) to
// this work item.
//
UL_FREE_POOL( pWorkItem, UL_WORK_ITEM_POOL_TAG );
}
/***************************************************************************++
Routine Description:
UlLogTimerDpcRoutine :
Arguments:
Ignored
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
VOID
UlLogTimerDpcRoutine(
PKDPC Dpc,
PVOID DeferredContext,
PVOID SystemArgument1,
PVOID SystemArgument2
)
{
PUL_WORK_ITEM pWorkItem;
UlAcquireSpinLockAtDpcLevel(
&g_LogTimerSpinLock
);
if( g_LogTimerInitialized )
{
//
// It's not possible to acquire the resource which protects
// the log list at DISPATCH_LEVEL therefore we will queue a
// work item for this.
//
pWorkItem = (PUL_WORK_ITEM) UL_ALLOCATE_POOL(
NonPagedPool,
sizeof(*pWorkItem),
UL_WORK_ITEM_POOL_TAG
);
if ( pWorkItem )
{
UL_QUEUE_WORK_ITEM(
pWorkItem,
&UlLogTimerHandler
);
}
else
{
UlTrace(LOGGING,("Ul!UlLogTimerDpcRoutine: Not enough memory.\n"));
}
}
UlReleaseSpinLockFromDpcLevel(
&g_LogTimerSpinLock
);
}
/***************************************************************************++
Routine Description:
UlBufferTimerDpcRoutine :
Arguments:
Ignored
--***************************************************************************/
VOID
UlBufferTimerDpcRoutine(
PKDPC Dpc,
PVOID DeferredContext,
PVOID SystemArgument1,
PVOID SystemArgument2
)
{
PUL_WORK_ITEM pWorkItem;
UlAcquireSpinLockAtDpcLevel( &g_BufferTimerSpinLock );
if( g_BufferTimerInitialized )
{
//
// It's not possible to acquire the resource which protects
// the log list at DISPATCH_LEVEL therefore we will queue a
// work item for this.
//
pWorkItem = (PUL_WORK_ITEM) UL_ALLOCATE_POOL(
NonPagedPool,
sizeof(*pWorkItem),
UL_WORK_ITEM_POOL_TAG
);
if ( pWorkItem )
{
UL_QUEUE_WORK_ITEM(
pWorkItem,
&UlBufferTimerHandler
);
}
else
{
UlTrace(LOGGING,("Ul!UlBufferTimerDpcRoutine: Not enough memory.\n"));
}
}
UlReleaseSpinLockFromDpcLevel( &g_BufferTimerSpinLock );
}
/***************************************************************************++
Routine Description:
UlLogBufferTimerHandler :
Work item for the threadpool that goes thru the log list and
flush the log's file buffers.
Arguments:
PUL_WORK_ITEM - Ignored but cleaned up at the end
--***************************************************************************/
VOID
UlBufferTimerHandler(
IN PUL_WORK_ITEM pWorkItem
)
{
NTSTATUS Status;
PLIST_ENTRY pLink;
PUL_LOG_FILE_ENTRY pEntry;
PAGED_CODE();
UlTrace(LOGGING,("Ul!UlBufferTimerHandler: scanning the log entries ...\n"));
UlAcquireResourceShared(&g_pUlNonpagedData->LogListResource, TRUE);
for (pLink = g_LogListHead.Flink;
pLink != &g_LogListHead;
pLink = pLink->Flink
)
{
pEntry = CONTAINING_RECORD(
pLink,
UL_LOG_FILE_ENTRY,
LogFileListEntry
);
UlAcquireResourceExclusive(&pEntry->EntryResource, TRUE);
Status = UlpFlushLogFile( pEntry );
// TODO: Handle STATUS_DISK_FULL
UlReleaseResource(&pEntry->EntryResource);
}
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
//
// Free the memory allocated (ByDpcRoutine below) to
// this work item.
//
UL_FREE_POOL( pWorkItem, UL_WORK_ITEM_POOL_TAG );
}
/***************************************************************************++
Routine Description:
UlpGrowLogEntry:
All it does is to grow the log file entry in the log list. So that
recycle function later on can successfully updates the new log file
name. This function get called only when Dir Name is changed. Because
only that case causes us to enlarge the already allocated name buffer.
Name changes originated from timer based or size based recycling
does not require us to enlarge as we initially allocates more than enough
buffer for those changes.
You should HAVE the log list resource exclusively before calling
this function.
Arguments:
too many
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpGrowLogEntry(
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup,
IN PUL_LOG_FILE_ENTRY pOldEntry
)
{
NTSTATUS Status;
USHORT FullPathFileNameLength;
UNICODE_STRING JunkFileName;
PUL_LOG_FILE_ENTRY pEntry;
UNICODE_STRING DosDevice;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pConfigGroup != NULL);
ASSERT(pOldEntry != NULL);
Status = STATUS_SUCCESS;
pEntry = NULL;
// TODO: Get rid of this function by making sure that log entry never grows.
// TODO: It means Allocating the Log Dir Name separately not inline.
UlTrace( LOGGING, ("Ul!UlpGrowLogEntry: old_entry %p\n", pOldEntry ));
RtlInitUnicodeString( &JunkFileName, L"\\none.log" );
FullPathFileNameLength= UL_MAX_PATH_PREFIX_LENGTH +
pConfigGroup->LoggingConfig.LogFileDir.Length +
(UL_MAX_FILE_NAME_SUFFIX_LENGTH+1) * sizeof(WCHAR);
pEntry = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_LOG_FILE_ENTRY,
FullPathFileNameLength,
UL_LOG_FILE_ENTRY_POOL_TAG
);
if ( pEntry == NULL )
{
Status = STATUS_INSUFFICIENT_RESOURCES;
goto end;
}
RtlZeroMemory( pEntry, sizeof(*pEntry) );
pEntry->Signature = UL_LOG_FILE_ENTRY_POOL_TAG;
*pEntry = *pOldEntry;
//
// Construct a default dir_name. Don't worry about the
// L"none.log", it will be overwritten by the recycler
// later on, as long as there's MAX_LOG_FILE_NAME_SIZE
// space for the time/type dependent part of the file
// name ( aka short file name ), it's all right.
//
pEntry->FileName.Length = 0;
pEntry->FileName.MaximumLength = FullPathFileNameLength;
pEntry->FileName.Buffer = (PWSTR) ( pEntry + 1 );
if (pConfigGroup->LoggingConfig.LogFileDir.Buffer[0] == L'\\')
{
if (pConfigGroup->LoggingConfig.LogFileDir.Buffer[1] == L'\\')
{
// UNC share: "\\alitudev\temp"
RtlInitUnicodeString( &DosDevice, UL_UNC_PATH_PREFIX );
}
else
{
// Local Directory name is missing the device i.e "\temp"
// It should be fully qualified name.
Status = STATUS_NOT_SUPPORTED;
goto end;
}
// Skip second backslash for the UNC path
RtlCopyUnicodeString( &pEntry->FileName, &DosDevice );
RtlCopyMemory(
&(pEntry->FileName.Buffer[pEntry->FileName.Length/sizeof(WCHAR)]),
&(pConfigGroup->LoggingConfig.LogFileDir.Buffer[1]),
pConfigGroup->LoggingConfig.LogFileDir.Length - sizeof(WCHAR)
);
pEntry->FileName.Length +=
(pConfigGroup->LoggingConfig.LogFileDir.Length - sizeof(WCHAR));
}
else
{
RtlInitUnicodeString( &DosDevice, UL_LOCAL_PATH_PREFIX );
RtlCopyUnicodeString( &pEntry->FileName, &DosDevice );
RtlAppendUnicodeStringToString(
&pEntry->FileName,
&(pConfigGroup->LoggingConfig.LogFileDir)
);
}
pEntry->pShortName =
(PWSTR) &pEntry->FileName.Buffer[pEntry->FileName.Length/sizeof(WCHAR)];
Status = RtlAppendUnicodeStringToString(
&pEntry->FileName,
&JunkFileName
);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Do the replacement here
//
RemoveEntryList(&pOldEntry->LogFileListEntry);
pOldEntry->LogFileListEntry.Flink =
pOldEntry->LogFileListEntry.Blink = NULL;
//
// Restore/Carry the buffer from the old entry
//
pEntry->Flags.LogTitleWritten = pOldEntry->Flags.LogTitleWritten;
pEntry->LogBuffer = pOldEntry->LogBuffer;
KeInitializeEvent(&pEntry->CloseEvent, NotificationEvent, FALSE);
// Create the new entries eresource
Status = UlInitializeResource(&pEntry->EntryResource,"EntryResource",0,
UL_LOG_FILE_ENTRY_POOL_TAG);
ASSERT(NT_SUCCESS(Status));
// Delete old entry eresource
Status = UlDeleteResource( &pOldEntry->EntryResource );
ASSERT(NT_SUCCESS(Status));
UL_FREE_POOL_WITH_SIG(pOldEntry,UL_LOG_FILE_ENTRY_POOL_TAG);
InsertHeadList(&g_LogListHead, &pEntry->LogFileListEntry);
pConfigGroup->pLogFileEntry = pEntry;
//
// Lets return happily
//
end:
if (!NT_SUCCESS(Status))
{
UlTrace(LOGGING,
("Ul!UlpGrowLogEntry: old_entry %p, file %S, failure %08lx\n",
pOldEntry,
pOldEntry->FileName.Buffer,
Status
));
if (pEntry)
{
NTSTATUS TempStatus;
// Delete the entry eresource
TempStatus = UlDeleteResource( &pEntry->EntryResource );
ASSERT(NT_SUCCESS(TempStatus));
UL_FREE_POOL_WITH_SIG(pEntry,UL_LOG_FILE_ENTRY_POOL_TAG);
}
}
return Status;
}
/***************************************************************************++
Routine Description:
UlReconfigureLogEntry :
This function implements the logging reconfiguration per attribute.
Everytime config changes happens we try to update the existing logging
parameters here.
Arguments:
pConfig - corresponding cgroup object
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlReConfigureLogEntry(
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgOld,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgNew
)
{
NTSTATUS Status ;
BOOLEAN HaveToReCycle;
//
// Sanity check first
//
PAGED_CODE();
Status = STATUS_SUCCESS;
HaveToReCycle = FALSE;
UlTrace( LOGGING, ("Ul!UlReConfigureLogEntry: entry %p\n",
pConfigGroup->pLogFileEntry ));
if ( pCfgOld->LoggingEnabled==FALSE && pCfgNew->LoggingEnabled==FALSE )
{
//
// Do Nothing. Not Even update the fields
// as soon as we get enable request,
// field update will take place anyway.
//
return Status;
}
if ( pCfgOld->LoggingEnabled==TRUE && pCfgNew->LoggingEnabled==FALSE )
{
//
// Stop logging but keep the entry in the
// list. CODEWORK do we have to keep this info
// in the entry itself and when we recycle the
// logs based on timer, we do not recycle the
// ones with logging disabled ??
//
pCfgOld->LoggingEnabled = FALSE;
return Status;
}
else
{
pCfgOld->LoggingEnabled = TRUE;
}
if ( pCfgNew->LogFormat >= HttpLoggingTypeMaximum ||
pCfgNew->LogPeriod >= HttpLoggingPeriodMaximum )
{
return STATUS_INVALID_PARAMETER;
}
//
// An important check to ensure that no infinite loop occurs because of
// ridiculusly small truncatesizes. Means smaller than maximum
// allowed log record line (10*1024)
//
if ( pCfgNew->LogPeriod == HttpLoggingPeriodMaxSize &&
pCfgNew->LogFileTruncateSize != HTTP_LIMIT_INFINITE &&
pCfgNew->LogFileTruncateSize < UL_MIN_ALLOWED_TRUNCATESIZE
)
{
UlTrace( LOGGING,
("Ul!UlReConfigureLogEntry: Truncate size too small %d !\n",
pCfgNew->LogFileTruncateSize
));
return STATUS_INVALID_PARAMETER;
}
//
// No matter what ReConfiguration should acquire the LogListResource
// exclusively.
//
UlAcquireResourceExclusive(&g_pUlNonpagedData->LogListResource, TRUE);
//
// Proceed down to see if anything else changed
// as well ...
//
if ( RtlCompareUnicodeString(
&pCfgNew->LogFileDir, &pCfgOld->LogFileDir, TRUE ) != 0
)
{
//
// Always force ReGrow otherwise dir name won't be updated
// Update the cgroup to hold the new log_dir name
//
if (pCfgOld->LogFileDir.Buffer != NULL)
{
UL_FREE_POOL(
pCfgOld->LogFileDir.Buffer,
UL_CG_LOGDIR_POOL_TAG
);
pCfgOld->LogFileDir.Buffer = NULL;
}
pCfgOld->LogFileDir.Buffer =
(PWSTR) UL_ALLOCATE_ARRAY(
PagedPool,
UCHAR,
pCfgNew->LogFileDir.MaximumLength,
UL_CG_LOGDIR_POOL_TAG
);
if (pCfgOld->LogFileDir.Buffer == NULL)
{
pCfgOld->LogFileDir.Length = 0;
pCfgOld->LogFileDir.MaximumLength = 0;
Status = STATUS_NO_MEMORY;
goto end;
}
RtlCopyMemory(
pCfgOld->LogFileDir.Buffer,
pCfgNew->LogFileDir.Buffer,
pCfgNew->LogFileDir.MaximumLength
);
pCfgOld->LogFileDir.Length = pCfgNew->LogFileDir.Length;
pCfgOld->LogFileDir.MaximumLength = pCfgNew->LogFileDir.MaximumLength;
UlpGrowLogEntry( pConfigGroup, pConfigGroup->pLogFileEntry );
// Need to find the proper sequence number after scanning the new
// log directory.
SET_SEQUNCE_NUMBER_STALE(pConfigGroup->pLogFileEntry);
HaveToReCycle = TRUE;
}
if ( pCfgNew->LogFormat != pCfgOld->LogFormat )
{
Status = UlpUpdateFormat(
pConfigGroup->pLogFileEntry,
pCfgOld,
pCfgNew
);
goto end;
}
if ( pCfgNew->LogPeriod != pCfgOld->LogPeriod )
{
Status = UlpUpdatePeriod(
pConfigGroup->pLogFileEntry,
pCfgOld,
pCfgNew
);
goto end;
}
//
// Both LogFormat and LogPeriod always trigger the recycle and they also
// handle changes to the LogFileTruncateSize and LogExtFileFlags before
// triggering the recycle. Therefore we can bail out safely.
//
if ( pCfgNew->LogFileTruncateSize != pCfgOld->LogFileTruncateSize )
{
Status = UlpUpdateLogTruncateSize(
pConfigGroup->pLogFileEntry,
pCfgOld,
pCfgNew,
&HaveToReCycle
);
}
if ( pCfgNew->LogExtFileFlags != pCfgOld->LogExtFileFlags )
{
//
// Just a change in the flags should not cause us to recyle.
// Unless something else is also changed. If that's the case
// then it's either handled above us, or will be handled down
//
Status = UlpUpdateLogFlags(
pConfigGroup->pLogFileEntry,
pCfgOld,
pCfgNew
);
}
if ( HaveToReCycle )
{
//
// If we are here that means directory name has been changed
// and nobody has recycled the log file yet. So time to go
//
Status = UlpRecycleLogFile(pConfigGroup->pLogFileEntry);
}
end:
if (NT_SUCCESS(Status) == FALSE)
{
UlTrace( LOGGING,
("Ul!UlReConfigureLogEntry: entry %p, failure %08lx\n",
pConfigGroup->pLogFileEntry,
Status
));
}
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
return Status;
} // UlReConfigureLogEntry
/***************************************************************************++
Routine Description:
UlpUpdateFormat :
Arguments:
pConfig - corresponding cgroup object
--***************************************************************************/
NTSTATUS
UlpUpdateFormat(
OUT PUL_LOG_FILE_ENTRY pEntry,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgOld,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgNew
)
{
NTSTATUS Status ;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
PAGED_CODE();
Status = STATUS_SUCCESS;
if ( pEntry == NULL )
{
return STATUS_INVALID_PARAMETER;
}
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pCfgOld->LogFormat==pEntry->Format);
pCfgOld->LogFormat = pCfgNew->LogFormat;
pEntry->Format = pCfgNew->LogFormat;
pCfgOld->LogPeriod = pCfgNew->LogPeriod;
pEntry->Period = (HTTP_LOGGING_PERIOD) pCfgNew->LogPeriod;
pEntry->TruncateSize = pCfgNew->LogFileTruncateSize;
pCfgOld->LogFileTruncateSize= pCfgNew->LogFileTruncateSize;
pEntry->LogExtFileFlags = pCfgNew->LogExtFileFlags;
pCfgOld->LogExtFileFlags = pCfgNew->LogExtFileFlags;
KeQuerySystemTime( &CurrentTimeStamp );
RtlTimeToTimeFields( &CurrentTimeStamp, &CurrentTimeFields );
if ( pEntry->Period != HttpLoggingPeriodMaxSize )
{
Status = UlpCalculateTimeToExpire(
&CurrentTimeFields,
pEntry->Period,
&pEntry->TimeToExpire
);
ASSERT(NT_SUCCESS(Status)==TRUE);
}
ASSERT(NT_SUCCESS(Status)==TRUE);
SET_SEQUNCE_NUMBER_STALE(pEntry);
Status = UlpRecycleLogFile( pEntry );
return Status;
}
/***************************************************************************++
Routine Description:
UlpUpdatePeriod :
Arguments:
pConfig - corresponding cgroup object
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpUpdatePeriod(
OUT PUL_LOG_FILE_ENTRY pEntry,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgOld,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgNew
)
{
NTSTATUS Status ;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
PAGED_CODE();
Status = STATUS_SUCCESS;
if ( pEntry == NULL )
{
return STATUS_INVALID_PARAMETER;
}
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pCfgOld->LogPeriod==(ULONG)pEntry->Period);
pCfgOld->LogPeriod = pCfgNew->LogPeriod;
pEntry->Period = (HTTP_LOGGING_PERIOD) pCfgNew->LogPeriod;
pEntry->TruncateSize = pCfgNew->LogFileTruncateSize;
pCfgOld->LogFileTruncateSize= pCfgNew->LogFileTruncateSize;
pEntry->LogExtFileFlags = pCfgNew->LogExtFileFlags;
pCfgOld->LogExtFileFlags = pCfgNew->LogExtFileFlags;
KeQuerySystemTime( &CurrentTimeStamp );
RtlTimeToTimeFields( &CurrentTimeStamp, &CurrentTimeFields );
if ( pEntry->Period != HttpLoggingPeriodMaxSize )
{
Status = UlpCalculateTimeToExpire(
&CurrentTimeFields,
pEntry->Period,
&pEntry->TimeToExpire
);
ASSERT(NT_SUCCESS(Status)==TRUE);
}
SET_SEQUNCE_NUMBER_STALE(pEntry);
Status = UlpRecycleLogFile( pEntry );
return Status;
}
/***************************************************************************++
Routine Description:
UlpUpdateLogTruncateSize :
Arguments:
pConfig - corresponding cgroup object
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpUpdateLogTruncateSize(
OUT PUL_LOG_FILE_ENTRY pEntry,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgOld,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgNew,
OUT BOOLEAN * pHaveToReCycle
)
{
NTSTATUS Status ;
//
// Sanity check first
//
PAGED_CODE();
Status = STATUS_SUCCESS;
//
// For MAX_SIZE period type we should check if
// limited => unlimited:
// we can still use the last log file
// unlimited => limited:
// we should open a new one if old size is larger than
// the new limitation
//
if ( pEntry == NULL )
{
return STATUS_INVALID_PARAMETER;
}
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pCfgOld->LogFileTruncateSize==pEntry->TruncateSize);
if ( pCfgOld->LogPeriod == HttpLoggingPeriodMaxSize )
{
if ( pCfgOld->LogFileTruncateSize == HTTP_LIMIT_INFINITE )
{
//
// Unlimited to Limited
//
if ( pEntry->TotalWritten.QuadPart >
(ULONGLONG)pCfgNew->LogFileTruncateSize )
{
// In case flags get changed lets take it too
pEntry->LogExtFileFlags = pCfgNew->LogExtFileFlags;
pCfgOld->LogExtFileFlags = pCfgNew->LogExtFileFlags;
UlpRecycleLogFile( pEntry );
*pHaveToReCycle = FALSE;
}
}
else
{
//
// Limited to Unlimited
// Nothing special to do
//
}
}
pEntry->TruncateSize = pCfgNew->LogFileTruncateSize;
pCfgOld->LogFileTruncateSize = pCfgNew->LogFileTruncateSize;
return Status;
}
/***************************************************************************++
Routine Description:
UlpUpdateLogFlags :
REQUIRES caller to have loglist resource exclusively.
Arguments:
pEntry - corresponding logfile entry
old & new configuration
--***************************************************************************/
NTSTATUS
UlpUpdateLogFlags(
OUT PUL_LOG_FILE_ENTRY pEntry,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgOld,
IN PHTTP_CONFIG_GROUP_LOGGING pCfgNew
)
{
NTSTATUS Status ;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
//
// No need to do anything else, we will
// just display a new title with the
// new fields.
//
PAGED_CODE();
Status = STATUS_SUCCESS;
if (pEntry == NULL)
{
return STATUS_INVALID_PARAMETER;
}
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
ASSERT(pCfgOld->LogExtFileFlags==pEntry->LogExtFileFlags);
pCfgOld->LogExtFileFlags = pCfgNew->LogExtFileFlags;
pEntry->LogExtFileFlags = pCfgNew->LogExtFileFlags;
if (pEntry->Format == HttpLoggingTypeW3C)
{
pEntry->Flags.LogTitleWritten = 0;
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpConstructLogFileEntry :
Finds out the correct file name for the newly created log file
from our - current time dependent- time-to-name converters. Also
allocates the necessary file entry from paged pool. This entry get
removed from the list when the corresponding config group object
has been destroyed. At that time RemoveLogFile entry called and
it frees this memory.
Arguments:
pConfig - corresponding cgroup object
ppEntry - will point to newly created entry
pDirectoryName - the directory name to store newcoming log file
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpConstructLogFileEntry(
IN PHTTP_CONFIG_GROUP_LOGGING pConfig,
OUT PUL_LOG_FILE_ENTRY * ppEntry,
OUT PUNICODE_STRING pDirectoryName,
IN PTIME_FIELDS pCurrentTimeFields
)
{
NTSTATUS Status,TmpStatus;
USHORT FullPathFileNameLength;
ULONG SequenceNumber;
PUL_LOG_FILE_ENTRY pEntry;
WCHAR _FileName[UL_MAX_FILE_NAME_SUFFIX_LENGTH+1];
UNICODE_STRING FileName =
{ 0, (UL_MAX_FILE_NAME_SUFFIX_LENGTH+1)*sizeof(WCHAR), _FileName };
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pConfig != NULL);
ASSERT(ppEntry != NULL);
Status = STATUS_SUCCESS;
SequenceNumber = 1;
pEntry = NULL;
FullPathFileNameLength = pDirectoryName->Length;
UlpConstructFileName(
(HTTP_LOGGING_PERIOD) pConfig->LogPeriod,
UL_GET_LOG_FILE_NAME_PREFIX(pConfig->LogFormat),
&FileName,
pCurrentTimeFields,
&SequenceNumber
);
FullPathFileNameLength += FileName.Length;
//
// Allocate a memory for our new logfile entry in the list.
// To avoid the frequent reallocs for the log entry - E.g.
// we receive a timer update and filename changes according to
// new time - , We will try to allocate a fixed amount here
// for all the possible file_names ( this doesn't include
// the log_dir changes may happen from WAS through cgroup
// in that case we will realloc a new entry ). It has to
// be nonpaged because it holds a eresource.
//
pEntry = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_LOG_FILE_ENTRY,
(UL_MAX_FILE_NAME_SUFFIX_LENGTH + 1) * sizeof(WCHAR) +
pDirectoryName->Length,
UL_LOG_FILE_ENTRY_POOL_TAG
);
if ( pEntry == NULL )
{
Status = STATUS_NO_MEMORY;
goto end;
}
RtlZeroMemory( pEntry, sizeof(*pEntry) );
pEntry->Signature = UL_LOG_FILE_ENTRY_POOL_TAG;
//
// Concat the directory & file name properly.
//
pEntry->FileName.Length = FullPathFileNameLength;
pEntry->FileName.MaximumLength= (UL_MAX_FILE_NAME_SUFFIX_LENGTH+1) * sizeof(WCHAR) +
pDirectoryName->Length;
pEntry->FileName.Buffer = (PWSTR) ( pEntry + 1 );
RtlCopyUnicodeString( &(pEntry->FileName), pDirectoryName );
Status = RtlAppendUnicodeStringToString( &(pEntry->FileName), &FileName );
ASSERT(NT_SUCCESS(Status));
pEntry->FileName.Buffer[FullPathFileNameLength/sizeof(WCHAR)] = UNICODE_NULL;
pEntry->pShortName = (PWSTR)
&(pEntry->FileName.Buffer[pDirectoryName->Length/sizeof(WCHAR)]);
//
// Create a log entry buffer of system dependent size
// typically 64K
//
Status = UlInitializeResource(&pEntry->EntryResource,"EntryResource",0,
UL_LOG_FILE_ENTRY_POOL_TAG);
ASSERT(NT_SUCCESS(Status));
//
// Initialize the file handle
//
pEntry->hFile = NULL;
KeInitializeEvent(&pEntry->CloseEvent, NotificationEvent, FALSE);
//
// Set the logging information from config group
// easier for other routines to use this values
// w/o reaching to the config-group
//
pEntry->Format = pConfig->LogFormat;
pEntry->Period = (HTTP_LOGGING_PERIOD) pConfig->LogPeriod;
pEntry->TruncateSize = pConfig->LogFileTruncateSize;
pEntry->LogExtFileFlags = pConfig->LogExtFileFlags;
//
// Time to initialize our Log Cycling parameter
//
pEntry->TimeToExpire = 0;
pEntry->SequenceNumber = SequenceNumber;
pEntry->TotalWritten.QuadPart = (ULONGLONG)0;
pEntry->Flags.Value = 0;
if (pEntry->Format == HttpLoggingTypeW3C)
{
pEntry->Flags.LogTitleWritten = 0;
}
else
{
pEntry->Flags.LogTitleWritten = 1;
}
pEntry->LogBuffer = NULL;
Status = UlpCalculateTimeToExpire(
pCurrentTimeFields,
pEntry->Period,
&pEntry->TimeToExpire
);
if ( !NT_SUCCESS(Status) )
{
goto end;
}
//
// Lets happily return our entry
//
*ppEntry = pEntry;
end:
if ( !NT_SUCCESS(Status) )
{
if ( pEntry )
{
NTSTATUS TempStatus;
if ( pEntry->LogBuffer )
{
UlPplFreeLogBuffer( pEntry->LogBuffer );
}
// Delete the entry eresource
TempStatus = UlDeleteResource( &pEntry->EntryResource );
ASSERT(NT_SUCCESS(TempStatus));
UL_FREE_POOL_WITH_SIG( pEntry, UL_LOG_FILE_ENTRY_POOL_TAG );
}
}
return Status;
}
/***************************************************************************++
Routine Description:
UlpConstructFileName:
A bunch of current_time TO file_name conversions comes here ...
Arguments:
period - period type of the log
prefix - any prefix to be added to the file name
filename - result file name
fields - time fields
Return Value:
VOID - No return value.
--***************************************************************************/
VOID
UlpConstructFileName(
IN HTTP_LOGGING_PERIOD period,
IN PCWSTR prefix,
OUT PUNICODE_STRING filename,
IN PTIME_FIELDS fields,
IN OUT PULONG sequenceNu //OPTIONAL
)
{
WCHAR _tmp[UL_MAX_FILE_NAME_SUFFIX_LENGTH+1];
UNICODE_STRING tmp = { 0, 0, _tmp };
CSHORT Year;
LONG WcharsCopied = 0L;
PAGED_CODE();
//
// Retain just last 2 digits of the Year
//
tmp.MaximumLength = (UL_MAX_FILE_NAME_SUFFIX_LENGTH+1) * sizeof(WCHAR);
if (fields)
{
Year = fields->Year % 100;
}
switch ( period )
{
case HttpLoggingPeriodHourly:
{
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
(UTF8_LOGGING_ENABLED() ?
L"%.5s%02.2d%02d%02d%02d.%s" :
L"%.3s%02.2d%02d%02d%02d.%s"),
prefix,
Year,
fields->Month,
fields->Day,
fields->Hour,
DEFAULT_LOG_FILE_EXTENSION
);
}
break;
case HttpLoggingPeriodDaily:
{
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
(UTF8_LOGGING_ENABLED() ?
L"%.5s%02.2d%02d%02d.%s" :
L"%.3s%02.2d%02d%02d.%s"),
prefix,
Year,
fields->Month,
fields->Day,
DEFAULT_LOG_FILE_EXTENSION
);
}
break;
case HttpLoggingPeriodWeekly:
{
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
(UTF8_LOGGING_ENABLED() ?
L"%.5s%02.2d%02d%02d.%s" :
L"%.3s%02.2d%02d%02d.%s"),
prefix,
Year,
fields->Month,
UlpWeekOfMonth(fields),
DEFAULT_LOG_FILE_EXTENSION
);
}
break;
case HttpLoggingPeriodMonthly:
{
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
(UTF8_LOGGING_ENABLED() ?
L"%.5s%02.2d%02d.%s" :
L"%.3s%02.2d%02d.%s"),
prefix,
Year,
fields->Month,
DEFAULT_LOG_FILE_EXTENSION
);
}
break;
case HttpLoggingPeriodMaxSize:
{
if ( sequenceNu != NULL )
{
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
(UTF8_LOGGING_ENABLED() ?
L"%.9s%d.%s" :
L"%.7s%d.%s"),
prefix,
(*sequenceNu),
DEFAULT_LOG_FILE_EXTENSION
);
(*sequenceNu) += 1;
}
else
{
ASSERT(!"Improper sequence number !");
}
}
break;
default:
{
//
// This should never happen ...
//
ASSERT(!"Unknown Log Format !");
WcharsCopied =
_snwprintf( _tmp,
UL_MAX_FILE_NAME_SUFFIX_LENGTH,
L"%.7s?.%s",
prefix,
DEFAULT_LOG_FILE_EXTENSION
);
}
}
//
// As long as we allocate an enough space for a possible
// log filename we should never hit to this assert here.
//
ASSERT(WcharsCopied >0 );
if ( WcharsCopied < 0 )
{
//
// This should never happen but lets cover it
// anyway.
//
WcharsCopied = UL_MAX_FILE_NAME_SUFFIX_LENGTH*sizeof(WCHAR);
tmp.Buffer[UL_MAX_FILE_NAME_SUFFIX_LENGTH] = UNICODE_NULL;
}
tmp.Length = (USHORT) WcharsCopied*sizeof(WCHAR);
tmp.MaximumLength = (UL_MAX_FILE_NAME_SUFFIX_LENGTH+1)*sizeof(WCHAR);
RtlCopyUnicodeString( filename, &tmp );
}
/***************************************************************************++
Routine Description:
UlpRecycleLogFile :
This function requires to have the loglist resource shared,as well as
the logfile entry mutex to be acquired.
We do not want anybody to Create/Remove/ReConfig to the entry while
we are working on it, therefore shared access to the loglist.
We do not want anybody to Hit/Flush to the entry, therefore
entry's mutex should be acquired.
Or otherwise caller might have the loglist resource exclusively and
this will automatically ensure the safety as well. As it is not
possible for anybody else to acquire entry mutex first w/o having
the loglist resource shared at least, according to the current
design.
Sometimes it may be necessary to scan the new directory to figure out
the correct sequence numbe rand the file name. Especially after dir
name reconfig and/or the period becomes MaskPeriod.
Arguments:
pEntry - Points to the existing entry.
NeedToReCalculate - shows if we have to recalculate the time-to-expire.
--***************************************************************************/
NTSTATUS
UlpRecycleLogFile(
IN PUL_LOG_FILE_ENTRY pEntry
)
{
NTSTATUS Status;
OBJECT_ATTRIBUTES ObjectAttributes;
IO_STATUS_BLOCK IoStatusBlock;
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeStamp;
PWCHAR pSrc;
PWCHAR pDst;
USHORT Index;
USHORT OldFileNameLength;
WCHAR _FileName[UL_MAX_FILE_NAME_SUFFIX_LENGTH];
UNICODE_STRING FileName =
{ 0, UL_MAX_FILE_NAME_SUFFIX_LENGTH*sizeof(WCHAR), _FileName };
//
// Sanity check.
//
PAGED_CODE();
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
Status = STATUS_SUCCESS;
//
// We have two criterions for the log file name
// its LogFormat and its LogPeriod
//
ASSERT(pEntry->Format < HttpLoggingTypeMaximum);
ASSERT(pEntry->Period < HttpLoggingPeriodMaximum);
ASSERT(pEntry->FileName.Length!=0);
UlTrace( LOGGING, ("Ul!UlpRecycleLogFile: pEntry %p \n", pEntry ));
//
// This value is computed for the GMT time zone.
//
KeQuerySystemTime( &CurrentTimeStamp );
RtlTimeToTimeFields( &CurrentTimeStamp, &CurrentTimeFields );
// Init total written to zero. It may get updated if we scan
// the directory down below.
pEntry->TotalWritten.QuadPart = (ULONGLONG)0;
// If we need to scan the directory. Sequence number should start
// from 1 again. Set this before constructing the log file name.
if (pEntry->Flags.StaleSequenceNumber &&
pEntry->Period==HttpLoggingPeriodMaxSize)
{
// Init otherwise if QueryDirectory doesn't find any it
// will not update this values
pEntry->SequenceNumber = 1;
}
//
// Now construct the filename using the lookup table
// And the current time
//
UlpConstructFileName(
pEntry->Period,
UL_GET_LOG_FILE_NAME_PREFIX(pEntry->Format),
&FileName,
&CurrentTimeFields,
&pEntry->SequenceNumber
);
if ( pEntry->FileName.MaximumLength <= FileName.Length )
{
ASSERT(FALSE);
Status = STATUS_INVALID_PARAMETER;
goto end;
}
//
// Do the magic and renew the filename. Replace the old file
// name with the new one.
//
ASSERT( pEntry->pShortName != NULL );
if ( pEntry->pShortName != NULL )
{
//
// Get rid of the old filename before flushing the
// directories and reconcataneting the new file name
// to the end again.
//
*((PWCHAR)pEntry->pShortName) = UNICODE_NULL;
pEntry->FileName.Length =
wcslen( pEntry->FileName.Buffer ) * sizeof(WCHAR);
//
// Create/Open the director(ies) first. This might be
// necessary if we get called after an entry reconfiguration
// and directory name change.
//
Status = UlpCreateSafeDirectory( &pEntry->FileName );
if (!NT_SUCCESS(Status))
goto end;
//
// Now Restore the short file name pointer back
//
pEntry->pShortName = (PWSTR)
&(pEntry->FileName.Buffer[pEntry->FileName.Length/sizeof(WCHAR)]);
//
// Append the new file name ( based on updated current time )
// to the end.
//
Status = RtlAppendUnicodeStringToString( &pEntry->FileName, &FileName );
if (!NT_SUCCESS(Status))
goto end;
}
//
// If the sequence is stale because of the nature of the recycle.
// And if our preiod is size based then rescan the new directory
// to figure out the proper file to open.
//
if (pEntry->Flags.StaleSequenceNumber &&
pEntry->Period==HttpLoggingPeriodMaxSize)
{
// This call may update the filename, the file size and the
// sequence number if there is an old file in the new dir.
Status = UlpQueryDirectory(pEntry);
if (!NT_SUCCESS(Status))
goto end;
}
//
// Time to close the old file and reopen a new one
//
if (pEntry->hFile != NULL)
{
//
// Before closing the old one we need to flush the buffer
//
UlpFlushLogFile(pEntry);
UlpLogCloseHandle(pEntry);
}
InitializeObjectAttributes(
&ObjectAttributes,
&pEntry->FileName, // Full path name
OBJ_CASE_INSENSITIVE | // Attributes
UL_KERNEL_HANDLE,
NULL, // RootDirectory
NULL // SecurityDescriptor
);
//
// Make the created file Aysnc by not picking the sync flag.
//
Status = ZwCreateFile(
&pEntry->hFile,
FILE_GENERIC_WRITE,
&ObjectAttributes,
&IoStatusBlock,
NULL,
FILE_ATTRIBUTE_NORMAL,
FILE_SHARE_READ,
FILE_OPEN_IF,
FILE_NON_DIRECTORY_FILE, // | FILE_SYNCHRONOUS_IO_NONALERT,
NULL,
0);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Get the file size, etc from the file.
//
Status = ZwQueryInformationFile(
pEntry->hFile,
&IoStatusBlock,
&pEntry->FileInfo,
sizeof(pEntry->FileInfo),
FileStandardInformation
);
if (NT_SUCCESS(Status) == FALSE)
goto end;
//
// Recalculate the time to expire.
//
if (pEntry->Flags.StaleTimeToExpire &&
pEntry->Period != HttpLoggingPeriodMaxSize)
{
UlpCalculateTimeToExpire(
&CurrentTimeFields,
pEntry->Period,
&pEntry->TimeToExpire
);
}
//
// By setting the flag to zero, we mark that we need to write title with the
// next incoming request But this only applies to W3C format.Otherwise the flag
// stays as set all the time, and the LogWriter doesn't attempt to write the
// title for NCSA and IIS log formats with the next incoming request.
//
if (pEntry->Format == HttpLoggingTypeW3C)
{
pEntry->Flags.LogTitleWritten = 0;
}
else
{
pEntry->Flags.LogTitleWritten = 1;
}
UlTrace( LOGGING, ("Ul!UlpRecycleLogFile: entry %p, file %S, handle %lx\n",
pEntry,
pEntry->FileName.Buffer,
pEntry->hFile
));
end:
// Mark fields non-stale again;
RESET_SEQUNCE_NUMBER_STALE(pEntry);
RESET_TIME_TO_EXPIRE_STALE(pEntry);
// TODO: Handle STATUS_DISK_FULL case gracefully.
if ( Status == STATUS_DISK_FULL )
{
UlTrace(LOGGING,("UlpRecycleLogFile: DISK FULL entry %p, failure %08lx\n",
pEntry,
Status
));
if (pEntry->hFile != NULL)
{
ZwClose( pEntry->hFile );
pEntry->hFile = NULL;
}
// TODO: pEntry->Flags.RecyclePending = 1;
// TODO: UlpFireDiskFullTimer();
}
if ( !NT_SUCCESS(Status) && Status != STATUS_DISK_FULL)
{
//
// If we made it to this point, then the create/open has failed.
//
UlTrace( LOGGING,("Ul!UlpRecycleLogFile: entry %p, failure %08lx\n",
pEntry,
Status
));
if (pEntry->hFile != NULL)
{
ZwClose( pEntry->hFile );
pEntry->hFile = NULL;
}
}
return Status;
}
/***************************************************************************++
Routine Description:
IsLogFileOverFlow:
Arguments:
--***************************************************************************/
__inline
BOOLEAN
UlpIsLogFileOverFlow(
IN PUL_LOG_FILE_ENTRY pEntry,
IN ULONG ReqdBytes
)
{
if (pEntry->Period != HttpLoggingPeriodMaxSize ||
pEntry->TruncateSize == HTTP_LIMIT_INFINITE)
{
return FALSE;
}
else
{
return((pEntry->TotalWritten.QuadPart + (ULONGLONG)ReqdBytes)
>=
(ULONGLONG)pEntry->TruncateSize
);
}
}
/***************************************************************************++
Routine Description:
UlpIncrementBytesWritten:
Arguments:
--***************************************************************************/
__inline
VOID
UlpIncrementBytesWritten(
IN PUL_LOG_FILE_ENTRY pEntry,
IN ULONG BytesWritten
)
{
UlInterlockedAdd64((PLONGLONG) &(pEntry->TotalWritten.QuadPart),
(ULONGLONG)BytesWritten);
};
/***************************************************************************++
Routine Description:
UlpWeekOfMonth : Ordinal Number of the week of the current month
Stolen from IIS 5.1 code base.
Example
July 2000 ... :
S M T W T F S WeekOfMonth
1 1
2 3 4 5 6 7 8 2
9 10 11 12 13 14 15 3
16 17 18 19 20 21 22 4
23 24 25 26 27 28 29 5
30 31 6
Finds the ordinal number of the week of current month.
The numbering of weeks starts from 1 and run through 6 per month (max).
The week number changes only on sundays.
The calculation to be use is:
1 + (dayOfMonth - 1)/7 + ((dayOfMonth - 1) % 7 > dayOfWeek);
(a) (b) (c) (d)
(a) to set the week numbers to begin from week numbered "1"
(b) used to calculate the rough number of the week on which a given
day falls based on the date.
(c) calculates what is the offset from the start of week for a given
day based on the fact that a week had 7 days.
(d) is the raw day of week given to us.
(c) > (d) indicates that the week is rolling forward and hence
the week count should be offset by 1 more.
Arguments:
PTIME_FIELDS - system time fields
Return Value:
ULONG - This func magically returns the week of the month
--***************************************************************************/
__inline
ULONG UlpWeekOfMonth(
IN PTIME_FIELDS fields
)
{
ULONG Tmp;
Tmp = (fields->Day - 1);
Tmp = ( 1 + Tmp/7 + (((Tmp % 7) > ((ULONG) fields->Weekday)) ? 1 : 0));
return Tmp;
}
/***************************************************************************++
Routine Description:
UlpInitializeGMTOffset :
Calculates and builds the time difference string.
Get called during the initialization.
And every hour after that.
--***************************************************************************/
VOID
UlpGetGMTOffset()
{
RTL_TIME_ZONE_INFORMATION Tzi;
NTSTATUS Status;
CHAR Sign;
LONG Bias;
ULONG Hour;
ULONG Minute;
ULONG DT = UL_TIME_ZONE_ID_UNKNOWN;
LONG BiasN = 0;
PAGED_CODE();
//
// get the timezone data from the system
//
Status = NtQuerySystemInformation(
SystemCurrentTimeZoneInformation,
(PVOID)&Tzi,
sizeof(Tzi),
NULL
);
if (!NT_SUCCESS(Status))
{
UlTrace(LOGGING,("Ul!UlpGetGMTOffset: failure %08lx\n", Status));
}
else
{
DT = UlCalcTimeZoneIdAndBias(&Tzi, &BiasN);
}
if ( BiasN > 0 )
{
//
// UTC = local time + bias
//
Bias = BiasN;
Sign = '-';
}
else
{
Bias = -1 * BiasN;
Sign = '+';
}
Minute = Bias % 60;
Hour = (Bias - Minute) / 60;
UlTrace( LOGGING,
("Ul!UlpGetGMTOffset: %c%02d:%02d (h:m) D/S %d BiasN %d\n",
Sign,
Hour,
Minute,
DT,
BiasN
) );
_snprintf( g_GMTOffset,
SIZE_OF_GMT_OFFSET,
"%c%02d%02d",
Sign,
Hour,
Minute
);
}
/***************************************************************************++
Routine Description:
UlpInitializeLogBufferGranularity :
This will determine the (MAX) size of the buffer we will be using
for the eac log file entry.
--***************************************************************************/
NTSTATUS
UlpInitializeLogBufferGranularity()
{
SYSTEM_BASIC_INFORMATION sbi;
NTSTATUS Status;
Status = STATUS_SUCCESS;
//
// Get the granularity from the system
//
Status = NtQuerySystemInformation(
SystemBasicInformation,
(PVOID)&sbi,
sizeof(sbi),
NULL
);
if ( !NT_SUCCESS(Status) )
{
UlTrace( LOGGING,
("Ul!UlpInitializeLogBufferGranularity: failure %08lx\n",
Status) );
return Status;
}
g_AllocationGranularity = sbi.AllocationGranularity;
UlTrace( LOGGING,
("Ul!UlpInitializeLogBufferGranularity: %d\n",
g_AllocationGranularity
) );
return Status;
}
/***************************************************************************++
Routine Description:
UlProbeLogField :
Probes the content of a user log field, including the terminating
null.
Arguments:
- the log field to be probed
--***************************************************************************/
__inline
VOID
UlpProbeLogField(
IN PVOID pField,
IN SIZE_T FieldLength,
IN ULONG Alignment
)
{
if ( pField )
{
ProbeTestForRead(
pField,
FieldLength + Alignment,
Alignment
);
}
}
/***************************************************************************++
Routine Description:
UlProbeLogData :
Probes the content of the user buffer of Log Data
Note: pUserLogData holds untrusted data sent down from user mode.
The caller MUST have a __try/__except block to catch any exceptions
or access violations that occur while probing this data.
Arguments:
PHTTP_LOG_FIELDS_DATA - The log data ( from WP ) to be probed and verified.
--***************************************************************************/
NTSTATUS
UlProbeLogData(
IN PHTTP_LOG_FIELDS_DATA pUserLogData
)
{
NTSTATUS Status;
PAGED_CODE();
Status = STATUS_SUCCESS;
if (pUserLogData)
{
UlTrace( LOGGING, ("Ul!UlProbeLogData: pUserLogData %p\n",
pUserLogData ));
//
// Check for the log fields data structure
//
ProbeTestForRead(
pUserLogData,
sizeof(HTTP_LOG_FIELDS_DATA),
sizeof(USHORT)
);
//
// Now check for the individual strings
//
UlpProbeLogField(pUserLogData->ClientIp,
pUserLogData->ClientIpLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->ServiceName,
pUserLogData->ServiceNameLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->ServerName,
pUserLogData->ServerNameLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->ServerIp,
pUserLogData->ServerIpLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->UriQuery,
pUserLogData->UriQueryLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->Host,
pUserLogData->HostLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->UserAgent,
pUserLogData->UserAgentLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->Cookie,
pUserLogData->CookieLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->Referrer,
pUserLogData->ReferrerLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->Method,
pUserLogData->MethodLength,
sizeof(CHAR));
UlpProbeLogField(pUserLogData->UserName,
pUserLogData->UserNameLength,
sizeof(WCHAR));
UlpProbeLogField(pUserLogData->UriStem,
pUserLogData->UriStemLength,
sizeof(WCHAR));
#if DBG
// CODEWORK: should we do this all the time? Remember, this is
// untrusted user-mode data
//
// Few more controls for chk bits
//
if ((pUserLogData->ClientIp &&
pUserLogData->ClientIpLength != strlen(pUserLogData->ClientIp))
||
(pUserLogData->ServiceName &&
pUserLogData->ServiceNameLength != strlen(pUserLogData->ServiceName))
||
(pUserLogData->ServerName &&
pUserLogData->ServerNameLength != strlen(pUserLogData->ServerName))
||
(pUserLogData->ServerIp &&
pUserLogData->ServerIpLength != strlen(pUserLogData->ServerIp))
||
(pUserLogData->Method &&
pUserLogData->MethodLength != strlen(pUserLogData->Method))
||
(pUserLogData->UriQuery &&
pUserLogData->UriQueryLength != strlen(pUserLogData->UriQuery))
||
(pUserLogData->Host &&
pUserLogData->HostLength != strlen(pUserLogData->Host))
||
(pUserLogData->UserAgent &&
pUserLogData->UserAgentLength != strlen(pUserLogData->UserAgent))
||
(pUserLogData->Cookie &&
pUserLogData->CookieLength != strlen(pUserLogData->Cookie))
||
(pUserLogData->Referrer &&
pUserLogData->ReferrerLength != strlen(pUserLogData->Referrer))
||
(pUserLogData->UserName &&
pUserLogData->UserNameLength != wcslen(pUserLogData->UserName)*sizeof(WCHAR))
//
// Disabled because UriSTem may not be null terminated.
// ||
// (pUserLogData->UriStem &&
// pUserLogData->UriStemLength != wcslen(pUserLogData->UriStem)*sizeof(WCHAR))
)
{
//
// Invalid log field has been pushed down by the WP.
// Complain and reject the request.
//
UlTrace(LOGGING,(
"Ul!UlProbeLogData: INVALID field rcvd from WP in pLogFields %p \n",
pUserLogData
));
Status = STATUS_INVALID_PARAMETER;
}
#endif // DBG
}
return Status;
}
/***************************************************************************++
Routine Description:
UlAllocateLogDataBuffer :
We capture the log fields from user ( WP ) buffer to our internal
buffer that we allocate here.
Also we set our pointer to pRequest here.
Arguments:
pLogData - The internal buffer to hold logging info. We will keep this
around until we are done with logging.
pRequest - Pointer to the currently logged request.
--***************************************************************************/
NTSTATUS
UlAllocateLogDataBuffer(
OUT PUL_LOG_DATA_BUFFER pLogData,
IN PUL_INTERNAL_REQUEST pRequest,
IN PUL_CONFIG_GROUP_OBJECT pConfigGroup // CG from cache or request
)
{
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pLogData);
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pRequest));
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
UlTrace(LOGGING, ("Ul!UlAllocateLogDataBuffer: pLogData %p \n",
pLogData
));
//
// Initialize Log Fields in the Log Buffer
//
UL_REFERENCE_INTERNAL_REQUEST(pRequest);
pLogData->pRequest = pRequest;
pLogData->CacheAndSendResponse= FALSE;
pLogData->BytesTransferred= 0;
pLogData->pConfigGroup= NULL;
pLogData->Line = pLogData->Buffer;
//
// Capture Format & Flags from the Request's Config Group. Or
// from the cache entries.
// There's a possiblity that the values inside the log entry
// maybe stale but that's acceptable. We do not want to acquire
// the log resource so that we can avoid the contention.
//
pLogData->Format= pConfigGroup->LoggingConfig.LogFormat;
pLogData->Flags = UL_GET_LOG_TYPE_MASK(
pConfigGroup->LoggingConfig.LogFormat,
pConfigGroup->LoggingConfig.LogExtFileFlags
);
pLogData->Used= 0; // For NCSA & W3C this is size of the line.
pLogData->Length= 0; // Allocation length,the default is 4k
pLogData->UsedOffset1= 0; // Used by all formats.
pLogData->UsedOffset2= 0; // This SHOULD only be nonzero if format is IIS
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
UlDestroyLogDataBuffer :
After we are done with writing this record we have to clean up
the internal log buffer here.
Arguments:
pLogData - The buffer to be destroyed
--***************************************************************************/
VOID
UlDestroyLogDataBufferWorker(
IN PUL_WORK_ITEM pWorkItem
)
{
PUL_LOG_DATA_BUFFER pLogData;
//
// Sanity check
//
ASSERT(pWorkItem);
pLogData = CONTAINING_RECORD(
pWorkItem,
UL_LOG_DATA_BUFFER,
WorkItem
);
//
// If we are keeping a private pointer to the cgroup release it
// as well
//
if (pLogData->pConfigGroup)
{
DEREFERENCE_CONFIG_GROUP(pLogData->pConfigGroup);
pLogData->pConfigGroup = NULL;
}
//
// Now release the possibly allocated large log line buffer
//
if (pLogData->Length > UL_LOG_LINE_BUFFER_SIZE)
{
// Large log line get allocated from paged pool
// we better be running on lowered IRQL for this case.
PAGED_CODE();
UL_FREE_POOL(pLogData->Line, UL_LOG_DATA_BUFFER_POOL_TAG);
}
//
// Last release our pointer to request structure here.
//
if (pLogData->pRequest)
{
PUL_INTERNAL_REQUEST pRequest;
pRequest = pLogData->pRequest;
pLogData->pRequest = NULL;
UL_DEREFERENCE_INTERNAL_REQUEST(pRequest);
}
UlTrace(LOGGING,("Ul!UlpDestroyLogDataBufferWorker: pLogData %p \n",
pLogData
));
}
/***************************************************************************++
Routine Description:
Captures and writes the log fields from user (WP) buffer to the log line.
Captures only those necessary fields according to the picked Flags.
Does UTF8 and LocalCode Page conversion for UserName and URI Stem.
Leaves enough space for Date & Time fields for late generation.
Does SpaceToPlus conversion for UserAgent, Cookie, Referrer & Host.
Arguments:
pLogData : User Buffer which holds the fields and their lengths
Version : Version information from Request
pLogBuffer : Structure which holds final log line and additional
information.
--***************************************************************************/
NTSTATUS
UlCaptureLogFieldsW3C(
IN PHTTP_LOG_FIELDS_DATA pLogData,
IN HTTP_VERSION Version,
OUT PUL_LOG_DATA_BUFFER pLogBuffer
)
{
NTSTATUS Status;
ULONG Flags;
PCHAR psz;
PCHAR pBuffer;
ULONG BytesConverted;
ULONG FastLength;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pLogBuffer);
Status = STATUS_SUCCESS;
Flags = pLogBuffer->Flags;
BytesConverted = 0;
//
// Try fast length calculation for the the default case. If this fails
// Recalc function will precisely calculate the required length by
// paying attention to the picked flags.
//
FastLength = pLogData->ClientIpLength
+ pLogData->UserNameLength
+ pLogData->ServiceNameLength
+ pLogData->ServerNameLength
+ pLogData->ServerIpLength
+ pLogData->MethodLength
+ pLogData->UriStemLength
+ pLogData->UriQueryLength
+ pLogData->UserAgentLength
+ pLogData->CookieLength
+ pLogData->ReferrerLength
+ pLogData->HostLength
+ MAX_W3C_FIX_FIELD_OVERHEAD
;
if (UTF8_LOGGING_ENABLED() || (FastLength > UL_LOG_LINE_BUFFER_SIZE))
{
FastLength = UlpCalcLogLineLengthW3C(
pLogData,
Flags,
(UTF8_LOGGING_ENABLED() ? 2 : 1)
);
if (FastLength > UL_LOG_LINE_BUFFER_SIZE)
{
Status = UlpReallocLogLine(pLogBuffer, FastLength);
if (!NT_SUCCESS(Status))
{
return Status;
}
}
}
//
// Leave enough space for the date & time fields: "2000-01-31 00:12:23 "
// For W3C format the field "Used" shows the size of the log line we have filled.
// UsedOffset1 shows how much space we have saved for date and/or time logfields.
// UsedOffset2 shows the size of the log line which will be stored in the cache
// entry, not including reserved space for date & time and the logfields after
// ServerPort. Therefore we only store the fragment which starts from logfield
// UserName to ServerPort. Others will be generated per cache hit.
//
psz = pBuffer = &pLogBuffer->Line[0];
if ( Flags & MD_EXTLOG_DATE ) psz += 11;
if ( Flags & MD_EXTLOG_TIME ) psz += 9;
// Generate all the fields except BytesSend, BytesReceived & TimeTaken
// They will be added when the send is complete and if they are picked
// We will only copy the fields after this point to the cache entry.
// Lets set and remember the size of the fields we are discarding.
pLogBuffer->UsedOffset1 = (USHORT) DIFF(psz - pBuffer);
if ( Flags & MD_EXTLOG_SITE_NAME )
{
if (pLogData->ServiceNameLength)
{
psz = UlStrPrintStr(psz, pLogData->ServiceName,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_COMPUTER_NAME )
{
if (pLogData->ServerNameLength)
{
psz = UlStrPrintStr(psz, pLogData->ServerName,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_SERVER_IP )
{
if (pLogData->ServerIpLength)
{
psz = UlStrPrintStr(psz, pLogData->ServerIp,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_METHOD )
{
if (pLogData->MethodLength)
{
psz = UlStrPrintStr(psz, pLogData->Method,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_URI_STEM )
{
BytesConverted = 0;
if (pLogData->UriStemLength)
{
PCHAR pszT = psz;
if (UTF8_LOGGING_ENABLED())
{
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UriStem,
pLogData->UriStemLength/sizeof(WCHAR),
psz,
pLogData->UriStemLength * 2
);
ASSERT(BytesConverted);
}
else
{
RtlUnicodeToMultiByteN(
psz,
pLogData->UriStemLength,
&BytesConverted,
pLogData->UriStem,
pLogData->UriStemLength
);
}
psz += BytesConverted;
// Do SpeaceToPlus conversion before writting out
// the terminator space.
while (pszT != psz)
{
if (*pszT == ' ') *pszT = '+';
pszT++;
}
*psz++ = ' ';
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_URI_QUERY )
{
if (pLogData->UriQueryLength)
{
psz = UlStrPrintStr(psz, pLogData->UriQuery,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_HTTP_STATUS )
{
psz = UlStrPrintProtocolStatus(psz,(USHORT)pLogData->ProtocolStatus,' ');
}
if ( Flags & MD_EXTLOG_WIN32_STATUS )
{
psz = UlStrPrintUlong(psz, pLogData->Win32Status,' ');
}
if ( Flags & MD_EXTLOG_SERVER_PORT )
{
psz = UlStrPrintUlong(psz, pLogData->ServerPort,' ');
}
//
// Cache builder won't be storing the fields after this line.
// They have to be generated per hit for cache hits.
//
pLogBuffer->UsedOffset2 = (USHORT) DIFF(psz - pBuffer);
if ( Flags & MD_EXTLOG_USERNAME )
{
BytesConverted = 0;
if (pLogData->UserNameLength)
{
// Do either UTF8 or LocalCodePage Conversion
// not including the terminating null.
PCHAR pszT = psz;
if (UTF8_LOGGING_ENABLED())
{
// UTF8 Conversion may require upto two times because of a
// possible 2 byte to 4 byte conversion.
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UserName,
pLogData->UserNameLength/sizeof(WCHAR),
psz,
pLogData->UserNameLength * 2
);
ASSERT(BytesConverted);
}
else
{
// Local codepage is normally closer to the half the length,
// but due to the possibility of pre-composed characters,
// the upperbound of the ANSI length is the UNICODE length
// in bytes
RtlUnicodeToMultiByteN(
psz,
pLogData->UserNameLength,
&BytesConverted,
pLogData->UserName,
pLogData->UserNameLength
);
}
// Forward the psz by BytesConverted
psz += BytesConverted;
// Do SpaceToPlus conversion
while (pszT != psz)
{
if (*pszT == ' ') *pszT = '+';
pszT++;
}
*psz++ = ' ';
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_CLIENT_IP )
{
if (pLogData->ClientIpLength)
{
psz = UlStrPrintStr(psz, pLogData->ClientIp,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_PROTOCOL_VERSION )
{
psz = UlStrPrintStr(psz, UL_GET_NAME_FOR_HTTP_VERSION(Version),' ');
}
if ( Flags & MD_EXTLOG_USER_AGENT )
{
if (pLogData->UserAgentLength)
{
psz = UlStrPrintStrC(psz, pLogData->UserAgent,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_COOKIE )
{
if (pLogData->CookieLength)
{
psz = UlStrPrintStrC(psz, pLogData->Cookie,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_REFERER )
{
if (pLogData->ReferrerLength)
{
psz = UlStrPrintStrC(psz, pLogData->Referrer,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_HOST )
{
if (pLogData->HostLength)
{
psz = UlStrPrintStrC(psz, pLogData->Host,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
// Finally calculate the used space
pLogBuffer->Used = (ULONG) DIFF(psz - pBuffer);
// Date & Time fields will be filled in when the LogHit completes.
// As well as the fields BytesSent,BytesReceived and TimeTaken will
// be added to the end of the log line, then.
UlTrace(LOGGING, ("Ul!UlCaptureLogFields: user %p kernel %p\n",
pLogData,pLogBuffer
));
return Status;
}
NTSTATUS
UlCaptureLogFieldsNCSA(
IN PHTTP_LOG_FIELDS_DATA pLogData,
IN HTTP_VERSION Version,
OUT PUL_LOG_DATA_BUFFER pLogBuffer
)
{
NTSTATUS Status;
ULONG BytesConverted;
PCHAR psz;
PCHAR pBuffer;
ULONG Utf8Multiplier;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pLogBuffer);
Status = STATUS_SUCCESS;
Utf8Multiplier = (UTF8_LOGGING_ENABLED() ? 2 : 1);
// Estimate the length and reallocate the log data buffer line
// if necessary
pLogBuffer->Length =
2 + pLogData->ClientIpLength +
2 + // For remote user log name
2 + pLogData->UserNameLength * Utf8Multiplier +
29 + // Fixed Date & Time Space
2 + pLogData->MethodLength + // "MTHD U-STEM?U-QUERY P-VER"
2 + pLogData->UriStemLength * Utf8Multiplier +
2 + pLogData->UriQueryLength +
2 + UL_HTTP_VERSION_LENGTH + // Version plus Quotes ""
2 + MAX_ULONG_STR + // ProtocolStatus
2 + MAX_ULONGLONG_STR + // BytesSend
3 // \r\n\0
;
if ( pLogBuffer->Length > UL_LOG_LINE_BUFFER_SIZE )
{
Status = UlpReallocLogLine(pLogBuffer, pLogBuffer->Length);
if (!NT_SUCCESS(Status))
{
return Status;
}
}
//
// UCIP - username [date:time offset] "MTHD U-STEM?U-QUERY P-VER" Status BSent
//
// Set and remember the beginning
psz = pBuffer = &pLogBuffer->Line[0];
// Client IP
if (pLogData->ClientIpLength)
{
psz = UlStrPrintStr(psz, pLogData->ClientIp,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
// Fixed dash
*psz++ = '-'; *psz++ = ' ';
// UserName
if (pLogData->UserNameLength)
{
BytesConverted = 0;
if (UTF8_LOGGING_ENABLED())
{
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UserName,
pLogData->UserNameLength/sizeof(WCHAR),
psz,
pLogData->UserNameLength * 2
);
ASSERT(BytesConverted);
}
else
{
RtlUnicodeToMultiByteN(
psz,
pLogData->UserNameLength,
&BytesConverted,
pLogData->UserName,
pLogData->UserNameLength
);
}
psz += BytesConverted; *psz++ = ' ';
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
// [Date:Time GmtOffset] -> "[07/Jan/2000:00:02:23 -0800] "
// Just leave the space for the time being. But remember the
// offset of the beginning of the reserved space of 29 bytes.
pLogBuffer->UsedOffset1 = (USHORT) DIFF(psz - pBuffer);
// Forward psz to bypass the reserved space
psz += NCSA_FIX_DATE_AND_TIME_FIELD_SIZE;
// "MTHD U-STEM?U-QUERY P-VER"
*psz++ = '\"';
if (pLogData->MethodLength)
{
psz = UlStrPrintStr(psz, pLogData->Method,' ');
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
if (pLogData->UriStemLength)
{
BytesConverted = 0;
if (UTF8_LOGGING_ENABLED())
{
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UriStem,
pLogData->UriStemLength/sizeof(WCHAR),
psz,
pLogData->UriStemLength * 2
);
ASSERT(BytesConverted);
}
else
{
RtlUnicodeToMultiByteN(
psz,
pLogData->UriStemLength,
&BytesConverted,
pLogData->UriStem,
pLogData->UriStemLength
);
}
psz += BytesConverted;
}
else
{
*psz++ = '-';
}
if (pLogData->UriQueryLength)
{
*psz++ = '?';
psz = UlStrPrintStr(psz, pLogData->UriQuery,' ');
}
else
{
*psz++ = ' ';
}
pLogBuffer->UsedOffset2 = (USHORT) DIFF(psz - pBuffer);
psz = UlStrPrintStr(psz, UL_GET_NAME_FOR_HTTP_VERSION(Version),'\"');
*psz++ = ' ';
// ProtocolStatus
psz = UlStrPrintProtocolStatus(psz, (USHORT)pLogData->ProtocolStatus,' ');
// Calculate the length upto now
pLogBuffer->Used += (ULONG) DIFF(psz - pBuffer);
// BytesSend will be filled later on during hit processing
return Status;
}
NTSTATUS
UlCaptureLogFieldsIIS(
IN PHTTP_LOG_FIELDS_DATA pLogData,
IN HTTP_VERSION Version,
OUT PUL_LOG_DATA_BUFFER pLogBuffer
)
{
NTSTATUS Status;
ULONG BytesConverted;
PCHAR psz;
PCHAR pBuffer;
ULONG Utf8Multiplier;
//
// Sanity check.
//
PAGED_CODE();
ASSERT(pLogBuffer);
Status = STATUS_SUCCESS;
Utf8Multiplier = (UTF8_LOGGING_ENABLED() ? 2 : 1);
// Estimate the length and reallocate the log data buffer line
// if necessary
pLogBuffer->Length =
2 + pLogData->ClientIpLength +
2 + pLogData->UserNameLength * Utf8Multiplier +
22 + // Fixed Date & Time Space
2 + pLogData->ServiceNameLength +
2 + pLogData->ServerNameLength +
2 + pLogData->ServerIpLength +
2 + MAX_ULONGLONG_STR + // TimeTaken
2 + MAX_ULONGLONG_STR + // BytesReceived
2 + MAX_ULONGLONG_STR + // BytesSend
2 + MAX_ULONG_STR + // ProtocolStatus
2 + MAX_ULONG_STR + // Win32 Status
2 + pLogData->MethodLength +
2 + pLogData->UriStemLength * Utf8Multiplier +
2 + pLogData->UriQueryLength +
3 // \r\n\0
;
if ( pLogBuffer->Length > UL_LOG_LINE_BUFFER_SIZE )
{
// TODO: Fix this and do the limit checks per field
if (pLogBuffer->Length > UL_MAX_LOG_LINE_BUFFER_SIZE)
{
return STATUS_INVALID_PARAMETER;
}
Status = UlpReallocLogLine(pLogBuffer, UL_MAX_LOG_LINE_BUFFER_SIZE);
if (!NT_SUCCESS(Status))
{
return Status;
}
}
//
// UIP,user,D,T,site,Server,SIP,Ttaken,BR,BS,PS,WS,M,URI,URIQUERY,
//
// We will store the fragmented IIS log line as follows. If captured fields
// won't fit to the default 4k buffer the fragments will be increased by mul
// tiplier of 2 until they fit. But this calculation happened earlier and now
// we know the actual size of each fragment. All the available fields will be
// written to the beginning of their corresponding fragments and unavailable ones
// for the first two fragments ( i.e. date and time for 1 and timetaken
// bytesreceived and bytessent for 2, none for the three ) will be appended later
// when the hit happens. To be able to do that we will remember used_offsets for
// fragments one and two, both in the log_data and in the cache entry. But for this
// schema we need a special LogWriter which expects and handles 3 fragments rather
// than one complete line.
// <- upto time -> <- from siteName to BytesSent -> <- From p status to query ->
// 0 511 512 1023 1024 4096
// FRAGMENT ONE
// -----------------------------------------------------
// Cache entry will not store this fragment, the fields in
// here have to be regenerated for the (pure) cache hits.
psz = pBuffer = &pLogBuffer->Line[0];
// Client IP
if (pLogData->ClientIpLength)
{
psz = UlStrPrintStr(psz, pLogData->ClientIp,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// UserName
if (pLogData->UserNameLength)
{
BytesConverted = 0;
if (UTF8_LOGGING_ENABLED())
{
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UserName,
pLogData->UserNameLength/sizeof(WCHAR),
psz,
pLogData->UserNameLength * 2
);
ASSERT(BytesConverted);
}
else
{
RtlUnicodeToMultiByteN(
psz,
pLogData->UserNameLength,
&BytesConverted,
pLogData->UserName,
pLogData->UserNameLength
);
}
psz += BytesConverted; *psz++ = ',';
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// Date and Time will be added later to the end of this fragment.
pLogBuffer->UsedOffset1 = (USHORT) DIFF(psz - pBuffer);
// FRAGMENT TWO
// -----------------------------------------------------
pBuffer = psz = &pLogBuffer->Line[512];
// SiteName
if (pLogData->ServiceNameLength)
{
psz = UlStrPrintStr(psz, pLogData->ServiceName,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// ServerName
if (pLogData->ServerNameLength)
{
psz = UlStrPrintStr(psz, pLogData->ServerName,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// ServerIp
if (pLogData->ServerIpLength)
{
psz = UlStrPrintStr(psz, pLogData->ServerIp,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// TimeTaken BytesSent and BytesReceived will be added later
// to the end of this fragment
pLogBuffer->UsedOffset2 = (USHORT) DIFF(psz - pBuffer);
// FRAGMENT THREE
// -----------------------------------------------------
pBuffer = psz = &pLogBuffer->Line[1024];
// ProtocolStatus
psz = UlStrPrintProtocolStatus(psz, (USHORT)pLogData->ProtocolStatus,',');
*psz++ = ' ';
// Win32 Status
psz = UlStrPrintUlong(psz, pLogData->Win32Status,',');
*psz++ = ' ';
// Method
if (pLogData->MethodLength)
{
psz = UlStrPrintStr(psz, pLogData->Method,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// URI Stem
if (pLogData->UriStemLength)
{
BytesConverted = 0;
if (UTF8_LOGGING_ENABLED())
{
BytesConverted =
HttpUnicodeToUTF8(
pLogData->UriStem,
pLogData->UriStemLength/sizeof(WCHAR),
psz,
pLogData->UriStemLength * 2
);
ASSERT(BytesConverted);
}
else
{
RtlUnicodeToMultiByteN(
psz,
pLogData->UriStemLength,
&BytesConverted,
pLogData->UriStem,
pLogData->UriStemLength
);
}
psz += BytesConverted; *psz++ = ',';
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = ' ';
// URI Query
if (pLogData->UriQueryLength)
{
psz = UlStrPrintStr(psz, pLogData->UriQuery,',');
}
else
{
*psz++ = '-'; *psz++ = ',';
}
*psz++ = '\r'; *psz++ = '\n';
// The size of the fragment 3 goes to Used
pLogBuffer->Used += (ULONG) DIFF(psz - pBuffer);
*psz++ = ANSI_NULL;
return Status;
}
/***************************************************************************++
Routine Description:
UlLogHttpHit :
This function ( or its cache pair ) gets called everytime a log hit
happens. Just before completing the SendResponse request to the user.
The most likely places for calling this API or its pair for cache
is just before the send completion when we were about the destroy
send trackers.
Means:
1. UlpCompleteSendRequestWorker for ORDINARY hits; before destroying
the PUL_CHUNK_TRACKER for send operation.
2. UlpCompleteSendCacheEntryWorker for both types of CACHE hits
(cache build&send or just pure cache hit) before destroying the
the PUL_FULL_TRACKER for cache send operation.
This function requires Request & Response structures ( whereas its
cache pair only requires the Request ) to successfully generate the
the log fields and even for referencing to the right log configuration
settings for this site ( thru pRequest's pConfigInfo pointer ).
Unfortunately the major concern is untimely resetting of the connection
(when client terminates the connection before we get a chance to reach
to the -above- mentioned places) In that case UlConnectionDestroyedWorker
will asynchrously destroy our pRequest pointer ( in the pHttpConnection
structure ) and cause us to miss log hits. So we cannot trust on
HttpConnection for pRequest pointer. To solve this issue we keeep our
own pointer to Request. No need to worry about pResponse as it's going
to be preserved by chunk_trucker until its own destruction.
Arguments:
pResponse - pointer to the internal response structure, surely
passed down by the chunk tracker. We will grap our Log Data
buffer from this structure which originaly allocated when
capturing the Response from the user.
See UlCaptureHttpResponse.
--***************************************************************************/
NTSTATUS
UlLogHttpHit(
IN PUL_LOG_DATA_BUFFER pLogBuffer
)
{
NTSTATUS Status;
PUL_CONFIG_GROUP_OBJECT pConfigGroup;
PUL_INTERNAL_REQUEST pRequest;
PCHAR psz;
PCHAR pBuffer;
ULONG BytesWritten;
LONGLONG LifeTime;
LARGE_INTEGER CurrentTimeStamp;
PUL_LOG_FILE_ENTRY pEntry;
//
// A LOT of sanity checks.
//
PAGED_CODE();
Status = STATUS_SUCCESS;
UlTrace( LOGGING, ("Ul!UlLogHttpHit: pLogData %p\n", pLogBuffer ));
ASSERT(pLogBuffer);
pRequest = pLogBuffer->pRequest;
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pRequest));
if (pRequest->ConfigInfo.pLoggingConfig == NULL ||
IS_LOGGING_DISABLED(pRequest->ConfigInfo.pLoggingConfig)
)
{
//
// If logging is disabled or log settings don't
// exist then do not proceed. Just exit out.
//
return STATUS_SUCCESS;
}
pConfigGroup = pRequest->ConfigInfo.pLoggingConfig;
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
//
// Construct the remaining log fields
//
switch(pLogBuffer->Format)
{
case HttpLoggingTypeW3C:
{
// First write the date & time fields to the beginning reserved
// space. Do not increment the used counter for date & time, b/c
// CaptureLogFields already did this when reserving the space.
psz = &pLogBuffer->Line[0];
if ( pLogBuffer->Flags & MD_EXTLOG_DATE )
{
UlpGetDateTimeFields(
HttpLoggingTypeW3C,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 10);
}
if ( pLogBuffer->Flags & MD_EXTLOG_TIME )
{
UlpGetDateTimeFields(
HttpLoggingTypeW3C,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 8);
}
// Set and remember where we started
pBuffer = psz = &pLogBuffer->Line[pLogBuffer->Used];
// Now proceed with the remaining fields, but add them to the end.
if ( pLogBuffer->Flags & MD_EXTLOG_BYTES_SENT )
{
psz = UlStrPrintUlonglong(psz, pRequest->BytesSent,' ');
}
if ( pLogBuffer->Flags & MD_EXTLOG_BYTES_RECV )
{
psz = UlStrPrintUlonglong(psz, pRequest->BytesReceived,' ');
}
if ( pLogBuffer->Flags & MD_EXTLOG_TIME_TAKEN )
{
KeQuerySystemTime( &CurrentTimeStamp );
LifeTime = CurrentTimeStamp.QuadPart - pRequest->TimeStamp.QuadPart;
if (LifeTime < 0)
{
LifeTime = 0;
UlTrace(LOGGING, ("CopyTimeStampField failure.\n"));
}
LifeTime /= (10*1000); // Conversion from 100-nanosecond to millisecs.
psz = UlStrPrintUlonglong(psz, (ULONGLONG)LifeTime,' ');
}
// Now calculate the space we have used
pLogBuffer->Used += (ULONG) DIFF(psz - pBuffer);
// Eat the last space and write the \r\n to the end.
// Only if we have any fields picked and written
if (pLogBuffer->Used)
{
psz = &pLogBuffer->Line[pLogBuffer->Used-1]; // Eat the last space
*psz++ = '\r'; *psz++ = '\n'; *psz++ = ANSI_NULL;
pLogBuffer->Used += 1;
}
else
{
return STATUS_SUCCESS; // No log fields nothing to log
}
// Cleanup the UsedOffsets otherwise length calculation will
// fail down below.
pLogBuffer->UsedOffset1 = pLogBuffer->UsedOffset2 = 0;
}
break;
case HttpLoggingTypeNCSA:
{
// [date:time GmtOffset] -> "[07/Jan/2000:00:02:23 -0800] "
// Restore the pointer to the reserved space first.
psz = &pLogBuffer->Line[pLogBuffer->UsedOffset1];
*psz++ = '[';
UlpGetDateTimeFields(
HttpLoggingTypeNCSA,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ':';
ASSERT(BytesWritten == 11);
UlpGetDateTimeFields(
HttpLoggingTypeNCSA,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 8);
UlAcquireResourceShared(&g_pUlNonpagedData->LogListResource, TRUE);
psz = UlStrPrintStr(psz, g_GMTOffset,']');
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
*psz++ = ' ';
ASSERT(((ULONG) DIFF(psz - &pLogBuffer->Line[pLogBuffer->UsedOffset1])) == 29);
// BytesSent
pBuffer = psz = &pLogBuffer->Line[pLogBuffer->Used];
psz = UlStrPrintUlonglong(psz, pRequest->BytesSent,'\r');
pLogBuffer->Used += (ULONG) DIFF(psz - pBuffer);
// \n\0
*psz++ = '\n'; *psz++ = ANSI_NULL;
pLogBuffer->Used += 1;
// Cleanup the UsedOffsets otherwise length calculation will
// fail down below.
pLogBuffer->UsedOffset1 = pLogBuffer->UsedOffset2 = 0;
}
break;
case HttpLoggingTypeIIS:
{
// At this time to construct the IIS log line is slightly
// different than the cache completion case. We can either
// use memmoves to get rid of the gaps between the three
// fragments. Or we provide a special log writer to actually
// handle three fragments when writting to the final log buffer
// I have picked the latter.
// Complete Fragment 1
pBuffer = psz = &pLogBuffer->Line[pLogBuffer->UsedOffset1];
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
pLogBuffer->UsedOffset1 += (USHORT) DIFF(psz - pBuffer);
// Complete Fragment 2
pBuffer = psz = &pLogBuffer->Line[512 + pLogBuffer->UsedOffset2];
KeQuerySystemTime( &CurrentTimeStamp );
LifeTime = CurrentTimeStamp.QuadPart - pRequest->TimeStamp.QuadPart;
if (LifeTime < 0)
{
LifeTime = 0;
UlTrace(LOGGING, ("CopyTimeStampField: failure.\n"));
}
LifeTime /= (10*1000); // Conversion from 100-nanosecond to millisecs.
psz = UlStrPrintUlonglong(psz, (ULONGLONG)LifeTime,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pRequest->BytesReceived,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pRequest->BytesSent,',');
*psz++ = ' ';
pLogBuffer->UsedOffset2 += (USHORT) DIFF(psz - pBuffer);
// Size of the final log line should be
// pLogBuffer->UsedOffset1 + pLogBuffer->UsedOffset2 + pLogBuffer->Used
// for IIS log format down below.
}
break;
default:
{
ASSERT(!"Unknown Log Format Type\n");
return STATUS_INVALID_PARAMETER;
}
}
//
// Finally this log line is ready to rock. Lets write it out.
//
UlAcquireResourceShared(&g_pUlNonpagedData->LogListResource, TRUE);
//
// Unless otherwise there's a problem in the set config group Ioctl
// we should never come here with having a null entry pointer. Even
// in the case of log reconfiguration, cgroup should acquire the log
// list resource to update its entry pointer.
//
pEntry = pConfigGroup->pLogFileEntry;
if (pEntry)
{
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
Status =
UlpWriteToLogFile (
pEntry,
pLogBuffer->UsedOffset1 + pLogBuffer->UsedOffset2 + pLogBuffer->Used,
&pLogBuffer->Line[0],
pLogBuffer->UsedOffset1,
pLogBuffer->UsedOffset2
);
}
#if DBG
else
{
//
// Practically every time we leak a url_cache entry we will not close
// the corresponding cgroup and the corresponding log entry. In that
// case next time we try to run http.sys we might see this assertion
// going on. Not usefull at this time because it's too late to cacth
// the original leak. Also if we run out of system resources when
// allocation the log file entry we will come here as well.
//
UlTrace(LOGGING,("Ul!UlLogHttpHit:null logfile entry !\n"));
Status = STATUS_INVALID_PARAMETER;
}
if (!NT_SUCCESS(Status))
{
UlTrace( LOGGING, ("Ul!UlLogHttpHit: entry %p, failure %08lx \n",
pEntry,
Status
));
}
#endif // DBG
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
return Status;
}
/***************************************************************************++
Routine Description:
Queues a work item for the actual log hit function.
Arguments:
pTracker - Supplies the tracker to complete.
--***************************************************************************/
NTSTATUS
UlLogHttpCacheHit(
PUL_FULL_TRACKER pTracker
)
{
NTSTATUS Status;
PUL_LOG_DATA_BUFFER pLogData;
ULONG NewLength;
PUL_INTERNAL_REQUEST pRequest;
//
// A Lot of sanity checks.
//
PAGED_CODE();
ASSERT(pTracker);
ASSERT(IS_VALID_FULL_TRACKER(pTracker));
ASSERT(pTracker->pLogData);
Status = STATUS_SUCCESS;
pLogData = pTracker->pLogData;
pLogData->BytesTransferred = pTracker->IoStatus.Information;
pRequest = pLogData->pRequest;
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pRequest));
UlTrace(LOGGING,("Ul!UlLogHttpCacheHit: pLogData %p\n",pTracker->pLogData));
//
// Restore the logging data from cache. If this hasn't been done by
// the BuildCacheEntry for cache&send responses.
//
if ( pLogData->CacheAndSendResponse == FALSE )
{
ASSERT(pTracker->pUriEntry);
// Make sure that internal buffer is big enough
// before proceeding.
// pLogData->Length = pTracker->pUriEntry->MaxLength;
switch( pLogData->Format )
{
case HttpLoggingTypeW3C:
{
// Recalculate the log line size to see if need to realloc
NewLength = UlpRecalcLogLineLengthW3C(
pLogData->Flags,
pLogData->pRequest,
pTracker->pUriEntry->LogDataLength
);
if ( NewLength > UL_LOG_LINE_BUFFER_SIZE )
{
Status = UlpReallocLogLine(pLogData, NewLength);
if (!NT_SUCCESS(Status))
{
// Worker won't get called. Cleanup immediately
UlDestroyLogDataBuffer(pLogData);
return Status;
}
}
if ( pTracker->pUriEntry->LogDataLength )
{
RtlCopyMemory( &pLogData->Line[pTracker->pUriEntry->UsedOffset1],
pTracker->pUriEntry->pLogData,
pTracker->pUriEntry->LogDataLength
);
}
// Cache data plus the reserved space for date and time
pLogData->Used = pTracker->pUriEntry->LogDataLength +
pTracker->pUriEntry->UsedOffset1;
}
break;
case HttpLoggingTypeNCSA:
{
PCHAR psz,pBuffer;
ASSERT( pTracker->pUriEntry->LogDataLength );
ASSERT( pTracker->pUriEntry->pLogData );
NewLength = MAX_NCSA_CACHE_FIELD_OVERHEAD
+ pTracker->pUriEntry->LogDataLength;
if ( NewLength > UL_LOG_LINE_BUFFER_SIZE )
{
Status = UlpReallocLogLine(pLogData, NewLength);
if (!NT_SUCCESS(Status))
{
// Worker won't get called. Cleanup immediately
UlDestroyLogDataBuffer(pLogData);
return Status;
}
}
psz = pBuffer = &pLogData->Line[0];
// Client IP
psz = UlStrPrintIP(
psz,
pRequest->pHttpConn->pConnection->RemoteAddress,
' '
);
// Fixed dash
*psz++ = '-'; *psz++ = ' ';
// Authenticated users cannot be served from cache.
*psz++ = '-'; *psz++ = ' ';
// Mark the beginning of the date & time fields
pLogData->UsedOffset1 = (USHORT) DIFF(psz - pBuffer);
// Forward psz to bypass the reserved space
psz += NCSA_FIX_DATE_AND_TIME_FIELD_SIZE;
RtlCopyMemory( psz,
pTracker->pUriEntry->pLogData,
pTracker->pUriEntry->LogDataLength
);
psz += pTracker->pUriEntry->LogDataLength;
// Protocol Version
psz = UlStrPrintStr(
psz,
UL_GET_NAME_FOR_HTTP_VERSION(pRequest->Version),
'\"'
);
*psz++ = ' ';
// ProtocolStatus
psz = UlStrPrintProtocolStatus(
psz,
pTracker->pUriEntry->StatusCode,
' '
);
// Calculate the length until now
pLogData->Used += (ULONG) DIFF(psz - pBuffer);
}
break;
case HttpLoggingTypeIIS:
{
PCHAR psz;
ULONG BytesWritten;
LONGLONG LifeTime;
LARGE_INTEGER CurrentTimeStamp;
ASSERT( pTracker->pUriEntry->LogDataLength );
ASSERT( pTracker->pUriEntry->pLogData );
NewLength = MAX_IIS_CACHE_FIELD_OVERHEAD
+ pTracker->pUriEntry->LogDataLength;
if ( NewLength > UL_LOG_LINE_BUFFER_SIZE )
{
Status = UlpReallocLogLine(pLogData, NewLength);
if (!NT_SUCCESS(Status))
{
// Worker won't get called. Cleanup immediately
UlDestroyLogDataBuffer(pLogData);
return Status;
}
}
BytesWritten = 0;
// Followings specify the size of iis log line fragments
// two and three. We are constructing the fragment one
// completely from scratch for pure cache hits.
pLogData->UsedOffset1 = pTracker->pUriEntry->UsedOffset1;
pLogData->UsedOffset2 = pTracker->pUriEntry->UsedOffset2;
ASSERT(pLogData->UsedOffset1 + pLogData->UsedOffset2
== pTracker->pUriEntry->LogDataLength );
psz = &pLogData->Line[0];
// Client IP
psz = UlStrPrintIP(
psz,
pRequest->pHttpConn->pConnection->RemoteAddress,
','
);
*psz++ = ' ';
// Authenticated users cannot be served from cache.
*psz++ = '-'; *psz++ = ','; *psz++ = ' ';
// Date & Time fields
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
// Fragment two
RtlCopyMemory( psz,
pTracker->pUriEntry->pLogData,
pLogData->UsedOffset1
);
psz += pLogData->UsedOffset1;
KeQuerySystemTime( &CurrentTimeStamp );
LifeTime = CurrentTimeStamp.QuadPart - pRequest->TimeStamp.QuadPart;
if (LifeTime < 0)
{
LifeTime = 0;
UlTrace(LOGGING, ("CopyTimeStampField: failure.\n"));
}
LifeTime /= (10*1000); // Conversion from 100-nanosecond to millisecs.
psz = UlStrPrintUlonglong(psz, (ULONGLONG)LifeTime,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pRequest->BytesReceived,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pLogData->BytesTransferred,',');
*psz++ = ' ';
// Fragment three
RtlCopyMemory( psz,
&pTracker->pUriEntry->pLogData[pLogData->UsedOffset1],
pLogData->UsedOffset2
);
psz += pLogData->UsedOffset2;
// Calculate the full size of the final log line
pLogData->Used += (ULONG) DIFF(psz - &pLogData->Line[0]);
// Reset the UsedOffset1 & 2 to zero to tell that the log line
// is not fragmented anymore but a complete line.
pLogData->UsedOffset1 = pLogData->UsedOffset2 = 0;
}
break;
default:
ASSERT(!"Unknown Log Format.\n");
}
}
//
// For cache hits we can get the corresponding cgroup from uri_cache
// entry to avoid the costly cgroup lookup. And also not to keep the
// entry around during the logging process we will have a direct ref
// to the actual cgroup. DestroyLogBuffer will release this refcount
//
if ( pTracker->pUriEntry->ConfigInfo.pLoggingConfig != NULL )
{
REFERENCE_CONFIG_GROUP(pTracker->pUriEntry->ConfigInfo.pLoggingConfig);
pLogData->pConfigGroup = pTracker->pUriEntry->ConfigInfo.pLoggingConfig;
}
pTracker->pLogData = NULL;
//
// We cannot allow send-response to resume parsing otherwise request buffers
// will be freed up. Therefore complete inline for the time being.
//
// UL_QUEUE_WORK_ITEM( &pLogData->WorkItem, &UlLogHttpCacheHitWorker);
UlLogHttpCacheHitWorker( &pLogData->WorkItem );
return Status;
}
/***************************************************************************++
Routine Description:
UlLogHttpCacheHitWorker :
Please read the description of UlLogHttpHit first.
Here where we log the cache hits. We use cache entry itself -
acquired from tracker - . Some fields are generated at this
time; i.e. timetaken,date,time...
Arguments:
PUL_WORK_ITEM : Will point us to the right pLogData
--***************************************************************************/
VOID
UlLogHttpCacheHitWorker(
IN PUL_WORK_ITEM pWorkItem
)
{
NTSTATUS Status;
PUL_INTERNAL_REQUEST pRequest;
PUL_LOG_DATA_BUFFER pLogData;
PCHAR psz;
PCHAR pBuffer;
LONGLONG LifeTime;
LARGE_INTEGER CurrentTimeStamp;
ULONG BytesWritten;
PUL_LOG_FILE_ENTRY pEntry;
ULONG Flags;
//
// A Lot of sanity checks.
//
PAGED_CODE();
pLogData = CONTAINING_RECORD(
pWorkItem,
UL_LOG_DATA_BUFFER,
WorkItem
);
ASSERT(pLogData);
UlTrace(LOGGING,("Ul!UlLogHttpCacheHitWorker: pLogData %p\n", pLogData));
pRequest = pLogData->pRequest;
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pRequest));
//
// Unlike the LogHttpHit function for non-cache hits we do not require
// the pResponse here. Tracker & pRequest already provides the info
// about how much data we sent & the time taken.
//
if (pLogData->pConfigGroup == NULL ||
IS_LOGGING_DISABLED(pLogData->pConfigGroup))
{
ASSERT(pLogData->pConfigGroup != NULL);
UlTrace( LOGGING,
("Http!UlLogHttpCacheHitWorker: Skipping pLogData->pConfigGroup %p\n",
pLogData->pConfigGroup
));
//
// If logging is disabled or log settings don't exist then do not
// proceed. Just exit out. But not before cleanup still goto end.
//
Status = STATUS_SUCCESS;
goto end;
}
ASSERT(IS_VALID_CONFIG_GROUP(pLogData->pConfigGroup));
//
// Construct the remaining fields before logging out this hit.
//
switch(pLogData->Format)
{
case HttpLoggingTypeW3C:
{
Flags = pLogData->Flags;
// First write the date & time fields to the beginning reserved
// space. Do not increment the used counter for date & time, b/c
// CaptureLogFields already did this when reserving the space.
psz = &pLogData->Line[0];
if ( Flags & MD_EXTLOG_DATE )
{
BytesWritten = 0;
UlpGetDateTimeFields(
HttpLoggingTypeW3C,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 10);
}
if ( Flags & MD_EXTLOG_TIME )
{
BytesWritten = 0;
UlpGetDateTimeFields(
HttpLoggingTypeW3C,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 8);
}
pBuffer = psz = &pLogData->Line[pLogData->Used];
// For pure cache hits we have to generate the few more fields
if ( pLogData->CacheAndSendResponse == FALSE )
{
// Capture log fields from Request alive
if ( Flags & MD_EXTLOG_USERNAME )
{
*psz++ = '-'; *psz++ = ' ';
}
if ( Flags & MD_EXTLOG_CLIENT_IP )
{
psz = UlStrPrintIP(
psz,
pRequest->pHttpConn->pConnection->RemoteAddress,
' '
);
}
if ( Flags & MD_EXTLOG_PROTOCOL_VERSION )
{
psz = UlStrPrintStr(
psz,
UL_GET_NAME_FOR_HTTP_VERSION(pRequest->Version),
' '
);
}
if ( Flags & MD_EXTLOG_USER_AGENT )
{
if (pRequest->HeaderValid[HttpHeaderUserAgent] &&
pRequest->Headers[HttpHeaderUserAgent].HeaderLength)
{
psz = UlStrPrintStrC(
psz,
(const CHAR *)pRequest->Headers[HttpHeaderUserAgent].pHeader,
' '
);
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_COOKIE )
{
if (pRequest->HeaderValid[HttpHeaderCookie] &&
pRequest->Headers[HttpHeaderCookie].HeaderLength)
{
psz = UlStrPrintStrC(
psz,
(const CHAR *)pRequest->Headers[HttpHeaderCookie].pHeader,
' '
);
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_REFERER )
{
if (pRequest->HeaderValid[HttpHeaderReferer] &&
pRequest->Headers[HttpHeaderReferer].HeaderLength)
{
psz = UlStrPrintStrC(
psz,
(const CHAR *)pRequest->Headers[HttpHeaderReferer].pHeader,
' '
);
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
if ( Flags & MD_EXTLOG_HOST )
{
if (pRequest->HeaderValid[HttpHeaderHost] &&
pRequest->Headers[HttpHeaderHost].HeaderLength)
{
psz = UlStrPrintStrC(
psz,
(const CHAR *)pRequest->Headers[HttpHeaderHost].pHeader,
' '
);
}
else
{
*psz++ = '-'; *psz++ = ' ';
}
}
}
// Now proceed with the remaining fields, but add them to the end.
if ( pLogData->Flags & MD_EXTLOG_BYTES_SENT )
{
psz = UlStrPrintUlonglong(psz, pLogData->BytesTransferred,' ');
}
if ( pLogData->Flags & MD_EXTLOG_BYTES_RECV )
{
psz = UlStrPrintUlonglong(psz, pRequest->BytesReceived,' ');
}
if ( pLogData->Flags & MD_EXTLOG_TIME_TAKEN )
{
KeQuerySystemTime( &CurrentTimeStamp );
LifeTime = CurrentTimeStamp.QuadPart - pRequest->TimeStamp.QuadPart;
if (LifeTime < 0)
{
LifeTime = 0;
UlTrace( LOGGING, ("CopyTimeStampField failure.\n"));
}
LifeTime /= (10*1000); // Conversion from 100-nanosecond to millisecs.
psz = UlStrPrintUlonglong(psz, (ULONGLONG)LifeTime,' ');
}
// Calculate the used space
pLogData->Used += (ULONG) DIFF(psz - pBuffer);
// Eat the last space and write the \r\n to the end.
// Only if we have any fields picked and written
if ( pLogData->Used )
{
psz = &pLogData->Line[pLogData->Used-1]; // Eat the last space
*psz++ = '\r'; *psz++ = '\n'; *psz++ = ANSI_NULL;
pLogData->Used += 1;
if ( pLogData->Length == 0 )
{
ASSERT( pLogData->Used <= UL_LOG_LINE_BUFFER_SIZE );
}
else
{
ASSERT( pLogData->Used <= pLogData->Length );
}
}
else
{
goto end; // No log fields nothing to log
}
// Cleanup the UsedOffsets
pLogData->UsedOffset1 = pLogData->UsedOffset2 = 0;
}
break;
case HttpLoggingTypeNCSA:
{
// [date:time GmtOffset] -> "[07/Jan/2000:00:02:23 -0800] "
// Restore the pointer to the reserved space first.
psz = &pLogData->Line[pLogData->UsedOffset1];
*psz++ = '[';
BytesWritten = 0;
UlpGetDateTimeFields(
HttpLoggingTypeNCSA,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ':';
ASSERT(BytesWritten == 11);
BytesWritten = 0;
UlpGetDateTimeFields(
HttpLoggingTypeNCSA,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ' ';
ASSERT(BytesWritten == 8);
UlAcquireResourceShared(&g_pUlNonpagedData->LogListResource, TRUE);
psz = UlStrPrintStr(psz, g_GMTOffset,']');
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
*psz++ = ' ';
ASSERT(((ULONG) DIFF(psz - &pLogData->Line[pLogData->UsedOffset1]) == 29));
// BytesSent
pBuffer = psz = &pLogData->Line[pLogData->Used];
psz = UlStrPrintUlonglong(psz, pLogData->BytesTransferred,'\r');
pLogData->Used += (ULONG) DIFF(psz - pBuffer);
// \n\0
*psz++ = '\n'; *psz++ = ANSI_NULL;
pLogData->Used += 1;
// Cleanup the UsedOffsets
pLogData->UsedOffset1 = pLogData->UsedOffset2 = 0;
}
break;
case HttpLoggingTypeIIS:
{
if ( pLogData->CacheAndSendResponse == TRUE )
{
// We need to work on the fragmented log line
// which is coming from the originaly allocated
// line but not from cache.
// Complete Fragment 1
pBuffer = psz = &pLogData->Line[pLogData->UsedOffset1];
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
psz,
&BytesWritten,
NULL,
NULL
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
UlpGetDateTimeFields(
HttpLoggingTypeIIS,
NULL,
NULL,
psz,
&BytesWritten
);
psz += BytesWritten; *psz++ = ','; *psz++ = ' ';
pLogData->UsedOffset1 += (USHORT) DIFF(psz - pBuffer);
// Complete Fragment 2
pBuffer = psz = &pLogData->Line[512 + pLogData->UsedOffset2];
KeQuerySystemTime( &CurrentTimeStamp );
LifeTime = CurrentTimeStamp.QuadPart - pRequest->TimeStamp.QuadPart;
if (LifeTime < 0)
{
LifeTime = 0;
UlTrace(LOGGING, ("CopyTimeStampField: failure.\n"));
}
LifeTime /= (10*1000); // Conversion from 100-nanosecond to millisecs.
psz = UlStrPrintUlonglong(psz, (ULONGLONG)LifeTime,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pRequest->BytesReceived,',');
*psz++ = ' ';
psz = UlStrPrintUlonglong(psz, pLogData->BytesTransferred,',');
*psz++ = ' ';
pLogData->UsedOffset2 += (USHORT) DIFF(psz - pBuffer);
// Size of the final log line is
// pLogData->UsedOffset1 + pLogData->UsedOffset2 + pLogData->Used
}
// Or else IIS log line is already done. We have completed it before
// relasing the cache entry.
}
break;
default:
{
ASSERT(!"Unknown Log Format Type\n");
Status = STATUS_INVALID_PARAMETER;
goto end;
}
}
//
// Finally this log line is ready to rock. Lets write it out.
//
UlAcquireResourceShared(&g_pUlNonpagedData->LogListResource, TRUE);
pEntry = pLogData->pConfigGroup->pLogFileEntry;
if (pEntry)
{
ASSERT(IS_VALID_LOG_FILE_ENTRY(pEntry));
Status =
UlpWriteToLogFile(
pEntry,
pLogData->UsedOffset1 + pLogData->UsedOffset2 + pLogData->Used,
&pLogData->Line[0],
pLogData->UsedOffset1,
pLogData->UsedOffset2
);
}
UlReleaseResource(&g_pUlNonpagedData->LogListResource);
end:
if (pLogData)
{
//
// Cleanup the references & the log buffer
//
UlDestroyLogDataBuffer(pLogData);
}
#if DBG
if (!NT_SUCCESS(Status))
{
UlTrace(LOGGING,("Http!UlLogHttpcacheHitWorker: failure 0x%08lx \n",
Status
));
}
#endif
return;
}
/***************************************************************************++
Routine Description:
Initializes the Log Date & Time Cache
--***************************************************************************/
VOID
UlpInitializeLogCache(
VOID
)
{
LARGE_INTEGER SystemTime;
ULONG LogType;
ExInitializeFastMutex( &g_LogCacheFastMutex);
KeQuerySystemTime(&SystemTime);
for ( LogType=0; LogType<HttpLoggingTypeMaximum; LogType++ )
{
UlpGenerateDateAndTimeFields( (HTTP_LOGGING_TYPE) LogType,
SystemTime,
g_UlDateTimeCache[LogType].Date,
&g_UlDateTimeCache[LogType].DateLength,
g_UlDateTimeCache[LogType].Time,
&g_UlDateTimeCache[LogType].TimeLength
);
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart = SystemTime.QuadPart;
}
}
/***************************************************************************++
Routine Description:
Generates all possible types of date/time fields from a LARGE_INTEGER.
Arguments:
CurrentTime: A 64 bit Time value to be converted.
--***************************************************************************/
VOID
UlpGenerateDateAndTimeFields(
IN HTTP_LOGGING_TYPE LogType,
IN LARGE_INTEGER CurrentTime,
OUT PCHAR pDate,
OUT PULONG pDateLength,
OUT PCHAR pTime,
OUT PULONG pTimeLength
)
{
TIME_FIELDS CurrentTimeFields;
LARGE_INTEGER CurrentTimeLoc;
TIME_FIELDS CurrentTimeFieldsLoc;
PCHAR psz;
LONG Length;
// This routine does touch to pageable memory if the default log buffer
// wasn't sufficent enough to hold log fields and get reallocated from
// paged pool. For this reason the date&time cache can not use SpinLocks.
PAGED_CODE();
ASSERT(LogType < HttpLoggingTypeMaximum);
RtlTimeToTimeFields( &CurrentTime, &CurrentTimeFields );
switch(LogType)
{
case HttpLoggingTypeW3C:
//
// Uses GMT with format as follows;
//
// 2000-01-31 00:12:23
//
if (pDate)
{
psz = pDate;
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Year, 4, '-' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Month,2, '-' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Day, 2, '\0');
*pDateLength = (ULONG) DIFF(psz - pDate);
}
if (pTime)
{
psz = pTime;
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Hour, 2, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Minute,2, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFields.Second,2, '\0');
*pTimeLength = (ULONG) DIFF(psz - pTime);
}
break;
case HttpLoggingTypeNCSA:
//
// Uses GMT Time with format as follows;
//
// 07/Jan/2000 00:02:23
//
ExSystemTimeToLocalTime( &CurrentTime, &CurrentTimeLoc );
RtlTimeToTimeFields( &CurrentTimeLoc, &CurrentTimeFieldsLoc );
if(pDate)
{
psz = pDate;
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Day, 2, '/' );
psz = UlStrPrintStr(psz, UL_GET_MONTH_AS_STR(CurrentTimeFieldsLoc.Month),'/');
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Year,4, '\0');
*pDateLength = (ULONG) DIFF(psz - pDate);
}
if(pTime)
{
psz = pTime;
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Hour, 2, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Minute,2, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Second,2, '\0');
*pTimeLength = (ULONG) DIFF(psz - pTime);
}
break;
case HttpLoggingTypeIIS:
//
// Uses LOCAL Time with format as follows;
// This should be localised if we can solve the problem.
//
// 1/31/2000 0:02:03
//
ExSystemTimeToLocalTime( &CurrentTime, &CurrentTimeLoc );
RtlTimeToTimeFields( &CurrentTimeLoc, &CurrentTimeFieldsLoc );
if (pDate)
{
psz = pDate;
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Month, 0, '/' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Day, 0, '/' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Year, 0, '\0');
*pDateLength = (ULONG) DIFF(psz - pDate);
}
if(pTime)
{
psz = pTime;
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Hour, 0, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Minute,2, ':' );
psz = UlStrPrintUlongPad(psz, CurrentTimeFieldsLoc.Second,2, '\0');
*pTimeLength = (ULONG) DIFF(psz - pTime);
}
break;
}
return;
}
/***************************************************************************++
Routine Description:
Generates a date header and updates cached value if required.
Caller should overwrite the terminating null by a space or comma.
Arguments:
Date and Time are optional. But one of them should be provided.
--***************************************************************************/
VOID
UlpGetDateTimeFields(
IN HTTP_LOGGING_TYPE LogType,
OUT PCHAR pDate,
OUT PULONG pDateLength,
OUT PCHAR pTime,
OUT PULONG pTimeLength
)
{
LARGE_INTEGER SystemTime;
LARGE_INTEGER CacheTime;
LONG Length;
LONGLONG Timediff;
PAGED_CODE();
ASSERT(LogType < HttpLoggingTypeMaximum);
ASSERT(pDate || pTime);
//
// Get the current time.
//
KeQuerySystemTime( &SystemTime );
CacheTime.QuadPart =
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart;
//
// Check the difference between the current time, and
// the cached time.
//
Timediff = SystemTime.QuadPart - CacheTime.QuadPart;
if (Timediff < ONE_SECOND)
{
//
// The cached date&time hasn't gone stale yet.We can copy.
// Force a barrier around reading the string into memory.
//
UL_READMOSTLY_READ_BARRIER();
if (pDate)
{
RtlCopyMemory( pDate,
g_UlDateTimeCache[LogType].Date,
g_UlDateTimeCache[LogType].DateLength
);
*pDateLength = g_UlDateTimeCache[LogType].DateLength;
}
if (pTime)
{
RtlCopyMemory( pTime,
g_UlDateTimeCache[LogType].Time,
g_UlDateTimeCache[LogType].TimeLength
);
*pTimeLength = g_UlDateTimeCache[LogType].TimeLength;
}
UL_READMOSTLY_READ_BARRIER();
//
// Get grab the cached time value again in case it has been changed.
// As you notice we do not have a lock around this part of the code.
//
if (CacheTime.QuadPart ==
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart)
{
//
// Value hasn't changed. We are all set.
//
return;
}
//
// Otherwise fall down and flush the cache, and then recopy.
//
}
//
// The cached date & time is stale. We need to update it.
//
ExAcquireFastMutex( &g_LogCacheFastMutex );
//
// Has someone else updated the time while we were blocked?
//
CacheTime.QuadPart =
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart;
Timediff = SystemTime.QuadPart - CacheTime.QuadPart;
if (Timediff >= ONE_SECOND)
{
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart = 0;
KeQuerySystemTime( &SystemTime );
UL_READMOSTLY_WRITE_BARRIER();
UlpGenerateDateAndTimeFields(
LogType,
SystemTime,
g_UlDateTimeCache[LogType].Date,
&g_UlDateTimeCache[LogType].DateLength,
g_UlDateTimeCache[LogType].Time,
&g_UlDateTimeCache[LogType].TimeLength
);
UL_READMOSTLY_WRITE_BARRIER();
g_UlDateTimeCache[LogType].LastSystemTime.QuadPart =
SystemTime.QuadPart;
}
//
// The time has been updated. Copy the new string into
// the caller's buffer.
//
if (pDate)
{
RtlCopyMemory( pDate,
g_UlDateTimeCache[LogType].Date,
g_UlDateTimeCache[LogType].DateLength
);
*pDateLength = g_UlDateTimeCache[LogType].DateLength;
}
if (pTime)
{
RtlCopyMemory( pTime,
g_UlDateTimeCache[LogType].Time,
g_UlDateTimeCache[LogType].TimeLength
);
*pTimeLength = g_UlDateTimeCache[LogType].TimeLength;
}
ExReleaseFastMutex( &g_LogCacheFastMutex );
return;
}