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

1294 lines
28 KiB
C++

/*++
Copyright (c) 2001-2001 Microsoft Corporation
Module Name:
timeouts.cxx
Abstract:
Implement connection timeout quality-of-service (QoS) functionality.
The following timers must be monitored during the lifetime of
a connection:
* Connection Timeout
* Header Wait Timeout
* Entity Body Receive Timeout
* Response Processing Timeout
* Minimum Bandwidth (implemented as a Timeout)
When any one of these timeout values expires, the connection should be
terminated.
The timer information is maintained in a timeout info block,
UL_TIMEOUT_INFO_ENTRY, as part of the UL_HTTP_CONNECTION object.
A timer can be Set or Reset. Setting a timer calculates when the specific
timer should expire, and updates the timeout info block. Resetting a timer
turns a specific timer off. Both Setting and Resettnig a timer will cause the
timeout block to be re-evaluated to find the least valued expiry time.
// TODO:
The timeout manager uses a Timer Wheel(c) technology, as used
by NT's TCP/IP stack for monitoring TCB timeouts. We will reimplement
and modify the logic they use. The linkage for the Timer Wheel(c)
queues is provided in the timeout info block.
// TODO: CONVERT TO USING Timer Wheel Ticks instead of SystemTime.
There are three separate units of time: SystemTime (100ns intervals), Timer
Wheel Ticks (SystemTime / slot interval length), and Timer Wheel Slot
(Timer Wheel Ticks modulo the number of slots in the Timer Wheel).
Author:
Eric Stenson (EricSten) 24-Mar-2001
Revision History:
This was originally implemented as the Connection Timeout Monitor.
--*/
#include "precomp.h"
#include "timeoutsp.h"
//
// Private globals.
//
#ifdef ALLOC_PRAGMA
#pragma alloc_text( INIT, UlInitializeTimeoutMonitor )
#pragma alloc_text( PAGE, UlTerminateTimeoutMonitor )
#pragma alloc_text( PAGE, UlSetTimeoutMonitorInformation )
#pragma alloc_text( PAGE, UlpTimeoutMonitorWorker )
#pragma alloc_text( PAGE, UlSetPerSiteConnectionTimeoutValue )
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlpSetTimeoutMonitorTimer
NOT PAGEABLE -- UlpTimeoutMonitorDpcRoutine
NOT PAGEABLE -- UlpTimeoutCheckExpiry
NOT PAGEABLE -- UlpTimeoutInsertTimerWheelEntry
NOT PAGEABLE -- UlTimeoutRemoveTimerWheelEntry
NOT PAGEABLE -- UlInitializeConnectionTimerInfo
NOT PAGEABLE -- UlSetConnectionTimer
NOT PAGEABLE -- UlSetMinKBSecTimer
NOT PAGEABLE -- UlResetConnectionTimer
NOT PAGEABLE -- UlEvaluateTimerState
#endif // 0
//
// Connection Timeout Montior globals
//
LONG g_TimeoutMonitorInitialized = FALSE;
KDPC g_TimeoutMonitorDpc;
KTIMER g_TimeoutMonitorTimer;
KEVENT g_TimeoutMonitorTerminationEvent;
KEVENT g_TimeoutMonitorAddListEvent;
UL_WORK_ITEM g_TimeoutMonitorWorkItem;
//
// Timeout constants
//
ULONG g_TM_MinKBSecDivisor; // Bytes/Sec
LONGLONG g_TM_ConnectionTimeout; // 100ns ticks (Global...can be overriden)
LONGLONG g_TM_HeaderWaitTimeout; // 100ns ticks
//
// NOTE: Must be in sync with the _CONNECTION_TIMEOUT_TIMERS enum
//
CHAR *g_aTimeoutTimerNames[] = {
"ConnectionIdle", // TimerConnectionIdle
"HeaderWait", // TimerHeaderWait
"MinKBSec", // TimerMinKBSec
"EntityBody", // TimerEntityBody
"Response", // TimerResponse
};
//
// Timer Wheel(c)
//
static LIST_ENTRY g_TimerWheel[TIMER_WHEEL_SLOTS+1]; // TODO: alloc on its own page.
static UL_SPIN_LOCK g_TimerWheelMutex;
static USHORT g_TimerWheelCurrentSlot;
//
// Public functions.
//
/***************************************************************************++
Routine Description:
Initializes the Timeout Monitor and kicks off the first polling interval
Arguments:
(none)
--***************************************************************************/
VOID
UlInitializeTimeoutMonitor(
VOID
)
{
int i;
LARGE_INTEGER Now;
//
// Sanity check.
//
PAGED_CODE();
UlTrace(TIMEOUTS, (
"http!UlInitializeTimeoutMonitor\n"
));
ASSERT( FALSE == g_TimeoutMonitorInitialized );
//
// Set default configuration information.
//
g_TM_ConnectionTimeout = 15 * 60 * C_NS_TICKS_PER_SEC; // 15 min
g_TM_HeaderWaitTimeout = 15 * 60 * C_NS_TICKS_PER_SEC; // 15 min
g_TM_MinKBSecDivisor = 0; // 0 == Disabled
//
// Init Timer Wheel(c) state
//
//
// Set current slot
//
KeQuerySystemTime(&Now);
g_TimerWheelCurrentSlot = UlpSystemTimeToTimerWheelSlot(Now.QuadPart);
//
// Init Timer Wheel(c) slots & mutex
//
ASSERT(KeGetCurrentIrql() == PASSIVE_LEVEL); // InitializeListHead requirement
for ( i = 0; i < TIMER_WHEEL_SLOTS ; i++ )
{
InitializeListHead( &(g_TimerWheel[i]) );
}
InitializeListHead( &(g_TimerWheel[TIMER_OFF_SLOT]) );
UlInitializeSpinLock( &(g_TimerWheelMutex), "TimeoutMonitor" );
//
// Init DPC object & set DPC routine
//
KeInitializeDpc(
&g_TimeoutMonitorDpc, // DPC object
&UlpTimeoutMonitorDpcRoutine, // DPC routine
NULL // context
);
KeInitializeTimer(
&g_TimeoutMonitorTimer
);
//
// Event to control rescheduling of the timeout monitor timer
//
KeInitializeEvent(
&g_TimeoutMonitorAddListEvent,
NotificationEvent,
TRUE
);
//
// Initialize the termination event.
//
KeInitializeEvent(
&g_TimeoutMonitorTerminationEvent,
NotificationEvent,
FALSE
);
//
// Init done!
//
InterlockedExchange( &g_TimeoutMonitorInitialized, TRUE );
//
// Kick-off the first monitor sleep period
//
UlpSetTimeoutMonitorTimer();
}
/***************************************************************************++
Routine Description:
Terminate the Timeout Monitor, including any pending timer events.
Arguments:
(none)
--***************************************************************************/
VOID
UlTerminateTimeoutMonitor(
VOID
)
{
//
// Sanity check.
//
PAGED_CODE();
UlTrace(TIMEOUTS, (
"http!UlTerminateTimeoutMonitor\n"
));
//
// Clear the "initialized" flag. If the timeout monitor runs soon,
// it will see this flag, set the termination event, and exit
// quickly.
//
if ( TRUE == InterlockedCompareExchange(
&g_TimeoutMonitorInitialized,
FALSE,
TRUE) )
{
//
// Cancel the timeout monitor timer. If it fails, the monitor
// is running. Wait for it to complete.
//
if ( !KeCancelTimer( &g_TimeoutMonitorTimer ) )
{
KeWaitForSingleObject(
(PVOID)&g_TimeoutMonitorTerminationEvent,
UserRequest,
KernelMode,
FALSE,
NULL
);
}
}
UlTrace(TIMEOUTS, (
"http!UlTerminateTimeoutMonitor: Done!\n"
));
} // UlpTerminateTimeoutMonitor
/***************************************************************************++
Routine Description:
Sets the global Timeout Monitor configuration information
Arguments:
pInfo pointer to HTTP_CONTROL_CHANNEL_TIMEOUT_LIMIT structure
--***************************************************************************/
VOID
UlSetTimeoutMonitorInformation(
IN PHTTP_CONTROL_CHANNEL_TIMEOUT_LIMIT pInfo
)
{
LONGLONG localValue;
LONGLONG newValue;
LONGLONG originalValue;
//
// Sanity check.
//
PAGED_CODE();
ASSERT( pInfo );
UlTrace(TIMEOUTS, (
"http!UlSetTimeoutMonitorInformation:\n"
" ConnectionTimeout: %d\n"
" HeaderWaitTimeout: %d\n"
" MinFileKbSec: %d\n",
pInfo->ConnectionTimeout,
pInfo->HeaderWaitTimeout,
pInfo->MinFileKbSec
));
if ( pInfo->ConnectionTimeout )
{
UlInterlockedExchange64(
&g_TM_ConnectionTimeout,
(LONGLONG)(pInfo->ConnectionTimeout * C_NS_TICKS_PER_SEC)
);
}
if ( pInfo->HeaderWaitTimeout )
{
UlInterlockedExchange64(
&g_TM_HeaderWaitTimeout,
(LONGLONG)(pInfo->HeaderWaitTimeout * C_NS_TICKS_PER_SEC)
);
}
if ( pInfo->MinFileKbSec )
{
InterlockedExchange( (PLONG)&g_TM_MinKBSecDivisor, pInfo->MinFileKbSec );
}
} // UlSetTimeoutMonitorInformation
/***************************************************************************++
Routine Description:
Sets up a timer event to fire after the polling interval has expired.
Arguments:
(none)
--***************************************************************************/
VOID
UlpSetTimeoutMonitorTimer(
VOID
)
{
LARGE_INTEGER TimeoutMonitorInterval;
ASSERT( TRUE == g_TimeoutMonitorInitialized );
UlTrace(TIMEOUTS, (
"http!UlpSetTimeoutMonitorTimer\n"
));
//
// Don't want to execute this more often than few seconds.
// In particular, do not want to execute this every 0 seconds, as the
// machine will become completely unresponsive.
//
//
// negative numbers mean relative time
//
TimeoutMonitorInterval.QuadPart = -DEFAULT_POLLING_INTERVAL;
KeSetTimer(
&g_TimeoutMonitorTimer,
TimeoutMonitorInterval,
&g_TimeoutMonitorDpc
);
}
/***************************************************************************++
Routine Description:
Dispatch routine called by the timer event that queues up the Timeout
Montior
Arguments:
(all ignored)
--***************************************************************************/
VOID
UlpTimeoutMonitorDpcRoutine(
IN PKDPC Dpc,
IN PVOID DeferredContext,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
)
{
if( g_TimeoutMonitorInitialized )
{
//
// Do that timeout monitor thang.
//
UL_QUEUE_WORK_ITEM(
&g_TimeoutMonitorWorkItem,
&UlpTimeoutMonitorWorker
);
}
} // UlpTimeoutMonitorDpcRoutine
/***************************************************************************++
Routine Description:
Timeout Monitor thread
Arguments:
pWorkItem (ignored)
--***************************************************************************/
VOID
UlpTimeoutMonitorWorker(
IN PUL_WORK_ITEM pWorkItem
)
{
ULONG TimeoutMonitorEntriesSeen;
LARGE_INTEGER Now;
//
// sanity check
//
PAGED_CODE();
UlTrace(TIMEOUTS, (
"http!UlpTimeoutMonitorWorker\n"
));
//
// Check for things to expire now.
//
UlpTimeoutCheckExpiry();
UlTrace(TIMEOUTS, (
"http!UlpTimeoutMonitorWorker: g_TimerWheelCurrentSlot is now %d\n",
g_TimerWheelCurrentSlot
));
if ( g_TimeoutMonitorInitialized )
{
//
// Reschedule ourselves
//
UlpSetTimeoutMonitorTimer();
}
else
{
//
// Signal shutdown event
//
KeSetEvent(
&g_TimeoutMonitorTerminationEvent,
0,
FALSE
);
}
} // UlpTimeoutMonitor
/***************************************************************************++
Routine Description:
Walks a given timeout watch list looking for items that should be expired
Returns:
Count of connections remaining on list (after all expired connections removed)
Notes:
Possible Issue: Since we use the system time to check to see if something
should be expired, it's possible we will mis-expire some connections if the
system time is set forward.
Similarly, if the clock is set backward, we may not expire connections as
expected.
--***************************************************************************/
ULONG
UlpTimeoutCheckExpiry(
VOID
)
{
LARGE_INTEGER Now;
KIRQL OldIrql;
PLIST_ENTRY pEntry;
PLIST_ENTRY pHead;
PUL_HTTP_CONNECTION pHttpConn;
PUL_TIMEOUT_INFO_ENTRY pInfo;
LIST_ENTRY ZombieList;
ULONG Entries;
USHORT Limit;
USHORT CurrentSlot;
PAGED_CODE();
//
// Init zombie list
//
InitializeListHead(&ZombieList);
//
// Get current time
//
KeQuerySystemTime(&Now);
Limit = UlpSystemTimeToTimerWheelSlot( Now.QuadPart );
ASSERT( TIMER_OFF_SLOT != Limit );
//
// Lock Timer Wheel(c)
//
UlAcquireSpinLock(
&g_TimerWheelMutex,
&OldIrql
);
CurrentSlot = g_TimerWheelCurrentSlot;
//
// Walk the slots up until Limit
//
Entries = 0;
while ( CurrentSlot != Limit )
{
pHead = &(g_TimerWheel[CurrentSlot]);
pEntry = pHead->Flink;
ASSERT( pEntry );
//
// Walk this slot's queue
//
while ( pEntry != pHead )
{
pInfo = CONTAINING_RECORD(
pEntry,
UL_TIMEOUT_INFO_ENTRY,
QueueEntry
);
ASSERT( MmIsAddressValid(pInfo) );
pHttpConn = CONTAINING_RECORD(
pInfo,
UL_HTTP_CONNECTION,
TimeoutInfo
);
ASSERT( (pHttpConn != NULL) && \
(pHttpConn->Signature == UL_HTTP_CONNECTION_POOL_TAG) );
//
// go to next node (in case we remove the current one from the list)
//
pEntry = pEntry->Flink;
Entries++;
if (0 == pHttpConn->RefCount)
{
//
// If the ref-count has gone to zero, the httpconn will be
// cleaned up soon; ignore this item and let the cleanup
// do its job.
//
Entries--;
continue; // inner while loop
}
//
// See if we should move this entry to a different slot
//
if ( pInfo->SlotEntry != CurrentSlot )
{
ASSERT( IS_VALID_TIMER_WHEEL_SLOT(pInfo->SlotEntry) );
ASSERT( pInfo->QueueEntry.Flink != NULL );
//
// Move to correct slot
//
RemoveEntryList(
&pInfo->QueueEntry
);
InsertTailList(
&(g_TimerWheel[pInfo->SlotEntry]),
&pInfo->QueueEntry
);
Entries--;
continue; // inner while loop
}
//
// See if we should expire this connection
//
UlAcquireSpinLockAtDpcLevel( &pInfo->Lock );
if ( pInfo->CurrentExpiry < Now.QuadPart )
{
UlTrace(TIMEOUTS, (
"http!UlpTimeoutCheckExpiry: pInfo %p expired because %s timer\n",
pInfo,
g_aTimeoutTimerNames[pInfo->CurrentTimer]
));
//
// Expired. Remove entry from list & move to Zombie list
//
RemoveEntryList(
&pInfo->QueueEntry
);
pInfo->QueueEntry.Flink = NULL;
InsertTailList(
&ZombieList,
&pInfo->ZombieEntry
);
//
// Add ref the pHttpConn to prevent it being killed before we
// can kill it ourselves. (zombifying)
//
UL_REFERENCE_HTTP_CONNECTION(pHttpConn);
}
UlReleaseSpinLockFromDpcLevel( &pInfo->Lock );
} // Walk slot queue
CurrentSlot = ((CurrentSlot + 1) % TIMER_WHEEL_SLOTS);
} // ( CurrentSlot != Limit )
g_TimerWheelCurrentSlot = Limit;
UlReleaseSpinLock(
&g_TimerWheelMutex,
OldIrql
);
//
// remove entries on zombie list
//
if ( !IsListEmpty(&ZombieList) )
{
pEntry = ZombieList.Flink;
while ( &ZombieList != pEntry )
{
pInfo = CONTAINING_RECORD(
pEntry,
UL_TIMEOUT_INFO_ENTRY,
ZombieEntry
);
ASSERT( MmIsAddressValid(pInfo) );
pHttpConn = CONTAINING_RECORD(
pInfo,
UL_HTTP_CONNECTION,
TimeoutInfo
);
ASSERT( UL_IS_VALID_HTTP_CONNECTION(pHttpConn) );
pEntry = pEntry->Flink;
UlCloseConnection(pHttpConn->pConnection, TRUE, NULL, NULL);
//
// Remove the reference we added when zombifying
//
UL_DEREFERENCE_HTTP_CONNECTION(pHttpConn);
Entries--;
}
}
return Entries;
} // UlpTimeoutCheckExpiry
//
// New Timer Wheel(c) primatives
//
/***************************************************************************++
Routine Description:
Arguments:
--***************************************************************************/
VOID
UlInitializeConnectionTimerInfo(
PUL_TIMEOUT_INFO_ENTRY pInfo
)
{
LARGE_INTEGER Now;
int i;
ASSERT( TRUE == g_TimeoutMonitorInitialized );
//
// Get current time
//
KeQuerySystemTime(&Now);
//
// Init Lock
//
UlInitializeSpinLock( &pInfo->Lock, "TimeoutInfoLock" );
//
// Timer state
//
ASSERT( 0 == TimerConnectionIdle );
pInfo->Timers[TimerConnectionIdle] = TIMER_WHEEL_TICKS(Now.QuadPart + g_TM_ConnectionTimeout);
for ( i = 1; i < TimerMaximumTimer; i++ )
{
pInfo->Timers[i] = TIMER_OFF_TICK;
}
pInfo->CurrentTimer = TimerConnectionIdle;
pInfo->CurrentExpiry = TIMER_WHEEL_TICKS(Now.QuadPart + g_TM_ConnectionTimeout);
pInfo->MinKBSecSystemTime = TIMER_OFF_SYSTIME;
pInfo->ConnectionTimeoutValue = g_TM_ConnectionTimeout;
//
// Wheel state
//
pInfo->SlotEntry = UlpTimerWheelTicksToTimerWheelSlot( pInfo->CurrentExpiry );
UlpTimeoutInsertTimerWheelEntry(pInfo);
} // UlInitializeConnectionTimerInfo
/***************************************************************************++
Routine Description:
Arguments:
--***************************************************************************/
VOID
UlpTimeoutInsertTimerWheelEntry(
PUL_TIMEOUT_INFO_ENTRY pInfo
)
{
KIRQL OldIrql;
ASSERT( NULL != pInfo );
ASSERT( TRUE == g_TimeoutMonitorInitialized );
ASSERT( IS_VALID_TIMER_WHEEL_SLOT(pInfo->SlotEntry) );
UlTrace(TIMEOUTS, (
"http!UlTimeoutInsertTimerWheelEntry: pInfo %p Slot %d\n",
pInfo,
pInfo->SlotEntry
));
UlAcquireSpinLock(
&g_TimerWheelMutex,
&OldIrql
);
InsertTailList(
&(g_TimerWheel[pInfo->SlotEntry]),
&pInfo->QueueEntry
);
UlReleaseSpinLock(
&g_TimerWheelMutex,
OldIrql
);
} // UlTimeoutInsertTimerWheel
/***************************************************************************++
Routine Description:
Arguments:
--***************************************************************************/
VOID
UlTimeoutRemoveTimerWheelEntry(
PUL_TIMEOUT_INFO_ENTRY pInfo
)
{
KIRQL OldIrql;
ASSERT( NULL != pInfo );
ASSERT( !IsListEmpty(&pInfo->QueueEntry) );
UlTrace(TIMEOUTS, (
"http!UlTimeoutRemoveTimerWheelEntry: pInfo %p\n",
pInfo
));
UlAcquireSpinLock(
&g_TimerWheelMutex,
&OldIrql
);
if (pInfo->QueueEntry.Flink != NULL)
{
RemoveEntryList(
&pInfo->QueueEntry
);
pInfo->QueueEntry.Flink = NULL;
}
UlReleaseSpinLock(
&g_TimerWheelMutex,
OldIrql
);
} // UlTimeoutRemoveTimerWheelEntry
/***************************************************************************++
Routine Description:
Set the per Site Connection Timeout Value override
Arguments:
pInfo Timeout info block
TimeoutValue Override value
--***************************************************************************/
VOID
UlSetPerSiteConnectionTimeoutValue(
PUL_TIMEOUT_INFO_ENTRY pInfo,
LONGLONG TimeoutValue
)
{
ASSERT( NULL != pInfo );
ASSERT( 0L != TimeoutValue );
PAGED_CODE();
UlTrace(TIMEOUTS, (
"http!UlSetPerSiteConnectionTimeoutValue: pInfo %p TimeoutValue %I64X.\n",
pInfo,
TimeoutValue
));
ExInterlockedCompareExchange64(
&pInfo->ConnectionTimeoutValue, // Destination
&TimeoutValue, // Exchange
&pInfo->ConnectionTimeoutValue, // Comperand
&pInfo->Lock.KSpinLock // Lock
);
} // UlSetPerSiteConnectionTimeoutValue
/***************************************************************************++
Routine Description:
Starts a given timer in the timer info block.
Arguments:
pInfo Timer info block
Timer Timer to set
--***************************************************************************/
VOID
UlSetConnectionTimer(
PUL_TIMEOUT_INFO_ENTRY pInfo,
CONNECTION_TIMEOUT_TIMER Timer
)
{
LARGE_INTEGER Now;
ASSERT( NULL != pInfo );
ASSERT( IS_VALID_TIMEOUT_TIMER(Timer) );
ASSERT( UlDbgSpinLockOwned( &pInfo->Lock ) );
UlTrace(TIMEOUTS, (
"http!UlSetConnectionTimer: pInfo %p Timer %s\n",
pInfo,
g_aTimeoutTimerNames[Timer]
));
//
// Get current time
//
KeQuerySystemTime(&Now);
//
// Set timer to apropriate value
//
switch ( Timer )
{
case TimerConnectionIdle:
case TimerEntityBody:
case TimerResponse:
// all can be handled with the same timeout value
pInfo->Timers[Timer] = TIMER_WHEEL_TICKS(Now.QuadPart + pInfo->ConnectionTimeoutValue);
break;
case TimerHeaderWait:
pInfo->Timers[TimerHeaderWait] = TIMER_WHEEL_TICKS(Now.QuadPart + g_TM_HeaderWaitTimeout);
break;
// NOTE: TimerMinKBSec is handled in UlSetMinKBSecTimer()
default:
UlTrace(TIMEOUTS, ( "http!UlSetConnectionTimer: Bad Timer! (%d)\n", Timer ));
ASSERT( FALSE );
}
} // UlSetConnectionTimer
/***************************************************************************++
Routine Description:
Turns on the MinKBSec timer, adds the number of secs given the minimum
bandwidth specified.
Arguments:
pInfo Timer info block
BytesToSend Bytes to be sent
--***************************************************************************/
VOID
UlSetMinKBSecTimer(
PUL_TIMEOUT_INFO_ENTRY pInfo,
LONGLONG BytesToSend
)
{
LONGLONG XmitTicks;
KIRQL OldIrql;
ULONG NewTick;
BOOLEAN bCallEvaluate = FALSE;
ASSERT( NULL != pInfo );
UlTrace(TIMEOUTS, (
"http!UlSetMinKBSecTimer: pInfo %p BytesToSend %ld\n",
pInfo,
BytesToSend
));
if ( g_TM_MinKBSecDivisor )
{
if ( 0L != BytesToSend )
{
//
// Calculate the estimated time required to send BytesToSend
//
XmitTicks = BytesToSend / g_TM_MinKBSecDivisor;
if (0 == XmitTicks)
{
XmitTicks = C_NS_TICKS_PER_SEC;
}
else
{
XmitTicks *= C_NS_TICKS_PER_SEC;
}
UlAcquireSpinLock(
&pInfo->Lock,
&OldIrql
);
if ( TIMER_OFF_SYSTIME == pInfo->MinKBSecSystemTime )
{
LARGE_INTEGER Now;
//
// Get current time
//
KeQuerySystemTime(&Now);
pInfo->MinKBSecSystemTime = (Now.QuadPart + XmitTicks);
}
else
{
pInfo->MinKBSecSystemTime += XmitTicks;
}
NewTick = TIMER_WHEEL_TICKS(pInfo->MinKBSecSystemTime);
if ( NewTick != pInfo->Timers[TimerMinKBSec] )
{
bCallEvaluate = TRUE;
pInfo->Timers[TimerMinKBSec] = NewTick;
}
UlReleaseSpinLock(
&pInfo->Lock,
OldIrql
);
if ( TRUE == bCallEvaluate )
{
UlEvaluateTimerState(pInfo);
}
}
}
} // UlSetMinKBSecTimer
/***************************************************************************++
Routine Description:
Turns off a given timer in the timer info block.
Arguments:
pInfo Timer info block
Timer Timer to reset
--***************************************************************************/
VOID
UlResetConnectionTimer(
PUL_TIMEOUT_INFO_ENTRY pInfo,
CONNECTION_TIMEOUT_TIMER Timer
)
{
ASSERT( NULL != pInfo );
ASSERT( IS_VALID_TIMEOUT_TIMER(Timer) );
ASSERT( UlDbgSpinLockOwned( &pInfo->Lock ) );
UlTrace(TIMEOUTS, (
"http!UlResetConnectionTimer: pInfo %p Timer %s\n",
pInfo,
g_aTimeoutTimerNames[Timer]
));
//
// Turn off timer
//
pInfo->Timers[Timer] = TIMER_OFF_TICK;
if (TimerMinKBSec == Timer)
{
pInfo->MinKBSecSystemTime = TIMER_OFF_SYSTIME;
}
} // UlResetConnectionTimer
/***************************************************************************++
Routine Description:
Turns off all timers
Arguments:
pInfo Timer info block
--***************************************************************************/
VOID
UlResetAllConnectionTimers(
PUL_TIMEOUT_INFO_ENTRY pInfo
)
{
int i;
ASSERT( NULL != pInfo );
ASSERT( UlDbgSpinLockOwned( &pInfo->Lock ) );
for ( i = 0; i < TimerMaximumTimer; i++ )
{
pInfo->Timers[i] = TIMER_OFF_TICK;
}
pInfo->CurrentTimer = TimerConnectionIdle;
pInfo->CurrentExpiry = TIMER_OFF_TICK;
pInfo->MinKBSecSystemTime = TIMER_OFF_SYSTIME;
}
//
// Private functions
//
/***************************************************************************++
Routine Description:
Arguments:
--***************************************************************************/
VOID
UlEvaluateTimerState(
PUL_TIMEOUT_INFO_ENTRY pInfo
)
{
int i;
ULONG MinTimeout = TIMER_OFF_TICK;
CONNECTION_TIMEOUT_TIMER MinTimeoutTimer = TimerConnectionIdle;
ASSERT( NULL != pInfo );
ASSERT( !UlDbgSpinLockOwned( &pInfo->Lock ) );
for ( i = 0; i < TimerMaximumTimer; i++ )
{
if (pInfo->Timers[i] < MinTimeout)
{
MinTimeout = pInfo->Timers[i];
MinTimeoutTimer = (CONNECTION_TIMEOUT_TIMER) i;
}
}
//
// If we've found a different expiry time, update expiry state.
//
if (pInfo->CurrentExpiry != MinTimeout)
{
KIRQL OldIrql;
#if DBG
LARGE_INTEGER Now;
KeQuerySystemTime(&Now);
ASSERT( MinTimeout >= TIMER_WHEEL_TICKS(Now.QuadPart) );
#endif // DBG
//
// Calculate new slot
//
InterlockedExchange(
(LONG *) &pInfo->SlotEntry,
UlpTimerWheelTicksToTimerWheelSlot(MinTimeout)
);
//
// Move to new slot if necessary
//
if ( (pInfo->SlotEntry != TIMER_OFF_SLOT) && (MinTimeout < pInfo->CurrentExpiry) )
{
//
// Only move if it's on the Wheel; If Flink is null, it's in
// the process of being expired.
//
if ( NULL != pInfo->QueueEntry.Flink )
{
UlAcquireSpinLock(
&g_TimerWheelMutex,
&OldIrql
);
UlTrace(TIMEOUTS, (
"http!UlEvaluateTimerInfo: pInfo %p: Moving to new slot %d\n",
pInfo,
pInfo->SlotEntry
));
RemoveEntryList(
&pInfo->QueueEntry
);
InsertTailList(
&(g_TimerWheel[pInfo->SlotEntry]),
&pInfo->QueueEntry
);
UlReleaseSpinLock(
&g_TimerWheelMutex,
OldIrql
);
}
}
//
// Update timer wheel state
//
pInfo->CurrentExpiry = MinTimeout;
pInfo->CurrentTimer = MinTimeoutTimer;
}
} // UlpEvaluateTimerState