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

255 lines
7.3 KiB
C

//-----------------------------------------------------------------------------
// Copyright (C) Microsoft Corporation, 1997 - 1999
//
// rpcisapi.cxx
//
// IIS ISAPI extension part of the RPC proxy over HTTP.
//
// Exports:
//
// BOOL WINAPI GetExtensionVersion( HSE_VERSION_INFO *pVer )
//
// Returns the version of the spec that this server was built with.
//
// BOOL WINAPI HttpExtensionProc( EXTENSION_CONTROL_BLOCK *pECB )
//
// This function does all of the work.
//
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// Includes:
//-----------------------------------------------------------------------------
#include <sysinc.h>
#include <winsock2.h>
#include <rpc.h>
#include <rpcdcep.h>
#include <rpcerrp.h>
#include <httpfilt.h>
#include <httpext.h>
#include <mbstring.h>
#include "ecblist.h"
#include "filter.h"
#include "olist.h"
#include "server.h"
//-----------------------------------------------------------------------------
// Globals:
//-----------------------------------------------------------------------------
extern SERVER_INFO *g_pServerInfo;
extern BOOL g_fIsIIS6;
//-----------------------------------------------------------------------------
// GetExtensionVersion()
//
//-----------------------------------------------------------------------------
BOOL WINAPI GetExtensionVersion( HSE_VERSION_INFO *pVer )
{
pVer->dwExtensionVersion = MAKELONG( HSE_VERSION_MINOR, HSE_VERSION_MAJOR );
ASSERT( sizeof(EXTENSION_DESCRIPTION) <= HSE_MAX_EXT_DLL_NAME_LEN );
lstrcpy( pVer->lpszExtensionDesc, EXTENSION_DESCRIPTION );
return TRUE;
}
//-----------------------------------------------------------------------------
// ReplyToClient()
//
//-----------------------------------------------------------------------------
BOOL ReplyToClient( EXTENSION_CONTROL_BLOCK *pECB,
char *pBuffer,
DWORD *pdwSize,
DWORD *pdwStatus )
{
DWORD dwFlags = (HSE_IO_SYNC | HSE_IO_NODELAY);
if (!pECB->WriteClient(pECB->ConnID,pBuffer,pdwSize,dwFlags))
{
*pdwStatus = GetLastError();
#ifdef DBG_ERROR
DbgPrint("ReplyToClient(): failed: %d\n",*pdwStatus);
#endif
return FALSE;
}
*pdwStatus = HSE_STATUS_SUCCESS;
return TRUE;
}
//-----------------------------------------------------------------------------
// ParseQueryString()
//
// The query string is in the format:
//
// <Index_of_pOverlapped>
//
// Where the index is in ASCII Hex. The index is read and converted back
// to a DWORD then used to locate the SERVER_OVERLAPPED. If its found,
// return TRUE, else return FALSE.
//
// NOTE: "&" is the parameter separator if multiple parameters are passed.
//-----------------------------------------------------------------------------
BOOL ParseQueryString( unsigned char *pszQuery,
SERVER_OVERLAPPED **ppOverlapped,
DWORD *pdwStatus )
{
DWORD dwIndex = 0;
pszQuery = AnsiHexToDWORD(pszQuery,&dwIndex,pdwStatus);
if (!pszQuery)
{
return FALSE;
}
*ppOverlapped = GetOverlapped(dwIndex);
if (*ppOverlapped == NULL)
{
return FALSE;
}
return TRUE;
}
//-----------------------------------------------------------------------------
// HttpExtensionProc()
//
//-----------------------------------------------------------------------------
DWORD WINAPI HttpExtensionProc( EXTENSION_CONTROL_BLOCK *pECB )
{
DWORD dwStatus;
DWORD dwFlags;
DWORD dwSize;
SERVER_INFO *pServerInfo = g_pServerInfo;
SERVER_OVERLAPPED *pOverlapped = NULL;
HSE_SEND_HEADER_EX_INFO HeaderEx;
CHAR *pVerb;
pECB->dwHttpStatusCode = STATUS_CONNECTION_OK;
if (g_fIsIIS6)
{
pVerb = RPC_CONNECT;
}
else
{
pVerb = POST_STR;
}
//
// The RPC request must be a post (or RPC_CONNECT for 6.0):
//
if (_mbsicmp(pECB->lpszMethod,pVerb))
{
dwSize = sizeof(STATUS_MUST_BE_POST_STR) - 1; // don't want to count trailing 0.
ReplyToClient(pECB,STATUS_MUST_BE_POST_STR,&dwSize,&dwStatus);
return HSE_STATUS_SUCCESS;
}
//
// Make sure there is no data from the initial BIND in the ECB data buffer:
//
// ASSERT(pECB->cbTotalBytes == 0);
//
// Get the connect parameters:
//
if (!ParseQueryString(pECB->lpszQueryString,&pOverlapped,&dwStatus))
{
dwSize = sizeof(STATUS_POST_BAD_FORMAT_STR) - 1; // don't want to count trailing 0.
ReplyToClient(pECB,STATUS_POST_BAD_FORMAT_STR,&dwSize,&dwStatus);
return HSE_STATUS_SUCCESS;
}
pOverlapped->pECB = pECB;
//
// Add the new ECB to the Active ECB List:
//
if (!AddToECBList(g_pServerInfo->pActiveECBList,pECB))
{
#ifdef DBG_ERROR
DbgPrint("HttpExtensionProc(): AddToECBList() failed\n");
#endif
FreeOverlapped(pOverlapped);
pECB->dwHttpStatusCode = STATUS_SERVER_ERROR;
return HSE_STATUS_ERROR;
}
//
// Submit the first client async read:
//
if (!StartAsyncClientRead(pECB,pOverlapped->pConn,&dwStatus))
{
#ifdef DBG_ERROR
DbgPrint("HttpExtensionProc(): StartAsyncClientRead() failed %d\n",dwStatus);
#endif
FreeOverlapped(pOverlapped);
CleanupECB(pECB);
pECB->dwHttpStatusCode = STATUS_SERVER_ERROR;
return HSE_STATUS_ERROR;
}
//
// Post the first server read on the new socket:
//
IncrementECBRefCount(pServerInfo->pActiveECBList,pECB);
if (!SubmitNewRead(pServerInfo,pOverlapped,&dwStatus))
{
#ifdef DBG_ERROR
DbgPrint("HttpExtensionProc(): SubmitNewRead() failed %d\n",dwStatus);
#endif
FreeOverlapped(pOverlapped);
CleanupECB(pECB);
pECB->dwHttpStatusCode = STATUS_SERVER_ERROR;
return HSE_STATUS_ERROR;
}
//
// Make sure the server receive thread is up and running:
//
if (!CheckStartReceiveThread(g_pServerInfo,&dwStatus))
{
#ifdef DBG_ERROR
DbgPrint("HttpExtensionProc(): CheckStartReceiveThread() failed %d\n",dwStatus);
#endif
FreeOverlapped(pOverlapped);
CleanupECB(pECB);
pECB->dwHttpStatusCode = STATUS_SERVER_ERROR;
return HSE_STATUS_ERROR;
}
//
// Send back connection Ok to client, and also set fKeepConn to FALSE.
//
dwSize = sizeof(HeaderEx);
dwFlags = 0;
memset(&HeaderEx,0,dwSize);
HeaderEx.fKeepConn = FALSE;
if (!pECB->ServerSupportFunction(pECB->ConnID,
HSE_REQ_SEND_RESPONSE_HEADER_EX,
&HeaderEx,
&dwSize,
&dwFlags))
{
#ifdef DBG_ERROR
DbgPrint("HttpExtensionProc(): SSF(HSE_REQ_SEND_RESPONSE_HEADER_EX) failed %d\n",dwStatus);
#endif
FreeOverlapped(pOverlapped);
CleanupECB(pECB);
pECB->dwHttpStatusCode = STATUS_SERVER_ERROR;
return HSE_STATUS_ERROR;
}
return HSE_STATUS_PENDING;
}