2020-09-30 16:53:55 +02:00

2560 lines
66 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1996-1997 Microsoft Corporation
Module Name:
servinfo.cxx
Abstract:
Class implementation for global server info list
Contents:
INTERNET_HANDLE_OBJECT::GetServerInfo
INTERNET_HANDLE_OBJECT::FindServerInfo
ReleaseServerInfo
INTERNET_HANDLE_OBJECT::PurgeServerInfoList
CServerInfo::CServerInfo
CServerInfo::~CServerInfo
CServerInfo::Reference
CServerInfo::Dereference
CServerInfo::UpdateConnectTime
CServerInfo::UpdateRTT
CServerInfo::GetConnection
CFsm_GetConnection::RunSM
CServerInfo::GetConnection_Fsm
CServerInfo::ReleaseConnection
CServerInfo::RemoveWaiter
(CServerInfo::FindKeepAliveConnection)
(CServerInfo::KeepAliveWaiters)
(CServerInfo::RunOutOfConnections)
(CServerInfo::UpdateConnectionLimit)
CServerInfo::PurgeKeepAlives
ContainingServerInfo
Author:
Richard L Firth (rfirth) 07-Oct-1996
Revision History:
07-Oct-1996 rfirth
Created
--*/
#include <wininetp.h>
#include <perfdiag.hxx>
//
// private macros
//
//#define CHECK_CONNECTION_COUNT() \
// INET_ASSERT(!UnlimitedConnections() \
// ? (TotalAvailableConnections() <= ConnectionLimit()) : TRUE)
#define CHECK_CONNECTION_COUNT() /* NOTHING */
//#define RLF_DEBUG 1
#if INET_DEBUG
#ifdef RLF_DEBUG
#define DPRINTF dprintf
#else
#define DPRINTF (void)
#endif
#else
#define DPRINTF (void)
#endif
//
// functions
//
DWORD
INTERNET_HANDLE_OBJECT::GetServerInfo(
IN LPSTR lpszHostName,
IN DWORD dwServiceType,
IN BOOL bDoResolution,
OUT CServerInfo * * lplpServerInfo
)
/*++
Routine Description:
Finds or creates a CServerInfo entry
Arguments:
lpszHostName - pointer to server name to get info for
dwServiceType - type of service for which CServerInfo requested
bDoResolution - TRUE if we are to resolve host name
lplpServerInfo - pointer to created/found CServerInfo if successful
Return Value:
DWORD
Success - ERROR_SUCCESS
Failure - ERROR_NOT_ENOUGH_MEMORY
Couldn't create the CServerInfo
ERROR_WINHTTP_NAME_NOT_RESOLVED
We were asked to resolve the name, but failed
--*/
{
DEBUG_ENTER((DBG_SESSION,
Dword,
"GetServerInfo",
"%q, %s (%d), %B, %#x",
lpszHostName,
InternetMapService(dwServiceType),
dwServiceType,
bDoResolution,
lplpServerInfo
));
ICSTRING hostName(lpszHostName);
CServerInfo * lpServerInfo = NULL;
BOOL bCreated = FALSE;
DWORD error = ERROR_SUCCESS;
if (hostName.HaveString()) {
hostName.MakeLowerCase();
LPSTR lpszHostNameLower = hostName.StringAddress();
if (!LockSerializedList(&_ServerInfoList))
{
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpServerInfo = FindServerInfo(lpszHostNameLower);
if (lpServerInfo == NULL)
{
lpServerInfo = New CServerInfo(&_ServerInfoList,
lpszHostNameLower,
&error,
dwServiceType,
GetMaxConnectionsPerServer(WINHTTP_OPTION_MAX_CONNS_PER_SERVER)
);
if (lpServerInfo != NULL)
{
if (error != ERROR_SUCCESS)
{
delete lpServerInfo;
lpServerInfo = NULL;
}
else
{
bCreated = TRUE;
// Reference this to keep it alive beyond the unlock/
lpServerInfo->Reference();
}
}
else
{
error = ERROR_NOT_ENOUGH_MEMORY;
}
}
UnlockSerializedList(&_ServerInfoList);
} else {
//
// failed to create ICSTRING
//
error = GetLastError();
INET_ASSERT(error != ERROR_SUCCESS);
lpServerInfo = NULL;
}
//
// if we created a new CServerInfo and we are instructed to resolve the host
// name then do it now, outside of the global server info list lock. This
// operation may take some time
//
if (bDoResolution && (lpServerInfo != NULL)) {
//error = lpServerInfo->ResolveHostName();
if (error != ERROR_SUCCESS) {
ReleaseServerInfo(lpServerInfo);
lpServerInfo = NULL;
}
}
quit:
*lplpServerInfo = lpServerInfo;
DEBUG_LEAVE(error);
return error;
}
CServerInfo *
INTERNET_HANDLE_OBJECT::FindServerInfo(
IN LPSTR lpszHostName
)
/*++
Routine Description:
Walks the server info list looking for the requested server
Arguments:
lpszHostName - pointer to server name to find (IN LOWER CASE!)
Return Value:
CServerInfo *
Success - pointer to found list entry
Failure - NULL
--*/
{
DEBUG_ENTER((DBG_SESSION,
Pointer,
"FindServerInfo",
"%q",
lpszHostName
));
DWORD hashHostName = CalculateHashValue(lpszHostName);
CServerInfo * lpServerInfo = NULL;
BOOL found = FALSE;
if (!LockSerializedList(&_ServerInfoList))
{
goto quit;
}
for (lpServerInfo = (CServerInfo *)HeadOfSerializedList(&_ServerInfoList);
lpServerInfo != (CServerInfo *)SlSelf(&_ServerInfoList);
lpServerInfo = lpServerInfo->Next()) {
if (lpServerInfo->Match(hashHostName, lpszHostName)) {
found = TRUE;
break;
}
}
if (!found)
{
lpServerInfo = NULL;
}
// Need to keep this alive beyond the lock.
if (lpServerInfo)
{
lpServerInfo->Reference();
}
UnlockSerializedList(&_ServerInfoList);
quit:
DEBUG_LEAVE(lpServerInfo);
return lpServerInfo;
}
VOID
ReleaseServerInfo(
IN CServerInfo * lpServerInfo
)
/*++
Routine Description:
Release a CServerInfo by dereferencing it. If the reference count goes to
zero, the CServerInfo will be destroyed
Arguments:
lpServerInfo - pointer to CServerInfo to release
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"ReleaseServerInfo",
"%#x [%q]",
lpServerInfo,
lpServerInfo->GetHostName()
));
lpServerInfo->Dereference();
DEBUG_LEAVE(0);
}
VOID
INTERNET_HANDLE_OBJECT::PurgeServerInfoList(
IN BOOL bForce
)
/*++
Routine Description:
Throw out any CServerInfo entries that have expired or any KEEP_ALIVE
entries (for any CServerInfo) that have expired
Arguments:
bForce - TRUE if we forcibly remove entries which have not yet expired but
which have a reference count of 1, else FALSE to remove only
entries that have expired
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"PurgeServerInfoList",
"%B",
bForce
));
if (!LockSerializedList(&_ServerInfoList))
{
// Can't purge list if unable to obtain the lock.
goto quit;
}
PLIST_ENTRY pEntry = HeadOfSerializedList(&_ServerInfoList);
PLIST_ENTRY pPrevious = (PLIST_ENTRY)SlSelf(&_ServerInfoList);
while (TRUE) {
if (pEntry == (PLIST_ENTRY)SlSelf(&_ServerInfoList)) {
break;
}
CServerInfo * pServerInfo;
//pServerInfo = (CServerInfo *)pEntry;
//pServerInfo = CONTAINING_RECORD(pEntry, CONNECTION_LIMIT, m_List);
pServerInfo = ContainingServerInfo(pEntry);
BOOL deleted = FALSE;
if (pServerInfo->ReferenceCount() == 1) {
if (bForce || pServerInfo->Expired()) {
//dprintf("purging server info entry for %q\n", pServerInfo->GetHostName());
deleted = pServerInfo->Dereference();
} else {
pServerInfo->PurgeKeepAlives(PKA_NO_FORCE);
}
}
if (!deleted) {
pPrevious = pEntry;
}
pEntry = pPrevious->Flink;
}
UnlockSerializedList(&_ServerInfoList);
quit:
DEBUG_LEAVE(0);
}
VOID
INTERNET_HANDLE_OBJECT::PurgeKeepAlives(
IN DWORD dwForce
)
/*++
Routine Description:
Throw out any KEEP_ALIVE entries from any CServerInfo that have expired or
which have failed authentication or which are unused, depending on dwForce
Arguments:
dwForce - force to apply when purging. Value can be:
PKA_NO_FORCE - only purge timed-out sockets or sockets in
close-wait state (default)
PKA_NOW - purge all sockets
PKA_AUTH_FAILED - purge sockets that have been marked as failing
authentication
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"PurgeKeepAlives",
"%s [%d]",
(dwForce == PKA_NO_FORCE) ? "NO_FORCE"
: (dwForce == PKA_NOW) ? "NOW"
: (dwForce == PKA_AUTH_FAILED) ? "AUTH_FAILED"
: "?",
dwForce
));
if (!LockSerializedList(&_ServerInfoList))
{
goto quit;
}
PLIST_ENTRY pEntry = HeadOfSerializedList(&_ServerInfoList);
while (pEntry != (PLIST_ENTRY)SlSelf(&_ServerInfoList)) {
CServerInfo * lpServerInfo = ContainingServerInfo(pEntry);
lpServerInfo->PurgeKeepAlives(dwForce);
pEntry = pEntry->Flink;
}
UnlockSerializedList(&_ServerInfoList);
quit:
DEBUG_LEAVE(0);
}
//
// methods
//
CServerInfo::CServerInfo(
IN SERIALIZED_LIST * ServerInfoList,
IN LPSTR lpszHostName,
OUT DWORD* pdwError,
IN DWORD dwService,
IN DWORD dwMaxConnections
)
/*++
Routine Description:
CServerInfo constructor
Arguments:
lpszHostName - server for which to create CServerInfo
dwService - which service to create CServerInfo for
dwMaxConnections - maximum number of simultaneous connections to this
server
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_OBJECTS,
None,
"CServerInfo::CServerInfo",
"%q, %s (%d), %d",
lpszHostName,
InternetMapService(dwService),
dwService,
dwMaxConnections
));
INIT_SERVER_INFO();
m_ServerInfoList = ServerInfoList;
*pdwError = ERROR_SUCCESS;
InitializeListHead(&m_List);
m_Expires = 0;
m_Wrap = 0;
m_ReferenceCount = 1;
m_HostName = lpszHostName;
if (!m_HostName.StringAddress())
{
goto error;
}
m_HostName.MakeLowerCase();
m_Hash = CalculateHashValue(m_HostName.StringAddress());
m_Services.Word = 0;
m_HttpSupport.Word = 0;
m_Flags.Word = 0;
m_ProxyLink = NULL;
INET_ASSERT(dwService == INTERNET_SERVICE_HTTP);
SetHTTP();
//
// only initialize the keep-alive and connection limit lists if we are
// creating the server info entry for a HTTP server (or CERN proxy)
//
//
// BUGBUG - we only want to do this on demand
//
//if (IsHTTP()) {
InitializeSerializedList(&m_KeepAliveList);
SetKeepAliveListInitialized();
//
// the maximum number of connections per server is initialized to the
// default (registry) value unless overridden by the caller
//
if (dwMaxConnections == 0)
{
dwMaxConnections = DEFAULT_MAX_CONNECTIONS_PER_SERVER;
}
m_ConnectionLimit = dwMaxConnections;
//} else {
// m_ConnectionLimit = UNLIMITED_CONNECTIONS;
//}
//dprintf("*** %s: limit = %d\n", GetHostName(), m_ConnectionLimit);
//
// BUGBUG - only create event if limiting connections. Need method to manage
// connection limit count/event creation
//
m_NewLimit = m_ConnectionLimit;
m_ConnectionsAvailable = m_ConnectionLimit;
//m_ActiveConnections = 0;
m_LastActiveTime = 0;
m_ConnectTime = (DWORD)-1;
m_RTT = 0;
m_dwError = ERROR_SUCCESS;
//
// add to the global list. We are assuming here that the caller has already
// checked for dupes
//
if (!InsertAtHeadOfSerializedList(m_ServerInfoList, &m_List))
*pdwError = ERROR_NOT_ENOUGH_MEMORY;
quit:
DEBUG_LEAVE(0);
return;
error:
*pdwError = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
CServerInfo::~CServerInfo()
/*++
Routine Description:
CServerInfo destructor
Arguments:
None.
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_OBJECTS,
None,
"CServerInfo::~CServerInfo",
"{%q}",
GetHostName()
));
CHECK_SERVER_INFO();
//GlobalServerInfoDeAllocCount++;
// unlink if we have a nested obj
if ( m_ProxyLink )
{
CServerInfo *pDerefObj = NULL;
// will leak if unable to dereference
if (LockSerializedList(m_ServerInfoList))
{
pDerefObj = m_ProxyLink;
m_ProxyLink = NULL;
UnlockSerializedList(m_ServerInfoList);
}
if (pDerefObj)
{
pDerefObj->Dereference();
}
}
RemoveFromSerializedList(m_ServerInfoList, &m_List);
INET_ASSERT(m_ReferenceCount == 0);
if (IsKeepAliveListInitialized() && LockSerializedList(&m_KeepAliveList))
{
while (!IsSerializedListEmpty(&m_KeepAliveList))
{
//dprintf("%#x ~S-I killing K-A %#x\n", GetCurrentThreadId(), HeadOfSerializedList(&m_KeepAliveList));
LPVOID pEntry = SlDequeueHead(&m_KeepAliveList);
INET_ASSERT(pEntry != NULL);
if (pEntry != NULL) {
ICSocket * pSocket = ContainingICSocket(pEntry);
//dprintf("~CServerInfo: destroying socket %#x\n", pSocket->GetSocket());
pSocket->Destroy();
}
}
UnlockSerializedList(&m_KeepAliveList);
TerminateSerializedList(&m_KeepAliveList);
}
DEBUG_LEAVE(0);
}
VOID
CServerInfo::Reference(
VOID
)
/*++
Routine Description:
Increments the reference count for the CServerInfo
Arguments:
None.
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::Reference",
"{%q}",
GetHostName()
));
CHECK_SERVER_INFO();
INET_ASSERT(m_ReferenceCount > 0);
InterlockedIncrement(&m_ReferenceCount);
//dprintf("CServerInfo %s - %d\n", GetHostName(), m_ReferenceCount);
DEBUG_PRINT(SESSION,
INFO,
("Reference count = %d\n",
ReferenceCount()
));
DEBUG_LEAVE(0);
}
BOOL
CServerInfo::Dereference(
VOID
)
/*++
Routine Description:
Dereferences the SESSION_INFO. If the reference count goes to zero then this
entry is deleted. If the reference count goes to 1 then the expiry timer is
started
Arguments:
None.
Return Value:
BOOL
TRUE - entry was deleted
FALSE - entry was not deleted
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::Dereference",
"{%q}",
GetHostName()
));
CHECK_SERVER_INFO();
INET_ASSERT(m_ReferenceCount > 0);
//
// we need to grab the list - we may be removing this entry or updating
// the reference count and expiry fields which must be done atomically
//
SERIALIZED_LIST * ServerInfoList = m_ServerInfoList;
BOOL deleted = FALSE;
if (!LockSerializedList(ServerInfoList))
goto quit;
LONG result = InterlockedDecrement(&m_ReferenceCount);
//dprintf("CServerInfo %s - %d\n", GetHostName(), m_ReferenceCount);
DEBUG_PRINT(SESSION,
INFO,
("Reference count = %d\n",
ReferenceCount()
));
if (result == 0) {
delete this;
deleted = TRUE;
} else if (result == 1) {
//
// start expiration proceedings...
//
SetExpiryTime();
}
UnlockSerializedList(ServerInfoList);
quit:
DEBUG_LEAVE(deleted);
return deleted;
}
DWORD
CServerInfo::SetCachedProxyServerInfo(
IN CServerInfo * pProxyServer,
IN DWORD dwProxyVersion,
IN BOOL fUseProxy,
IN INTERNET_SCHEME HostScheme,
IN INTERNET_PORT HostPort,
IN INTERNET_SCHEME ProxyScheme,
IN INTERNET_PORT ProxyPort
)
/*++
Routine Description:
If the Version information match up, copies
the proxy information and links this server object
to the appopriate proxy server object
Assumes that this is called on successful use of the proxy
object.
Arguments:
None.
Return Value:
DWORD
ERROR_SUCCESS
ERROR_NOT_ENOUGH_MEMORY - entry was not deleted because there
wasn't available memory to obtain lock
--*/
{
DWORD error=ERROR_SUCCESS;
if (!LockSerializedList(m_ServerInfoList))
{
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
if ( dwProxyVersion != GlobalProxyVersionCount )
{
SetProxyScriptCached(FALSE);
goto cleanup; // bail, we don't accept out of date additions to the cache
}
if ( m_ProxyLink )
{
if ( IsProxyScriptCached() &&
HostScheme == m_HostScheme &&
HostPort == m_HostPort &&
fUseProxy )
{
if ( pProxyServer == m_ProxyLink ) {
INET_ASSERT(dwProxyVersion == GlobalProxyVersionCount);
m_dwProxyVersion = dwProxyVersion; // we're now up to date
goto cleanup; // match, no version or host changes
}
INET_ASSERT(pProxyServer != m_ProxyLink );
}
//
// unlink, because we have a new entry to save,
// and the previous entry is bad
//
m_ProxyLink->Dereference();
m_ProxyLink = NULL;
}
//
// Add new cached entry
//
SetProxyScriptCached(TRUE);
m_HostScheme = HostScheme;
m_HostPort = HostPort;
m_dwProxyVersion = dwProxyVersion; // we're now up to date
if ( fUseProxy )
{
INET_ASSERT(this != pProxyServer);
m_ProxyLink = pProxyServer;
m_ProxyLink->Reference();
m_ProxyLink->m_HostScheme = ProxyScheme;
m_ProxyLink->m_HostPort = ProxyPort;
switch (ProxyScheme)
{
case INTERNET_SCHEME_HTTP:
m_ProxyLink->SetCernProxy();
break;
case INTERNET_SCHEME_SOCKS:
m_ProxyLink->SetSocksGateway();
break;
}
}
cleanup:
UnlockSerializedList(m_ServerInfoList);
quit:
return error;
}
CServerInfo *
CServerInfo::GetCachedProxyServerInfo(
IN INTERNET_SCHEME HostScheme,
IN INTERNET_PORT HostPort,
OUT BOOL *pfCachedEntry
)
/*++
Routine Description:
Retrieves a cached server object, that indicates
a probable proxy to use
On Success, the return has an additional increment
on its ref count, assumition that caller derefs
Arguments:
None.
Return Value:
CServerInfo *
NULL on failure
--*/
{
CServerInfo *pProxyServer = NULL;
if (!LockSerializedList(m_ServerInfoList))
return NULL;
*pfCachedEntry = FALSE;
if ( IsProxyScriptCached() )
{
//
// Examine Version Count
//
if ( GlobalProxyVersionCount == m_dwProxyVersion &&
HostScheme == m_HostScheme &&
HostPort == m_HostPort
)
{
*pfCachedEntry = TRUE;
if ( m_ProxyLink ) {
// matched cached entry
m_ProxyLink->Reference();
pProxyServer = m_ProxyLink;
}
}
else
{
// version is expired, remove reference
SetProxyScriptCached(FALSE);
if ( m_ProxyLink ) {
m_ProxyLink->Dereference();
m_ProxyLink = NULL;
}
}
}
UnlockSerializedList(m_ServerInfoList);
return pProxyServer;
}
BOOL
CServerInfo::CopyCachedProxyInfoToProxyMsg(
IN OUT AUTO_PROXY_ASYNC_MSG *pQueryForProxyInfo
)
/*++
Routine Description:
Retrieves Cached Proxy info from object
Arguments:
None.
Return Value:
BOOL
TRUE - sucess
--*/
{
BOOL fSuccess = FALSE;
// really only need to lock to proctect m_HostPort && m_HostScheme
if (!LockSerializedList(m_ServerInfoList))
return FALSE;
pQueryForProxyInfo->SetUseProxy(FALSE);
pQueryForProxyInfo->_lpszProxyHostName =
m_HostName.StringAddress() ?
NewString(m_HostName.StringAddress()) :
NULL;
if ( pQueryForProxyInfo->_lpszProxyHostName != NULL ) {
// copy out cached entry to proxy message structure
pQueryForProxyInfo->_nProxyHostPort = m_HostPort;
pQueryForProxyInfo->_tProxyScheme = m_HostScheme;
pQueryForProxyInfo->_bFreeProxyHostName = TRUE;
pQueryForProxyInfo->_dwProxyHostNameLength =
strlen((pQueryForProxyInfo)->_lpszProxyHostName);
pQueryForProxyInfo->SetUseProxy(TRUE);
fSuccess = TRUE; // success
}
UnlockSerializedList(m_ServerInfoList);
return fSuccess;
}
VOID
CServerInfo::UpdateConnectTime(
IN DWORD dwConnectTime
)
/*++
Routine Description:
Calculates average connect time
Arguments:
dwConnectTime - current connect time
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::UpdateConnectTime",
"{%q} %d",
GetHostName(),
dwConnectTime
));
DWORD connectTime = m_ConnectTime;
if (connectTime == (DWORD)-1) {
connectTime = dwConnectTime;
} else {
connectTime = (connectTime + dwConnectTime) / 2;
}
//dprintf("%s: connect time = %d, ave = %d\n", GetHostName(), dwConnectTime, connectTime);
DEBUG_PRINT(SESSION,
INFO,
("average connect time = %d mSec\n",
connectTime
));
InterlockedExchange((LPLONG)&m_ConnectTime, connectTime);
DEBUG_LEAVE(0);
}
VOID
CServerInfo::UpdateRTT(
IN DWORD dwRTT
)
/*++
Routine Description:
Calculates rolling average round-trip time
Arguments:
dwRTT - current round-trip time
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::UpdateRTT",
"{%q} %d",
GetHostName(),
dwRTT
));
DWORD RTT = m_RTT;
if (RTT == 0) {
RTT = dwRTT;
} else {
RTT = (RTT + dwRTT) / 2;
}
//dprintf("%s: RTT = %d, ave = %d\n", GetHostName(), dwRTT, RTT);
DEBUG_PRINT(SESSION,
INFO,
("average round trip time = %d mSec\n",
RTT
));
InterlockedExchange((LPLONG)&m_RTT, RTT);
DEBUG_LEAVE(0);
}
DWORD
CFsm_GetConnection::RunSM(
IN CFsm * Fsm
)
/*++
Routine Description:
Runs next CFsm_GetConnection state
Arguments:
Fsm - FSM controlling operation
Return Value:
DWORD
Success - ERROR_SUCCESS
Failure -
--*/
{
//dprintf("%#x: %s FSM %#x state %s\n", GetCurrentThreadId(), Fsm->MapType(), Fsm, Fsm->MapState());
DEBUG_ENTER((DBG_SESSION,
Dword,
"CFsm_GetConnection::RunSM",
"%#x",
Fsm
));
CServerInfo * pServerInfo = (CServerInfo *)Fsm->GetContext();
CFsm_GetConnection * stateMachine = (CFsm_GetConnection *)Fsm;
DWORD error;
switch (Fsm->GetState()) {
case FSM_STATE_INIT:
stateMachine->StartTimer();
//
// fall through
//
case FSM_STATE_CONTINUE:
#ifdef NEW_CONNECTION_SCHEME
case FSM_STATE_ERROR:
#endif
error = pServerInfo->GetConnection_Fsm(stateMachine);
break;
#ifndef NEW_CONNECTION_SCHEME
case FSM_STATE_ERROR:
INET_ASSERT((Fsm->GetError() == ERROR_WINHTTP_TIMEOUT)
|| (Fsm->GetError() == ERROR_WINHTTP_OPERATION_CANCELLED));
pServerInfo->RemoveWaiter((DWORD_PTR)Fsm);
error = Fsm->GetError();
Fsm->SetDone();
//dprintf("%#x: FSM_STATE_ERROR - %d\n", GetCurrentThreadId(), error);
break;
#endif
default:
error = ERROR_WINHTTP_INTERNAL_ERROR;
Fsm->SetDone(ERROR_WINHTTP_INTERNAL_ERROR);
INET_ASSERT(FALSE);
break;
}
DEBUG_LEAVE(error);
return error;
}
DWORD
CServerInfo::GetConnection_Fsm(
IN CFsm_GetConnection * Fsm
)
/*++
Routine Description:
Tries to get a connection of requested type for caller. If no connection is
available then one of the following happens:
* If there are available keep-alive connections of a different type then
one is closed and the caller allowed to create a new connection
* If this is an async request, the FSM is blocked and the thread returns
to the pool if a worker, or back to the app if an app thread
* If this is a sync request, we wait on an event for a connection to be
made available, or the connect timeout to elapse
Arguments:
Fsm - get connection FSM
Return Value:
DWORD
Success - ERROR_SUCCESS
Depending on *lplpSocket, we either returned the socket to
use, or its okay to create a new connection
ERROR_IO_PENDING
Request will complete asynchronously
Failure - ERROR_WINHTTP_TIMEOUT
Failed to get connection in time allowed
ERROR_WINHTTP_INTERNAL_ERROR
Something unexpected happened
--*/
{
DEBUG_ENTER((DBG_SESSION,
Dword,
"CServerInfo::GetConnection_Fsm",
"{%q [%d+%d/%d]} %#x(%#x, %d, %d)",
GetHostName(),
m_ConnectionsAvailable,
ElementsOnSerializedList(&m_KeepAliveList),
m_ConnectionLimit,
Fsm,
Fsm->m_dwSocketFlags,
Fsm->m_nPort,
Fsm->m_dwTimeout
));
PERF_ENTER(GetConnection);
BOOL bFound = FALSE;
DWORD error = ERROR_SUCCESS;
CFsm_GetConnection & fsm = *Fsm;
ICSocket * pSocket = NULL;
LPINTERNET_THREAD_INFO lpThreadInfo = fsm.GetThreadInfo();
HANDLE hEvent = NULL;
BOOL bUnlockList = TRUE;
BOOL bKeepAliveWaiters;
INET_ASSERT(lpThreadInfo != NULL);
INET_ASSERT(lpThreadInfo->hObjectMapped != NULL);
INET_ASSERT(((HANDLE_OBJECT *)lpThreadInfo->hObjectMapped)->
GetHandleType() == TypeHttpRequestHandle);
if ((lpThreadInfo == NULL) || (lpThreadInfo->hObjectMapped == NULL)) {
error = ERROR_WINHTTP_INTERNAL_ERROR;
goto quit;
}
BOOL bAsyncRequest;
bAsyncRequest = lpThreadInfo->IsAsyncWorkerThread
|| ((INTERNET_HANDLE_OBJECT *)lpThreadInfo->hObjectMapped)->
IsAsyncHandle();
*fsm.m_lplpSocket = NULL;
try_again:
bUnlockList = TRUE;
//
// use m_Waiters to serialize access. N.B. - we will acquire m_KeepAliveList
// from within m_Waiters
//
if (!m_Waiters.Acquire()) {
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
if (IsNewLimit()) {
UpdateConnectionLimit();
}
bKeepAliveWaiters = KeepAliveWaiters();
if (fsm.m_dwSocketFlags & SF_KEEP_ALIVE) {
//
// maintain requester order - if there are already waiters then queue
// this request, else try to satisfy the requester. HOWEVER, only check
// for existing requesters the FIRST time through. If we're here with
// FSM_STATE_CONTINUE then we've been unblocked and we can ignore any
// waiters that came after us
//
if ((fsm.GetState() == FSM_STATE_CONTINUE) || !bKeepAliveWaiters) {
DEBUG_PRINT(SESSION,
INFO,
("no current waiters for K-A connections\n"
));
while (pSocket = FindKeepAliveConnection(fsm.m_dwSocketFlags,
fsm.m_nPort,
fsm.m_lpszSecureTunnelHost)) {
if (pSocket->IsReset() || pSocket->HasExpired()) {
DPRINTF("%#x: %#x: ********* socket %#x is closed already\n",
GetCurrentThreadId(),
Fsm,
pSocket->GetSocket()
);
DEBUG_PRINT(SESSION,
INFO,
("K-A connection %#x [%#x/%d] is reset (%B) or expired (%B)\n",
pSocket,
pSocket->GetSocket(),
pSocket->GetSourcePort(),
pSocket->IsReset(),
pSocket->HasExpired()
));
pSocket->SetLinger(FALSE, 0);
pSocket->Shutdown(2);
//dprintf("GetConnection: destroying reset socket %#x\n", pSocket->GetSocket());
pSocket->Destroy();
pSocket = NULL;
if (!UnlimitedConnections()) {
++m_ConnectionsAvailable;
}
CHECK_CONNECTION_COUNT();
} else {
DPRINTF("%#x: %#x: *** matched %#x, %#x\n",
GetCurrentThreadId(),
Fsm,
pSocket->GetSocket(),
pSocket->GetFlags()
);
break;
}
}
if (pSocket == NULL) {
DEBUG_PRINT(SESSION,
INFO,
("no available K-A connections\n"
));
/*
//
// if all connections are in use as keep-alive connections then
// since we're here, we want a keep-alive connection that doesn't
// match the currently available keep-alive connections. Terminate
// the oldest keep-alive connection (at the head of the queue)
// and generate a new connection
//
LockSerializedList(&m_KeepAliveList);
if (ElementsOnSerializedList(&m_KeepAliveList) == m_ConnectionLimit) {
pSocket = ContainingICSocket(SlDequeueHead(&m_KeepAliveList));
pSocket->SetLinger(FALSE, 0);
pSocket->Shutdown(2);
pSocket->Destroy();
if (!UnlimitedConnections()) {
++m_ConnectionsAvailable;
}
CHECK_CONNECTION_COUNT();
}
UnlockSerializedList(&m_KeepAliveList);
*/
}
} else {
DEBUG_PRINT(SESSION,
INFO,
("%d waiters for K-A connection to %q\n",
ElementsOnSerializedList(&m_KeepAliveList),
GetHostName()
));
}
}
//
// if we found a matching keep-alive connection or we are not limiting
// connections then we're done
//
if ((pSocket != NULL) || UnlimitedConnections()) {
INET_ASSERT(error == ERROR_SUCCESS);
goto exit;
}
//
// no keep-alive connections matched, or there are already waiters for
// keep-alive connections
//
INET_ASSERT(m_ConnectionsAvailable <= m_ConnectionLimit);
if (m_ConnectionsAvailable > 0) {
if (fsm.m_lpszSecureTunnelHost)
goto exit; // don't create a connection here for SSL tunneling
//
// can create a connection
//
DEBUG_PRINT(SESSION,
INFO,
("OK to create new connection\n"
));
DPRINTF("%#x: %#x: *** %s OK to create connection %d/%d\n",
GetCurrentThreadId(),
Fsm,
GetHostName(),
m_ConnectionsAvailable,
m_ConnectionLimit
);
--m_ConnectionsAvailable;
} else if (fsm.GetElapsedTime() > fsm.m_dwTimeout) {
error = ERROR_WINHTTP_TIMEOUT;
} else {
//
// if there are keep-alive connections but no keep-alive waiters
// then either we don't want a keep-alive connection, or the ones
// available don't match our requirements.
// If we need a connection of a different type - e.g. SSL when all
// we have is non-SSL then close a connection & generate a new one.
// If we need a non-keep-alive connection then its okay to return
// a current keep-alive connection, the understanding being that the
// caller will not add Connection: Keep-Alive header (HTTP 1.0) or
// will add Connection: Close header (HTTP 1.1)
//
//
// BUGBUG - what about waiters for non-keep-alive connections?
//
// scenario - limit of 1 connection:
//
// A. request for k-a
// continue & create connection
// B. request non-k-a
// none available; wait
// C. release k-a connection; unblock sync waiter B
// D. request non-k-a
// k-a available; return it; caller converts to non-k-a
// E. unblocked waiter B request non-k-a
// none available; wait
//
// If this situation continues, eventually B will time-out, whereas it
// could have had the connection taken by D. Request D is younger and
// therefore can afford to wait while B continues with the connection
//
BOOL fHaveConnection = FALSE;
if (!bKeepAliveWaiters || (fsm.GetState() == FSM_STATE_CONTINUE)) {
if (!LockSerializedList(&m_KeepAliveList)) {
error = ERROR_NOT_ENOUGH_MEMORY;
goto exit;
}
if (ElementsOnSerializedList(&m_KeepAliveList) != 0) {
pSocket = ContainingICSocket(SlDequeueHead(&m_KeepAliveList));
fHaveConnection = TRUE;
#define SOCK_FLAGS (SF_ENCRYPT | SF_DECRYPT | SF_SECURE | SF_TUNNEL)
DWORD dwSocketTypeFlags = pSocket->GetFlags() & SOCK_FLAGS;
DWORD dwRequestTypeFlags = fsm.m_dwSocketFlags & SOCK_FLAGS;
if ((dwSocketTypeFlags ^ dwRequestTypeFlags)
|| (fsm.m_nPort != pSocket->GetPort())) {
DEBUG_PRINT(SESSION,
INFO,
("different socket types (%#x, %#x) or ports (%d, %d) requested\n",
fsm.m_dwSocketFlags,
pSocket->GetFlags(),
fsm.m_nPort,
pSocket->GetPort()
));
DPRINTF("%#x: %#x: *** closing socket %#x: %#x vs. %#x\n",
GetCurrentThreadId(),
Fsm,
pSocket->GetSocket(),
pSocket->GetFlags(),
fsm.m_dwSocketFlags
);
pSocket->SetLinger(FALSE, 0);
pSocket->Shutdown(2);
//dprintf("GetConnection: destroying different type socket %#x\n", pSocket->GetSocket());
pSocket->Destroy();
pSocket = NULL;
// If we were trying to wait for established SSL tunnel,
// but one wasn't found, then this connection is open
// for anyone.
if (!UnlimitedConnections() && fsm.m_lpszSecureTunnelHost) {
++m_ConnectionsAvailable;
}
} else {
DPRINTF("%#x: %#x: *** returning k-a connection %#x as non-k-a\n",
GetCurrentThreadId(),
Fsm,
pSocket->GetSocket()
);
}
CHECK_CONNECTION_COUNT();
}
UnlockSerializedList(&m_KeepAliveList);
if (fHaveConnection) {
goto exit;
}
}
DPRINTF("%#x: %#x: blocking %s FSM %#x state %s %d/%d\n",
GetCurrentThreadId(),
Fsm,
Fsm->MapType(),
Fsm,
Fsm->MapState(),
m_ConnectionsAvailable,
m_ConnectionLimit
);
//
// we have to wait for a connection to become available. If we are an
// async request then we queue this FSM & return the thread to the pool
// or, if app thread, return pending indication to the app. If this is
// a sync request (in an app thread) then we block on an event waiting
// for a connection to become available
//
if (!bAsyncRequest) {
//
// create unnamed, initially unsignalled, auto-reset event
//
hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);
if (hEvent == NULL) {
error = GetLastError();
goto exit;
}
}
CConnectionWaiter * pWaiter;
DWORD dwStatus = ERROR_SUCCESS;
#if INET_DEBUG
for (pWaiter = (CConnectionWaiter *)m_Waiters.Head();
pWaiter != (CConnectionWaiter *)m_Waiters.Self();
pWaiter = (CConnectionWaiter *)pWaiter->Next()) {
INET_ASSERT(pWaiter->Id() != (DWORD_PTR)(bAsyncRequest ? (DWORD_PTR)Fsm : lpThreadInfo->ThreadId));
}
#endif
pWaiter = New CConnectionWaiter(&m_Waiters,
!bAsyncRequest,
(fsm.m_dwSocketFlags & SF_KEEP_ALIVE)
? TRUE
: FALSE,
bAsyncRequest
? (DWORD_PTR)Fsm
: lpThreadInfo->ThreadId,
hEvent,
//
// priority in request handle object
// controls relative position in list
// of waiters
//
((HTTP_REQUEST_HANDLE_OBJECT *)
lpThreadInfo->hObjectMapped)->
GetPriority(),
&dwStatus
);
DPRINTF("%#x: %#x: new waiter %#x: as=%B, K-A=%B, id=%#x, hE=%#x, pri=%d, status=%#x, sf=%#x, preq=%#x ssl=%s\n",
GetCurrentThreadId(),
Fsm,
pWaiter,
bAsyncRequest,
(fsm.m_dwSocketFlags & SF_KEEP_ALIVE)
? TRUE
: FALSE,
bAsyncRequest
? (DWORD_PTR)Fsm
: lpThreadInfo->ThreadId,
hEvent,
((HTTP_REQUEST_HANDLE_OBJECT *)lpThreadInfo->hObjectMapped)->
GetPriority(),
dwStatus,
fsm.m_dwSocketFlags,
((HTTP_REQUEST_HANDLE_OBJECT *)lpThreadInfo->hObjectMapped),
fsm.m_lpszSecureTunnelHost ? fsm.m_lpszSecureTunnelHost : ""
);
if (pWaiter == NULL) {
error = ERROR_NOT_ENOUGH_MEMORY;
goto exit;
}
else if (dwStatus != ERROR_SUCCESS) {
error = dwStatus;
delete pWaiter; // free since it wasn't inserted
goto exit;
}
if (bAsyncRequest) {
//
// ensure that when the FSM is unblocked normally, the new state
// is STATE_CONTINUE
//
Fsm->SetState(FSM_STATE_CONTINUE);
error = BlockWorkItem(Fsm,
(DWORD_PTR)pWaiter,
fsm.m_dwTimeout
);
if (error == ERROR_SUCCESS) {
error = ERROR_IO_PENDING;
}
}
else
{
m_Waiters.Release();
bUnlockList = FALSE;
DPRINTF("%#x: %#x: %s FSM %#x %s waiting %d msec\n",
GetCurrentThreadId(),
Fsm,
Fsm->MapType(),
Fsm,
Fsm->MapState(),
fsm.m_dwTimeout
);
DWORD dwWaitTime = (fsm.m_dwTimeout != INFINITE) ?
(fsm.m_dwTimeout - fsm.GetElapsedTime()) :
INFINITE;
if (((int)dwWaitTime <= 0) && (dwWaitTime != INFINITE)) {
DEBUG_PRINT(SESSION,
ERROR,
("SYNC wait timed out (%d mSec)\n",
fsm.m_dwTimeout
));
error = ERROR_WINHTTP_TIMEOUT;
} else {
DEBUG_PRINT(SESSION,
INFO,
("waiting %d mSec for SYNC event %#x\n",
dwWaitTime,
hEvent
));
//
// we'd better not be doing a sync wait if we are in the
// context of an app thread making an async request
//
INET_ASSERT(lpThreadInfo->IsAsyncWorkerThread
|| !((INTERNET_HANDLE_OBJECT *)lpThreadInfo->
hObjectMapped)->IsAsyncHandle());
//INET_ASSERT(dwWaitTime <= 60000);
error = WaitForSingleObject(hEvent, dwWaitTime);
DPRINTF("%#x: %#x: sync waiter unblocked - error = %d\n",
GetCurrentThreadId(),
Fsm,
error
);
}
if (error == STATUS_TIMEOUT) {
DPRINTF("%#x: %#x: %s: %d+%d/%d: timed out %#x (%s FSM %#x %s)\n",
GetCurrentThreadId(),
Fsm,
GetHostName(),
m_ConnectionsAvailable,
ElementsOnSerializedList(&m_KeepAliveList),
m_ConnectionLimit,
GetCurrentThreadId(),
Fsm->MapType(),
Fsm,
Fsm->MapState()
);
RemoveWaiter(lpThreadInfo->ThreadId);
error = ERROR_WINHTTP_TIMEOUT;
}
BOOL bOk;
bOk = CloseHandle(hEvent);
INET_ASSERT(bOk);
if (error == WAIT_OBJECT_0) {
DPRINTF("%#x: %#x: sync requester trying again\n",
GetCurrentThreadId(),
Fsm
);
fsm.SetState(FSM_STATE_CONTINUE);
goto try_again;
}
}
}
exit:
//
// if we are returning a (keep-alive) socket that has a different blocking
// mode from that requested, change it
//
if (pSocket != NULL) {
if ((pSocket->GetFlags() & SF_NON_BLOCKING)
^ (fsm.m_dwSocketFlags & SF_NON_BLOCKING)) {
DEBUG_PRINT(SESSION,
INFO,
("different blocking modes requested: %#x, %#x\n",
fsm.m_dwSocketFlags,
pSocket->GetFlags()
));
DPRINTF("%#x: %#x: *** changing socket %#x to %sBLOCKING\n",
GetCurrentThreadId(),
Fsm,
pSocket->GetSocket(),
fsm.m_dwSocketFlags & SF_NON_BLOCKING ? "NON-" : ""
);
if (!(GlobalRunningNovellClient32 && !GlobalNonBlockingClient32)) {
pSocket->SetNonBlockingMode(fsm.m_dwSocketFlags & SF_NON_BLOCKING);
}
}
*fsm.m_lplpSocket = pSocket;
}
if (bUnlockList) {
m_Waiters.Release();
}
quit:
if (error != ERROR_IO_PENDING) {
fsm.SetDone();
}
DPRINTF("%#x: %#x: %s: %d+%d/%d: get: %d, %#x, %d\n",
GetCurrentThreadId(),
Fsm,
GetHostName(),
m_ConnectionsAvailable,
ElementsOnSerializedList(&m_KeepAliveList),
m_ConnectionLimit,
error,
pSocket ? pSocket->GetSocket() : 0,
m_Waiters.Count()
);
PERF_LEAVE(GetConnection);
DEBUG_LEAVE(error);
return error;
}
DWORD
CServerInfo::ReleaseConnection(
IN ICSocket * lpSocket OPTIONAL
)
/*++
Routine Description:
Returns a keep-alive connection to the pool, or allows another requester to
create a connection
Arguments:
lpSocket - pointer to ICSocket if we are returning a keep-alive connection
Return Value:
DWORD
Success - ERROR_SUCCESS
Failure -
--*/
{
DEBUG_ENTER((DBG_SESSION,
Dword,
"CServerInfo::ReleaseConnection",
"{%q [%d+%d/%d]} %#x [%#x]",
GetHostName(),
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit(),
lpSocket,
lpSocket ? lpSocket->GetSocket() : 0
));
PERF_ENTER(ReleaseConnection);
DWORD error = ERROR_SUCCESS;
BOOL bRelease = FALSE;
if (!m_Waiters.Acquire()) {
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
//
// quite often (at least with catapult proxy based on IIS) the server may
// drop the connection even though it indicated it would keep it open. This
// typically happens on 304 (frequent) and 302 (less so) responses. If we
// determine the server has dropped the connection then throw it away and
// allow the app to create a new one
//
if (lpSocket != NULL) {
if (lpSocket->IsClosed() || lpSocket->IsReset()) {
DEBUG_PRINT(SESSION,
INFO,
("socket %#x already dead - throwing it out\n",
lpSocket->GetSocket()
));
DPRINTF("%#x: socket %#x: already reset\n",
GetCurrentThreadId(),
lpSocket->GetSocket()
);
//dprintf("ReleaseConnection: destroying already closed socket %#x\n", lpSocket->GetSocket());
BOOL bDestroyed = lpSocket->Dereference();
INET_ASSERT(bDestroyed);
lpSocket = NULL;
} else {
//
// if we are returning a keep-alive socket, put it in non-blocking
// mode if not already. Typically, Internet Explorer uses non-blocking
// sockets. In the infrequent cases where we want a blocking socket
// - mainly when doing java downloads - we will convert the socket
// to blocking mode when we get it from the pool
//
if (!lpSocket->IsNonBlocking()) {
DPRINTF("%#x: ***** WARNING: releasing BLOCKING k-a socket %#x\n",
GetCurrentThreadId(),
lpSocket->GetSocket()
);
if (!(GlobalRunningNovellClient32 && !GlobalNonBlockingClient32)) {
lpSocket->SetNonBlockingMode(TRUE);
}
}
}
}
if (lpSocket != NULL) {
DPRINTF("%#x: releasing K-A %#x (%d+%d/%d)\n",
GetCurrentThreadId(),
lpSocket ? lpSocket->GetSocket() : 0,
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit()
);
INET_ASSERT(lpSocket->IsOpen());
INET_ASSERT(!lpSocket->IsOnList());
//INET_ASSERT(!lpSocket->IsReset());
lpSocket->SetKeepAlive();
DEBUG_PRINT(SESSION,
INFO,
("releasing keep-alive socket %#x\n",
lpSocket->GetSocket()
));
lpSocket->SetExpiryTime(GlobalKeepAliveSocketTimeout);
INET_ASSERT(!IsOnSerializedList(&m_KeepAliveList, lpSocket->List()));
if (!InsertAtTailOfSerializedList(&m_KeepAliveList, lpSocket->List()))
{
DEBUG_PRINT(SESSION,
INFO,
("not enough memory to release %#x to k-a pool\n",
lpSocket->GetSocket()
));
lpSocket->Dereference();
if (!UnlimitedConnections())
{
++m_ConnectionsAvailable;
}
}
lpSocket = NULL;
INET_ASSERT(UnlimitedConnections()
? TRUE
: (KeepAliveConnections() <= ConnectionLimit()));
bRelease = TRUE;
} else {
DPRINTF("%#x: releasing connection (%d+%d/%d)\n",
GetCurrentThreadId(),
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit()
);
if (!UnlimitedConnections()) {
++m_ConnectionsAvailable;
}
CHECK_CONNECTION_COUNT();
bRelease = TRUE;
}
if (bRelease && !UnlimitedConnections()) {
CHECK_CONNECTION_COUNT();
CConnectionWaiter * pWaiter = (CConnectionWaiter *)m_Waiters.RemoveHead();
if (pWaiter != NULL) {
DEBUG_PRINT(SESSION,
INFO,
("unblocking %s waiter %#x, pri=%d\n",
pWaiter->IsSync() ? "SYNC" : "ASYNC",
pWaiter->Id(),
pWaiter->GetPriority()
));
DPRINTF("%#x: Unblocking %s connection waiter %#x, pri=%d\n",
GetCurrentThreadId(),
pWaiter->IsSync() ? "Sync" : "Async",
pWaiter->Id(),
pWaiter->GetPriority()
);
if (pWaiter->IsSync()) {
pWaiter->Signal();
} else {
DWORD n = UnblockWorkItems(1, (DWORD_PTR)pWaiter, ERROR_SUCCESS);
//INET_ASSERT(n == 1);
}
delete pWaiter;
} else {
DEBUG_PRINT(SESSION,
INFO,
("no waiters\n"
));
DPRINTF("%#x: !!! NOT unblocking connection waiter\n",
GetCurrentThreadId()
);
}
} else {
DPRINTF("%#x: !!! NOT releasing or unlimited?\n",
GetCurrentThreadId()
);
DEBUG_PRINT(SESSION,
INFO,
("bRelease = %B, UnlimitedConnections() = %B\n",
bRelease,
UnlimitedConnections()
));
}
DEBUG_PRINT(SESSION,
INFO,
("avail+k-a/limit = %d+%d/%d\n",
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit()
));
if (IsNewLimit()) {
UpdateConnectionLimit();
}
m_Waiters.Release();
quit:
PERF_LEAVE(ReleaseConnection);
DEBUG_LEAVE(error);
DPRINTF("%#x: %s: %d+%d/%d: rls %#x: %d, %d\n",
GetCurrentThreadId(),
GetHostName(),
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit(),
lpSocket ? lpSocket->GetSocket() : 0,
error,
m_Waiters.Count()
);
return error;
}
VOID
CServerInfo::RemoveWaiter(
IN DWORD_PTR dwId
)
/*++
Routine Description:
Removes a CConnectionWaiter corresponding to the FSM
Arguments:
dwId - waiter id to match
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::RemoveWaiter",
"%#x",
dwId
));
if (!m_Waiters.Acquire())
goto quit;
CConnectionWaiter * pWaiter;
BOOL found = FALSE;
for (pWaiter = (CConnectionWaiter *)m_Waiters.Head();
pWaiter != (CConnectionWaiter *)m_Waiters.Self();
pWaiter = (CConnectionWaiter *)pWaiter->Next()) {
if (pWaiter->Id() == dwId) {
m_Waiters.Remove((CPriorityListEntry *)pWaiter);
delete pWaiter;
found = TRUE;
break;
}
}
m_Waiters.Release();
quit:
//INET_ASSERT(found);
DEBUG_LEAVE(0);
}
//
// private CServerInfo methods
//
ICSocket *
CServerInfo::FindKeepAliveConnection(
IN DWORD dwSocketFlags,
IN INTERNET_PORT nPort,
IN LPSTR pszTunnelServer
)
/*++
Routine Description:
Find a keep-alive connection with the requested attributes and port number
Arguments:
dwSocketFlags - socket type flags (e.g. SF_SECURE)
nPort - port to server
pszTunnelServer - hostname of server through SSL tunnel, or
NULL if not checked.
Return Value:
ICSocket *
--*/
{
DPRINTF("%#x: *** looking for K-A connection\n", GetCurrentThreadId());
DEBUG_ENTER((DBG_SESSION,
Pointer,
"CServerInfo::FindKeepAliveConnection",
"{%q} %#x, %d",
GetHostName(),
dwSocketFlags,
nPort
));
ICSocket * pSocket = NULL;
BOOL bFound = FALSE;
//
// don't check whether socket is non-blocking - we only really want to match
// on secure/non-secure. Possible flags to check on are:
//
// SF_ENCRYPT - should be subsumed by SF_SECURE
// SF_DECRYPT - should be subsumed by SF_SECURE
// SF_NON_BLOCKING - this isn't criterion for match
// SF_CONNECTIONLESS - not implemented?
// SF_AUTHORIZED - must be set if authorized & in pool
// SF_SECURE - opened for SSL/PCT if set
// SF_KEEP_ALIVE - must be set
// SF_TUNNEL - must be set if we're looking for a CONNECT tunnel to proxy
//
dwSocketFlags &= ~SF_NON_BLOCKING;
if (!LockSerializedList(&m_KeepAliveList))
goto quit;
PLIST_ENTRY pEntry;
for (pEntry = HeadOfSerializedList(&m_KeepAliveList);
pEntry != (PLIST_ENTRY)SlSelf(&m_KeepAliveList);
pEntry = pEntry->Flink) {
pSocket = ContainingICSocket(pEntry);
INET_ASSERT(pSocket->IsKeepAlive());
//
// We make sure the socket we request is the correct socket,
// Match() is a bit confusing and needs a bit of explaining,
// Match IS NOT AN EXACT MATCH, it mearly checks to make sure
// that the requesting flags (dwSocketFlags) are found in the
// socket flags. So this can lead to a secure socket being returned
// on a non-secure open request, now realistically this doesn't happen
// because of the port number. But in the case of tunnelling this may be
// an issue, so we add an additional check to make sure that we only
// get a tunneled socket to a proxy if we specifically request one.
//
if (pSocket->Match(dwSocketFlags)
&& (pSocket->GetPort() == nPort)
&& pSocket->MatchTunnelSemantics(dwSocketFlags, pszTunnelServer)
&& RemoveFromSerializedList(&m_KeepAliveList, pSocket->List())) {
INET_ASSERT(!IsOnSerializedList(&m_KeepAliveList, pSocket->List()));
bFound = TRUE;
DEBUG_PRINT(SESSION,
INFO,
("returning keep-alive socket %#x\n",
pSocket->GetSocket()
));
DPRINTF("%#x: *** %s keep-alive connection %#x (%d/%d), wantf=%#x, gotf=%#x\n",
GetCurrentThreadId(),
GetHostName(),
pSocket->GetSocket(),
AvailableConnections(),
ConnectionLimit(),
dwSocketFlags,
pSocket->GetFlags()
);
break;
}
}
UnlockSerializedList(&m_KeepAliveList);
if (!bFound) {
pSocket = NULL;
}
quit:
DEBUG_LEAVE(pSocket);
return pSocket;
}
BOOL
CServerInfo::KeepAliveWaiters(
VOID
)
/*++
Routine Description:
Determine if any of the waiters on the list are for keep-alive connections
Arguments:
None.
Return Value:
BOOL
--*/
{
DEBUG_ENTER((DBG_SESSION,
Bool,
"CServerInfo::KeepAliveWaiters",
NULL
));
BOOL found = FALSE;
CConnectionWaiter * pWaiter;
if (!m_Waiters.Acquire())
goto quit;
for (pWaiter = (CConnectionWaiter *)m_Waiters.Head();
pWaiter != (CConnectionWaiter *)m_Waiters.Self();
pWaiter = (CConnectionWaiter *)pWaiter->Next()) {
if (pWaiter->IsKeepAlive()) {
found = TRUE;
break;
}
}
m_Waiters.Release();
quit:
DEBUG_LEAVE(found);
return found;
}
VOID
CServerInfo::UpdateConnectionLimit(
VOID
)
/*++
Routine Description:
Change connection limit to new limit
Assumes: 1. Caller has acquired this object before calling this function
Arguments:
None.
Return Value:
None.
--*/
{
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::UpdateConnectionLimit",
"{%q: %d=>%d (%d+%d)}",
GetHostName(),
ConnectionLimit(),
GetNewLimit(),
AvailableConnections(),
KeepAliveConnections()
));
LONG difference = GetNewLimit() - ConnectionLimit();
//
// BUGBUG - only handling increases in limit for now
//
INET_ASSERT(difference > 0);
if (difference > 0) {
m_ConnectionsAvailable += difference;
}
m_ConnectionLimit = m_NewLimit;
DEBUG_PRINT(SESSION,
INFO,
("%q: new: %d+%d/%d\n",
GetHostName(),
AvailableConnections(),
KeepAliveConnections(),
ConnectionLimit()
));
DEBUG_LEAVE(0);
}
VOID
CServerInfo::PurgeKeepAlives(
IN DWORD dwForce
)
/*++
Routine Description:
Purges any timed-out keep-alive connections
Arguments:
dwForce - force to apply when purging. Value can be:
PKA_NO_FORCE - only purge timed-out sockets or sockets in
close-wait state (default)
PKA_NOW - purge all sockets
PKA_AUTH_FAILED - purge sockets that have been marked as failing
authentication
Return Value:
None.
--*/
{
//dprintf("%#x PurgeKeepAlives(%d)\n", GetCurrentThreadId(), dwForce);
DEBUG_ENTER((DBG_SESSION,
None,
"CServerInfo::PurgeKeepAlives",
"{%q [ref=%d, k-a=%d]} %s [%d]",
GetHostName(),
ReferenceCount(),
KeepAliveConnections(),
(dwForce == PKA_NO_FORCE) ? "NO_FORCE"
: (dwForce == PKA_NOW) ? "NOW"
: (dwForce == PKA_AUTH_FAILED) ? "AUTH_FAILED"
: "?",
dwForce
));
if (IsKeepAliveListInitialized()) {
INET_ASSERT(ReferenceCount() >= 1);
if (!m_Waiters.Acquire())
goto quit;
if (!LockSerializedList(&m_KeepAliveList))
goto Cleanup;
PLIST_ENTRY last = (PLIST_ENTRY)SlSelf(&m_KeepAliveList);
DWORD ticks = GetTickCountWrap();
for (PLIST_ENTRY pEntry = HeadOfSerializedList(&m_KeepAliveList);
pEntry != (PLIST_ENTRY)SlSelf(&m_KeepAliveList);
pEntry = last->Flink) {
ICSocket * pSocket = ContainingICSocket(pEntry);
BOOL bDelete;
if (pSocket->IsReset()) {
//dprintf("%q: socket %#x/%d CLOSE-WAIT\n", GetHostName(), pSocket->GetSocket(), pSocket->GetSourcePort());
bDelete = TRUE;
} else if (dwForce == PKA_NO_FORCE) {
bDelete = pSocket->HasExpired(ticks);
} else if (dwForce == PKA_NOW) {
bDelete = TRUE;
} else if (dwForce == PKA_AUTH_FAILED) {
bDelete = pSocket->IsAuthorized();
} else {
INET_ASSERT(FALSE); // invalid value for dwForce!
bDelete = TRUE;
}
if (bDelete) {
//dprintf("%q: socket %#x/%d. Close-Wait=%B, Expired=%B, Now=%B, Auth=%B\n",
// GetHostName(),
// pSocket->GetSocket(),
// pSocket->GetSourcePort(),
// pSocket->IsReset(),
// pSocket->HasExpired(ticks),
// (dwForce == PKA_NOW),
// (dwForce == PKA_AUTH_FAILED) && pSocket->IsAuthorized()
// );
DEBUG_PRINT(SESSION,
INFO,
("purging keep-alive socket %#x/%d: Close-Wait=%B, Expired=%B, Now=%B, Auth=%B\n",
pSocket->GetSocket(),
pSocket->GetSourcePort(),
pSocket->IsReset(),
pSocket->HasExpired(ticks),
(dwForce == PKA_NOW),
(dwForce == PKA_AUTH_FAILED) && pSocket->IsAuthorized()
));
if (RemoveFromSerializedList(&m_KeepAliveList, pEntry))
{
BOOL bDestroyed;
bDestroyed = pSocket->Dereference();
INET_ASSERT(bDestroyed);
if (!UnlimitedConnections()) {
++m_ConnectionsAvailable;
INET_ASSERT(m_ConnectionsAvailable <= m_ConnectionLimit);
}
}
else
{
DEBUG_PRINT(SESSION,
INFO,
("k-a socket %#x couldn't be removed from the list\n",
pSocket->GetSocket()
));
}
} else {
DEBUG_PRINT(SESSION,
INFO,
("socket %#x/%d expires in %d mSec\n",
pSocket->GetSocket(),
pSocket->GetSourcePort(),
pSocket->GetExpiryTime() - ticks
));
last = pEntry;
}
}
UnlockSerializedList(&m_KeepAliveList);
Cleanup:
m_Waiters.Release();
}
quit:
DEBUG_LEAVE(0);
}
//
// friend functions
//
CServerInfo *
ContainingServerInfo(
IN LPVOID lpAddress
)
/*++
Routine Description:
Returns address of CServerInfo given address of m_List
Arguments:
lpAddress - address of m_List
Return Value:
CServerInfo *
--*/
{
return CONTAINING_RECORD(lpAddress, CServerInfo, m_List);
}