Windows2000/private/inet/urlmon/perf/oinetperf/oinet.cxx
2020-09-30 17:12:32 +02:00

813 lines
22 KiB
C++

#include <windows.h>
#include <ole2.h>
#include <stdio.h>
#include <urlmon.h>
#include <string.h>
#include <malloc.h>
#include <urlmon.hxx>
#define IE5
#ifdef PRODUCT_PROF
extern "C" void _stdcall StartCAP(void);
extern "C" void _stdcall StopCAP(void);
extern "C" void _stdcall SuspendCAP(void);
extern "C" void _stdcall ResumeCAP(void);
extern "C" void _stdcall StartCAPAll(void);
extern "C" void _stdcall StopCAPAll(void);
#else
#define StartCAP()
#define StopCAP()
#define SuspendCAP()
#define ResumeCAP()
#define StartCAPAll()
#define StopCAPAll()
#endif
#define FLAG_TRACE 1
#define FLAG_DUMPDATA 2
#define MAX_DOWNLOADS 2000
#define MAX_URL INTERNET_MAX_URL_LENGTH
const INT BUF_SIZE = 2 * 1024;
const INT MAX_BUF_SIZE = 1024 * 16;
BOOL g_dwVerbose = 0;
DWORD g_dwNumDownloads = 1;
DWORD g_dwDownloads = 1;
DWORD g_dwTotalBytes = 0;
DWORD g_dwCacheFlag = BINDF_NOWRITECACHE | BINDF_GETNEWESTVERSION;
DWORD dwBuf_Size = BUF_SIZE;
LPWSTR g_pwzUrl = NULL;
LPCSTR g_pInfile = NULL;
LPCSTR g_pTitle = NULL;
LPCSTR g_pRun = NULL;
LPCSTR g_pModule = NULL;
HANDLE g_hCompleted;
__int64 g_ibeg = 0, g_iend, g_ifreq;
// Class: COInetProtocolHook
// Purpose: Sample Implementation of ProtocolSink and BindInfo
// interface for simplified urlmon async download
// Interfaces:
// [Needed For All]
// IOInetProtocolSink
// - provide sink for pluggable prot's callback
// IOInetBindInfo
// - provide the bind options
// [Needed For Http]
// IServiceProvider
// - used to query http specific services
// e.g. HttpNegotiate, Authentication, UIWindow
// IHttpNegotiate
// - http negotiation service, it has two methods,
// one is the http pluggable protocol asks the
// client for additional headers, the other is
// callback for returned http server status
// e.g 200, 401, etc.
// Author: DanpoZ (Danpo Zhang)
// History: 11-20-97 Created
// 05-19-98 Modified to act as performance test
// NOTE: IOInetXXXX == IInternetXXXX
// on the SDK, you will see IInternetXXXX, these are same
// interfaces
class COInetProtocolHook : public IOInetProtocolSink,
public IOInetBindInfo,
public IHttpNegotiate,
public IServiceProvider
{
public:
COInetProtocolHook(HANDLE g_hCompleted, IOInetProtocol* pProt);
virtual ~COInetProtocolHook();
// IUnknown methods
STDMETHODIMP QueryInterface(REFIID iid, void **ppvObj);
STDMETHODIMP_(ULONG) AddRef(void);
STDMETHODIMP_(ULONG) Release(void);
// IOInetProtocolSink methods
STDMETHODIMP Switch(PROTOCOLDATA *pStateInfo);
STDMETHODIMP ReportProgress(ULONG ulStatusCode, LPCWSTR szStatusText);
STDMETHODIMP ReportData(DWORD grfBSCF, ULONG ulProgress, ULONG ulProgressMax);
STDMETHODIMP ReportResult(HRESULT hrResult, DWORD dwError, LPCWSTR wzResult);
//IOInetBindInfo methods
STDMETHODIMP GetBindInfo(DWORD *grfBINDF, BINDINFO * pbindinfo);
STDMETHODIMP GetBindString(ULONG ulStringType, LPOLESTR *ppwzStr, ULONG cEl, ULONG *pcElFetched);
//IService Provider methods
STDMETHODIMP QueryService(REFGUID guidService, REFIID riid, void **ppvObj);
//IHttpNegotiate methods
STDMETHODIMP BeginningTransaction(LPCWSTR szURL, LPCWSTR szHeaders, DWORD dwReserved, LPWSTR *pszAdditionalHeaders);
STDMETHODIMP OnResponse(DWORD dwResponseCode, LPCWSTR szResponseHeaders, LPCWSTR szRequestHeaders, LPWSTR *pszAdditionalHeaders);
private:
IOInetProtocol* _pProt;
HANDLE _hCompleted;
CRefCount _CRefs;
};
typedef struct tagInfo
{
WCHAR wzUrl[INTERNET_MAX_URL_LENGTH];
IOInetProtocol* pProt;
COInetProtocolHook* pHook;
IOInetProtocolSink* pSink;
IOInetBindInfo* pBindInfo;
} INFO, *PINFO;
typedef BOOL(WINAPI *PFNSPA)(HANDLE, DWORD);
INFO Info[MAX_DOWNLOADS];
void MylstrcpyW(WCHAR *pwd, WCHAR *pws)
{
while (*pws)
{
*pwd++ = *pws++;
}
*pwd = 0;
}
WCHAR *MyDupA2W(LPSTR pa)
{
int i;
WCHAR *pw, *pwd;
i = lstrlen(pa);
pw = (WCHAR *)CoTaskMemAlloc((i + 1) * sizeof(WCHAR));
pwd = pw;
while (*pa)
{
*pwd++ = (WCHAR)*pa++;
}
*pwd++ = 0;
return pw;
}
void SetSingleProcessorAffinity()
{
PFNSPA pfn;
pfn = (PFNSPA)GetProcAddress(GetModuleHandleA("KERNEL32.DLL"), "SetProcessAffinityMask");
if (pfn)
{
pfn(GetCurrentProcess(), 1);
}
}
VOID DisplayUsage(char **argv)
{
printf("Usage: %s /u:url [/n:# /t:Title /r:RunStr /v:#]\n", argv[0]);
printf(" %s /i:infile [/t:Title /r:RunStr /v:#]\n", argv[0]);
printf(" /c - write to cache (default is NOWRITECACHE)\n");
printf(" /g - read from cache (default is GETNEWESTVERSION)\n");
printf(" /d - direct read (default uses QueryDataAvailable)\n");
printf(" /l:# - read buffer length\n");
printf(" /m:module - pre load module\n");
printf(" /n:# - download # times.\n");
printf(" /1 - single processor affinity (default multiprocessor)\n");
printf(" /v:# - verbose level 1=trace 2=dump data\n");
}
BOOL ProcessCommandLine(int argcIn, char **argvIn)
{
BOOL bRC = FALSE;
int argc = argcIn;
char **argv = argvIn;
argv++; argc--;
if (argc == 0)
{
DisplayUsage(argvIn);
return(FALSE);
}
while (argc > 0 && argv[0][0] == '/')
{
switch (argv[0][1])
{
case 'c':
case 'C':
g_dwCacheFlag &= ~BINDF_NOWRITECACHE;
break;
case 'g':
case 'G':
g_dwCacheFlag &= ~BINDF_GETNEWESTVERSION;
break;
case 'd':
case 'D':
g_dwCacheFlag |= BINDF_DIRECT_READ;
break;
case 'i':
case 'I':
g_pInfile = &argv[0][3];
bRC = TRUE;
break;
case 'l':
case 'L':
dwBuf_Size = atoi(&argv[0][3]);
if (dwBuf_Size > MAX_BUF_SIZE)
dwBuf_Size = MAX_BUF_SIZE;
break;
case 'm':
case 'M':
g_pModule = &argv[0][3];
break;
case 'n':
case 'N':
g_dwNumDownloads = (DWORD)atoi(&argv[0][3]);
g_dwNumDownloads = max(1, g_dwNumDownloads);
g_dwNumDownloads = min(MAX_DOWNLOADS, g_dwNumDownloads);
break;
case 'r':
case 'R':
g_pRun = &argv[0][3];
break;
case 't':
case 'T':
g_pTitle = &argv[0][3];
break;
case 'u':
case 'U':
g_pwzUrl = MyDupA2W(&argv[0][3]);
bRC = TRUE;
break;
case 'v':
case 'V':
g_dwVerbose = (DWORD)atoi(&argv[0][3]);
break;
case '1':
SetSingleProcessorAffinity();
break;
case '?':
default:
DisplayUsage(argvIn);
bRC = FALSE;
}
argv++; argc--;
}
return(bRC);
}
BOOL BuildInfoList(PINFO pInfo, DWORD dwNumDownloads)
{
DWORD i = 0;
if (g_pInfile != NULL)
{
TCHAR szName[INTERNET_MAX_URL_LENGTH + 1];
FILE *fp;
if ((fp = fopen(g_pInfile, "r")) == NULL)
{
printf("error opening file:%s GLE=%d\n", g_pInfile, GetLastError());
return NULL;
}
while (fgets(szName, INTERNET_MAX_URL_LENGTH, fp) != NULL)
{
if (szName[0] != '#')
{
szName[strlen(szName) - sizeof(char)] = '\0';
int rc;
rc = MultiByteToWideChar(CP_ACP, 0, szName, -1, (pInfo + i)->wzUrl, INTERNET_MAX_URL_LENGTH);
if (!rc)
{
(pInfo + i)->wzUrl[INTERNET_MAX_URL_LENGTH - 1] = 0;
wprintf(L"BuildInfoList:string too long; truncated to %s\n", (pInfo + i)->wzUrl);
}
i++;
}
}
g_dwNumDownloads = i;
fclose(fp);
}
else
{
for (i = 0; i < dwNumDownloads; i++)
{
MylstrcpyW((pInfo + i)->wzUrl, g_pwzUrl);
}
}
g_dwDownloads = g_dwNumDownloads;
return(TRUE);
}
HRESULT StartDownloads(IOInetSession* pSession, PINFO pInfo, DWORD dwNumDownloads)
{
HRESULT hr = NOERROR;
for (DWORD i = 0; i < dwNumDownloads; i++)
{
// Create a pluggable protocol
hr = pSession->CreateBinding(
NULL, // [in ] BindCtx, always NULL
(pInfo + i)->wzUrl, // [in ] url
NULL, // [in ] IUnknown for Aggregration
NULL, // [out] IUNknown for Aggregration
&(pInfo + i)->pProt, // [out] return pProt pointer
0 // [in ] bind option, pass 0
);
if (g_dwVerbose & FLAG_TRACE)
printf("MAIN: Session->CreateBinding: %lx\n", hr);
// Create a protocolHook (sink) and Start the async operation
if (hr == NOERROR)
{
(pInfo + i)->pHook = new COInetProtocolHook(g_hCompleted, (pInfo + i)->pProt);
(pInfo + i)->pSink = NULL;
(pInfo + i)->pBindInfo = NULL;
if ((pInfo + i)->pHook)
{
hr = (pInfo + i)->pHook->QueryInterface(IID_IOInetProtocolSink, (void**)&(pInfo + i)->pSink);
hr = (pInfo + i)->pHook->QueryInterface(IID_IOInetBindInfo, (void**)&(pInfo + i)->pBindInfo);
}
if ((pInfo + i)->pProt && (pInfo + i)->pSink && (pInfo + i)->pBindInfo)
{
hr = (pInfo + i)->pProt->Start((pInfo + i)->wzUrl, (pInfo + i)->pSink, (pInfo + i)->pBindInfo, PI_FORCE_ASYNC, 0);
if (g_dwVerbose & FLAG_TRACE)
printf("MAIN: pProtocol->Start: %lx\n", hr);
}
}
}
return(hr);
}
VOID CleanInfoList(PINFO pInfo, DWORD dwNumDownloads)
{
for (DWORD i = 0; i < dwNumDownloads; i++)
{
if ((pInfo + i)->pProt)
(pInfo + i)->pProt->Terminate(0);
if ((pInfo + i)->pSink)
{
(pInfo + i)->pSink->Release();
}
if ((pInfo + i)->pBindInfo)
{
(pInfo + i)->pBindInfo->Release();
}
if ((pInfo + i)->pHook)
{
(pInfo + i)->pHook->Release();
}
// release COM objects
if ((pInfo + i)->pProt)
{
// BUG (POSSIBLE RESOURCE LEAK)
// If the pProt is IE's http/gopher/ftp implementation,
// calling pProt->Release() now might cause resource leak
// since pProt (although finished the download), might
// be still waiting wininet to call back about the
// confirmation of the handle closing.
// The correct time to release pProt is to wait after
// pProtSink get destroyed.
(pInfo + i)->pProt->Release();
}
}
}
// Purpose: create a session object
// Get a pluggable procotol from the session
// Start the pluggable protocol async download
// Author: DanpoZ (Danpo Zhang)
// History: 11-20-97 Created
int _cdecl main(int argc, char** argv)
{
IOInetSession* pSession = NULL;
IOInetProtocol* pProt = NULL;
HRESULT hr = NOERROR;
DWORD dwLoadTime;
HMODULE hMod = NULL;
if (!ProcessCommandLine(argc, argv))
return 0;
// Init COM
CoInitialize(NULL);
if (g_pModule != NULL)
{
hMod = LoadLibrary(g_pModule);
}
g_hCompleted = CreateEvent(NULL, FALSE, FALSE, NULL);
// Get a Session
hr = CoInternetGetSession(0, &pSession, 0);
if (g_dwVerbose & FLAG_TRACE)
printf("MAIN: Created Session: %lx\n", hr);
if (hr == NOERROR)
{
if (!BuildInfoList(&Info[0], g_dwNumDownloads))
return(2);
hr = StartDownloads(pSession, &Info[0], g_dwNumDownloads);
if (hr == NOERROR)
{
// wait until the async download finishes
WaitForSingleObject(g_hCompleted, INFINITE);
}
StopCAP();
QueryPerformanceCounter((LARGE_INTEGER *)&g_iend);
QueryPerformanceFrequency((LARGE_INTEGER *)&g_ifreq);
dwLoadTime = (LONG)(((g_iend - g_ibeg) * 1000) / g_ifreq);
float fKB;
float fSec;
float fKBSec;
if (dwLoadTime == 0)
dwLoadTime = 1;
fKB = ((float)g_dwTotalBytes) / 1024;
fSec = ((float)dwLoadTime) / 1000;
fKBSec = fKB / fSec;
printf("%s,%s,%d,%d,%2.0f\n", g_pTitle ? g_pTitle : "Oinetperf", g_pRun ? g_pRun : "1", dwLoadTime, g_dwTotalBytes, fKBSec);
CleanInfoList(&Info[0], g_dwNumDownloads);
}
if (pSession)
{
pSession->Release();
}
CoTaskMemFree(g_pwzUrl);
if ((g_pModule != NULL) && (hMod != NULL))
{
FreeLibrary(hMod);
}
// kill COM
CoUninitialize();
return(0);
}
COInetProtocolHook::COInetProtocolHook(HANDLE g_hCompleted, IOInetProtocol* pProt)
{
_hCompleted = g_hCompleted;
_pProt = pProt;
}
COInetProtocolHook::~COInetProtocolHook()
{
CloseHandle(_hCompleted);
}
HRESULT COInetProtocolHook::QueryInterface(REFIID iid, void **ppvObj)
{
HRESULT hr = NOERROR;
*ppvObj = NULL;
if (iid == IID_IUnknown || iid == IID_IOInetProtocolSink)
{
*ppvObj = static_cast<IOInetProtocolSink*>(this);
}
else if (iid == IID_IOInetBindInfo)
{
*ppvObj = static_cast<IOInetBindInfo*>(this);
}
else if (iid == IID_IServiceProvider)
{
*ppvObj = static_cast<IServiceProvider*>(this);
}
else if (iid == IID_IHttpNegotiate)
{
*ppvObj = static_cast<IHttpNegotiate*>(this);
}
else
{
hr = E_NOINTERFACE;
}
if (*ppvObj)
{
AddRef();
}
return hr;
}
ULONG COInetProtocolHook::AddRef(void)
{
LONG lRet = ++_CRefs;
return lRet;
}
ULONG COInetProtocolHook::Release(void)
{
LONG lRet = --_CRefs;
if (lRet == 0)
{
delete this;
}
return lRet;
}
HRESULT COInetProtocolHook::Switch(PROTOCOLDATA *pStateInfo)
{
printf("Are you crazy? I don't know how to do Thread switching!!!\n");
return E_NOTIMPL;
}
HRESULT COInetProtocolHook::ReportProgress(ULONG ulStatusCode, LPCWSTR szStatusText)
{
switch (ulStatusCode)
{
case BINDSTATUS_FINDINGRESOURCE:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): Resolving name %s\n", szStatusText);
break;
case BINDSTATUS_CONNECTING:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): Connecting to %s\n", szStatusText);
break;
case BINDSTATUS_SENDINGREQUEST:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): Sending request\n");
break;
case BINDSTATUS_CACHEFILENAMEAVAILABLE:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): cache filename available\n");
break;
case BINDSTATUS_MIMETYPEAVAILABLE:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): mimetype available = %s\n", szStatusText);
break;
case BINDSTATUS_REDIRECTING:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): Redirecting to %s\n", szStatusText);
break;
default:
if (g_dwVerbose & FLAG_TRACE)
wprintf(L"CALLBACK(ReportProgress): others...\n");
break;
}
return NOERROR;
}
HRESULT COInetProtocolHook::ReportData(DWORD grfBSCF, ULONG ulProgress, ULONG ulProgressMax)
{
if (g_dwVerbose & FLAG_TRACE)
printf("CALLBACK(ReportData) %d, %d, %d \n", grfBSCF, ulProgress, ulProgressMax);
// Pull data via pProt->Read(), here are the possible returned
// HRESULT values and how we should act upon:
// if E_PENDING is returned:
// client already get all the data in buffer, there is nothing
// can be done here, client should walk away and wait for the
// next chuck of data, which will be notified via ReportData()
// callback.
// if S_FALSE is returned:
// this is EOF, everything is done, however, client must wait
// for ReportResult() callback to indicate that the pluggable
// protocol is ready to shutdown.
// if S_OK is returned:
// keep on reading, until you hit E_PENDING/S_FALSE/ERROR, the deal
// is that the client is supposed to pull ALL the available
// data in the buffer
// if none of the above is returning:
// Error occured, client should decide how to handle it, most
// commonly, client will call pProt->Abort() to abort the download
char *pBuf = (char *)_alloca(dwBuf_Size);
HRESULT hr = NOERROR;
ULONG cbRead;
while (hr == S_OK)
{
cbRead = 0;
if (g_ibeg == 0)
{
QueryPerformanceCounter((LARGE_INTEGER *)&g_ibeg);
StartCAP();
}
// pull data
if (g_dwVerbose & FLAG_TRACE)
{
printf("MAIN: pProtocol->Read attempting %d bytes\n", dwBuf_Size);
}
hr = _pProt->Read((void*)pBuf, dwBuf_Size, &cbRead);
if ((hr == S_OK || hr == E_PENDING || hr == S_FALSE) && cbRead)
{
if (g_dwVerbose & FLAG_DUMPDATA)
{
for (ULONG i = 0; i < cbRead; i++)
{
printf("%c", pBuf[i]);
}
}
if (g_dwVerbose & FLAG_TRACE)
{
printf("MAIN: pProtocol->Read %d bytes\n", cbRead);
}
g_dwTotalBytes += cbRead;
}
}
if (hr == S_FALSE)
{
if (g_dwVerbose & FLAG_TRACE)
printf("MAIN: pProtocol->Read returned EOF \n");
}
else if (hr != E_PENDING)
{
if (g_dwVerbose & FLAG_TRACE)
{
printf("MAIN: pProtocol->Read returned Error %1x \n, hr");
printf("MAIN: pProtocol->Abort called \n", hr);
}
_pProt->Abort(hr, 0);
}
return NOERROR;
}
HRESULT COInetProtocolHook::ReportResult(HRESULT hrResult, DWORD dwError, LPCWSTR wzResult)
{
// This is the last call back from the pluggable protocol,
// this call is equivlant to the IBindStatusCallBack::OnStopBinding()
// it basically tells you that the pluggable protocol is ready
// to shutdown
if (g_dwVerbose & FLAG_TRACE)
printf("CALLBACK(ReportResult): Download completed with status %1x\n", hrResult);
// set event to the main thread
if (InterlockedDecrement((LONG*)&g_dwDownloads) == 0)
SetEvent(g_hCompleted);
return NOERROR;
}
HRESULT COInetProtocolHook::GetBindInfo(DWORD *grfBINDF, BINDINFO * pbindinfo)
{
HRESULT hr = NOERROR;
// *grfBINDF = BINDF_DIRECT_READ | BINDF_ASYNCHRONOUS | BINDF_PULLDATA;
// *grfBINDF = BINDF_ASYNCHRONOUS | BINDF_ASYNCSTORAGE | BINDF_PULLDATA;
*grfBINDF = BINDF_ASYNCHRONOUS | BINDF_ASYNCSTORAGE | BINDF_PULLDATA | BINDF_IGNORESECURITYPROBLEM;
*grfBINDF |= g_dwCacheFlag;
// for HTTP GET, VERB is the only field we interested
// for HTTP POST, BINDINFO will point to Storage structure which
// contains data
BINDINFO bInfo;
ZeroMemory(&bInfo, sizeof(BINDINFO));
// all we need is size and verb field
bInfo.cbSize = sizeof(BINDINFO);
bInfo.dwBindVerb = BINDVERB_GET;
// src -> dest
hr = CopyBindInfo(&bInfo, pbindinfo);
return hr;
}
static LPSTR g_szAcceptStrAll = "*/*";
HRESULT COInetProtocolHook::GetBindString(ULONG ulStringType, LPOLESTR *ppwzStr, ULONG cEl, ULONG *pcElFetched)
{
HRESULT hr = INET_E_USE_DEFAULT_SETTING;
switch (ulStringType)
{
case BINDSTRING_HEADERS:
case BINDSTRING_EXTRA_URL:
case BINDSTRING_LANGUAGE:
case BINDSTRING_USERNAME:
case BINDSTRING_PASSWORD:
case BINDSTRING_ACCEPT_ENCODINGS:
case BINDSTRING_URL:
case BINDSTRING_USER_AGENT:
case BINDSTRING_POST_COOKIE:
case BINDSTRING_POST_DATA_MIME:
break;
case BINDSTRING_ACCEPT_MIMES:
// IE4's http pluggable protocol implementation does not
// honer INET_E_USE_DEFAULT_SETTING returned by this function
// starting from IE5, client can just return the USE_DEFAULT
// use for ie5 so we don't need a seperate bin for ie4 // #ifndef IE5
// this will be freed by the caller
*(ppwzStr + 0) = MyDupA2W(g_szAcceptStrAll);
*(ppwzStr + 1) = NULL;
*pcElFetched = 1;
hr = NOERROR;
//#endif
break;
default:
break;
}
return hr;
}
HRESULT COInetProtocolHook::QueryService(REFGUID guidService, REFIID riid, void **ppvObj)
{
HRESULT hr = E_NOINTERFACE;
*ppvObj = NULL;
if (guidService == IID_IHttpNegotiate)
{
*ppvObj = static_cast<IHttpNegotiate*>(this);
}
if (*ppvObj)
{
AddRef();
hr = NOERROR;
}
return hr;
}
HRESULT COInetProtocolHook::BeginningTransaction(LPCWSTR szURL, LPCWSTR szHeaders, DWORD dwReserved, LPWSTR *pszAdditionalHeaders)
{
if (g_dwVerbose & FLAG_TRACE)
printf("HTTPNEGOTIATE: Additional Headers? - No \n");
*pszAdditionalHeaders = NULL;
return NOERROR;
}
HRESULT COInetProtocolHook::OnResponse(DWORD dwResponseCode, LPCWSTR szResponseHeaders, LPCWSTR szRequestHeaders, LPWSTR *pszAdditionalHeaders)
{
if (g_dwVerbose & FLAG_TRACE)
printf("HTTPNEGOTIATE: Http server response code %d\n", dwResponseCode);
return NOERROR;
}