333 lines
7.3 KiB
C++
333 lines
7.3 KiB
C++
/*++
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
perfdiag.hxx
|
|
|
|
Abstract:
|
|
Performance diagnostics
|
|
|
|
Author:
|
|
Richard L Firth (rfirth) 24-Jan-1997
|
|
|
|
Revision History:
|
|
|
|
24-Jan-1997 rfirth
|
|
Created
|
|
|
|
--*/
|
|
|
|
#if !defined(_PERFDIAG_)
|
|
#define _PERFDIAG_
|
|
|
|
|
|
// perf event (PE_) definitions
|
|
|
|
|
|
//#define USE_ICECAP_FOR_DOWNLOAD_PROFILING 1
|
|
|
|
#define PE_START 1
|
|
#define PE_END 2
|
|
#define PE_CLIENT_REQUEST_START 3
|
|
#define PE_CLIENT_REQUEST_END 4
|
|
#define PE_CLIENT_REQUEST_QUEUED 5
|
|
#define PE_WORKER_REQUEST_START 6
|
|
#define PE_WORKER_REQUEST_END 7
|
|
#define PE_APP_CALLBACK_START 8
|
|
#define PE_APP_CALLBACK_END 9
|
|
#define PE_NAMERES_START 10
|
|
#define PE_NAMERES_END 11
|
|
#define PE_CONNECT_START 12
|
|
#define PE_CONNECT_END 13
|
|
#define PE_SEND_START 14
|
|
#define PE_SEND_END 15
|
|
#define PE_RECEIVE_START 16
|
|
#define PE_RECEIVE_END 17
|
|
#define PE_PEEK_RECEIVE_START 18
|
|
#define PE_PEEK_RECEIVE_END 19
|
|
#define PE_SOCKET_CLOSE_START 20
|
|
#define PE_SOCKET_CLOSE_END 21
|
|
#define PE_ACQUIRE_KEEP_ALIVE 22
|
|
#define PE_RELEASE_KEEP_ALIVE 23
|
|
#define PE_SOCKET_ERROR 24
|
|
#define PE_CACHE_READ_CHECK_START 25
|
|
#define PE_CACHE_READ_CHECK_END 26
|
|
#define PE_CACHE_WRITE_CHECK_START 27
|
|
#define PE_CACHE_WRITE_CHECK_END 28
|
|
#define PE_CACHE_RETRIEVE_START 29
|
|
#define PE_CACHE_RETRIEVE_END 30
|
|
#define PE_CACHE_READ_START 31
|
|
#define PE_CACHE_READ_END 32
|
|
#define PE_CACHE_WRITE_START 33
|
|
#define PE_CACHE_WRITE_END 34
|
|
#define PE_CACHE_CREATE_FILE_START 35
|
|
#define PE_CACHE_CREATE_FILE_END 36
|
|
#define PE_CACHE_CLOSE_FILE_START 37
|
|
#define PE_CACHE_CLOSE_FILE_END 38
|
|
#define PE_CACHE_EXPIRY_CHECK_START 39
|
|
#define PE_CACHE_EXPIRY_CHECK_END 40
|
|
#define PE_YIELD_SELECT_START 41
|
|
#define PE_YIELD_SELECT_END 42
|
|
#define PE_YIELD_OBJECT_WAIT_START 43
|
|
#define PE_YIELD_OBJECT_WAIT_END 44
|
|
#define PE_YIELD_SLEEP_START 45
|
|
#define PE_YIELD_SLEEP_END 46
|
|
#define PE_TRACE 47
|
|
#define PE_ENTER_PATH 48
|
|
#define PE_LEAVE_PATH 49
|
|
#define PE_TRACE_PATH 50
|
|
|
|
|
|
// API prototypes
|
|
|
|
|
|
#if defined(__cplusplus)
|
|
extern "C" {
|
|
#endif
|
|
|
|
VOID
|
|
#if defined(_WINX32_)
|
|
__declspec(dllexport)
|
|
#else
|
|
__declspec(dllimport)
|
|
#endif
|
|
WINAPI
|
|
WininetPerfLog(
|
|
IN DWORD dwEvent,
|
|
IN DWORD dwInfo1,
|
|
IN DWORD dwInfo2,
|
|
IN HINTERNET hInternet
|
|
);
|
|
|
|
#if defined(__cplusplus)
|
|
}
|
|
#endif
|
|
|
|
#if defined(_WINX32_)
|
|
#if defined(USE_PERF_DIAG)
|
|
|
|
|
|
// manifests
|
|
|
|
|
|
//#define DEFAULT_PE_LOG_RECORDS 4096
|
|
#define DEFAULT_PE_LOG_RECORDS 16384
|
|
|
|
|
|
// types
|
|
|
|
|
|
typedef struct {
|
|
LARGE_INTEGER liTime;
|
|
DWORD dwThreadId;
|
|
DWORD dwThreadPriority;
|
|
HINTERNET hInternet;
|
|
DWORD dwEvent;
|
|
DWORD dwInfo;
|
|
DWORD dwInfo2;
|
|
} PERF_INFO, * LPPERF_INFO;
|
|
|
|
|
|
// prototypes
|
|
|
|
|
|
VOID PerfSleep(DWORD dwMilliseconds);
|
|
|
|
int PerfSelect(
|
|
int nfds,
|
|
fd_set FAR * readfds,
|
|
fd_set FAR * writefds,
|
|
fd_set FAR * exceptfds,
|
|
const struct timeval FAR * timeout
|
|
);
|
|
|
|
DWORD PerfWaitForSingleObject(HANDLE hObject, DWORD dwTimeout);
|
|
|
|
// classes
|
|
|
|
|
|
class CPerfDiag {
|
|
|
|
private:
|
|
|
|
LPBYTE m_lpbPerfBuffer;
|
|
DWORD m_dwPerfBufferLen;
|
|
LPBYTE m_lpbEnd;
|
|
LPBYTE m_lpbNext;
|
|
BOOL m_bFull;
|
|
BOOL m_bStarted;
|
|
BOOL m_bStartFinished;
|
|
LARGE_INTEGER m_liStartTime;
|
|
|
|
LPPERF_INFO get_next_record(VOID);
|
|
|
|
VOID get_perf_time(PLARGE_INTEGER pTime) {
|
|
QueryPerformanceCounter(pTime);
|
|
}
|
|
|
|
VOID get_time(LPPERF_INFO lpInfo) {
|
|
get_perf_time(&lpInfo->liTime);
|
|
}
|
|
|
|
VOID free_perf_buffer(VOID) {
|
|
if (m_lpbPerfBuffer) {
|
|
m_lpbPerfBuffer = (LPBYTE)FREE_MEMORY(m_lpbPerfBuffer);
|
|
|
|
INET_ASSERT(!m_lpbPerfBuffer);
|
|
|
|
}
|
|
}
|
|
|
|
VOID perf_start(VOID) {
|
|
while (!m_bStartFinished) {
|
|
if (InterlockedExchange((LPLONG)&m_bStarted, TRUE) == FALSE) {
|
|
get_perf_time(&m_liStartTime);
|
|
Init();
|
|
m_bStartFinished = TRUE;
|
|
}
|
|
if (!m_bStartFinished) {
|
|
Sleep(0);
|
|
}
|
|
}
|
|
}
|
|
|
|
public:
|
|
|
|
CPerfDiag();
|
|
~CPerfDiag();
|
|
|
|
VOID Init(DWORD dwNumberOfRecords = DEFAULT_PE_LOG_RECORDS) {
|
|
CreateBuffer(dwNumberOfRecords);
|
|
}
|
|
|
|
VOID CreateBuffer(LPBYTE lpbBuffer, DWORD dwBufferLength) {
|
|
}
|
|
|
|
VOID CreateBuffer(DWORD dwNumberOfRecords = DEFAULT_PE_LOG_RECORDS) {
|
|
free_perf_buffer();
|
|
m_dwPerfBufferLen = dwNumberOfRecords * sizeof(PERF_INFO);
|
|
m_lpbPerfBuffer = (LPBYTE)ALLOCATE_MEMORY(LMEM_FIXED, m_dwPerfBufferLen);
|
|
if (m_lpbPerfBuffer) {
|
|
m_lpbEnd = m_lpbPerfBuffer + m_dwPerfBufferLen;
|
|
m_lpbNext = m_lpbPerfBuffer;
|
|
m_bFull = FALSE;
|
|
} else {
|
|
OutputDebugString("*** Cannot create performance buffer\n");
|
|
}
|
|
}
|
|
|
|
VOID Log(DWORD dwEvent, DWORD dwInfo = 0);
|
|
VOID Log(DWORD dwEvent, DWORD dwInfo, DWORD dwThreadId, HINTERNET hInternet);
|
|
VOID Log(DWORD dwEvent, DWORD dwInfo, DWORD dwInfo2, DWORD dwThreadId, HINTERNET hInternet);
|
|
VOID Dump(VOID);
|
|
};
|
|
|
|
|
|
// global data
|
|
|
|
|
|
extern CPerfDiag * GlobalPerfDiag;
|
|
|
|
|
|
// macros
|
|
|
|
|
|
#define PERF_INIT() if (!GlobalPerfDiag) GlobalPerfDiag = new CPerfDiag
|
|
#define PERF_END() if (GlobalPerfDiag) delete GlobalPerfDiag
|
|
#define PERF_LOG if (GlobalPerfDiag) GlobalPerfDiag->Log
|
|
#define PERF_DUMP() if (GlobalPerfDiag) GlobalPerfDiag->Dump()
|
|
#define PERF_Sleep(n) PerfSleep(n)
|
|
#define PERF_Select(n, pReadfds, pWritefds, pExceptfds, pTimeout) PerfSelect(n, pReadfds, pWritefds, pExceptfds, pTimeout)
|
|
#define PERF_WaitForSingleObject(hObject, dwTimeout) PerfWaitForSingleObject(hObject, dwTimeout)
|
|
|
|
#else
|
|
|
|
#define PERF_INIT() \
|
|
/* NOTHING */
|
|
|
|
#define PERF_END() \
|
|
/* NOTHING */
|
|
|
|
#define PERF_LOG \
|
|
(VOID)
|
|
|
|
#define PERF_DUMP() \
|
|
/* NOTHING */
|
|
|
|
#define PERF_Sleep(n) Sleep(n)
|
|
|
|
#define PERF_Select(n, pReadfds, pWritefds, pExceptfds, pTimeout) \
|
|
_I_select(n, pReadfds, pWritefds, pExceptfds, pTimeout)
|
|
|
|
#define PERF_WaitForSingleObject(hObject, dwTimeout) \
|
|
WaitForSingleObject(hObject, dwTimeout)
|
|
|
|
#endif // defined(USE_PERF_DIAG)
|
|
#endif // defined(_WINX32_)
|
|
|
|
//#define PERF_ENTER(Name) \
|
|
// StartCAP()
|
|
|
|
//#define PERF_LEAVE(Name) \
|
|
// StopCAP()
|
|
|
|
|
|
#if defined(USE_PERF_DIAG)
|
|
|
|
#define PERF_LOG_EX \
|
|
WininetPerfLog
|
|
|
|
#define PERF_ID(Name) \
|
|
(LPCSTR)(# Name)
|
|
|
|
#define PERF_TRACE(Name, Info) \
|
|
WininetPerfLog(PE_TRACE_PATH, Info, (DWORD)PERF_ID(Name), 0)
|
|
|
|
#define PERF_ENTER(Name) \
|
|
WininetPerfLog(PE_ENTER_PATH, 0, (DWORD)PERF_ID(Name), 0)
|
|
|
|
#define PERF_LEAVE(Name) \
|
|
WininetPerfLog(PE_LEAVE_PATH, 0, (DWORD)PERF_ID(Name), 0)
|
|
|
|
#else
|
|
|
|
#define PERF_LOG_EX \
|
|
/* NOTHING */
|
|
|
|
#define PERF_ID(Name) \
|
|
/* NOTHING */
|
|
|
|
#define PERF_TRACE(Name, Info) \
|
|
/* NOTHING */
|
|
|
|
|
|
#ifdef USE_ICECAP_FOR_DOWNLOAD_PROFILING
|
|
|
|
#define PERF_ENTER(Name) \
|
|
StartCAP()
|
|
|
|
#define PERF_LEAVE(Name) \
|
|
StopCAP()
|
|
|
|
#else
|
|
|
|
#define PERF_ENTER(Name) \
|
|
/* NOTHING */
|
|
|
|
#define PERF_LEAVE(Name) \
|
|
/* NOTHING */
|
|
|
|
#endif // defined(USE_ICECAP_FOR_DOWNLOAD_PROFILING)
|
|
|
|
|
|
#endif // defined(USE_PERF_DIAG)
|
|
|
|
|
|
#define STOP_SENDREQ_PERF() \
|
|
/* nothing */
|
|
|
|
#define START_SENDREQ_PERF() \
|
|
/* nothing */
|
|
|
|
#endif // defined(_PERFDIAG_)
|