2020-09-30 17:12:32 +02:00

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_)