1891 lines
50 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
Copyright (c) 2000 Microsoft Corporation
Module Name:
worker.c
Abstract:
This module defines functions for the worker thread pool.
Author:
Gurdeep Singh Pall (gurdeep) Nov 13, 1997
Revision History:
lokeshs - extended/modified threadpool.
Rob Earhart (earhart) September 29, 2000
Split off from threads.c
Environment:
These routines are statically linked in the caller's executable
and are callable only from user mode. They make use of Nt system
services.
--*/
#include <ntos.h>
#include <ntrtl.h>
#include <wow64t.h>
#include "ntrtlp.h"
#include "threads.h"
// Worker Thread Pool
// ------------------
// Clients can submit functions to be executed by a worker thread. Threads are
// created if the work queue exceeds a threshold. Clients can request that the
// function be invoked in the context of a I/O thread. I/O worker threads
// can be used for initiating asynchronous I/O requests. They are not terminated if
// there are pending IO requests. Worker threads terminate if inactivity exceeds a
// threshold.
// Clients can also associate IO completion requests with the IO completion port
// waited upon by the non I/O worker threads. One should not post overlapped IO requests
// in worker threads.
ULONG StartedWorkerInitialization ; // Used for Worker thread startup synchronization
ULONG CompletedWorkerInitialization ; // Used to check if Worker thread pool is initialized
ULONG NumFutureWorkItems = 0 ; // Future work items (timers, waits, &c to exec in workers)
ULONG NumFutureIOWorkItems = 0 ; // Future IO work items (timers, waits, &c to exec in IO workers)
ULONG NumIOWorkerThreads ; // Count of IO Worker Threads alive
ULONG NumWorkerThreads ; // Count of Worker Threads alive
ULONG NumMinWorkerThreads ; // Min worker threads should be alive: 1 if ioCompletion used, else 0
ULONG NumIOWorkRequests ; // Count of IO Work Requests pending
ULONG NumLongIOWorkRequests ; // IO Worker threads executing long worker functions
ULONG NumWorkRequests ; // Count of Work Requests pending.
ULONG NumQueuedWorkRequests; // Count of work requests pending on IO completion
ULONG NumLongWorkRequests ; // Worker threads executing long worker functions
ULONG NumExecutingWorkerThreads ; // Worker threads currently executing worker functions
ULONG TotalExecutedWorkRequests ; // Total worker requests that were picked up
ULONG OldTotalExecutedWorkRequests ; // Total worker requests since last timeout.
HANDLE WorkerThreadTimerQueue = NULL ; // Timer queue used by worker threads
HANDLE WorkerThreadTimer = NULL ; // Timer used by worker threads
RTL_CRITICAL_SECTION WorkerTimerCriticalSection; // Synchronizes access to the worker timer
ULONG LastThreadCreationTickCount ; // Tick count at which the last thread was created
LIST_ENTRY IOWorkerThreads ; // List of IOWorkerThreads
PRTLP_IOWORKER_TCB PersistentIOTCB ; // ptr to TCB of persistest IO worker thread
HANDLE WorkerCompletionPort ; // Completion port used for queuing tasks to Worker threads
RTL_CRITICAL_SECTION WorkerCriticalSection ; // Exclusion used by worker threads
NTSTATUS
RtlpStartWorkerThread (
VOID
);
VOID
RtlpWorkerThreadCancelTimer(
VOID
)
{
if (! RtlTryEnterCriticalSection(&WorkerTimerCriticalSection)) {
//
// Either another thread is setting a timer, or clearing it.
// Either way, there's no reason for us to clear the timer --
// return immediately.
//
return;
}
__try {
if (WorkerThreadTimer) {
ASSERT(WorkerThreadTimerQueue);
RtlDeleteTimer(WorkerThreadTimerQueue,
WorkerThreadTimer,
NULL);
}
} __finally {
WorkerThreadTimer = NULL;
RtlLeaveCriticalSection(&WorkerTimerCriticalSection);
}
}
VOID
RtlpWorkerThreadTimerCallback(
PVOID Context,
BOOLEAN NotUsed
)
/*++
Routine Description:
This routine checks if new worker thread has to be created
Arguments:
None
Return Value:
None
--*/
{
IO_COMPLETION_BASIC_INFORMATION Info ;
BOOLEAN bCreateThread = FALSE ;
NTSTATUS Status ;
ULONG QueueLength, Threshold ;
Status = NtQueryIoCompletion(
WorkerCompletionPort,
IoCompletionBasicInformation,
&Info,
sizeof(Info),
NULL
) ;
if (!NT_SUCCESS(Status))
return ;
QueueLength = Info.Depth ;
if (!QueueLength) {
OldTotalExecutedWorkRequests = TotalExecutedWorkRequests ;
return ;
}
RtlEnterCriticalSection (&WorkerCriticalSection) ;
// if there are queued work items and no new work items have been scheduled
// in the last 30 seconds then create a new thread.
// this will take care of deadlocks.
// this will create a problem only if some thread is running for a long time
if (TotalExecutedWorkRequests == OldTotalExecutedWorkRequests) {
bCreateThread = TRUE ;
}
// if there are a lot of queued work items, then create a new thread
{
ULONG NumEffWorkerThreads = NumWorkerThreads > NumLongWorkRequests
? NumWorkerThreads - NumLongWorkRequests
: 0;
ULONG ShortWorkRequests ;
ULONG CapturedNumExecutingWorkerThreads;
ULONG ThreadCreationDampingTime = NumWorkerThreads < NEW_THREAD_THRESHOLD
? THREAD_CREATION_DAMPING_TIME1
: (NumWorkerThreads < 30
? THREAD_CREATION_DAMPING_TIME2
: (NumWorkerThreads << 13)); // *100ms
Threshold = (NumWorkerThreads < MAX_WORKER_THREADS
? (NumEffWorkerThreads < 7
? NumEffWorkerThreads*NumEffWorkerThreads
: ((NumEffWorkerThreads<40)
? NEW_THREAD_THRESHOLD * NumEffWorkerThreads
: NEW_THREAD_THRESHOLD2 * NumEffWorkerThreads))
: 0xffffffff) ;
CapturedNumExecutingWorkerThreads = NumExecutingWorkerThreads;
ShortWorkRequests = QueueLength + CapturedNumExecutingWorkerThreads > NumLongWorkRequests
? QueueLength + CapturedNumExecutingWorkerThreads - NumLongWorkRequests
: 0;
if (LastThreadCreationTickCount > NtGetTickCount())
LastThreadCreationTickCount = NtGetTickCount() ;
if (ShortWorkRequests > Threshold
&& (LastThreadCreationTickCount + ThreadCreationDampingTime
< NtGetTickCount()))
{
bCreateThread = TRUE ;
}
}
if (bCreateThread && NumWorkerThreads<MaxThreads) {
RtlpStartWorkerThread();
RtlpWorkerThreadCancelTimer();
}
OldTotalExecutedWorkRequests = TotalExecutedWorkRequests ;
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
}
NTSTATUS
RtlpWorkerThreadSetTimer(
VOID
)
{
NTSTATUS Status;
HANDLE NewTimerQueue;
HANDLE NewTimer;
Status = STATUS_SUCCESS;
if (! RtlTryEnterCriticalSection(&WorkerTimerCriticalSection)) {
//
// Either another thread is setting a timer, or clearing it.
// Either way, there's no reason for us to set the timer --
// return immediately.
//
return STATUS_SUCCESS;
}
__try {
if (! WorkerThreadTimerQueue) {
Status = RtlCreateTimerQueue(&NewTimerQueue);
if (! NT_SUCCESS(Status)) {
__leave;
}
WorkerThreadTimerQueue = NewTimerQueue;
}
ASSERT(WorkerThreadTimerQueue != NULL);
if (! WorkerThreadTimer) {
Status = RtlCreateTimer(
WorkerThreadTimerQueue,
&NewTimer,
RtlpWorkerThreadTimerCallback,
NULL,
60000,
60000,
WT_EXECUTEINTIMERTHREAD
) ;
if (! NT_SUCCESS(Status)) {
__leave;
}
WorkerThreadTimer = NewTimer;
}
} __finally {
RtlLeaveCriticalSection(&WorkerTimerCriticalSection);
}
return Status;
}
#if _MSC_FULL_VER >= 13008827
#pragma warning(push)
#pragma warning(disable:4715) // Not all control paths return (due to infinite loop)
#endif
LONG
RtlpWorkerThread (
PVOID Parameter
)
/*++
Routine Description:
All non I/O worker threads execute in this routine. Worker thread will try to
terminate when it has not serviced a request for
STARTING_WORKER_SLEEP_TIME +
STARTING_WORKER_SLEEP_TIME << 1 +
...
STARTING_WORKER_SLEEP_TIME << MAX_WORKER_SLEEP_TIME_EXPONENT
Arguments:
HandlePtr - Pointer to our handle.
N.B. This is closed by RtlpStartWorkerThread, but
we're still responsible for the memory.
Return Value:
--*/
{
NTSTATUS Status ;
PVOID WorkerProc ;
PVOID Context ;
IO_STATUS_BLOCK IoSb ;
ULONG SleepTime ;
LARGE_INTEGER TimeOut ;
ULONG Terminate ;
PVOID Overlapped ;
UNREFERENCED_PARAMETER(Parameter);
#if DBG
DbgPrintEx(DPFLTR_RTLTHREADPOOL_ID,
RTLP_THREADPOOL_TRACE_MASK,
"Starting worker thread\n");
#endif
// Set default sleep time for 40 seconds.
#define WORKER_IDLE_TIMEOUT 40000 // In Milliseconds
SleepTime = WORKER_IDLE_TIMEOUT ;
// Loop servicing I/O completion requests
for ( ; ; ) {
TimeOut.QuadPart = Int32x32To64( SleepTime, -10000 ) ;
Status = NtRemoveIoCompletion(
WorkerCompletionPort,
(PVOID) &WorkerProc,
&Overlapped,
&IoSb,
&TimeOut
) ;
if (Status == STATUS_SUCCESS) {
TotalExecutedWorkRequests ++ ;//interlocked op not req
InterlockedIncrement(&NumExecutingWorkerThreads) ;
// Call the work item.
// If IO APC, context1 contains number of IO bytes transferred, and context2
// contains the overlapped structure.
// If (IO)WorkerFunction, context1 contains the actual WorkerFunction to be
// executed and context2 contains the actual context
Context = (PVOID) IoSb.Information ;
RtlpApcCallout(WorkerProc,
IoSb.Status,
Context,
Overlapped);
SleepTime = WORKER_IDLE_TIMEOUT ;
InterlockedDecrement(&NumExecutingWorkerThreads) ;
RtlpWorkerThreadCancelTimer();
} else if (Status == STATUS_TIMEOUT) {
// NtRemoveIoCompletion timed out. Check to see if have hit our limit
// on waiting. If so terminate.
Terminate = FALSE ;
RtlEnterCriticalSection (&WorkerCriticalSection) ;
// The thread terminates if there are > 1 threads and the queue is small
// OR if there is only 1 thread and there is no request pending
if (NumWorkerThreads > 1) {
ULONG NumEffWorkerThreads = NumWorkerThreads > NumLongWorkRequests
? NumWorkerThreads - NumLongWorkRequests
: 0;
if (NumEffWorkerThreads<=NumMinWorkerThreads) {
Terminate = FALSE ;
} else {
//
// have been idle for very long time. terminate irrespective of number of
// work items. (This is useful when the set of runnable threads is taking
// care of all the work items being queued). dont terminate if
// (NumEffWorkerThreads == 1)
//
if (NumEffWorkerThreads > 1) {
Terminate = TRUE ;
} else {
Terminate = FALSE ;
}
}
} else {
if ( NumMinWorkerThreads == 0
&& NumWorkRequests == 0
&& NumFutureWorkItems == 0) {
Terminate = TRUE ;
} else {
Terminate = FALSE ;
}
}
if (Terminate) {
THREAD_BASIC_INFORMATION ThreadInfo;
ULONG IsIoPending ;
Terminate = FALSE ;
Status = NtQueryInformationThread( NtCurrentThread(),
ThreadIsIoPending,
&IsIoPending,
sizeof( IsIoPending ),
NULL
);
if (NT_SUCCESS( Status )) {
if (! IsIoPending )
Terminate = TRUE ;
}
}
if (Terminate) {
ASSERT(NumWorkerThreads > 0);
NumWorkerThreads--;
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
RtlpExitThreadFunc( 0 );
} else {
// This is the condition where a request was queued *after* the
// thread woke up - ready to terminate because of inactivity. In
// this case dont terminate - service the completion port.
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
}
} else {
ASSERTMSG ("NtRemoveIoCompletion failed",
(Status != STATUS_SUCCESS) && (Status != STATUS_TIMEOUT)) ;
}
}
return 1 ;
}
#if _MSC_FULL_VER >= 13008827
#pragma warning(pop)
#endif
NTSTATUS
RtlpStartWorkerThread (
VOID
)
/*++
Routine Description:
This routine starts a regular worker thread
Arguments:
Return Value:
NTSTATUS error codes resulting from attempts to create a thread
STATUS_SUCCESS
--*/
{
HANDLE ThreadHandle;
ULONG CurrentTickCount;
NTSTATUS Status;
// Create worker thread
Status = RtlpStartThreadpoolThread (RtlpWorkerThread,
NULL,
&ThreadHandle);
if (NT_SUCCESS(Status)) {
#if DBG
DbgPrintEx(DPFLTR_RTLTHREADPOOL_ID,
RTLP_THREADPOOL_TRACE_MASK,
"Created worker thread; handle %d (closing)\n",
ThreadHandle);
#endif
// We don't care about worker threads' handles.
NtClose(ThreadHandle);
// Update the time at which the current thread was created
LastThreadCreationTickCount = NtGetTickCount() ;
// Increment the count of the thread type created
NumWorkerThreads++;
} else {
#if DBG
DbgPrintEx(DPFLTR_RTLTHREADPOOL_ID,
RTLP_THREADPOOL_TRACE_MASK,
"Failed to create worker thread; status %p\n",
Status);
#endif
// Thread creation failed. If there is even one thread present do not return
// failure - else queue the request anyway.
if (NumWorkerThreads <= NumLongWorkRequests) {
return Status ;
}
}
return STATUS_SUCCESS ;
}
NTSTATUS
RtlpInitializeWorkerThreadPool (
)
/*++
Routine Description:
This routine initializes all aspects of the thread pool.
Arguments:
None
Return Value:
None
--*/
{
NTSTATUS Status = STATUS_SUCCESS ;
LARGE_INTEGER TimeOut ;
SYSTEM_BASIC_INFORMATION BasicInfo;
ASSERT(! RtlIsImpersonating());
// Initialize the timer component if it hasnt been done already
if (CompletedTimerInitialization != 1) {
Status = RtlpInitializeTimerThreadPool () ;
if ( !NT_SUCCESS(Status) )
return Status ;
}
// In order to avoid an explicit RtlInitialize() function to initialize the thread pool
// we use StartedInitialization and CompletedInitialization to provide us the necessary
// synchronization to avoid multiple threads from initializing the thread pool.
// This scheme does not work if RtlInitializeCriticalSection() fails - but in this case the
// caller has no choices left.
if (!InterlockedExchange(&StartedWorkerInitialization, 1L)) {
if (CompletedWorkerInitialization)
InterlockedExchange( &CompletedWorkerInitialization, 0 ) ;
do {
// Initialize Critical Sections
Status = RtlInitializeCriticalSection( &WorkerCriticalSection );
if (!NT_SUCCESS(Status))
break ;
Status = RtlInitializeCriticalSection( &WorkerTimerCriticalSection );
if (! NT_SUCCESS(Status)) {
RtlDeleteCriticalSection( &WorkerCriticalSection );
break;
}
InitializeListHead (&IOWorkerThreads) ;
// get number of processors
Status = NtQuerySystemInformation (
SystemBasicInformation,
&BasicInfo,
sizeof(BasicInfo),
NULL
) ;
if ( !NT_SUCCESS(Status) ) {
BasicInfo.NumberOfProcessors = 1 ;
}
// Create completion port used by worker threads
Status = NtCreateIoCompletion (
&WorkerCompletionPort,
IO_COMPLETION_ALL_ACCESS,
NULL,
BasicInfo.NumberOfProcessors
);
if (!NT_SUCCESS(Status)) {
RtlDeleteCriticalSection( &WorkerCriticalSection );
RtlDeleteCriticalSection( &WorkerTimerCriticalSection );
break;
}
} while ( FALSE ) ;
if (!NT_SUCCESS(Status) ) {
StartedWorkerInitialization = 0 ;
InterlockedExchange( &CompletedWorkerInitialization, ~0 ) ;
return Status ;
}
// Signal that initialization has completed
InterlockedExchange (&CompletedWorkerInitialization, 1L) ;
} else {
LARGE_INTEGER Timeout ;
// Sleep 1 ms and see if the other thread has completed initialization
ONE_MILLISECOND_TIMEOUT(TimeOut) ;
while (!*((ULONG volatile *)&CompletedWorkerInitialization)) {
NtDelayExecution (FALSE, &TimeOut) ;
}
if (CompletedWorkerInitialization != 1)
return STATUS_NO_MEMORY ;
}
return NT_SUCCESS(Status) ? STATUS_SUCCESS : Status ;
}
LONG
RtlpIOWorkerThread (
PVOID Parameter
)
/*++
Routine Description:
All I/O worker threads execute in this routine. All the work requests execute as APCs
in this thread.
Arguments:
HandlePtr - Pointer to our handle.
Return Value:
--*/
{
#define IOWORKER_IDLE_TIMEOUT 40000 // In Milliseconds
LARGE_INTEGER TimeOut ;
ULONG SleepTime = IOWORKER_IDLE_TIMEOUT ;
PRTLP_IOWORKER_TCB ThreadCB ; // Control Block allocated on the
// heap by the parent thread
NTSTATUS Status ;
BOOLEAN Terminate ;
ASSERT(Parameter != NULL);
ThreadCB = (PRTLP_IOWORKER_TCB) Parameter;
#if DBG
DbgPrintEx(DPFLTR_RTLTHREADPOOL_ID,
RTLP_THREADPOOL_TRACE_MASK,
"Starting IO worker thread\n");
#endif
// Sleep alertably so that all the activity can take place
// in APCs
for ( ; ; ) {
// Set timeout for IdleTimeout
TimeOut.QuadPart = Int32x32To64( SleepTime, -10000 ) ;
Status = NtDelayExecution (TRUE, &TimeOut) ;
// Status is STATUS_SUCCESS only when it has timed out
if (Status != STATUS_SUCCESS) {
continue ;
}
//
// idle timeout. check if you can terminate the thread
//
Terminate = FALSE ;
RtlEnterCriticalSection (&WorkerCriticalSection) ;
// dont terminate if it is a persistent thread
if (ThreadCB->Flags & WT_EXECUTEINPERSISTENTIOTHREAD) {
TimeOut.LowPart = 0x0;
TimeOut.HighPart = 0x80000000;
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
continue ;
}
// The thread terminates if there are > 1 threads and the queue is small
// OR if there is only 1 thread and there is no request pending
if (NumIOWorkerThreads > 1) {
ULONG NumEffIOWorkerThreads = NumIOWorkerThreads > NumLongIOWorkRequests
? NumIOWorkerThreads - NumLongIOWorkRequests
: 0;
ULONG Threshold;
if (NumEffIOWorkerThreads == 0) {
Terminate = FALSE ;
} else {
// Check if we need to shrink worker thread pool
Threshold = NEW_THREAD_THRESHOLD * (NumEffIOWorkerThreads-1);
if (NumIOWorkRequests-NumLongIOWorkRequests < Threshold) {
Terminate = TRUE ;
} else {
Terminate = FALSE ;
SleepTime <<= 1 ;
}
}
} else {
if (NumIOWorkRequests == 0
&& NumFutureIOWorkItems == 0) {
// delay termination of last thread
if (SleepTime < 4*IOWORKER_IDLE_TIMEOUT) {
SleepTime <<= 1 ;
Terminate = FALSE ;
} else {
Terminate = TRUE ;
}
} else {
Terminate = FALSE ;
}
}
//
// terminate only if no io is pending
//
if (Terminate) {
NTSTATUS xStatus;
THREAD_BASIC_INFORMATION ThreadInfo;
ULONG IsIoPending ;
Terminate = FALSE ;
xStatus = NtQueryInformationThread( ThreadCB->ThreadHandle,
ThreadIsIoPending,
&IsIoPending,
sizeof( IsIoPending ),
NULL
);
if (NT_SUCCESS( xStatus )) {
if (! IsIoPending )
Terminate = TRUE ;
}
}
if (Terminate) {
ASSERT(NumIOWorkerThreads > 0);
NumIOWorkerThreads--;
RemoveEntryList (&ThreadCB->List) ;
NtClose( ThreadCB->ThreadHandle ) ;
RtlpFreeTPHeap( ThreadCB );
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
RtlpExitThreadFunc( 0 );
} else {
// This is the condition where a request was queued *after* the
// thread woke up - ready to terminate because of inactivity. In
// this case dont terminate - service the completion port.
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
}
}
return 0 ; // Keep compiler happy
}
NTSTATUS
RtlpStartIOWorkerThread (
)
/*++
Routine Description:
This routine starts an I/O worker thread
N.B. Callers MUST hold the WorkerCriticalSection.
Arguments:
Return Value:
NTSTATUS error codes resulting from attempts to create a thread
STATUS_SUCCESS
--*/
{
ULONG CurrentTickCount ;
NTSTATUS Status ;
PRTLP_IOWORKER_TCB ThreadCB;
// Create the worker's control block
ThreadCB = (PRTLP_IOWORKER_TCB) RtlpAllocateTPHeap(sizeof(RTLP_IOWORKER_TCB), 0);
if (! ThreadCB) {
return STATUS_NO_MEMORY;
}
// Fill in the control block
ThreadCB->Flags = 0;
ThreadCB->LongFunctionFlag = FALSE;
// Create worker thread
Status = RtlpStartThreadpoolThread (RtlpIOWorkerThread,
ThreadCB,
&ThreadCB->ThreadHandle);
if (NT_SUCCESS(Status)) {
// Update the time at which the current thread was created,
// and insert the ThreadCB into the IO worker thread list.
LastThreadCreationTickCount = NtGetTickCount() ;
NumIOWorkerThreads++;
InsertHeadList(&IOWorkerThreads, &ThreadCB->List);
} else {
// Thread creation failed.
RtlpFreeTPHeap(ThreadCB);
// If there is even one thread present do not return
// failure since we can still service the work request.
if (NumIOWorkerThreads <= NumLongIOWorkRequests) {
return Status ;
}
}
return STATUS_SUCCESS ;
}
VOID
RtlpExecuteLongIOWorkItem (
PVOID WorkEntryPtr,
PVOID Context,
PVOID ThreadCB
)
/*++
Routine Description:
Executes an IO Work function. RUNs in a APC in the IO Worker thread.
Arguments:
WorkEntryPtr - Work entry to run
Context - Context for the work entry
ThreadCB - The thread running this APC
Return Value:
None.
--*/
{
PRTLP_WORK WorkEntry = (PRTLP_WORK) WorkEntryPtr;
RtlpWorkerCallout(WorkEntry->Function,
Context,
WorkEntry->ActivationContext,
WorkEntry->ImpersonationToken);
((PRTLP_IOWORKER_TCB)ThreadCB)->LongFunctionFlag = FALSE ;
RtlEnterCriticalSection(&WorkerCriticalSection);
// Decrement pending IO requests count
NumIOWorkRequests--;
// decrement pending long funcitons
NumLongIOWorkRequests--;
RtlLeaveCriticalSection(&WorkerCriticalSection);
if (WorkEntry->ActivationContext != INVALID_ACTIVATION_CONTEXT) {
RtlReleaseActivationContext(WorkEntry->ActivationContext);
}
if (WorkEntry->ImpersonationToken) {
NtClose(WorkEntry->ImpersonationToken);
}
RtlpFreeTPHeap(WorkEntry);
}
VOID
RtlpExecuteIOWorkItem (
PVOID WorkEntryPtr,
PVOID Context,
PVOID Parameter
)
/*++
Routine Description:
Executes an IO Work function. RUNs in a APC in the IO Worker thread.
Arguments:
WorkEntryPtr - Work entry to run
Context - Context for the work entry
Parameter - Argument is not used in this function.
Return Value:
--*/
{
PRTLP_WORK WorkEntry = (PRTLP_WORK) WorkEntryPtr;
UNREFERENCED_PARAMETER(Parameter);
RtlpWorkerCallout(WorkEntry->Function,
Context,
WorkEntry->ActivationContext,
WorkEntry->ImpersonationToken);
RtlEnterCriticalSection(&WorkerCriticalSection);
// Decrement pending IO requests count
NumIOWorkRequests--;
RtlLeaveCriticalSection(&WorkerCriticalSection);
if (WorkEntry->ActivationContext != INVALID_ACTIVATION_CONTEXT) {
RtlReleaseActivationContext(WorkEntry->ActivationContext);
}
if (WorkEntry->ImpersonationToken) {
NtClose(WorkEntry->ImpersonationToken);
}
RtlpFreeTPHeap(WorkEntry);
}
NTSTATUS
RtlpQueueIOWorkerRequest (
WORKERCALLBACKFUNC Function,
PVOID Context,
ULONG Flags,
HANDLE Token
)
/*++
Routine Description:
This routine queues up the request to be executed in an IO worker thread.
Arguments:
Function - Routine that is called by the worker thread
Context - Opaque pointer passed in as an argument to WorkerProc
Flags - Flags passed to RtlQueueWorkItem
Token - Impersonation token to use
Return Value:
--*/
{
NTSTATUS Status ;
PRTLP_IOWORKER_TCB TCB ;
BOOLEAN LongFunction = (Flags & WT_EXECUTELONGFUNCTION) ? TRUE : FALSE ;
PLIST_ENTRY ple ;
PRTLP_WORK WorkEntry ;
WorkEntry = (PRTLP_WORK) RtlpAllocateTPHeap(sizeof(RTLP_WORK),
HEAP_ZERO_MEMORY);
if (! WorkEntry) {
return STATUS_NO_MEMORY;
}
WorkEntry->Function = Function;
WorkEntry->Flags = Flags;
Status = RtlpThreadPoolGetActiveActivationContext(&WorkEntry->ActivationContext);
if (!NT_SUCCESS(Status)) {
if (Status == STATUS_SXS_THREAD_QUERIES_DISABLED) {
WorkEntry->ActivationContext = INVALID_ACTIVATION_CONTEXT;
Status = STATUS_SUCCESS;
} else {
goto cleanup_workentry;
}
}
if (Token) {
Status = NtDuplicateToken(Token,
TOKEN_IMPERSONATE,
NULL,
FALSE,
TokenImpersonation,
&WorkEntry->ImpersonationToken);
if (! NT_SUCCESS(Status)) {
goto cleanup_actctx;
}
} else {
WorkEntry->ImpersonationToken = NULL;
}
if (Flags & WT_EXECUTEINPERSISTENTIOTHREAD) {
if (!PersistentIOTCB) {
for (ple=IOWorkerThreads.Flink; ple!=&IOWorkerThreads; ple=ple->Flink) {
TCB = CONTAINING_RECORD (ple, RTLP_IOWORKER_TCB, List) ;
if (! TCB->LongFunctionFlag)
break;
}
if (ple == &IOWorkerThreads) {
Status = STATUS_NO_MEMORY;
goto cleanup_token;
}
PersistentIOTCB = TCB ;
TCB->Flags |= WT_EXECUTEINPERSISTENTIOTHREAD ;
} else {
TCB = PersistentIOTCB ;
}
} else {
for (ple=IOWorkerThreads.Flink; ple!=&IOWorkerThreads; ple=ple->Flink) {
TCB = CONTAINING_RECORD (ple, RTLP_IOWORKER_TCB, List) ;
// do not queue to the thread if it is executing a long function, or
// if you are queueing a long function and the thread is a persistent thread
if (! TCB->LongFunctionFlag
&& (! ((TCB->Flags&WT_EXECUTEINPERSISTENTIOTHREAD)
&& (Flags&WT_EXECUTELONGFUNCTION)))) {
break ;
}
}
if ((ple == &IOWorkerThreads) && (NumIOWorkerThreads<1)) {
#if DBG
DbgPrintEx(DPFLTR_RTLTHREADPOOL_ID,
RTLP_THREADPOOL_WARNING_MASK,
"Out of memory. "
"Could not execute IOWorkItem(%x)\n", (ULONG_PTR)Function);
#endif
Status = STATUS_NO_MEMORY;
goto cleanup_token;
}
else {
ple = IOWorkerThreads.Flink;
TCB = CONTAINING_RECORD (ple, RTLP_IOWORKER_TCB, List) ;
// treat it as a short function so that counters work fine.
LongFunction = FALSE;
}
// In order to implement "fair" assignment of work items between IO worker threads
// each time remove the entry and reinsert at back.
RemoveEntryList (&TCB->List) ;
InsertTailList (&IOWorkerThreads, &TCB->List) ;
}
// Increment the outstanding work request counter
NumIOWorkRequests++;
if (LongFunction) {
NumLongIOWorkRequests++;
TCB->LongFunctionFlag = TRUE ;
}
// Queue an APC to the IoWorker Thread
Status = NtQueueApcThread(
TCB->ThreadHandle,
LongFunction? (PPS_APC_ROUTINE)RtlpExecuteLongIOWorkItem:
(PPS_APC_ROUTINE)RtlpExecuteIOWorkItem,
(PVOID)WorkEntry,
Context,
TCB
);
if (! NT_SUCCESS( Status ) ) {
goto cleanup_counters;
}
return STATUS_SUCCESS;
cleanup_counters:
NumIOWorkRequests--;
if (LongFunction)
NumLongIOWorkRequests--;
cleanup_token:
if (WorkEntry->ImpersonationToken) {
NtClose(WorkEntry->ImpersonationToken);
}
cleanup_actctx:
if (WorkEntry->ActivationContext != INVALID_ACTIVATION_CONTEXT) {
RtlReleaseActivationContext(WorkEntry->ActivationContext);
}
cleanup_workentry:
RtlpFreeTPHeap(WorkEntry);
return Status;
}
NTSTATUS
RtlSetIoCompletionCallback (
IN HANDLE FileHandle,
IN APC_CALLBACK_FUNCTION CompletionProc,
IN ULONG Flags
)
/*++
Routine Description:
This routine binds an Handle and an associated callback function to the
IoCompletionPort which queues work items to worker threads.
Arguments:
Handle - handle to be bound to the IO completion port
CompletionProc - callback function to be executed when an IO request
pending on the IO handle completes.
Flags - Reserved. pass 0.
--*/
{
IO_STATUS_BLOCK IoSb ;
FILE_COMPLETION_INFORMATION CompletionInfo ;
NTSTATUS Status;
HANDLE Token = NULL;
if (LdrpShutdownInProgress) {
return STATUS_UNSUCCESSFUL;
}
if (Flags) {
return STATUS_INVALID_PARAMETER_3;
}
Status = RtlpCaptureImpersonation(FALSE, &Token);
if (! NT_SUCCESS(Status)) {
return Status;
}
// Make sure that the worker thread pool is initialized as the file handle
// is bound to IO completion port.
if (CompletedWorkerInitialization != 1) {
Status = RtlpInitializeWorkerThreadPool () ;
if (! NT_SUCCESS(Status) ) {
goto cleanup;
}
}
//
// from now on NumMinWorkerThreads should be 1. If there is only 1 worker thread
// create a new one.
//
if ( NumMinWorkerThreads == 0 ) {
// Take lock for the global worker thread pool
RtlEnterCriticalSection (&WorkerCriticalSection) ;
if ((NumWorkerThreads-NumLongWorkRequests) == 0) {
Status = RtlpStartWorkerThread () ;
if ( ! NT_SUCCESS(Status) ) {
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
goto cleanup;
}
}
// from now on, there will be at least 1 worker thread
NumMinWorkerThreads = 1 ;
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
}
// bind to IoCompletionPort, which queues work items to worker threads
CompletionInfo.Port = WorkerCompletionPort ;
CompletionInfo.Key = (PVOID) CompletionProc ;
Status = NtSetInformationFile (
FileHandle,
&IoSb, //not initialized
&CompletionInfo,
sizeof(CompletionInfo),
FileCompletionInformation //enum flag
) ;
cleanup:
if (Token) {
RtlpRestartImpersonation(Token);
NtClose(Token);
}
return Status ;
}
VOID
RtlpExecuteWorkerRequest (
NTSTATUS StatusIn, //not used
PVOID Context,
PVOID WorkContext
)
/*++
Routine Description:
This routine executes a work item.
Arguments:
Context - contains context to be passed to the callback function.
WorkContext - contains callback function ptr and flags
Return Value:
Notes:
This function executes in a worker thread or a timer thread if
WT_EXECUTEINTIMERTHREAD flag is set.
--*/
{
PRTLP_WORK WorkEntry = (PRTLP_WORK) WorkContext;
NTSTATUS Status;
if (! (WorkEntry->Flags & WT_EXECUTEINPERSISTENTTHREAD)
&& InterlockedDecrement(&NumQueuedWorkRequests)
&& NumExecutingWorkerThreads == NumWorkerThreads) {
RtlpWorkerThreadSetTimer();
}
RtlpWorkerCallout(WorkEntry->Function,
Context,
WorkEntry->ActivationContext,
WorkEntry->ImpersonationToken);
RtlEnterCriticalSection(&WorkerCriticalSection);
NumWorkRequests--;
if (WorkEntry->Flags & WT_EXECUTELONGFUNCTION) {
NumLongWorkRequests--;
}
RtlLeaveCriticalSection(&WorkerCriticalSection);
if (WorkEntry->ActivationContext != INVALID_ACTIVATION_CONTEXT)
RtlReleaseActivationContext(WorkEntry->ActivationContext);
if (WorkEntry->ImpersonationToken) {
NtClose(WorkEntry->ImpersonationToken);
}
RtlpFreeTPHeap( WorkEntry ) ;
}
NTSTATUS
RtlpQueueWorkerRequest (
WORKERCALLBACKFUNC Function,
PVOID Context,
ULONG Flags,
HANDLE Token
)
/*++
Routine Description:
This routine queues up the request to be executed in a worker thread.
Arguments:
Function - Routine that is called by the worker thread
Context - Opaque pointer passed in as an argument to WorkerProc
Flags - Flags passed to RtlQueueWorkItem
Token - Impersonation token to use
Return Value:
--*/
{
NTSTATUS Status ;
PRTLP_WORK WorkEntry ;
WorkEntry = (PRTLP_WORK) RtlpAllocateTPHeap ( sizeof (RTLP_WORK),
HEAP_ZERO_MEMORY) ;
if (! WorkEntry) {
return STATUS_NO_MEMORY;
}
Status = RtlpThreadPoolGetActiveActivationContext(&WorkEntry->ActivationContext);
if (!NT_SUCCESS(Status)) {
if (Status == STATUS_SXS_THREAD_QUERIES_DISABLED) {
WorkEntry->ActivationContext = INVALID_ACTIVATION_CONTEXT;
Status = STATUS_SUCCESS;
} else {
goto cleanup_workentry;
}
}
if (Token) {
Status = NtDuplicateToken(Token,
TOKEN_IMPERSONATE,
NULL,
FALSE,
TokenImpersonation,
&WorkEntry->ImpersonationToken);
if (! NT_SUCCESS(Status)) {
goto cleanup_actctx;
}
} else {
WorkEntry->ImpersonationToken = NULL;
}
// Increment the outstanding work request counter
NumWorkRequests++;
if (Flags & WT_EXECUTELONGFUNCTION) {
NumLongWorkRequests++;
}
WorkEntry->Function = Function ;
WorkEntry->Flags = Flags ;
if (Flags & WT_EXECUTEINPERSISTENTTHREAD) {
// Queue APC to timer thread
Status = NtQueueApcThread(
TimerThreadHandle,
(PPS_APC_ROUTINE)RtlpExecuteWorkerRequest,
(PVOID) STATUS_SUCCESS,
(PVOID) Context,
(PVOID) WorkEntry
) ;
} else {
InterlockedIncrement(&NumQueuedWorkRequests);
Status = NtSetIoCompletion (
WorkerCompletionPort,
RtlpExecuteWorkerRequest,
(PVOID) WorkEntry,
STATUS_SUCCESS,
(ULONG_PTR)Context
);
if (! NT_SUCCESS(Status)) {
InterlockedDecrement(&NumQueuedWorkRequests);
} else {
if (NumExecutingWorkerThreads == NumWorkerThreads) {
RtlpWorkerThreadSetTimer();
}
}
}
if ( ! NT_SUCCESS(Status) ) {
goto cleanup_counters;
}
return STATUS_SUCCESS;
cleanup_counters:
NumWorkRequests--;
if (Flags & WT_EXECUTELONGFUNCTION) {
NumLongWorkRequests--;
}
if (WorkEntry->ImpersonationToken) {
NtClose(WorkEntry->ImpersonationToken);
}
cleanup_actctx:
if (WorkEntry->ActivationContext != INVALID_ACTIVATION_CONTEXT) {
RtlReleaseActivationContext(WorkEntry->ActivationContext);
}
cleanup_workentry:
RtlpFreeTPHeap( WorkEntry ) ;
return Status;
}
NTSTATUS
RtlQueueWorkItem(
IN WORKERCALLBACKFUNC Function,
IN PVOID Context,
IN ULONG Flags
)
/*++
Routine Description:
This routine queues up the request to be executed in a worker thread.
Arguments:
Function - Routine that is called by the worker thread
Context - Opaque pointer passed in as an argument to WorkerProc
Flags - Can be:
WT_EXECUTEINIOTHREAD - Specifies that the WorkerProc should be invoked
by a thread that is never destroyed when there are pending IO requests.
This can be used by threads that invoke I/O and/or schedule APCs.
The below flag can also be set:
WT_EXECUTELONGFUNCTION - Specifies that the function might block for a
long duration.
Return Value:
STATUS_SUCCESS - Queued successfully.
STATUS_NO_MEMORY - There was not sufficient heap to perform the
requested operation.
--*/
{
ULONG Threshold ;
ULONG CurrentTickCount ;
NTSTATUS Status = STATUS_SUCCESS ;
HANDLE Token = NULL;
HANDLE RequestToken;
if (LdrpShutdownInProgress) {
return STATUS_UNSUCCESSFUL;
}
Status = RtlpCaptureImpersonation(Flags & WT_TRANSFER_IMPERSONATION,
&Token);
if (! NT_SUCCESS(Status)) {
return Status;
}
if (Flags & WT_TRANSFER_IMPERSONATION) {
RequestToken = Token;
} else {
RequestToken = NULL;
}
// Make sure the worker thread pool is initialized
if (CompletedWorkerInitialization != 1) {
Status = RtlpInitializeWorkerThreadPool () ;
if (! NT_SUCCESS(Status) ) {
goto cleanup;
}
}
// Take lock for the global worker thread pool
RtlEnterCriticalSection (&WorkerCriticalSection) ;
if (Flags&0xffff0000) {
MaxThreads = (Flags & 0xffff0000)>>16;
}
if (NEEDS_IO_THREAD(Flags)) {
//
// execute in IO Worker thread
//
ULONG NumEffIOWorkerThreads = NumIOWorkerThreads > NumLongIOWorkRequests
? NumIOWorkerThreads - NumLongIOWorkRequests
: 0;
ULONG ThreadCreationDampingTime = NumIOWorkerThreads < NEW_THREAD_THRESHOLD
? THREAD_CREATION_DAMPING_TIME1
: THREAD_CREATION_DAMPING_TIME2 ;
if (NumEffIOWorkerThreads && PersistentIOTCB && (Flags&WT_EXECUTELONGFUNCTION))
NumEffIOWorkerThreads -- ;
// Check if we need to grow I/O worker thread pool
Threshold = (NumEffIOWorkerThreads < MAX_WORKER_THREADS
? NEW_THREAD_THRESHOLD * NumEffIOWorkerThreads
: 0xffffffff) ;
if (LastThreadCreationTickCount > NtGetTickCount())
LastThreadCreationTickCount = NtGetTickCount() ;
if (NumEffIOWorkerThreads == 0
|| ((NumIOWorkRequests - NumLongIOWorkRequests > Threshold)
&& (LastThreadCreationTickCount + ThreadCreationDampingTime
< NtGetTickCount()))) {
// Grow the IO worker thread pool
Status = RtlpStartIOWorkerThread () ;
}
if (NT_SUCCESS(Status)) {
// Queue the work request
Status = RtlpQueueIOWorkerRequest(Function,
Context,
Flags,
RequestToken);
}
} else {
//
// execute in regular worker thread
//
ULONG NumEffWorkerThreads = NumWorkerThreads > NumLongWorkRequests
? NumWorkerThreads - NumLongWorkRequests
: 0;
ULONG ThreadCreationDampingTime = NumWorkerThreads < NEW_THREAD_THRESHOLD
? THREAD_CREATION_DAMPING_TIME1
: (NumWorkerThreads < 30
? THREAD_CREATION_DAMPING_TIME2
: NumWorkerThreads << 13);
// if io completion set, then have 1 more thread
if (NumMinWorkerThreads && NumEffWorkerThreads)
NumEffWorkerThreads -- ;
// Check if we need to grow worker thread pool
Threshold = (NumWorkerThreads < MAX_WORKER_THREADS
? (NumEffWorkerThreads < 7
? NumEffWorkerThreads*NumEffWorkerThreads
: ((NumEffWorkerThreads<40)
? NEW_THREAD_THRESHOLD * NumEffWorkerThreads
: NEW_THREAD_THRESHOLD2 * NumEffWorkerThreads))
: 0xffffffff) ;
if (LastThreadCreationTickCount > NtGetTickCount())
LastThreadCreationTickCount = NtGetTickCount() ;
if (NumEffWorkerThreads == 0 ||
( (NumWorkRequests - NumLongWorkRequests >= Threshold)
&& (LastThreadCreationTickCount + ThreadCreationDampingTime
< NtGetTickCount())))
{
// Grow the worker thread pool
if (NumWorkerThreads<MaxThreads) {
Status = RtlpStartWorkerThread () ;
}
}
// Queue the work request
if (NT_SUCCESS(Status)) {
Status = RtlpQueueWorkerRequest(Function,
Context,
Flags,
RequestToken);
}
}
// Release lock on the worker thread pool
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
cleanup:
if (Token) {
RtlpRestartImpersonation(Token);
NtClose(Token);
}
return Status ;
}
NTSTATUS
RtlpWorkerCleanup(
VOID
)
{
PLIST_ENTRY Node;
ULONG i;
HANDLE TmpHandle;
BOOLEAN Cleanup;
IS_COMPONENT_INITIALIZED( StartedWorkerInitialization,
CompletedWorkerInitialization,
Cleanup ) ;
if ( Cleanup ) {
RtlEnterCriticalSection (&WorkerCriticalSection) ;
if ( (NumWorkRequests != 0) || (NumIOWorkRequests != 0) ) {
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
return STATUS_UNSUCCESSFUL ;
}
// queue a cleanup for each worker thread
for (i = 0 ; i < NumWorkerThreads ; i ++ ) {
NtSetIoCompletion (
WorkerCompletionPort,
RtlpThreadCleanup,
NULL,
STATUS_SUCCESS,
0
);
}
// queue an apc to cleanup all IO worker threads
for (Node = IOWorkerThreads.Flink ; Node != &IOWorkerThreads ;
Node = Node->Flink )
{
PRTLP_IOWORKER_TCB ThreadCB ;
ThreadCB = CONTAINING_RECORD (Node, RTLP_IOWORKER_TCB, List) ;
RemoveEntryList( &ThreadCB->List) ;
TmpHandle = ThreadCB->ThreadHandle ;
NtQueueApcThread(
ThreadCB->ThreadHandle,
(PPS_APC_ROUTINE)RtlpThreadCleanup,
NULL,
NULL,
NULL
);
NtClose( TmpHandle ) ;
}
NumWorkerThreads = NumIOWorkerThreads = 0 ;
RtlLeaveCriticalSection (&WorkerCriticalSection) ;
}
return STATUS_SUCCESS;
}
NTSTATUS
RtlpThreadPoolGetActiveActivationContext(
PACTIVATION_CONTEXT* ActivationContext
)
{
ACTIVATION_CONTEXT_BASIC_INFORMATION ActivationContextInfo = {0};
NTSTATUS Status = STATUS_SUCCESS;
ASSERT(ActivationContext != NULL);
*ActivationContext = NULL;
Status =
RtlQueryInformationActivationContext(
RTL_QUERY_INFORMATION_ACTIVATION_CONTEXT_FLAG_USE_ACTIVE_ACTIVATION_CONTEXT,
NULL,
0,
ActivationContextBasicInformation,
&ActivationContextInfo,
sizeof(ActivationContextInfo),
NULL);
if (!NT_SUCCESS(Status)) {
goto Exit;
}
if ((ActivationContextInfo.Flags & ACTIVATION_CONTEXT_FLAG_NO_INHERIT) != 0) {
RtlReleaseActivationContext(ActivationContextInfo.ActivationContext);
ActivationContextInfo.ActivationContext = NULL;
// fall through
}
*ActivationContext = ActivationContextInfo.ActivationContext;
Status = STATUS_SUCCESS;
Exit:
return Status;
}
NTSTATUS
RtlpAcquireWorker(ULONG Flags)
{
NTSTATUS Status = STATUS_SUCCESS;
if (CompletedWorkerInitialization != 1) {
Status = RtlpInitializeWorkerThreadPool () ;
if (! NT_SUCCESS(Status) )
return Status ;
}
if (NEEDS_IO_THREAD(Flags)) {
RtlEnterCriticalSection(&WorkerCriticalSection);
InterlockedIncrement(&NumFutureIOWorkItems);
if (NumIOWorkerThreads == 0) {
Status = RtlpStartIOWorkerThread();
}
RtlLeaveCriticalSection(&WorkerCriticalSection);
} else {
RtlEnterCriticalSection(&WorkerCriticalSection);
InterlockedIncrement(&NumFutureWorkItems);
if (NumWorkerThreads == 0) {
Status = RtlpStartWorkerThread();
}
RtlLeaveCriticalSection(&WorkerCriticalSection);
}
return Status;
}
VOID
RtlpReleaseWorker(ULONG Flags)
{
if (NEEDS_IO_THREAD(Flags)) {
ASSERT(NumFutureIOWorkItems > 0);
InterlockedDecrement(&NumFutureIOWorkItems);
} else {
ASSERT(NumFutureWorkItems > 0);
InterlockedDecrement(&NumFutureWorkItems);
}
}