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

463 lines
11 KiB
C++

class ICSocket;
//
// class implementations
//
/*++
Class Description:
This class defines the INTERNET_HANDLE_OBJECT.
Private Member functions:
None.
Public Member functions:
--*/
class INTERNET_HANDLE_BASE : public HANDLE_OBJECT {
friend class INTERNET_HANDLE_OBJECT;
private:
//
// Passport Auth package's "Session Handle"
//
PP_CONTEXT _PPContext;
//
// _IsCopy - TRUE if this is part of a derived object handle (e.g. a
// connect handle object)
// BUGBUG - post-beta cleanup - combine into bitfield
//
BOOL _IsCopy;
//
// _UserAgent - name by why which the application wishes to be known to
// HTTP servers. Provides the User-Agent header unless overridden by
// specific User-Agent header from app
//
ICSTRING _UserAgent;
//
// _ProxyInfo - maintains the proxy server and bypass lists
//
PROXY_INFO * _ProxyInfo;
//
// _ProxyInfoResourceLock - must acquire for exclusive access in order to
// modify the proxy info
//
RESOURCE_LOCK _ProxyInfoResourceLock;
BOOL AcquireProxyInfo(BOOL bExclusiveMode) {
return _ProxyInfoResourceLock.Acquire(bExclusiveMode);
}
VOID ReleaseProxyInfo(VOID) {
_ProxyInfoResourceLock.Release();
}
VOID SafeDeleteProxyInfo(VOID) {
DEBUG_ENTER((DBG_OBJECTS,
None,
"SafeDeleteProxyInfo",
""
));
if ((_ProxyInfo != NULL) && (_ProxyInfo != PROXY_INFO_DIRECT)) {
if (AcquireProxyInfo(TRUE)) { // must check since asking for exclusive access
if (!IsProxyGlobal()) {
if (_ProxyInfo != NULL) {
delete _ProxyInfo;
}
}
_ProxyInfo = NULL;
ReleaseProxyInfo();
}
}
DEBUG_LEAVE(0);
}
//
// _dwInternetOpenFlags - flags from InternetOpen()
//
// BUGBUG - there should only be ONE flags DWORD for all handles descended
// from this one. This is it
// Rename to just _Flags, or _OpenFlags
//
DWORD _dwInternetOpenFlags;
//
// _WinsockLoaded - TRUE if we managed to successfully load winsock
//
//
// BUGBUG - post-beta cleanup - combine into bitfield
//
BOOL _WinsockLoaded;
//
// _pICSocket - pointer to ICSocket for new HTTP async code
//
ICSocket * _pICSocket;
protected:
//
// _Async - TRUE if the InternetOpen() handle, and all handles descended
// from it, support asynchronous I/O
//
//
// BUGBUG - post-beta cleanup - get from flags
//
BOOL _Async;
DWORD _MaxConnectionsPerServer;
DWORD _MaxConnectionsPer1_0Server;
//
// _DataAvailable - the number of bytes that can be read from this handle
// (i.e. only protocol handles) immediately. This avoids a read request
// being made asynchronously if it can be satisfied immediately
//
DWORD _DataAvailable;
//
// _EndOfFile - TRUE when we have received all data for this request. This
// is used to avoid the API having to perform an extraneous read (possibly
// asynchronously) just to discover that we reached end-of-file already
//
//
// BUGBUG - post-beta cleanup - combine into bitfield
//
BOOL _EndOfFile;
//
// _StatusCallback - we now maintain callbacks on a per-handle basis. The
// callback address comes from the parent handle or the DLL if this is an
// InternetOpen() handle. The status callback can be changed for an
// individual handle object using the ExchangeStatusCallback() method
// (called from InternetSetStatusCallback())
//
//
// BUGBUG - this should go in HANDLE_OBJECT
//
WINHTTP_STATUS_CALLBACK _StatusCallback;
BOOL _StatusCallbackType;
DWORD _dwStatusCallbackFlags;
// Codepage: required for conversion of object from unicode to mbcs in WinHttpOpenRequest.
DWORD _dwCodePage;
HANDLE _ThreadToken;
public:
INTERNET_HANDLE_BASE(
LPCSTR UserAgent,
DWORD AccessMethod,
LPSTR ProxyName,
LPSTR ProxyBypass,
DWORD Flags
);
INTERNET_HANDLE_BASE(INTERNET_HANDLE_BASE *INetObj);
virtual ~INTERNET_HANDLE_BASE(VOID);
//
// BUGBUG - rfirth 04/05/96 - remove virtual functions
//
// For the most part, these functions aren't required to be
// virtual. They should just be moved to the relevant handle type
// (e.g. FTP_FILE_HANDLE_OBJECT). Even GetHandleType() is overkill.
// Replacing with a method that just returns
// HANDLE_OBJECT::_ObjectType would be sufficient
//
virtual HINTERNET_HANDLE_TYPE GetHandleType(VOID)
{
return TypeInternetHandle;
}
HANDLE GetThreadToken(void) const { return _ThreadToken; }
PP_CONTEXT GetPPContext(void) const {
return _PPContext;
}
void SetPPContext(PP_CONTEXT PPContext) {
_PPContext = PPContext;
}
BOOL IsCopy(VOID) const {
return _IsCopy;
}
VOID GetUserAgent(LPSTR Buffer, LPDWORD BufferLength) {
_UserAgent.CopyTo(Buffer, BufferLength);
}
LPSTR GetUserAgent(VOID) {
return _UserAgent.StringAddress();
}
LPSTR GetUserAgent(LPDWORD lpdwLength) {
*lpdwLength = _UserAgent.StringLength();
return _UserAgent.StringAddress();
}
VOID SetUserAgent(LPSTR lpszUserAgent) {
INET_ASSERT(lpszUserAgent != NULL);
_UserAgent = lpszUserAgent;
}
BOOL IsProxy(VOID) const {
//
// we can return this info without acquiring the critical section
//
return ((_ProxyInfo != NULL) && (_ProxyInfo != PROXY_INFO_DIRECT)) ?
_ProxyInfo->IsProxySettingsConfigured()
: FALSE;
}
BOOL IsProxyGlobal(VOID) const {
INET_ASSERT(g_pGlobalProxyInfo != NULL);
return (_ProxyInfo == g_pGlobalProxyInfo) ? TRUE : FALSE;
}
PROXY_INFO * GetProxyInfo(VOID) const {
return _ProxyInfo;
}
VOID SetProxyInfo(PROXY_INFO * ProxyInfo) {
_ProxyInfo = ProxyInfo;
}
VOID ResetProxyInfo(VOID) {
SetProxyInfo(NULL);
}
DWORD
Refresh();
DWORD
SetProxyInfo(
IN DWORD dwAccessType,
IN LPCSTR lpszProxy OPTIONAL,
IN LPCSTR lpszProxyBypass OPTIONAL
);
DWORD
GetProxyStringInfo(
OUT LPVOID lpBuffer,
IN OUT LPDWORD lpdwBufferLength
);
DWORD
GetProxyInfo(
IN AUTO_PROXY_ASYNC_MSG **ppQueryForProxyInfo
);
BOOL
RedoSendRequest(
IN OUT LPDWORD lpdwError,
IN DWORD dwSecureStatus,
IN AUTO_PROXY_ASYNC_MSG *pQueryForProxyInfo,
IN CServerInfo *pOriginServer,
IN CServerInfo *pProxyServer
);
VOID SetContext(DWORD_PTR NewContext) {
_Context = NewContext;
}
BOOL IsAsyncHandle(VOID) {
return _Async;
}
VOID SetAvailableDataLength(DWORD Amount) {
INET_ASSERT((int)Amount >= 0);
_DataAvailable = Amount;
}
DWORD AvailableDataLength(VOID) const {
INET_ASSERT((int)_DataAvailable >= 0);
return _DataAvailable;
}
BOOL IsDataAvailable(VOID) {
INET_ASSERT((int)_DataAvailable >= 0);
return (_DataAvailable != 0) ? TRUE : FALSE;
}
VOID ReduceAvailableDataLength(DWORD Amount) {
//
// why would Amount be > _DataAvailable?
//
if (Amount > _DataAvailable) {
_DataAvailable = 0;
} else {
_DataAvailable -= Amount;
}
INET_ASSERT((int)_DataAvailable >= 0);
}
VOID IncreaseAvailableDataLength(DWORD Amount) {
_DataAvailable += Amount;
INET_ASSERT((int)_DataAvailable >= 0);
}
VOID SetEndOfFile(VOID) {
_EndOfFile = TRUE;
}
VOID ResetEndOfFile(VOID) {
_EndOfFile = FALSE;
}
BOOL IsEndOfFile(VOID) const {
return _EndOfFile;
}
WINHTTP_STATUS_CALLBACK GetStatusCallback(VOID)
{
return _StatusCallback;
}
BOOL IsUnicodeStatusCallback()
{
return _StatusCallbackType;
}
BOOL IsNotificationEnabled(DWORD dwStatus)
{
return (_dwStatusCallbackFlags & dwStatus);
}
VOID ResetStatusCallback(VOID) {
_StatusCallback = NULL;
_StatusCallbackType = FALSE;
_dwStatusCallbackFlags = 0;
}
//VOID AcquireAsyncSpinLock(VOID);
//
//VOID ReleaseAsyncSpinLock(VOID);
DWORD ExchangeStatusCallback(LPWINHTTP_STATUS_CALLBACK lpStatusCallback, BOOL fType, DWORD dwFlags);
//DWORD AddAsyncRequest(BOOL fNoCallbackOK);
//
//VOID RemoveAsyncRequest(VOID);
//
//DWORD GetAsyncRequestCount(VOID) {
//
// //
// // it doesn't matter about locking this variable - it can change before
// // we have returned it to the caller anyway
// //
//
// return _PendingAsyncRequests;
//}
// random methods on flags
DWORD GetInternetOpenFlags() {
return _dwInternetOpenFlags;
}
VOID
SetAbortHandle(
IN ICSocket * pSocket
);
ICSocket * GetAbortHandle(VOID) const {
return _pICSocket;
}
VOID
ResetAbortHandle(
VOID
);
BOOL IsFromCacheTimeoutSet(VOID) const {
return FALSE;
}
DWORD
GetMaxConnectionsPerServer(DWORD dwOption)
{
INET_ASSERT(dwOption == WINHTTP_OPTION_MAX_CONNS_PER_SERVER ||
dwOption == WINHTTP_OPTION_MAX_CONNS_PER_1_0_SERVER);
return (dwOption == WINHTTP_OPTION_MAX_CONNS_PER_SERVER) ?
_MaxConnectionsPerServer
: _MaxConnectionsPer1_0Server;
}
void
SetMaxConnectionsPerServer(DWORD dwOption, DWORD dwMaxConnections)
{
INET_ASSERT(dwOption == WINHTTP_OPTION_MAX_CONNS_PER_SERVER ||
dwOption == WINHTTP_OPTION_MAX_CONNS_PER_1_0_SERVER);
if (dwOption == WINHTTP_OPTION_MAX_CONNS_PER_SERVER)
_MaxConnectionsPerServer = dwMaxConnections;
else
_MaxConnectionsPer1_0Server = dwMaxConnections;
}
DWORD GetCodePage() { return _dwCodePage; }
void SetCodePage(DWORD dwCodePage) { _dwCodePage = dwCodePage; }
void AbortSocket (void);
};