Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

4060 lines
152 KiB

/*++
Copyright (c) 1997-2000 Microsoft Corporation
Module Name:
report.c
Abstract:
Manipulation routines for cpdata structures.
Author:
Melur Raghuraman (mraghu) 03-Oct-1997
Environment:
Revision History:
Corey Morgan (coreym) 04-June-2002
Reformatted report output to XML.
--*/
#include <stdlib.h>
#include <stdio.h>
#include "cpdata.h"
#include "tracectr.h"
#include <ntverp.h>
#include "item.h"
#define MODULE_STRING_SIZE 256
#define FILE_NAME_COLUMN_SIZE 80
#define URL_NAME_COLUMN_SIZE 79
#define MAX_GUID_STRING_SIZE 64
#define DISPLAY_SIZE 10
extern PTRACE_CONTEXT_BLOCK TraceContext;
extern ULONG TotalEventsLost;
extern ULONG TotalEventCount;
extern ULONG TimerResolution;
extern __int64 ElapseTime;
extern ULONG TotalBuffersRead;
extern BOOLEAN XPorHigher;
extern FARPROC EtwpIpv4ToStringA;
extern FARPROC EtwpIpv4ToStringW;
extern FARPROC EtwpIpv6ToStringA;
extern FARPROC EtwpIpv6ToStringW;
static FILE* procFile;
static void PrintDiskTotals();
static void PrintProcessCpuTime();
static void PrintProcessData();
static void PrintPerThreadPerDiskTable();
static void WriteTransactionStatistics();
static void WriteTransactionCPUTime();
static void PrintProcessSubDataInclusive();
static void PrintProcessSubDataExclusive();
void TransInclusive(
PLIST_ENTRY TrHead,
ULONG level
);
static void ReportHotFileInfo();
static void ReportPrintJobInfo();
PIIS_REPORT_RECORD IIS = NULL;
ULONG RequestsDiscarded = 0;
ULONG SendErrorRequests = 0;
void ProcessIisRequest(HTTP_REQUEST_RECORD *pReq);
static void ReportIisEvents();
extern GUID PrintJobGuid;
extern GUID UlGuid;
#ifdef DBG
BOOLEAN TracectrDbgEnabled = FALSE;
#endif
PWCHAR CpdiGuidToString(
PWCHAR s,
ULONG len,
LPGUID piid
);
PCHAR RemoveCtrlCharA(
PCHAR String,
ULONG NumChars
);
PCHAR ReduceStringA(
PCHAR OutString,
ULONG NumChars,
PCHAR LongString
);
PWCHAR ReduceStringW(
PWCHAR OutString,
ULONG NumChars,
PWCHAR LongString
);
ULONG ReportFlags = 0;
void DecodeIpAddressA(
USHORT AddrType,
PULONG IpAddrV4,
PUSHORT IpAddrV6,
PCHAR pszA
)
{
if (AddrType == TDI_ADDRESS_TYPE_IP) {
if (XPorHigher && EtwpIpv4ToStringA != NULL) {
struct in_addr IPv4Addr
= * (struct in_addr UNALIGNED*) IpAddrV4;
pszA = (PCHAR)(*EtwpIpv4ToStringA)(&IPv4Addr, pszA);
*pszA = '\0';
}
else {
StringCchCopyA(pszA, MAX_ADDRESS_LENGTH, "Undecodable IP Address");
}
}
else if (AddrType == TDI_ADDRESS_TYPE_IP6) {
if (XPorHigher && EtwpIpv6ToStringA != NULL) {
struct in6_addr IPv6Addr
= * (struct in6_addr UNALIGNED*) IpAddrV6;
pszA = (PCHAR)(*EtwpIpv6ToStringA)(&IPv6Addr, pszA);
*pszA = '\0';
}
else {
StringCchCopyA(pszA, MAX_ADDRESS_LENGTH, "Undecodable IP Address");
}
}
else {
StringCchCopyA(pszA, MAX_ADDRESS_LENGTH, "Unknown IP Address Type");
}
}
void DecodeIpAddressW(
USHORT AddrType,
PULONG IpAddrV4,
PUSHORT IpAddrV6,
PWCHAR pszW
)
{
if (AddrType == TDI_ADDRESS_TYPE_IP) {
if (XPorHigher && EtwpIpv4ToStringW != NULL) {
struct in_addr IPv4Addr
= * (struct in_addr UNALIGNED*) IpAddrV4;
pszW = (PWCHAR)(*EtwpIpv4ToStringW)(&IPv4Addr, pszW);
*pszW = L'\0';
}
else {
StringCchCopyW(pszW, MAX_ADDRESS_LENGTH, L"Undecodable IP Address");
}
}
else if (AddrType == TDI_ADDRESS_TYPE_IP6) {
if (XPorHigher && EtwpIpv6ToStringW != NULL) {
struct in6_addr IPv6Addr
= * (struct in6_addr UNALIGNED*) IpAddrV6;
pszW = (PWCHAR)(*EtwpIpv6ToStringW)(&IPv6Addr, pszW);
*pszW = L'\0';
}
else {
StringCchCopyW(pszW, MAX_ADDRESS_LENGTH, L"Undecodable IP Address");
}
}
else {
StringCchCopyW(pszW, MAX_ADDRESS_LENGTH, L"Unknown IP Address Type");
}
}
// URLs may contain non-printable characters. This routine removes them.
PCHAR RemoveCtrlCharA(
PCHAR String,
ULONG NumChars
)
{
ULONG i;
for (i = 0 ; i <= NumChars && String[i] != '\0'; i++) {
if ( isprint(String[i]) == 0 ||
String[i] == '\t' ||
String[i] == '\b' ||
String[i] == '\n' ||
String[i] == '\'' ||
String[i] == '\"') {
String[i] = '?';
}
}
return String;
}
PCHAR ReduceStringA(
PCHAR OutString,
ULONG NumChars,
PCHAR LongString
)
{
ULONG Size;
ULONG i;
if (LongString == NULL) {
return NULL;
}
// We assume here that LongString is not junk.
Size = strlen(LongString);
if (OutString == NULL) {
return NULL;
}
RtlZeroMemory(OutString, NumChars);
// This function is only useful when the length of LongString exceeds NumChars.
// However, it still works when the length of LongString is smaller than NumChars.
if (Size <= (NumChars - 1)) {
StringCchCopyA(OutString, NumChars, LongString);
return OutString;
}
i = Size - 1;
while (LongString[i] != '\\' && LongString[i] != '/' && i > 0) {
i--;
}
if (i == 0) { // there's no /s or \s. Just truncate.
StringCchCopyA(OutString, NumChars, LongString);
return OutString;
}
else {
if ((Size - i) >= NumChars - 3) { // only name exceeds given chars.
StringCchPrintfA(OutString, NumChars, "..%s", &LongString[1]);
return OutString;
}
else {
ULONG SpareChars = (NumChars - 3) - (Size - i);
StringCchCopyA(OutString, SpareChars + 1, LongString);
StringCchCatA(OutString, NumChars + 1, "..");
StringCchCatA(OutString, NumChars + 1, &LongString[i]);
return OutString;
}
}
}
PWCHAR ReduceStringW(
PWCHAR OutString,
ULONG NumChars,
PWCHAR LongString
)
{
ULONG Size;
ULONG i;
if (LongString == NULL) {
return NULL;
}
// We assume here that LongString is not junk.
Size = wcslen(LongString);
if (OutString == NULL) {
return NULL;
}
RtlZeroMemory(OutString, NumChars * sizeof(WCHAR));
// This function is only useful when the length of LongString exceeds NumChars.
// However, it still works when the length of LongString is smaller than NumChars.
if (Size <= (NumChars - 1)) {
StringCchCopyW(OutString, NumChars, LongString);
return OutString;
}
i = Size - 1;
while (LongString[i] != L'\\' && LongString[i] != L'/' && i > 0) {
i--;
}
if (i == 0) { // there's no /s or \s. Just truncate.
StringCchCopyW(OutString, NumChars, LongString);
return OutString;
}
else {
if ((Size - i) >= NumChars - 3) { // only name exceeds given chars.
StringCchPrintfW(OutString, NumChars, L"..%ws", &LongString[1]);
return OutString;
}
else {
ULONG SpareChars = (NumChars - 3) - (Size - i);
StringCchCopyW(OutString, SpareChars + 1, LongString);
StringCchCatW(OutString, NumChars + 1, L"..");
StringCchCatW(OutString, NumChars + 1, &LongString[i]);
return OutString;
}
}
}
void CollapseTree(
PLIST_ENTRY OldTree,
PLIST_ENTRY NewTree,
BOOL flat
)
{
PLIST_ENTRY OldNext;
PTRANS_RECORD pTrans;
PTRANS_RECORD pNewTrans;
OldNext = OldTree->Flink;
while (OldNext != OldTree)
{
pTrans = CONTAINING_RECORD(OldNext, TRANS_RECORD, Entry);
OldNext = OldNext->Flink;
pNewTrans = FindTransByList(NewTree, pTrans->pGuid, 0);
if( NULL != pNewTrans ){
pNewTrans->KCpu += pTrans->KCpu;
pNewTrans->UCpu += pTrans->UCpu;
pNewTrans->RefCount += pTrans->RefCount;
pNewTrans->RefCount1 += pTrans->RefCount1;
if (flat)
{
CollapseTree(&pTrans->SubTransListHead, NewTree, TRUE );
}
else
{
CollapseTree(& pTrans->SubTransListHead,
& pNewTrans->SubTransListHead,
FALSE);
}
}
}
}
#define BARLEN 70
char*
TimeWindowBar(
char* buffer,
ULONGLONG min,
ULONGLONG max
)
{
double unit;
ULONGLONG duration;
int pre, bar, count, index;
if(buffer == NULL){
return NULL;
}
duration = ((CurrentSystem.EndTime - CurrentSystem.StartTime) / 10000000);
unit = (double)BARLEN/(ULONG)duration;
pre = (int)((ULONG)((min - CurrentSystem.StartTime)/10000000) * unit);
bar = (int)((ULONG)((max - min)/10000000) * unit);
buffer[0] = '\0';
count = 0;
index = 0;
while(count < pre) {
buffer[index] = ' ';
index++;
count++;
}
buffer[index] = '|';
index++;
count = 0;
while(count < bar) {
buffer[index] = '_';
index++;
count++;
}
buffer[index] = '|';
buffer[index + 1] = '\0';
return buffer;
}
char*
GetDateString( char* buffer, size_t cchBuffer, ULONGLONG llTime )
{
FILETIME ft, lft;
SYSTEMTIME st;
LARGE_INTEGER LargeTmp;
BOOL bResult;
HRESULT hr;
WCHAR wDate[128];
CHAR aDate[128];
LargeTmp.QuadPart = llTime;
ft.dwHighDateTime = LargeTmp.HighPart;
ft.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&ft, &lft);
bResult = FileTimeToSystemTime (
&lft,
&st
);
if( ! bResult || st.wMonth > 12 ){
buffer[0] = '\0';
}else{
GetDateFormatW(
LOCALE_USER_DEFAULT, DATE_LONGDATE, &st, NULL, wDate, 128 );
WideCharToMultiByte(
CP_UTF8, 0, wDate, wcslen(wDate)+1, aDate, 128, NULL, NULL );
hr = StringCchCopyA( buffer, cchBuffer, aDate );
hr = StringCchCatA( buffer, cchBuffer, " " );
GetTimeFormatW(
LOCALE_USER_DEFAULT, 0, &st, NULL, wDate, 128 );
WideCharToMultiByte(
CP_UTF8, 0, wDate, wcslen(wDate)+1, aDate, 128, NULL, NULL );
hr = StringCchCatA( buffer, cchBuffer, aDate );
}
return buffer;
}
void
WriteProc(
LPWSTR ProcFileName,
ULONG flags,
PVOID pUserContext
)
{
ULONGLONG duration;
PLIST_ENTRY Next, Head;
PPROCESS_FILE_RECORD pFileRec;
char buffer[MAXSTR];
BOOL bResult;
ReportFlags = flags;
procFile = _wfopen(ProcFileName, L"w");
if (procFile == NULL)
return;
fprintf( procFile, "<?xml version=\"1.0\" encoding='UTF-8'?>\n" );
if( !(TraceContext->Flags & TRACE_TRANSFORM_XML ) && TraceContext->XSLDocName != NULL ){
fprintf( procFile, "<?xml-stylesheet type=\"text/xsl\" href=\"%ws\"?>\n", TraceContext->XSLDocName );
}
fprintf( procFile, "<report>\n<header>\n" );
fprintf( procFile, "<version>%d</version>\n", VER_PRODUCTBUILD );
fprintf( procFile, "<type>%s</type>\n", ( ReportFlags & TRACE_TOTALS_REPORT ) ? "Total" : "Default" );
fprintf( procFile, "<build>%d</build>\n",CurrentSystem.BuildNumber );
fprintf( procFile, "<processors>%d</processors>\n", CurrentSystem.NumberOfProcessors );
if (CurrentSystem.CpuSpeed > 0) {
fprintf(procFile,
"<cpu_speed units='MHz'>%d</cpu_speed>\n",
CurrentSystem.CpuSpeed
);
} else {
fprintf(procFile, "<cpu_speed/>\n" );
}
if (CurrentSystem.MemorySize > 0) {
fprintf(procFile,
"<memory units='Mb'>%d</memory>\n",
CurrentSystem.MemorySize
);
} else {
fprintf(procFile, "<memory/>\n" );
}
if (CurrentSystem.ComputerName != NULL) {
fprintf(procFile,
"<computer_name>%ws</computer_name>\n",
CurrentSystem.ComputerName
);
} else {
fprintf(procFile, "<computer_name/>\n" );
}
fprintf( procFile, "<start>%s</start>\n",
GetDateString( buffer, MAXSTR, CurrentSystem.StartTime ) );
fprintf( procFile, "<end>%s</end>\n",
GetDateString( buffer, MAXSTR, CurrentSystem.EndTime) );
duration = (CurrentSystem.EndTime - CurrentSystem.StartTime) / 10000000;
fprintf( procFile, "<duration>%I64u</duration>\n",duration );
Head = &CurrentSystem.ProcessFileListHead;
Next = Head->Flink;
while (Next != Head) {
pFileRec = CONTAINING_RECORD( Next, PROCESS_FILE_RECORD, Entry );
Next = Next->Flink;
fprintf( procFile, "<trace name=\"%ws\">\n", pFileRec->TraceName ? pFileRec->TraceName : L"-" );
fprintf( procFile, "<file>%ws</file>\n", pFileRec->FileName ? pFileRec->FileName : L"-" );
if (pFileRec->StartTime == 0){
pFileRec->StartTime = CurrentSystem.StartTime;
}
if (pFileRec->EndTime == 0){
pFileRec->EndTime = CurrentSystem.EndTime;
}
fprintf( procFile, "<start>%s</start>\n",
GetDateString( buffer, MAXSTR, pFileRec->StartTime ) );
duration = (pFileRec->EndTime - pFileRec->StartTime) / 10000000;
fprintf( procFile,
"<end>%s</end>\n"
"<duration>%I64u</duration>\n",
GetDateString( buffer, MAXSTR, pFileRec->EndTime ),
duration );
fprintf( procFile,
"<duration_window xml:space='preserve'>%s</duration_window>\n",
TimeWindowBar(buffer, pFileRec->StartTime, pFileRec->EndTime) );
fprintf( procFile, "</trace>\n" );
}
fprintf( procFile, "</header>\n" );
if (flags & (TRACE_BASIC_REPORT|TRACE_TOTALS_REPORT)){
PMOF_INFO pMofInfo;
WriteTransactionStatistics();
WriteTransactionCPUTime();
pMofInfo = GetMofInfoHead ((LPCGUID)&PrintJobGuid);
if ( pMofInfo != NULL ){
if (pMofInfo->EventCount > 0) {
ReportPrintJobInfo();
}
}
pMofInfo = GetMofInfoHead ((LPCGUID)&UlGuid);
if ( pMofInfo != NULL ){
if (pMofInfo->EventCount > 0) {
ReportIisEvents();
}
}
// PrintProcessData() must be run before others to set the process
// time from the added thread times
PrintProcessData();
PrintProcessSubDataExclusive();
PrintProcessSubDataInclusive();
/*
PrintProcessCpuTime();
*/
PrintDiskTotals();
PrintPerThreadPerDiskTable();
pMofInfo = GetMofInfoHead ((LPCGUID)&FileIoGuid);
if ( pMofInfo != NULL ) {
pMofInfo = GetMofInfoHead ((LPCGUID)&DiskIoGuid);
if ( pMofInfo != NULL ) {
ReportHotFileInfo();
}
}
}
fprintf( procFile, "</report>\n" );
fclose(procFile);
}
static void
PrintDiskTotals()
{
// Print the Disk Table.
PTDISK_RECORD pDisk;
PLIST_ENTRY Next, Head;
ULONG rio, wio;
ULONGLONG Duration = (ULONGLONG)((CurrentSystem.EndTime - CurrentSystem.StartTime) / 10000000);
if (Duration == 0) {
return;
}
Head = &CurrentSystem.GlobalDiskListHead;
Next = Head->Flink;
if( Next == Head ){
return;
}
fprintf(procFile, "<table title='Disk Totals'>\n" );
Head = &CurrentSystem.GlobalDiskListHead;
Next = Head->Flink;
while (Next != Head) {
pDisk = CONTAINING_RECORD( Next, TDISK_RECORD, Entry );
rio = pDisk->ReadCount + pDisk->HPF;
wio = pDisk->WriteCount;
fprintf( procFile, "<disk number=\"%d\">\n", pDisk->DiskNumber );
fprintf( procFile, "<read_rate>%1.3f</read_rate>\n", (double)rio / ((double)Duration) );
fprintf( procFile, "<read_size>%d</read_size>\n", (rio == 0) ? 0 : (pDisk->ReadSize + pDisk->HPFSize) / rio );
fprintf( procFile, "<write_rate>%1.3f</write_rate>\n", (double)wio / ((double)Duration));
fprintf( procFile, "<write_size>%d</write_size>\n", (wio == 0) ? 0 : pDisk->WriteSize / wio );
fprintf( procFile, "</disk>\n" );
Next = Next->Flink;
}
fprintf(procFile, "</table>\n" );
}
void TotalTransChildren(
PLIST_ENTRY Head,
ULONG *Kernel,
ULONG *User,
LONG level )
{
PTRANS_RECORD pTrans;
PLIST_ENTRY Next;
Next = Head->Flink;
while( Next != Head ){
pTrans = CONTAINING_RECORD( Next, TRANS_RECORD, Entry );
Next = Next->Flink;
TotalTransChildren( &pTrans->SubTransListHead, Kernel, User, level+1 );
*Kernel += pTrans->KCpu;
*User += pTrans->UCpu;
}
}
void PrintTransList( PLIST_ENTRY TrHead, LONG level )
{
PTRANS_RECORD pTrans;
PMOF_INFO pMofInfo;
ULONG Kernel;
ULONG User;
WCHAR str[MAXSTR];
WCHAR buffer[MAXSTR];
PLIST_ENTRY TrNext = TrHead->Flink;
while( TrNext != TrHead ){
int count = 0;
pTrans = CONTAINING_RECORD( TrNext, TRANS_RECORD, Entry );
TrNext = TrNext->Flink;
Kernel = pTrans->KCpu;
User = pTrans->UCpu;
pMofInfo = GetMofInfoHead( pTrans->pGuid);
if (pMofInfo == NULL) {
return;
}
StringCchCopyW ( buffer,
MAXSTR,
( pMofInfo->strDescription ?
pMofInfo->strDescription :
CpdiGuidToString( str, MAXSTR, &pMofInfo->Guid ) ));
TotalTransChildren( &pTrans->SubTransListHead, &Kernel, &User, 0 );
fprintf( procFile, "<transaction name=\"%ws\">\n", buffer );
fprintf( procFile, "<count>%d</count>\n", pTrans->RefCount );
fprintf( procFile, "<kernel>%d</kernel>\n", Kernel );
fprintf( procFile, "<user>%d</user>\n", User );
fprintf( procFile, "\n<transaction>\n" );
if( level <= MAX_TRANS_LEVEL ){
PrintTransList( &pTrans->SubTransListHead, level+1 );
}
}
}
static void PrintProcessCpuTime()
{
PTHREAD_RECORD pThread;
PPROCESS_RECORD pProcess;
ULONGLONG lLifeTime;
ULONG TotalUserTime = 0;
ULONG TotalKernelTime = 0;
ULONG TotalCPUTime = 0;
PLIST_ENTRY Next, Head;
PLIST_ENTRY ThNext, ThHead;
BOOL titled;
ULONG usedThreadCount;
ULONG ThreadCount;
ULONG ProcessUTime;
ULONG ProcessKTime;
ULONG ThreadKCPU;
ULONG ThreadUCPU;
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
if( Head == Next ){
return;
}
fprintf( procFile, "<table title='Process/Thread CPU Time Statistics'>\n" );
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
lLifeTime = CalculateProcessLifeTime(pProcess);
fprintf(procFile,
"<process name=\"%ws\">"
"<pid>0x%04X</pid>"
"<kernel>%d</kernel>"
"<user>%d</user>"
"<read>%d</read>"
"<read_size>%1.2f</read_size>"
"<write>%d</write>"
"<write_size>%1.2f</write_size>"
"<send>%d</send>"
"<send_size>%1.2f</send_size>"
"<recv>%5d</recv>"
"<recv_size>%.2f</recv_size>"
"<duation>%I64u</duation>\n",
(pProcess->ImageName) ? pProcess->ImageName : L"Idle",
pProcess->PID,
CalculateProcessKCPU(pProcess),
CalculateProcessUCPU(pProcess),
pProcess->ReadIO + pProcess->HPF,
(pProcess->ReadIO + pProcess->HPF) ?
(double)(pProcess->ReadIOSize + pProcess->HPFSize)/
(double)(pProcess->ReadIO + pProcess->HPF) : 0,
pProcess->WriteIO,
pProcess->WriteIO ? (double)pProcess->WriteIOSize/(double)pProcess->WriteIO : 0,
pProcess->SendCount,
pProcess->SendCount ?
(double)(pProcess->SendSize / 1024)/(double)pProcess->SendCount : 0,
pProcess->RecvCount,
pProcess->RecvCount ?
(double)(pProcess->RecvSize / 1024) / (double)pProcess->RecvCount : 0,
(lLifeTime / 10000000 )
);
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
titled = FALSE;
usedThreadCount = 0;
ThreadCount = 0;
ProcessUTime = 0;
ProcessKTime = 0;
while (ThNext != ThHead) {
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
ThreadKCPU = (pThread->KCPUEnd - pThread->KCPUStart)
* CurrentSystem.TimerResolution;
ThreadUCPU = (pThread->UCPUEnd - pThread->UCPUStart)
* CurrentSystem.TimerResolution;
if( pThread->ReadIO
|| pThread->WriteIO
|| ThreadKCPU
|| ThreadUCPU
|| pThread->SendCount
|| pThread->RecvCount
|| pThread->HPF ){
fprintf(procFile,
"<thread>"
"<tid>0x%04I64X</tid>"
"<kernel>%d</kernel>"
"<user>%d</user>"
"<read>%d</read>"
"<read_size>%1.2f</read_size>"
"<write>%d</write>"
"<write_size>%1.2f</write_size>"
"<send>%d</send>"
"<send_size>%1.2f</send_size>"
"<recv>%d</recv>"
"<recv_size>%1.2f</recv_size>"
"<duration>%I64u</duration>"
"</thread>\n",
pThread->TID,
ThreadKCPU,
ThreadUCPU,
pThread->ReadIO + pThread->HPF,
pThread->ReadIO + pThread->HPF ?
(double)(pThread->ReadIOSize + pThread->HPFSize)/
(double)(pThread->ReadIO + pThread->HPF) : 0,
pThread->WriteIO,
pThread->WriteIO ? (double)pThread->WriteIOSize/
(double)pThread->WriteIO : 0,
pThread->SendCount,
pThread->SendCount ?
(double)(pThread->SendSize / 1024)/(double)pThread->SendCount : 0,
pThread->RecvCount,
pThread->RecvCount ?
(double)(pThread->RecvSize / 1024)/(double)pThread->RecvCount : 0,
((pThread->TimeEnd - pThread->TimeStart) / 10000000)
);
}
PrintTransList( &pThread->TransListHead, 0 );
TotalUserTime += ThreadUCPU;
TotalKernelTime += ThreadKCPU;
TotalCPUTime += ThreadKCPU + ThreadUCPU;
ThNext = ThNext->Flink;
}
fprintf( procFile, "</process>\n" );
Next = Next->Flink;
}
fprintf( procFile, "</table>\n" );
}
static void PrintProcessData()
{
PTHREAD_RECORD pThread;
PPROCESS_RECORD pProcess;
PTRANS_RECORD pTrans;
PLIST_ENTRY Next, Head;
PLIST_ENTRY ThNext, ThHead;
PLIST_ENTRY TrNext, TrHead;
ULONG usedThreadCount;
ULONG ThreadCount;
ULONG TotalusedThreadCount = 0;
ULONG TotalThreadCount = 0;
ULONG ThreadUTime;
ULONG ThreadKTime;
ULONG ThreadUTimeTrans;
ULONG ThreadKTimeTrans;
ULONG ThreadUTimeNoTrans;
ULONG ThreadKTimeNoTrans;
ULONG CountThreadNoTrans;
ULONG TotalKCPUThreadNoTrans;
ULONG TotalUCPUThreadNoTrans;
double PercentThreadNoTrans;
ULONG CountThreadTrans;
ULONG TotalKCPUThreadTrans;
ULONG TotalUCPUThreadTrans;
double PercentThreadTrans;
ULONG TransUTime;
ULONG TransKTime;
ULONG Processors;
ULONG TotalKThread = 0;
ULONG TotalUThread = 0;
ULONG TotalKTrans = 0;
ULONG TotalUTrans = 0;
double PerTotal = 0.0;
double IdlePercent = 0.0;
double percent;
double percentTrans;
double lDuration;
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
if( Head == Next ){
return;
}
Processors = CurrentSystem.NumberOfProcessors ? CurrentSystem.NumberOfProcessors : 1;
lDuration = ((double)((LONGLONG)(CurrentSystem.EndTime - CurrentSystem.StartTime)))
/ 10000000.00;
fprintf( procFile, "<table title='Image Statistics'>\n" );
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
usedThreadCount = 0;
ThreadCount = 0;
ThreadUTime = 0;
ThreadKTime = 0;
ThreadUTimeTrans = 0;
ThreadKTimeTrans = 0;
ThreadUTimeNoTrans = 0;
ThreadKTimeNoTrans = 0;
TransKTime = 0;
TransUTime = 0;
percent = 0;
CountThreadNoTrans = 0;
TotalKCPUThreadNoTrans = 0;
TotalUCPUThreadNoTrans = 0;
CountThreadTrans = 0;
TotalKCPUThreadTrans = 0;
TotalUCPUThreadTrans = 0;
while (ThNext != ThHead) {
LIST_ENTRY NewTransList;
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
if( pThread->ReadIO
|| pThread->WriteIO
|| pThread->KCPUEnd > pThread->KCPUStart
|| pThread->UCPUEnd > pThread->UCPUStart
|| pThread->SendCount
|| pThread->RecvCount
|| pThread->HPF )
{
usedThreadCount++;
TotalusedThreadCount++;
}
ThreadCount++;
TotalThreadCount++;
ThreadUTime += (pThread->UCPUEnd - pThread->UCPUStart)
* CurrentSystem.TimerResolution;
ThreadKTime += (pThread->KCPUEnd - pThread->KCPUStart)
* CurrentSystem.TimerResolution;
ThreadKTimeTrans += pThread->KCPU_Trans
* CurrentSystem.TimerResolution;
ThreadUTimeTrans += pThread->UCPU_Trans
* CurrentSystem.TimerResolution;
ThreadKTimeNoTrans += pThread->KCPU_NoTrans
* CurrentSystem.TimerResolution;
ThreadUTimeNoTrans += pThread->UCPU_NoTrans
* CurrentSystem.TimerResolution;
if (pThread->KCPU_Trans + pThread->UCPU_Trans == 0)
{
CountThreadNoTrans ++;
TotalKCPUThreadNoTrans += pThread->KCPU_NoTrans
* CurrentSystem.TimerResolution;
TotalUCPUThreadNoTrans += pThread->UCPU_NoTrans
* CurrentSystem.TimerResolution;
}
else
{
CountThreadTrans ++;
TotalKCPUThreadTrans += ( pThread->KCPU_Trans
+ pThread->KCPU_NoTrans)
* CurrentSystem.TimerResolution;
TotalUCPUThreadTrans += ( pThread->UCPU_Trans
+ pThread->UCPU_NoTrans)
* CurrentSystem.TimerResolution;
}
InitializeListHead(& NewTransList);
CollapseTree(& pThread->TransListHead, & NewTransList, TRUE);
TrHead = & NewTransList;
TrNext = TrHead->Flink;
while( TrNext != TrHead ){
pTrans = CONTAINING_RECORD( TrNext, TRANS_RECORD, Entry );
TransUTime += pTrans->UCpu;
TransKTime += pTrans->KCpu;
TrNext = TrNext->Flink;
}
DeleteTransList(& NewTransList, 0);
ThNext = ThNext->Flink;
}
TotalKThread += ThreadKTime;
TotalUThread += ThreadUTime;
TotalKTrans += TransKTime;
TotalUTrans += TransUTime;
percent = (((ThreadKTime + ThreadUTime + 0.0)/lDuration)/1000.0) * 100.0;
if (ThreadKTime + ThreadUTime == 0)
{
percentTrans = 0.0;
PercentThreadTrans = 0.0;
PercentThreadNoTrans = 0.0;
}
else
{
percentTrans = ( (ThreadKTimeTrans + ThreadUTimeTrans + 0.0)
/ (ThreadKTime + ThreadUTime + 0.0))
* 100.00;
PercentThreadTrans = ((TotalKCPUThreadTrans + TotalUCPUThreadTrans + 0.0)
/ (ThreadKTime + ThreadUTime + 0.0)) * 100.00;
PercentThreadNoTrans = ((TotalKCPUThreadNoTrans + TotalUCPUThreadNoTrans + 0.0)
/ (ThreadKTime + ThreadUTime + 0.0)) * 100.00;
}
PerTotal += percent;
fprintf(procFile,
"<image name=\"%ws\">"
"<pid>0x%08X</pid>"
"<threads>%d</threads>"
"<used_threads>%d</used_threads>"
"<process_kernel>%d</process_kernel>"
"<process_user>%d</process_user>"
"<transaction_kernel>%d</transaction_kernel>"
"<transaction_user>%d</transaction_user>"
"<cpu>%1.2f</cpu>"
"</image>\n",
(pProcess->ImageName) ? pProcess->ImageName : L"Idle",
pProcess->PID,
ThreadCount,
usedThreadCount,
ThreadKTime,
ThreadUTime,
ThreadKTimeTrans,
ThreadUTimeTrans,
(percent / Processors)
);
if(pProcess->PID == 0){
IdlePercent += (percent / Processors );
}
Next = Next->Flink;
}
fprintf( procFile, "</table>\n" );
}
void TransInclusive(
PLIST_ENTRY TrHead,
ULONG level
)
{
ULONG Kernel, User;
PLIST_ENTRY TrNext = TrHead->Flink;
PMOF_INFO pMofInfo;
PTRANS_RECORD pTrans;
WCHAR buffer[MAXSTR];
WCHAR str[MAXSTR];
while( TrNext != TrHead ){
ULONG count = 0;
pTrans = CONTAINING_RECORD( TrNext, TRANS_RECORD, Entry );
TrNext = TrNext->Flink;
pMofInfo = GetMofInfoHead( pTrans->pGuid);
if (pMofInfo == NULL) {
return;
}
Kernel = pTrans->KCpu;
User = pTrans->UCpu;
TotalTransChildren( &pTrans->SubTransListHead, &Kernel, &User, 0 );
StringCchCopyW ( buffer,
MAXSTR,
( pMofInfo->strDescription ?
pMofInfo->strDescription :
CpdiGuidToString( str, MAXSTR, &pMofInfo->Guid ) ));
fprintf(procFile,
"<transaction level='%d' name=\"%ws\">"
"<count>%d</count>"
"<kernel>%d</kernel>"
"<user>%d</user>",
level,
buffer,
pTrans->RefCount,
Kernel,
User
);
TransInclusive( &pTrans->SubTransListHead, level+1 );
fprintf(procFile, "</transaction>\n" );
}
}
static void PrintProcessSubDataInclusive()
{
PPROCESS_RECORD pProcess;
PTHREAD_RECORD pThread;
PLIST_ENTRY Next, Head;
PLIST_ENTRY TrNext;
PLIST_ENTRY ThNext, ThHead;
LIST_ENTRY NewHead;
BOOL bTable = FALSE;
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head && !bTable ) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
Next = Next->Flink;
if( pProcess->PID == 0 ){
continue;
}
// Total up all the threads into one list
InitializeListHead( &NewHead );
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
while( ThNext != ThHead ){
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
ThNext = ThNext->Flink;
CollapseTree(&pThread->TransListHead, &NewHead, FALSE );
}
TrNext = NewHead.Flink;
if( TrNext != &NewHead ){
bTable = TRUE;
}
DeleteTransList( &NewHead, 0 );
}
if( !bTable ){
return;
}
// Walk through the Process List and Print the report.
fprintf(procFile, "<table title='Inclusive Transactions Per Process'>\n" );
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
Next = Next->Flink;
if( pProcess->PID == 0 ){
continue;
}
// Total up all the threads into one list
InitializeListHead( &NewHead );
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
while( ThNext != ThHead ){
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
ThNext = ThNext->Flink;
CollapseTree(&pThread->TransListHead, &NewHead, FALSE );
}
TrNext = NewHead.Flink;
if( TrNext != &NewHead ){
fprintf(procFile,
"<process name=\"%ws\">"
"<pid>0x%04X</pid>\n",
(pProcess->ImageName) ? pProcess->ImageName : L"Idle",
pProcess->PID
);
TransInclusive( &NewHead, 0 );
fprintf( procFile, "</process>\n" );
}
DeleteTransList( &NewHead, 0 );
}
fprintf(procFile, "</table>\n" );
}
static void PrintProcessSubDataExclusive()
{
PPROCESS_RECORD pProcess;
PTRANS_RECORD pTrans;
PTHREAD_RECORD pThread;
PLIST_ENTRY ThNext, ThHead;
PMOF_INFO pMofInfo;
PLIST_ENTRY Next, Head;
PLIST_ENTRY TrNext;
LIST_ENTRY NewHead;
double percent, percentCPU, totalPerCPU;
double processPart;
double transPart;
double totalPercent;
double trans, KCPU, UCPU;
WCHAR str[MAXSTR];
double duration = ((double)((LONGLONG)(CurrentSystem.EndTime - CurrentSystem.StartTime))) / 10000000.00;
BOOL bTable = FALSE;
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head && !bTable ) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
Next = Next->Flink;
if( pProcess->PID == 0 ){
continue;
}
InitializeListHead( &NewHead );
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
while( ThNext != ThHead ){
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
ThNext = ThNext->Flink;
CollapseTree(&pThread->TransListHead, &NewHead, TRUE );
}
TrNext = NewHead.Flink;
if( TrNext != &NewHead ){
bTable = TRUE;
}
DeleteTransList( &NewHead, 0 );
}
if( !bTable ){
return;
}
// Walk through the Process List and Print the report.
fprintf(procFile, "<table title='Exclusive Transactions Per Process'>\n" );
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
BOOL titled = FALSE;
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
Next = Next->Flink;
if( pProcess->PID == 0 ){
continue;
}
InitializeListHead( &NewHead );
ThHead = &pProcess->ThreadListHead;
ThNext = ThHead->Flink;
while( ThNext != ThHead ){
pThread = CONTAINING_RECORD( ThNext, THREAD_RECORD, Entry );
ThNext = ThNext->Flink;
CollapseTree(&pThread->TransListHead, &NewHead, TRUE );
}
TrNext = NewHead.Flink;
totalPercent = 0.0;
totalPerCPU = 0.0;
while( TrNext != &NewHead ){
if(!titled){
fprintf(procFile,
"<process name=\"%ws\">"
"<pid>0x%04X</pid>\n",
(pProcess->ImageName) ? pProcess->ImageName : L"Idle",
pProcess->PID
);
titled = TRUE;
}
pTrans = CONTAINING_RECORD( TrNext, TRANS_RECORD, Entry );
TrNext = TrNext->Flink;
pMofInfo = GetMofInfoHead( pTrans->pGuid);
if (pMofInfo == NULL) {
return;
}
transPart = pTrans->UCpu + pTrans->KCpu;
processPart = CalculateProcessKCPU(pProcess)
+ CalculateProcessUCPU(pProcess);
percentCPU = ((((double)pTrans->KCpu + (double)pTrans->UCpu ) / 10.0 ) / duration) / ((double) CurrentSystem.NumberOfProcessors);
totalPerCPU += percentCPU;
if(processPart)
percent = (transPart/processPart) * 100.0;
else
percent = 0;
totalPercent += percent;
if (!(ReportFlags & TRACE_TOTALS_REPORT) ){
if (pTrans->RefCount == 0 && pTrans->RefCount1 == 0)
KCPU = UCPU = 0.0;
else if (pTrans->RefCount == 0) {
KCPU = (double) pTrans->KCpu;
UCPU = (double) pTrans->UCpu;
}
else {
KCPU = (double) pTrans->KCpu / (double) pTrans->RefCount;
UCPU = (double) pTrans->UCpu / (double) pTrans->RefCount;
}
}
else{
KCPU = (double)pTrans->KCpu;
UCPU = (double)pTrans->UCpu;
}
trans = (double)pTrans->RefCount / duration;
fprintf(procFile,
"<transaction name=\"%ws\">"
"<count>%d</count>"
"<rate>%1.2f</rate>"
"<kernel>%1.0f</kernel>"
"<user>%1.0f</user>"
"<process_cpu>%1.2f</process_cpu>"
"<cpu>%1.2f</cpu>"
"</transaction>\n",
(pMofInfo->strDescription != NULL) ? pMofInfo->strDescription : CpdiGuidToString( str, MAXSTR, &pMofInfo->Guid ),
pTrans->RefCount,
trans,
KCPU,
UCPU,
percent,
percentCPU
);
}
if( titled ){
fprintf( procFile, "</process>\n" );
}
DeleteTransList( &NewHead, 0 );
}
fprintf(procFile, "</table>\n" );
}
static void PrintPerThreadPerDiskTable( )
{
PPROCESS_RECORD pProcess;
PTDISK_RECORD pDisk;
PLIST_ENTRY Next, Head;
PLIST_ENTRY GNext, GHead;
PLIST_ENTRY DiNext, DiHead;
ULONG rio, wio, DiskNumber;
BOOL bTable = FALSE;
ULONGLONG Duration = (ULONGLONG)((CurrentSystem.EndTime - CurrentSystem.StartTime) / 10000000);
if (Duration == 0) {
return;
}
// Walk through the Process List and Print the report.
GHead = &CurrentSystem.GlobalDiskListHead;
GNext = GHead->Flink;
while (GNext != GHead && !bTable) {
pDisk = CONTAINING_RECORD( GNext, TDISK_RECORD, Entry);
DiskNumber = pDisk->DiskNumber;
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
DiHead = &pProcess->DiskListHead;
DiNext = DiHead->Flink;
while (DiNext != DiHead && !bTable) {
pDisk = CONTAINING_RECORD( DiNext, TDISK_RECORD, Entry );
if (DiskNumber != pDisk->DiskNumber) {
DiNext = DiNext->Flink;
continue;
}else{
bTable = TRUE;
break;
}
}
Next = Next->Flink;
}
GNext = GNext->Flink;
}
if( !bTable ){
return;
}
GHead = &CurrentSystem.GlobalDiskListHead;
GNext = GHead->Flink;
while (GNext != GHead) {
pDisk = CONTAINING_RECORD( GNext, TDISK_RECORD, Entry);
DiskNumber = pDisk->DiskNumber;
fprintf( procFile, "<table title='Disk' number='%d'>\n", DiskNumber );
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
while (Next != Head) {
pProcess = CONTAINING_RECORD( Next, PROCESS_RECORD, Entry );
DiHead = &pProcess->DiskListHead;
DiNext = DiHead->Flink;
while (DiNext != DiHead) {
pDisk = CONTAINING_RECORD( DiNext, TDISK_RECORD, Entry );
if (DiskNumber != pDisk->DiskNumber) {
DiNext = DiNext->Flink;
continue;
}
rio = pDisk->ReadCount + pDisk->HPF;
wio = pDisk->WriteCount;
fprintf(procFile,
"<image name='%ws'>"
"<pid>0x%08X</pid>"
"<authority>%ws</authority>"
"<read_rate>%1.3f</read_rate>"
"<read_size>%d</read_size>"
"<write_rate>%1.3f</write_rate>"
"<write_size>%d</write_size>"
"</image>\n",
(pProcess->ImageName) ? pProcess->ImageName : L"-",
pProcess->PID,
(pProcess->UserName) ? pProcess->UserName : L"-",
(double)rio / ((double)Duration),
(rio == 0) ? 0 : (pDisk->ReadSize + pDisk->HPFSize) / rio,
(double)wio / ((double)Duration),
(wio == 0) ? 0 : pDisk->WriteSize / wio
);
DiNext = DiNext->Flink;
}
Next = Next->Flink;
}
fprintf( procFile, "</table>\n" );
GNext = GNext->Flink;
}
}
static void WriteTransactionStatistics()
{
PLIST_ENTRY Head, Next;
PLIST_ENTRY Dhead, DNext;
ULONG trans;
double KCPU, UCPU, PerCpu;
double RIO, WIO, Send, Recv;
PMOF_INFO pMofInfo;
PMOF_DATA pMofData;
double AvgRT;
double TransRate;
WCHAR str[MAXSTR];
double duration = ((double)((LONGLONG)(CurrentSystem.EndTime - CurrentSystem.StartTime)))
/ 10000000.00;
BOOL bTable = FALSE;
Head = &CurrentSystem.EventListHead;
Next = Head->Flink;
while (Head != Next && !bTable ) {
pMofInfo = CONTAINING_RECORD(Next, MOF_INFO, Entry);
Dhead = &pMofInfo->DataListHead;
DNext = Dhead->Flink;
while(DNext!=Dhead){
pMofData = CONTAINING_RECORD(DNext, MOF_DATA, Entry);
trans = pMofData->CompleteCount;
if (trans > 0) {
bTable = TRUE;
break;
}
DNext = DNext->Flink;
}
Next = Next->Flink;
}
if( !bTable ){
return;
}
fprintf( procFile, "<table title='Transaction Statistics'>\n" );
Head = &CurrentSystem.EventListHead;
Next = Head->Flink;
while (Head != Next) {
pMofInfo = CONTAINING_RECORD(Next, MOF_INFO, Entry);
Dhead = &pMofInfo->DataListHead;
DNext = Dhead->Flink;
while(DNext!=Dhead){
pMofData = CONTAINING_RECORD(DNext, MOF_DATA, Entry);
trans = pMofData->CompleteCount;
if (trans > 0) {
UCPU = pMofData->UserCPU;
KCPU = pMofData->KernelCPU;
PerCpu = (((UCPU + KCPU)/1000.0)/duration) * 100.0;
UCPU /= trans;
KCPU /= trans;
if(CurrentSystem.NumberOfProcessors)
PerCpu/=CurrentSystem.NumberOfProcessors;
if( ReportFlags & TRACE_TOTALS_REPORT ){
RIO = pMofData->ReadCount;
WIO = pMofData->WriteCount;
Send = pMofData->SendCount;
Recv = pMofData->RecvCount;
}else{
RIO = pMofData->ReadCount / trans;
WIO = pMofData->WriteCount / trans;
Send = pMofData->SendCount / trans;
Recv = pMofData->RecvCount / trans;
}
AvgRT = (double)pMofData->TotalResponseTime;
AvgRT /= trans;
TransRate = ( (float)trans / duration );
// TODO: NOT /trans if TRACE_TOTALS_REPORT
fprintf(procFile,
"<transaction name='%ws'>"
"<count>%d</count>"
"<response_time>%1.0f</response_time>"
"<rate>%1.2f</rate>"
"<cpu>%1.2f</cpu>"
"<disk_read_per_trans>%1.2f</disk_read_per_trans>"
"<disk_write_per_trans>%1.2f</disk_write_per_trans>"
"<tcp_send_per_trans>%1.2f</tcp_send_per_trans>"
"<tcp_recv_per_trans>%1.2f</tcp_recv_per_trans>"
"</transaction>\n",
(pMofInfo->strDescription) ? pMofInfo->strDescription : CpdiGuidToString( str, MAXSTR, &pMofInfo->Guid ),
pMofData->CompleteCount,
AvgRT,
TransRate,
PerCpu,
RIO,
WIO,
Send,
Recv
);
}
DNext = DNext->Flink;
}
Next = Next->Flink;
}
fprintf(procFile, "</table>" );
}
static void WriteTransactionCPUTime()
{
PLIST_ENTRY Head, Next;
PLIST_ENTRY Dhead, DNext;
double KCPU, UCPU;
PMOF_INFO pMofInfo;
PMOF_DATA pMofData;
double trans;
double PerCpu;
WCHAR str[MAXSTR];
double duration = ((double)((LONGLONG)(CurrentSystem.EndTime - CurrentSystem.StartTime))) / 10000000.00;
BOOL bTable = FALSE;
Head = &CurrentSystem.EventListHead;
Next = Head->Flink;
while (Head != Next && !bTable ) {
pMofInfo = CONTAINING_RECORD(Next, MOF_INFO, Entry);
Dhead = &pMofInfo->DataListHead;
DNext = Dhead->Flink;
while(DNext!=Dhead){
pMofData = CONTAINING_RECORD(DNext, MOF_DATA, Entry);
trans = (double)pMofData->CompleteCount;
if (trans > 0) {
bTable = TRUE;
break;
}
DNext = DNext->Flink;
}
Next = Next->Flink;
}
if( !bTable ){
return;
}
fprintf( procFile, "<table title='Transaction CPU Utilization'>\n" );
Head = &CurrentSystem.EventListHead;
Next = Head->Flink;
while (Head != Next) {
pMofInfo = CONTAINING_RECORD(Next, MOF_INFO, Entry);
Dhead = &pMofInfo->DataListHead;
DNext = Dhead->Flink;
while(DNext!=Dhead){
pMofData = CONTAINING_RECORD(DNext, MOF_DATA, Entry);
trans = (double)pMofData->CompleteCount;
if (trans > 0) {
UCPU = pMofData->UserCPU;
UCPU /= trans;
KCPU = pMofData->KernelCPU;
KCPU /= trans;
PerCpu = (((pMofData->UserCPU + pMofData->KernelCPU + 0.0)/1000.0)/duration) * 100.0;
if( !(ReportFlags & TRACE_TOTALS_REPORT) ){
trans /= duration;
}
if(CurrentSystem.NumberOfProcessors){
PerCpu/=CurrentSystem.NumberOfProcessors;
}
// NOTE: RATE should be COUNT if TRACE_TOTALS_REPORT
fprintf(procFile,
"<transaction name='%ws'>"
"<rate>%1.2f</rate>"
"<min_kernel>%d</min_kernel>"
"<min_user>%d</min_user>"
"<max_kernel>%d</max_kernel>"
"<max_user>%d</max_user>"
"<per_trans_kernel>%1.0f</per_trans_kernel>"
"<per_trans_user>%1.0f</per_trans_user>"
"<total_kernel>%d</total_kernel>"
"<total_user>%d</total_user>"
"<cpu>%1.2f</cpu>"
"</transaction>\n",
(pMofInfo->strDescription) ? pMofInfo->strDescription : CpdiGuidToString( str, MAXSTR, &pMofInfo->Guid ),
trans,
pMofData->MinKCpu,
pMofData->MinUCpu,
pMofData->MaxKCpu,
pMofData->MaxUCpu,
KCPU,
UCPU,
pMofData->KernelCPU,
pMofData->UserCPU,
PerCpu
);
}
DNext = DNext->Flink;
}
Next = Next->Flink;
}
fprintf(procFile, "</table>\n" );
}
PWCHAR CpdiGuidToString(
PWCHAR s,
ULONG len,
LPGUID piid
)
{
StringCchPrintf(s, len,
L"{%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",
piid->Data1, piid->Data2,
piid->Data3,
piid->Data4[0], piid->Data4[1],
piid->Data4[2], piid->Data4[3],
piid->Data4[4], piid->Data4[5],
piid->Data4[6], piid->Data4[7]);
return(s);
}
#define MAX_LOGS 1024 * 1024
typedef struct _PTR_RECORD
{
PVOID ptrRecord;
ULONG keySort;
} PTR_RECORD, * PPTR_RECORD;
static PPTR_RECORD PtrBuffer = NULL;
static ULONG PtrMax = MAX_LOGS;
static ULONG PtrTotal = 0;
static ULONG PtrIndex;
int __cdecl
CompareFileRecord(const void * p1, const void * p2)
{
PPTR_RECORD pp1 = (PPTR_RECORD) p1;
PPTR_RECORD pp2 = (PPTR_RECORD) p2;
PFILE_RECORD pFile1 = (PFILE_RECORD) pp1->ptrRecord;
PFILE_RECORD pFile2 = (PFILE_RECORD) pp2->ptrRecord;
LONG diffFault = pp2->keySort - pp1->keySort;
if (diffFault == 0)
{
diffFault = pFile1->DiskNumber - pFile2->DiskNumber;
if (diffFault == 0)
{
diffFault = wcscmp(pFile1->FileName, pFile2->FileName);
}
}
return diffFault;
}
void
ReportHotFileInfo()
{
PLIST_ENTRY pHead = & CurrentSystem.HotFileListHead;
PLIST_ENTRY pNext = pHead->Flink;
PFILE_RECORD pFile;
BOOLEAN fDone = FALSE;
ULONGLONG Duration = ((CurrentSystem.EndTime - CurrentSystem.StartTime) / 10000000);
PMOF_INFO pMofInfo;
pMofInfo = GetMofInfoHead ((LPCGUID)&ProcessGuid);
if ( pMofInfo == NULL ) {
return;
}
pMofInfo = GetMofInfoHead ((LPCGUID)&ThreadGuid);
if ( pMofInfo == NULL ) {
return;
}
ASSERT(!PtrBuffer);
PtrBuffer = (PPTR_RECORD) VirtualAlloc(
NULL,
sizeof(PTR_RECORD) * PtrMax,
MEM_COMMIT,
PAGE_READWRITE);
if( NULL == PtrBuffer ){
goto Cleanup;
}
while (!fDone)
{
for (PtrTotal = 0, fDone = TRUE;
pNext != pHead;
pNext = pNext->Flink)
{
if (PtrTotal == PtrMax)
{
fDone = FALSE;
break;
}
pFile = CONTAINING_RECORD(pNext, FILE_RECORD, Entry);
if (pFile->ReadCount + pFile->WriteCount > 0)
{
PtrBuffer[PtrTotal].ptrRecord = (PVOID) pFile;
PtrBuffer[PtrTotal].keySort =
pFile->ReadCount + pFile->WriteCount;
PtrTotal ++;
}
}
if (!fDone)
{
VirtualFree(PtrBuffer, 0, MEM_RELEASE);
PtrMax += MAX_LOGS;
PtrBuffer = (PPTR_RECORD) VirtualAlloc(
NULL,
sizeof(PTR_RECORD) * PtrMax,
MEM_COMMIT,
PAGE_READWRITE);
if (PtrBuffer == NULL)
{
goto Cleanup;
}
}
}
if (PtrTotal > 1) {
qsort((void *) PtrBuffer,
(size_t) PtrTotal,
(size_t) sizeof(PTR_RECORD),
CompareFileRecord);
}
else {
return;
}
if (PtrTotal > DISPLAY_SIZE) {
PtrTotal = DISPLAY_SIZE;
}
// output HotFile report title
//
fprintf( procFile, "<table title='Files Causing Most Disk IOs' top='%d'>\n", DISPLAY_SIZE );
for (PtrIndex = 0; PtrIndex < PtrTotal; PtrIndex ++)
{
PLIST_ENTRY pProtoHead;
PLIST_ENTRY pProtoNext;
PPROTO_PROCESS_RECORD pProto;
WCHAR ReducedFileName[FILE_NAME_COLUMN_SIZE + 1];
LPWSTR szFile;
PWCHAR szLongFile;
LPWSTR szDrive = L"";
pFile = (PFILE_RECORD) PtrBuffer[PtrIndex].ptrRecord;
if (NULL != pFile->Drive) {
szDrive = pFile->Drive;
}
if (_wcsnicmp(pFile->FileName, L"\\Device\\", 8) == 0) {
szLongFile = (PWCHAR)(pFile->FileName) + 8;
while (*szLongFile != L'\\' && *szLongFile != L'\0') {
szLongFile++;
}
}
else {
szLongFile = (PWCHAR)(pFile->FileName);
}
if (wcslen(szLongFile) > FILE_NAME_COLUMN_SIZE) {
ReduceStringW(ReducedFileName, FILE_NAME_COLUMN_SIZE + 1, szLongFile);
szFile = ReducedFileName;
}else{
szFile = szLongFile;
}
fprintf(procFile,
"<file name='%ws'>"
"<disk>%d</disk>"
"<drive>%ws</drive>"
"<read_rate>%1.3f</read_rate>"
"<read_size>%d</read_size>"
"<write_rate>%1.3f</write_rate>"
"<write_size>%d</write_size>\n",
szFile,
pFile->DiskNumber,
szDrive,
Duration ? ((double)pFile->ReadCount / (double)Duration) : 0.0,
(pFile->ReadCount) ? (pFile->ReadSize / pFile->ReadCount) : 0,
Duration ? ((double)pFile->WriteCount / (double)Duration) : 0.0,
(pFile->WriteCount) ? (pFile->WriteSize / pFile->WriteCount) : 0);
pProtoHead = & pFile->ProtoProcessListHead;
pProtoNext = pProtoHead->Flink;
while (pProtoNext != pProtoHead)
{
pProto = CONTAINING_RECORD(pProtoNext, PROTO_PROCESS_RECORD, Entry);
pProtoNext = pProtoNext->Flink;
if (pProto->ReadCount + pProto->WriteCount > 0)
{
fprintf(procFile,
"<image name='%ws'>"
"<pid>0x%08X</pid>"
"<read_rate>%1.3f</read_rate>"
"<read_size>%d</read_size>"
"<write_rate>%1.3f</write_rate>"
"<write_size>%d</write_size>"
"</image>\n",
pProto->ProcessRecord->ImageName,
pProto->ProcessRecord->PID,
Duration ? ((double)pProto->ReadCount / (double)Duration) : 0.0,
(pProto->ReadCount) ? (pProto->ReadSize / pProto->ReadCount) : 0,
Duration ? ((double)pProto->WriteCount / (double)Duration) : 0.0,
(pProto->WriteCount) ? (pProto->WriteSize / pProto->WriteCount) : 0);
}
}
fprintf( procFile, "</file>\n" );
}
fprintf( procFile, "</table>\n" );
Cleanup:
if (PtrBuffer)
{
VirtualFree(PtrBuffer, 0, MEM_RELEASE);
}
}
#define CHECKTOK( x ) if( NULL == x ) { continue; }
static void ReportPrintJobInfo2(void)
{
PRINT_JOB_RECORD Job, *pJob;
char* s;
char line[MAXSTR];
ULONG TotalCount = 0;
ULONGLONG TotalRT = 0;
ULONG TotalCPUTime = 0;
pJob = &Job;
if( NULL == CurrentSystem.TempPrintFile ){
return;
}
rewind( CurrentSystem.TempPrintFile );
while ( fgets(line, MAXSTR, CurrentSystem.TempPrintFile) != NULL ) {
s = strtok( line, (","));
CHECKTOK( s );
pJob->JobId = atol(s);
if (pJob == NULL){
return;
}
}
fprintf(procFile, "<table title='Spooler Transaction Instance (Job) Data'>\n");
RtlZeroMemory(pJob, sizeof(PRINT_JOB_RECORD));
RtlZeroMemory(line, MAXSTR * sizeof(char));
rewind( CurrentSystem.TempPrintFile );
while ( fgets(line, MAXSTR, CurrentSystem.TempPrintFile) != NULL ) {
s = strtok( line, (","));
CHECKTOK( s );
pJob->JobId = atol(s);
if (pJob == NULL){
continue;
}
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->KCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->UCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->ReadIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->StartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->EndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->ResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->PrintJobTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->WriteIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->DataType = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->JobSize = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Pages = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->PagesPerSide = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->FilesOpened = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->GdiJobSize = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Color = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->XRes = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->YRes = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Quality = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Copies = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->TTOption = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->NumberOfThreads = atol(s);
fprintf(procFile,
"<job id='%d'>"
"<type>%d</type>"
"<size>%d</size>"
"<pages>%d</pages>"
"<PPS>%d</PPS>"
"<files>%hd</files>"
"<gdisize>%d</gdisize>"
"<color>%hd</color>"
"<xres>%hd</xres>"
"<yres>%hd</yres>"
"<qlty>%hd</qlty>"
"<copies>%hd</copies>"
"<ttopt>%hd</ttopt>"
"<threads>%d</threads>"
"</job>\n",
pJob->JobId,
pJob->DataType,
((pJob->JobSize) / 1024),
pJob->Pages,
pJob->PagesPerSide,
pJob->FilesOpened,
pJob->GdiJobSize,
pJob->Color,
pJob->XRes,
pJob->YRes,
pJob->Quality,
pJob->Copies,
pJob->TTOption,
pJob->NumberOfThreads
);
}
fprintf(procFile, "</table>\n" );
}
static void ReportPrintJobInfo(void)
{
PRINT_JOB_RECORD Job, *pJob;
char* s;
char line[MAXSTR];
FILETIME StTm, StlTm;
LARGE_INTEGER LargeTmp;
SYSTEMTIME stStart, stEnd, stDequeue;
ULONG TotalCount = 0;
ULONGLONG TotalRT = 0;
ULONG TotalCPUTime = 0;
if( NULL == CurrentSystem.TempPrintFile ){
return;
}
pJob = &Job;
rewind( CurrentSystem.TempPrintFile );
while ( fgets(line, MAXSTR, CurrentSystem.TempPrintFile) != NULL ) {
s = strtok( line, (","));
CHECKTOK( s );
pJob->JobId = atol(s);
if (pJob == NULL){
return;
}
}
fprintf(procFile, "<table title='Transaction Instance (Job) Statistics'>\n" );
RtlZeroMemory(pJob, sizeof(PRINT_JOB_RECORD));
RtlZeroMemory(line, MAXSTR * sizeof(char));
rewind( CurrentSystem.TempPrintFile );
while ( fgets(line, MAXSTR, CurrentSystem.TempPrintFile) != NULL ) {
s = strtok( line, (","));
CHECKTOK( s );
pJob->JobId = atol(s);
if (pJob == NULL){
continue;
}
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->KCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->UCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->ReadIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->StartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->EndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->ResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->PrintJobTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->WriteIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->DataType = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->JobSize = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Pages = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->PagesPerSide = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->FilesOpened = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->GdiJobSize = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Color = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->XRes = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->YRes = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Quality = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->Copies = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->TTOption = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pJob->NumberOfThreads = atol(s);
LargeTmp.QuadPart = pJob->StartTime;
StTm.dwHighDateTime = LargeTmp.HighPart;
StTm.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&StTm, &StlTm);
FileTimeToSystemTime (
&StlTm,
&stStart
);
LargeTmp.QuadPart = pJob->EndTime;
StTm.dwHighDateTime = LargeTmp.HighPart;
StTm.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&StTm, &StlTm);
FileTimeToSystemTime (
&StlTm,
&stEnd
);
LargeTmp.QuadPart = pJob->PrintJobTime;
StTm.dwHighDateTime = LargeTmp.HighPart;
StTm.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&StTm, &StlTm);
FileTimeToSystemTime (
&StlTm,
&stDequeue
);
fprintf(procFile,
"<job id='%d'>"
"<start>%2d:%02d:%02d.%03d</start>"
"<dequeue>%2d:%02d:%02d.%03d</dequeue>"
"<end>%2d:%02d:%02d.%03d</end>"
"<response_time>%I64u</response_time>"
"<cpu>%d</cpu>"
"</job>\n",
pJob->JobId,
stStart.wHour, stStart.wMinute, stStart.wSecond, stStart.wMilliseconds,
stDequeue.wHour, stDequeue.wMinute, stDequeue.wSecond, stDequeue.wMilliseconds,
stEnd.wHour, stEnd.wMinute, stEnd.wSecond, stEnd.wMilliseconds,
pJob->ResponseTime,
pJob->KCPUTime + pJob->UCPUTime
);
TotalCount++;
TotalRT += pJob->ResponseTime;
TotalCPUTime += (pJob->KCPUTime + pJob->UCPUTime);
}
if (TotalCount > 0) {
TotalRT /= TotalCount;
TotalCPUTime /= TotalCount;
}
fprintf(procFile,
"<summary>"
"<count>%d</count>"
"<total_response_time>%I64u</total_response_time>"
"<total_cpu>%d</total_cpu>"
"</summary>\n",
TotalCount, TotalRT, TotalCPUTime );
fprintf(procFile, "</table>\n" );
ReportPrintJobInfo2();
}
PUCHAR CopyUrlStr(PUCHAR UrlStr) {
PUCHAR CopiedStr;
if (UrlStr == NULL) {
return NULL;
}
CopiedStr = (PUCHAR)malloc(strlen(UrlStr) + 1);
if (CopiedStr != NULL) {
RtlCopyMemory(CopiedStr, UrlStr, strlen(UrlStr) + 1);
return CopiedStr;
}
else {
return NULL;
}
}
void ProcessIisRequest(HTTP_REQUEST_RECORD *pReq)
{
// Global variable IIS must not be NULL here.
PURL_RECORD pUrl;
PCLIENT_RECORD pClient;
PSITE_RECORD pSite;
pUrl = FindOrAddUrlRecord(pReq->URL);
if (pUrl != NULL) {
pUrl->SiteId = pReq->SiteId;
pUrl->Requests++;
pUrl->TotalResponseTime += pReq->ULResponseTime;
pUrl->KCPUTime += pReq->KCPUTime;
pUrl->UCPUTime += pReq->UCPUTime;
pUrl->ReadIO += pReq->ReadIO;
pUrl->WriteIO += pReq->WriteIO;
pUrl->BytesSent += pReq->BytesSent;
// Cached responses don't have this field filled
// Thus need to check for Hits field as well
pUrl->UrlType = pReq->W3ProcessType;
}
pClient = FindOrAddClientRecord(pReq->IpAddrType, pReq->IpAddrV4, pReq->IpAddrV6);
if (pClient != NULL) {
pClient->Requests++;
pClient->TotalResponseTime += pReq->ULResponseTime;
pClient->BytesSent += pReq->BytesSent;
}
pSite = FindOrAddSiteRecord(pReq->SiteId);
if (pSite != NULL) {
pSite->Requests++;
pSite->TotalResponseTime += pReq->ULResponseTime;
pSite->KCPUTime += pReq->KCPUTime;
pSite->UCPUTime += pReq->UCPUTime;
pSite->ReadIO += pReq->ReadIO;
pSite->WriteIO += pReq->WriteIO;
pSite->BytesSent += pReq->BytesSent;
}
if (pReq->ULEndType == EVENT_TRACE_TYPE_UL_CACHEDEND) {
IIS->CachedResponses++;
IIS->TotalCachedResponseTime += pReq->ULResponseTime;
IIS->CachedCPUTime += pReq->KCPUTime + pReq->UCPUTime;
if (pUrl != NULL) {
pUrl->Hits++;
}
if (pClient != NULL) {
pClient->Hits++;
}
if (pSite != NULL) {
pSite->Hits++;
}
}
else {
ULONGLONG localCGIEndTime = pReq->CGIEndTime;
ULONGLONG localISAPIEndTime = pReq->ISAPIEndTime;
ULONGLONG localASPEndTime = pReq->ASPEndTime;
ULONGLONG W3ResponseTime;
#ifdef DBG
// we'll check the validity of HTTP Requests Here.
ULONGLONG ULSum = 0, W3Res = 0;
if (pReq->ULDeliverTime == 0 || pReq->ULReceiveTime == 0) {
TrctrDbgPrint(("TRACERPT Error Req: %I64u UL DeliverTime nad/or ReceiveTime not available, should throw this away.\n", pReq->RequestId));
}
else {
ULSum = (pReq->ULDeliverTime - pReq->ULStartTime) + (pReq->ULEndTime - pReq->ULReceiveTime);
}
if (pReq->W3StartTime == 0) {
TrctrDbgPrint(("TRACERPT Error Req: %I64u W3 StartTime not available, should throw this away.\n", pReq->RequestId));
}
if (pReq->W3EndTime == 0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u W3 EndTime not available.\n", pReq->RequestId));
}
if (pReq->W3StartTime != 0 && pReq->W3EndTime != 0 && pReq->W3EndTime < pReq->W3StartTime) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u W3 EndTime smaller than pReq->W3StartTime.\n", pReq->RequestId));
}
if (ULSum != 0 && pReq->W3StartTime != 0 && pReq->W3EndTime != 0) {
double ULRatio = 0.0;
W3Res = pReq->W3EndTime - pReq->W3StartTime;
ULRatio = (double)ULSum / (double)(pReq->ULResponseTime - W3Res);
if (ULRatio < 0.5 || ULRatio > 2.0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u UL Ratio is unreal.\n", pReq->RequestId));
}
}
if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_CGIREQ) {
if (pReq->CGIStartTime == 0) {
TrctrDbgPrint(("TRACERPT Error Req: %I64u CGI StartTime not available, should throw this away.\n", pReq->RequestId));
}
if (pReq->CGIEndTime == 0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u CGI EndTime not available.\n", pReq->RequestId));
if (pReq->W3EndTime != 0 && (pReq->CGIStartTime > pReq->W3EndTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u CGI StartTime > pReq->W3EndTime.\n", pReq->RequestId));
}
else if (pReq->W3EndTime == 0 && (pReq->CGIStartTime > pReq->ULReceiveTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u CGI StartTime > pReq->ULReceiveTime.\n", pReq->RequestId));
}
}
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_ISAPIREQ) {
if (pReq->ISAPIStartTime == 0) {
TrctrDbgPrint(("TRACERPT Error Req: %I64u ISAPI StartTime not available, should throw this away.\n", pReq->RequestId));
}
if (pReq->ISAPIEndTime == 0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ISAPI EndTime not available.\n", pReq->RequestId));
if (pReq->W3EndTime != 0 && (pReq->ISAPIStartTime > pReq->W3EndTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ISAPI StartTime > pReq->W3EndTime.\n", pReq->RequestId));
}
else if (pReq->W3EndTime == 0 && (pReq->ISAPIStartTime > pReq->ULReceiveTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ISAPI StartTime > pReq->ULReceiveTime.\n", pReq->RequestId));
}
}
if (pReq->ASPStartTime == 0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ASP StartTime not available.\n", pReq->RequestId));
}
if (pReq->ASPEndTime == 0) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ASP EndTime not available.\n", pReq->RequestId));
if (pReq->ISAPIEndTime != 0 && (pReq->ASPStartTime > pReq->ISAPIEndTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ASP StartTime > pReq->ISAPIEndTime.\n", pReq->RequestId));
}
else if (pReq->W3EndTime != 0 && (pReq->ASPStartTime > pReq->W3EndTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ASP StartTime > pReq->W3EndTime.\n", pReq->RequestId));
}
else if (pReq->W3EndTime == 0 && (pReq->ASPStartTime > pReq->ULReceiveTime)) {
TrctrDbgPrint(("TRACERPT Warning Req: %I64u ASP StartTime > pReq->ULReceiveTime.\n", pReq->RequestId));
}
}
}
#endif
// Fix the time inconsistency in transactions
if (pReq->W3StartTime == 0 || (pReq->W3EndTime != 0 && pReq->W3EndTime < pReq->W3StartTime)) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
if (pReq->W3EndTime == 0 || pReq->W3EndTime > pReq->ULReceiveTime) {
pReq->W3EndTime = pReq->ULReceiveTime;
if (pReq->W3EndTime == 0 || pReq->W3EndTime < pReq->W3StartTime) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
}
W3ResponseTime = pReq->W3EndTime - pReq->W3StartTime;
if (W3ResponseTime < pReq->W3FilterResponseTime) {
pReq->W3FilterResponseTime = W3ResponseTime;
}
if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_CGIREQ) {
if (pReq->CGIStartTime == 0) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
else if (pReq->CGIEndTime == 0 || pReq->CGIEndTime > pReq->W3EndTime) {
localCGIEndTime = pReq->W3EndTime;
if (pReq->CGIStartTime > localCGIEndTime) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
}
else {
localCGIEndTime = pReq->CGIEndTime;
}
if (pReq->CGIStartTime < pReq->W3StartTime) {
pReq->CGIStartTime = pReq->W3StartTime;
}
if (localCGIEndTime < pReq->CGIStartTime) {
localCGIEndTime = pReq->CGIStartTime;
}
if ((pReq->W3FilterResponseTime + (localCGIEndTime - pReq->CGIStartTime)) > W3ResponseTime) {
pReq->W3FilterResponseTime = W3ResponseTime - (localCGIEndTime - pReq->CGIStartTime);
}
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_ISAPIREQ) {
if (pReq->ISAPIStartTime == 0) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
if (pReq->ISAPIEndTime == 0 || pReq->ISAPIEndTime > pReq->W3EndTime) {
localISAPIEndTime = pReq->W3EndTime;
if (pReq->ISAPIStartTime > localISAPIEndTime) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
}
else {
localISAPIEndTime = pReq->ISAPIEndTime;
}
if (pReq->ISAPIStartTime < pReq->W3StartTime) {
pReq->ISAPIStartTime = pReq->W3StartTime;
}
if (localISAPIEndTime < pReq->ISAPIStartTime) {
localISAPIEndTime = pReq->ISAPIStartTime;
}
if ((pReq->W3FilterResponseTime + (localISAPIEndTime - pReq->ISAPIStartTime)) > W3ResponseTime) {
pReq->W3FilterResponseTime = W3ResponseTime - (localISAPIEndTime - pReq->ISAPIStartTime);
}
if (pReq->ASPStartTime != 0) {
if (pReq->ASPEndTime == 0) {
localASPEndTime = localISAPIEndTime;
if (pReq->ASPStartTime > localASPEndTime) {
RequestsDiscarded++;
if (pReq->URL != NULL) {
free(pReq->URL);
}
return;
}
}
else {
localASPEndTime = pReq->ASPEndTime;
}
if (pReq->ASPStartTime < pReq->ISAPIStartTime) {
pReq->ASPStartTime = pReq->ISAPIStartTime;
}
if (localASPEndTime < pReq->ASPStartTime) {
localASPEndTime = pReq->ASPStartTime;
}
if ((localISAPIEndTime - pReq->ISAPIStartTime) < (localASPEndTime - pReq->ASPStartTime)) {
localASPEndTime = localISAPIEndTime;
}
}
} // Fix done. All end times are fixed.
IIS->TotalNonCachedResponseTime += pReq->ULResponseTime;
IIS->NonCachedCPUTime += pReq->KCPUTime + pReq->UCPUTime;
if (pReq->W3FilterResponseTime != 0) {
IIS->W3FilterRequests += pReq->W3FilterVisits;
IIS->TotalW3FilterResponseTime += pReq->W3FilterResponseTime;
IIS->TotalW3FilterCPUTime += pReq->W3FltrCPUTime;
}
if (pReq->ULEndType == EVENT_TRACE_TYPE_UL_SENDERROR && pReq->ISAPIStartTime == 0) {
// This is a request ended with SENDERROR.
SendErrorRequests++;
}
if (pReq->HttpStatus != 0 && pReq->HttpStatus >= 400) {
// This is a request with http error.
IIS->W3Error++;
IIS->TotalErrorResponseTime += pReq->ULResponseTime;
IIS->ErrorCPUTime += pReq->KCPUTime + pReq->UCPUTime;
IIS->TotalErrorULOnlyCPUTime += pReq->ULCPUTime;
IIS->TotalErrorW3OnlyCPUTime += pReq->W3CPUTime;
IIS->TotalErrorW3FilterCPUTime += pReq->W3FltrCPUTime;
IIS->TotalErrorCGIOnlyCPUTime += pReq->CGICPUTime;
IIS->TotalErrorISAPIOnlyCPUTime += pReq->ISAPICPUTime;
IIS->TotalErrorASPOnlyCPUTime += pReq->ASPCPUTime;
IIS->TotalErrorW3OnlyResponseTime += (pReq->W3EndTime - pReq->W3StartTime)
- pReq->W3FilterResponseTime;
if (pReq->ULStartTime != 0 && pReq->ULEndTime != 0 && pReq->ULDeliverTime != 0 && pReq->ULReceiveTime != 0) {
// ????????????????????????????????????????????????????????????????????????
// TotalErrorULOnlyResponseTime += (pReq->ULDeliverTime - pReq->ULStartTime)
// + (pReq->ULEndTime - pReq->ULReceiveTime);
IIS->TotalErrorULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
else {
IIS->TotalErrorULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
IIS->TotalErrorW3FilterResponseTime += pReq->W3FilterResponseTime;
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_FILEREQ) {
IIS->W3FileRequests++;
IIS->TotalFileResponseTime += pReq->ULResponseTime;
IIS->FileCPUTime += pReq->KCPUTime + pReq->UCPUTime;
IIS->TotalFileULOnlyCPUTime += pReq->ULCPUTime;
IIS->TotalFileW3OnlyCPUTime += pReq->W3CPUTime;
IIS->TotalFileW3FilterCPUTime += pReq->W3FltrCPUTime;
if (pSite != NULL) {
pSite->FileRequests++;
}
IIS->TotalFileW3OnlyResponseTime += (pReq->W3EndTime - pReq->W3StartTime)
- pReq->W3FilterResponseTime;
if (pReq->ULStartTime != 0 && pReq->ULEndTime != 0 && pReq->ULDeliverTime != 0 && pReq->ULReceiveTime != 0) {
// ????????????????????????????????????????????????????????????????????????
// TotalFileULOnlyResponseTime += (pReq->ULDeliverTime - pReq->ULStartTime)
// + (pReq->ULEndTime - pReq->ULReceiveTime);
IIS->TotalFileULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
else {
IIS->TotalFileULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
IIS->TotalFileW3FilterResponseTime += pReq->W3FilterResponseTime;
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_CGIREQ) {
IIS->W3CGIRequests++;
IIS->TotalCGIResponseTime += pReq->ULResponseTime;
IIS->TotalCGIOnlyResponseTime += (localCGIEndTime - pReq->CGIStartTime);
IIS->CGICPUTime += pReq->KCPUTime + pReq->UCPUTime;
IIS->TotalCGIOnlyCPUTime += pReq->CGICPUTime;
IIS->TotalCGIULOnlyCPUTime += pReq->ULCPUTime;
IIS->TotalCGIW3OnlyCPUTime += pReq->W3CPUTime;
IIS->TotalCGIW3FilterCPUTime += pReq->W3FltrCPUTime;
if (pSite != NULL) {
pSite->CGIRequests++;
}
IIS->TotalCGIW3OnlyResponseTime += (pReq->W3EndTime - pReq->W3StartTime)
- (localCGIEndTime - pReq->CGIStartTime)
- pReq->W3FilterResponseTime;
if (pReq->ULStartTime != 0 && pReq->ULEndTime != 0 && pReq->ULDeliverTime != 0 && pReq->ULReceiveTime != 0) {
// ????????????????????????????????????????????????????????????????????????
// TotalCGIULOnlyResponseTime += (pReq->ULDeliverTime - pReq->ULStartTime)
// + (pReq->ULEndTime - pReq->ULReceiveTime);
IIS->TotalCGIULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
else {
IIS->TotalCGIULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
IIS->TotalCGIW3FilterResponseTime += pReq->W3FilterResponseTime;
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_ISAPIREQ) {
ULONGLONG TimeSpentInAsp = 0;
ULONGLONG TimeSpentInISAPI = 0;
if (pReq->ASPStartTime != 0 && localASPEndTime != 0) {
TimeSpentInAsp = localASPEndTime - pReq->ASPStartTime;
}
if (pReq->ISAPIStartTime != 0 && localISAPIEndTime != 0) {
TimeSpentInISAPI = localISAPIEndTime - pReq->ISAPIStartTime;
if (TimeSpentInAsp > TimeSpentInISAPI) {
TimeSpentInISAPI = 0;
}
else {
TimeSpentInISAPI -= TimeSpentInAsp;
}
}
IIS->W3ISAPIRequests++;
IIS->TotalISAPIResponseTime += pReq->ULResponseTime;
IIS->ISAPICPUTime += pReq->KCPUTime + pReq->UCPUTime;
if (pSite != NULL) {
pSite->ISAPIRequests++;
}
IIS->TotalASPW3OnlyResponseTime += (((pReq->W3EndTime - pReq->W3StartTime) - pReq->W3FilterResponseTime)
- (TimeSpentInISAPI + TimeSpentInAsp));
IIS->W3ASPRequests++;
IIS->TotalASPResponseTime += pReq->ULResponseTime;
IIS->TotalASPOnlyResponseTime += TimeSpentInAsp;
IIS->ASPCPUTime += pReq->KCPUTime + pReq->UCPUTime;
IIS->TotalASPOnlyCPUTime += pReq->ASPCPUTime;
IIS->TotalASPULOnlyCPUTime += pReq->ULCPUTime;
IIS->TotalASPW3OnlyCPUTime += pReq->W3CPUTime;
IIS->TotalASPW3FilterCPUTime += pReq->W3FltrCPUTime;
IIS->TotalASPISAPIOnlyCPUTime += pReq->ISAPICPUTime;
if (pSite != NULL) {
pSite->ASPRequests++;
}
IIS->TotalASPISAPIOnlyResponseTime += TimeSpentInISAPI;
if (pReq->ULStartTime != 0 && pReq->ULEndTime != 0 && pReq->ULDeliverTime != 0 && pReq->ULReceiveTime != 0) {
// ????????????????????????????????????????????????????????????????????????
// TotalASPULOnlyResponseTime += (pReq->ULDeliverTime - pReq->ULStartTime)
// + (pReq->ULEndTime - pReq->ULReceiveTime);
IIS->TotalASPULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
else {
IIS->TotalASPULOnlyResponseTime += pReq->ULResponseTime - (pReq->W3EndTime - pReq->W3StartTime);
}
IIS->TotalASPW3FilterResponseTime += pReq->W3FilterResponseTime;
}
else if (pReq->W3ProcessType == EVENT_TRACE_TYPE_W3CORE_OOPREQ) { // ???
IIS->W3OOPRequests++;
IIS->TotalOOPResponseTime += pReq->ULResponseTime;
IIS->OOPCPUTime += pReq->KCPUTime + pReq->UCPUTime;
if (pSite != NULL) {
pSite->OOPRequests++;
}
}
}
IIS->TotalRequests++;
IIS->TotalCPUTime += pReq->KCPUTime + pReq->UCPUTime;
if (pReq->URL != NULL) {
free(pReq->URL);
}
}
static void ReportIisEvents(void)
{
HTTP_REQUEST_RECORD Req, *pReq;
char* s;
char line[MAXSTR];
URL_RECORD TopHitURLs[DISPLAY_SIZE];
URL_RECORD TopHitStaticURLs[DISPLAY_SIZE];
URL_RECORD TopSlowURLs[DISPLAY_SIZE];
URL_RECORD TopConsumingURLs[DISPLAY_SIZE];
URL_RECORD TopBytesURLs[DISPLAY_SIZE];
PURL_RECORD pUrl;
CLIENT_RECORD TopHitClients[DISPLAY_SIZE];
CLIENT_RECORD TopSlowClients[DISPLAY_SIZE];
CLIENT_RECORD TopBytesClients[DISPLAY_SIZE];
PCLIENT_RECORD pClient;
SITE_RECORD TopHitSites[DISPLAY_SIZE];
SITE_RECORD TopSlowSites[DISPLAY_SIZE];
SITE_RECORD TopConsumingSites[DISPLAY_SIZE];
SITE_RECORD TopBytesSites[DISPLAY_SIZE];
PSITE_RECORD pSite;
ULONG i, k;
ULONG Duration = (ULONG)((CurrentSystem.IISEndTime - CurrentSystem.IISStartTime) / 10000000);
ULONG MilDuration = (ULONG)((CurrentSystem.IISEndTime - CurrentSystem.IISStartTime) * CurrentSystem.NumberOfProcessors / 10000);
double Rates = 0.0;
PLIST_ENTRY Next, Head;
ULONG PrintCPUUsage = TRUE;
if( NULL == CurrentSystem.TempIisFile ){
return;
}
IIS = (PIIS_REPORT_RECORD)malloc(sizeof(IIS_REPORT_RECORD));
if (IIS == NULL) {
return;
}
RtlZeroMemory(IIS, sizeof(IIS_REPORT_RECORD));
Head = &CurrentSystem.ProcessListHead;
Next = Head->Flink;
if( Head == Next ){
PrintCPUUsage = FALSE;
}
pReq = &Req;
RtlZeroMemory(pReq, sizeof(HTTP_REQUEST_RECORD));
RtlZeroMemory(line, MAXSTR * sizeof(char));
// Process requests written in the file
rewind( CurrentSystem.TempIisFile );
while ( fgets(line, MAXSTR, CurrentSystem.TempIisFile) != NULL ) {
s = strtok( line, (","));
CHECKTOK( s );
pReq->RequestId = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->SiteId = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->KCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->UCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ReadIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->WriteIO = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULStartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULEndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULParseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULDeliverTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULReceiveTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULReceiveType = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULEndType = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3StartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3EndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3FilterResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3ProcessType = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3EndType = (SHORT) atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->FileReqTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->CGIStartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->CGIEndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ISAPIStartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ISAPIEndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ASPStartTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ASPEndTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->SSLResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->StrmFltrResponseTime = _atoi64(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->HttpStatus = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IsapiExt = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrType = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV4 = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[0] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[1] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[2] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[3] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[4] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[5] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[6] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->IpAddrV6[7] = (USHORT) atol(s);;
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->NumberOfThreads = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->BytesSent = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ULCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3CPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->W3FltrCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ISAPICPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->ASPCPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
pReq->CGICPUTime = atol(s);
s = strtok( NULL, (","));
CHECKTOK( s );
if (strlen(s) > 0) {
LPSTR strptr = (LPSTR)malloc(strlen(s) + 1);
if (strptr != NULL) {
RtlCopyMemory((PUCHAR)strptr, (PUCHAR)s, strlen(s));
if (*(strptr + strlen(s) - 1) == '\n') {
*(strptr + strlen(s) - 1) = '\0';
}
else {
*(strptr + strlen(s)) = '\0';
}
pReq->URL = (PCHAR)strptr;
}
else {
continue;
}
}
else {
continue;
}
ProcessIisRequest(pReq);
}
if (IIS->TotalRequests != 0) {
if (Duration != 0) {
Rates = (double)(IIS->TotalRequests) / (double)Duration;
}
else {
Rates = 0.0;
}
}
else {
free(IIS);
return;
}
fprintf(procFile, "<table title='Http Requests Response Time Statistics'>\n");
fprintf(procFile,
"<requests cached='true' type='Static HTTP'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%3d.0</component>"
"<component name='W3'>0.0</component>"
"<component name='W3Fltr'>0.0</component>"
"<component name='ISAPI'>0.0</component>"
"<component name='ASP'>0.0</component>"
"<component name='CGI'>0.0</component>"
"</requests>\n",
Duration ? (double)(IIS->CachedResponses) / (double)Duration : 0.0,
IIS->CachedResponses ? (double)(IIS->TotalCachedResponseTime / IIS->CachedResponses) / 10000.0 : 0.0,
IIS->CachedResponses ? 100 : 0);
fprintf(procFile,
"<requests cached='false' type='ASP'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3ISAPIRequests) / (double)Duration : 0.0,
(IIS->W3ISAPIRequests) ? ((double)IIS->TotalISAPIResponseTime / (double)IIS->W3ISAPIRequests) / 10000.0 : 0.0,
((IIS->TotalASPResponseTime) ? ((double)(IIS->TotalASPULOnlyResponseTime) / (double)(IIS->TotalASPResponseTime)) * 100.0 : 0.0),
((IIS->TotalASPResponseTime) ? ((double)(IIS->TotalASPW3OnlyResponseTime) / (double)(IIS->TotalASPResponseTime)) * 100.0 : 0.0),
((IIS->TotalASPResponseTime) ? ((double)(IIS->TotalASPW3FilterResponseTime) / (double)(IIS->TotalASPResponseTime)) * 100.0 : 0.0),
((IIS->TotalASPResponseTime) ? ((double)(IIS->TotalASPISAPIOnlyResponseTime) / (double)(IIS->TotalASPResponseTime)) * 100.0 : 0.0),
((IIS->TotalASPResponseTime) ? ((double)(IIS->TotalASPOnlyResponseTime) / (double)(IIS->TotalASPResponseTime)) * 100.0 : 0.0),
0.0);
fprintf(procFile,
"<requests cached='false' type='Static HTTP'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3FileRequests) / (double)Duration : 0.0,
(IIS->W3FileRequests) ? ((double)IIS->TotalFileResponseTime / (double)IIS->W3FileRequests) / 10000.0 : 0.0,
(IIS->TotalFileResponseTime ? ((double)(IIS->TotalFileULOnlyResponseTime) / (double)(IIS->TotalFileResponseTime)) * 100.0 : 0.0),
(IIS->TotalFileResponseTime ? ((double)(IIS->TotalFileW3OnlyResponseTime) / (double)(IIS->TotalFileResponseTime)) * 100.0 : 0.0),
(IIS->TotalFileResponseTime ? ((double)(IIS->TotalFileW3FilterResponseTime) / (double)(IIS->TotalFileResponseTime)) * 100.0 : 0.0),
0.0,
0.0,
0.0);
fprintf(procFile,
"<requests cached='false' type='CGI'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3CGIRequests) / (double)Duration : 0.0,
(IIS->W3CGIRequests) ? ((double)IIS->TotalCGIResponseTime / (double)IIS->W3CGIRequests) / 10000.0 : 0.0,
((IIS->TotalCGIResponseTime) ? ((double)(IIS->TotalCGIULOnlyResponseTime) / (double)(IIS->TotalCGIResponseTime)) * 100.0 : 0.0),
((IIS->TotalCGIResponseTime) ? ((double)(IIS->TotalCGIW3OnlyResponseTime) / (double)(IIS->TotalCGIResponseTime)) * 100.0 : 0.0),
((IIS->TotalCGIResponseTime) ? ((double)(IIS->TotalCGIW3FilterResponseTime) / (double)(IIS->TotalCGIResponseTime)) * 100.0 : 0.0),
0.0,
0.0,
((IIS->TotalCGIResponseTime) ? ((double)(IIS->TotalCGIOnlyResponseTime) / (double)(IIS->TotalCGIResponseTime)) * 100.0 : 0.0));
if (IIS->W3Error != 0) {
fprintf(procFile,
"<requests cached='false' type='Error'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3Error) / (double)Duration : 0.0,
(IIS->W3Error) ? ((double)IIS->TotalErrorResponseTime / (double)IIS->W3Error) / 10000.0 : 0.0,
((IIS->TotalErrorResponseTime) ? ((double)(IIS->TotalErrorULOnlyResponseTime) / (double)(IIS->TotalErrorResponseTime)) * 100.0 : 0.0),
((IIS->TotalErrorResponseTime) ? ((double)(IIS->TotalErrorW3OnlyResponseTime) / (double)(IIS->TotalErrorResponseTime)) * 100.0 : 0.0),
((IIS->TotalErrorResponseTime) ? ((double)(IIS->TotalErrorW3FilterResponseTime) / (double)(IIS->TotalErrorResponseTime)) * 100.0 : 0.0),
0.0,
0.0,
0.0);
}
fprintf(procFile,
"<summary cached='true'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%3d.0</component>"
"<component name='W3'>0.0</component>"
"<component name='W3Fltr'>0.0</component>"
"<component name='ISAPI'>0.0</component>"
"<component name='ASP'>0.0</component>"
"<component name='CGI'>0.0</component>"
"</summary>\n",
Duration ? (double)(IIS->CachedResponses) / (double)Duration : 0.0,
IIS->CachedResponses ? (double)(IIS->TotalCachedResponseTime / IIS->CachedResponses) / 10000.0 : 0.0,
IIS->CachedResponses ? 100 : 0);
fprintf(procFile,
"<summary cached='false'>"
"<rate percent='%1.3f'>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</summary>\n",
((double)(IIS->TotalRequests - IIS->CachedResponses) / ((double)IIS->TotalRequests)) * 100.0,
Duration ? (double)(IIS->TotalRequests - IIS->CachedResponses) / (double)Duration : 0.0,
(IIS->TotalRequests - IIS->CachedResponses) ? ((double)IIS->TotalNonCachedResponseTime / (double)(IIS->TotalRequests - IIS->CachedResponses)) / 10000.0 : 0.0,
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalFileULOnlyResponseTime + IIS->TotalCGIULOnlyResponseTime + IIS->TotalASPULOnlyResponseTime + IIS->TotalErrorULOnlyResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalFileW3OnlyResponseTime + IIS->TotalCGIW3OnlyResponseTime + IIS->TotalASPW3OnlyResponseTime + IIS->TotalErrorW3OnlyResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalW3FilterResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalASPISAPIOnlyResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalASPOnlyResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0: 0.0),
((IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalCGIOnlyResponseTime) / (double)(IIS->TotalNonCachedResponseTime)) * 100.0: 0.0));
fprintf(procFile,
"<summary type='totals'>"
"<rate>%1.3f</rate>"
"<response_time>%1.3f</response_time>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</summary>\n",
Duration ? (double)(IIS->TotalRequests) / (double)Duration : 0.0,
((double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) / (double)IIS->TotalRequests) / 10000.0,
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalCachedResponseTime + IIS->TotalFileULOnlyResponseTime + IIS->TotalCGIULOnlyResponseTime + IIS->TotalASPULOnlyResponseTime + IIS->TotalErrorULOnlyResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalFileW3OnlyResponseTime + IIS->TotalCGIW3OnlyResponseTime + IIS->TotalASPW3OnlyResponseTime + IIS->TotalErrorW3OnlyResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalW3FilterResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalASPISAPIOnlyResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalASPOnlyResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0),
((IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime) ? ((double)(IIS->TotalCGIOnlyResponseTime) / (double)(IIS->TotalCachedResponseTime + IIS->TotalNonCachedResponseTime)) * 100.0 : 0.0));
fprintf(procFile, "</table>\n");
if (PrintCPUUsage) {
fprintf(procFile, "<table title='Http Requests CPU Time Usage Statistics'>\n");
fprintf(procFile,
"<requests cached='true' type='Static HTTP'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%3d.0</component>"
"<component name='W3'>0.0</component>"
"<component name='W3Fltr'>0.0</component>"
"<component name='ISAPI'>0.0</component>"
"<component name='ASP'>0.0</component>"
"<component name='CGI'>0.0</component>"
"</requests>\n",
Duration ? (double)(IIS->CachedResponses) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->CachedCPUTime) / (double)MilDuration) * 100.0) : 0.0,
IIS->CachedResponses ? 100 : 0);
fprintf(procFile,
"<requests cached='false' type='ASP'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3ISAPIRequests) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->ISAPICPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->ASPCPUTime) ? ((double)(IIS->TotalASPULOnlyCPUTime) / (double)(IIS->ASPCPUTime)) * 100.0 : 0.0),
((IIS->ASPCPUTime) ? ((double)(IIS->TotalASPW3OnlyCPUTime) / (double)(IIS->ASPCPUTime)) * 100.0 : 0.0),
((IIS->ASPCPUTime) ? ((double)(IIS->TotalASPW3FilterCPUTime) / (double)(IIS->ASPCPUTime)) * 100.0 : 0.0),
((IIS->ASPCPUTime) ? ((double)(IIS->TotalASPISAPIOnlyCPUTime) / (double)(IIS->ASPCPUTime)) * 100.0 : 0.0),
((IIS->ASPCPUTime) ? ((double)(IIS->TotalASPOnlyCPUTime) / (double)(IIS->ASPCPUTime)) * 100.0: 0.0),
0.0);
fprintf(procFile,
"<requests cached='false' type='Static HTTP'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3FileRequests) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->FileCPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->FileCPUTime) ? ((double)(IIS->TotalFileULOnlyCPUTime) / (double)(IIS->FileCPUTime)) * 100.0 : 0.0),
((IIS->FileCPUTime) ? ((double)(IIS->TotalFileW3OnlyCPUTime) / (double)(IIS->FileCPUTime)) * 100.0 : 0.0),
((IIS->FileCPUTime) ? ((double)(IIS->TotalFileW3FilterCPUTime) / (double)(IIS->FileCPUTime)) * 100.0 : 0.0),
0.0,
0.0,
0.0);
fprintf(procFile,
"<requests cached='false' type='CGI'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3CGIRequests) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->CGICPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->CGICPUTime) ? ((double)(IIS->TotalCGIULOnlyCPUTime) / (double)(IIS->CGICPUTime)) * 100.0 : 0.0),
((IIS->CGICPUTime) ? ((double)(IIS->TotalCGIW3OnlyCPUTime) / (double)(IIS->CGICPUTime)) * 100.0 : 0.0),
((IIS->CGICPUTime) ? ((double)(IIS->TotalCGIW3FilterCPUTime) / (double)(IIS->CGICPUTime)) * 100.0 : 0.0),
0.0,
0.0,
((IIS->CGICPUTime) ? ((double)(IIS->TotalCGIOnlyCPUTime) / (double)(IIS->CGICPUTime)) * 100.0 : 0.0));
if (IIS->W3Error != 0) {
fprintf(procFile,
"<requests cached='false' type='Error'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</requests>\n",
Duration ? (double)(IIS->W3Error) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->ErrorCPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->ErrorCPUTime) ? ((double)(IIS->TotalErrorULOnlyCPUTime) / (double)(IIS->ErrorCPUTime)) * 100.0 : 0.0),
((IIS->ErrorCPUTime) ? ((double)(IIS->TotalErrorW3OnlyCPUTime) / (double)(IIS->ErrorCPUTime)) * 100.0 : 0.0),
((IIS->ErrorCPUTime) ? ((double)(IIS->TotalErrorW3FilterCPUTime) / (double)(IIS->ErrorCPUTime)) * 100.0 : 0.0),
0.0,
0.0,
0.0);
}
fprintf(procFile,
"<summary cached='true'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%3d.0</component>"
"<component name='W3'>0.0</component>"
"<component name='W3Fltr'>0.0</component>"
"<component name='ISAPI'>0.0</component>"
"<component name='ASP'>0.0</component>"
"<component name='CGI'>0.0</component>"
"</summary>\n",
Duration ? (double)(IIS->CachedResponses) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->CachedCPUTime) / (double)MilDuration) * 100.0) : 0.0,
IIS->CachedResponses ? 100 : 0);
fprintf(procFile,
"<summary cached='false'>"
"<rate percent='%1.1f'>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</summary>\n",
((double)(IIS->TotalRequests - IIS->CachedResponses) / ((double)IIS->TotalRequests)) * 100.0,
Duration ? (double)(IIS->TotalRequests - IIS->CachedResponses) / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->NonCachedCPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalFileULOnlyCPUTime + IIS->TotalCGIULOnlyCPUTime + IIS->TotalASPULOnlyCPUTime + IIS->TotalErrorULOnlyCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0 : 0.0),
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalFileW3OnlyCPUTime + IIS->TotalCGIW3OnlyCPUTime + IIS->TotalASPW3OnlyCPUTime + IIS->TotalErrorW3OnlyCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0 : 0.0),
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalW3FilterCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0 : 0.0),
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalASPISAPIOnlyCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0 : 0.0),
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalASPOnlyCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0: 0.0),
((IIS->NonCachedCPUTime) ? ((double)(IIS->TotalCGIOnlyCPUTime) / (double)(IIS->NonCachedCPUTime)) * 100.0: 0.0));
fprintf(procFile,
"<summary type='totals'>"
"<rate>%1.3f</rate>"
"<cpu>%1.1f</cpu>"
"<component name='UL'>%1.1f</component>"
"<component name='W3'>%1.1f</component>"
"<component name='W3Fltr'>%1.1f</component>"
"<component name='ISAPI'>%1.1f</component>"
"<component name='ASP'>%1.1f</component>"
"<component name='CGI'>%1.1f</component>"
"</summary>\n",
Duration ? (double)IIS->TotalRequests / (double)Duration : 0.0,
MilDuration ? (((double)(IIS->TotalCPUTime) / (double)MilDuration) * 100.0) : 0.0,
((IIS->TotalCPUTime) ? ((double)(IIS->CachedCPUTime + IIS->TotalFileULOnlyCPUTime + IIS->TotalCGIULOnlyCPUTime + IIS->TotalASPULOnlyCPUTime + IIS->TotalErrorULOnlyCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0 : 0.0),
((IIS->TotalCPUTime) ? ((double)(IIS->TotalFileW3OnlyCPUTime + IIS->TotalCGIW3OnlyCPUTime + IIS->TotalASPW3OnlyCPUTime + IIS->TotalErrorW3OnlyCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0 : 0.0),
((IIS->TotalCPUTime) ? ((double)(IIS->TotalW3FilterCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0 : 0.0),
((IIS->TotalCPUTime) ? ((double)(IIS->TotalASPISAPIOnlyCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0 : 0.0),
((IIS->TotalCPUTime) ? ((double)(IIS->TotalASPOnlyCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0: 0.0),
((IIS->TotalCPUTime) ? ((double)(IIS->TotalCGIOnlyCPUTime) / (double)(IIS->TotalCPUTime)) * 100.0: 0.0));
fprintf(procFile, "</table>\n");
}
RtlZeroMemory(TopHitURLs, sizeof(URL_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopHitStaticURLs, sizeof(URL_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopSlowURLs, sizeof(URL_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopConsumingURLs, sizeof(URL_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopBytesURLs, sizeof(URL_RECORD) * DISPLAY_SIZE);
for (k = 0; k < URL_HASH_TABLESIZE; k++) {
pUrl = GetHeadUrlRecord(k);
while (pUrl != NULL) {
ULONGLONG AverageURLResponseTime = 0;
ULONG Found;
URL_RECORD tmpUrl1, tmpUrl2;
if (pUrl->Requests != 0) {
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl2, sizeof(URL_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpUrl1.Requests != 0) {
RtlCopyMemory(&tmpUrl2, &TopHitURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopHitURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlCopyMemory(&tmpUrl1, &tmpUrl2, sizeof(URL_RECORD));
}
else if (!Found) {
if (TopHitURLs[i].Requests == 0) {
RtlCopyMemory(&TopHitURLs[i], pUrl, sizeof(URL_RECORD));
TopHitURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopHitURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlZeroMemory(&TopHitURLs[i], sizeof(URL_RECORD));
}
}
else if (pUrl->Requests > TopHitURLs[i].Requests) {
RtlCopyMemory(&tmpUrl1, &TopHitURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopHitURLs[i], pUrl, sizeof(URL_RECORD));
TopHitURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopHitURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlCopyMemory(&TopHitURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
}
}
}
}
if (tmpUrl1.Requests != 0) {
free(tmpUrl1.URL);
}
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl2, sizeof(URL_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpUrl1.Requests != 0) {
RtlCopyMemory(&tmpUrl2, &TopHitStaticURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopHitStaticURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlCopyMemory(&tmpUrl1, &tmpUrl2, sizeof(URL_RECORD));
}
else if (!Found) {
if ((pUrl->UrlType == EVENT_TRACE_TYPE_W3CORE_FILEREQ || pUrl->Hits > 0) &&
TopHitStaticURLs[i].Requests == 0) {
RtlCopyMemory(&TopHitStaticURLs[i], pUrl, sizeof(URL_RECORD));
TopHitStaticURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopHitStaticURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlZeroMemory(&TopHitStaticURLs[i], sizeof(URL_RECORD));
}
}
else if ((pUrl->UrlType == EVENT_TRACE_TYPE_W3CORE_FILEREQ || pUrl->Hits > 0) &&
pUrl->Requests > TopHitStaticURLs[i].Requests) {
RtlCopyMemory(&tmpUrl1, &TopHitStaticURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopHitStaticURLs[i], pUrl, sizeof(URL_RECORD));
TopHitStaticURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopHitStaticURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlCopyMemory(&TopHitStaticURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
}
}
}
}
if (tmpUrl1.Requests != 0) {
free(tmpUrl1.URL);
}
AverageURLResponseTime = pUrl->TotalResponseTime / pUrl->Requests;
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl2, sizeof(URL_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpUrl1.Requests != 0) {
RtlCopyMemory(&tmpUrl2, &TopSlowURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopSlowURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlCopyMemory(&tmpUrl1, &tmpUrl2, sizeof(URL_RECORD));
}
else if (!Found) {
if (TopSlowURLs[i].Requests == 0) {
RtlCopyMemory(&TopSlowURLs[i], pUrl, sizeof(URL_RECORD));
TopSlowURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopSlowURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlZeroMemory(&TopSlowURLs[i], sizeof(URL_RECORD));
}
}
else if (AverageURLResponseTime > (TopSlowURLs[i].TotalResponseTime / TopSlowURLs[i].Requests)) {
RtlCopyMemory(&tmpUrl1, &TopSlowURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopSlowURLs[i], pUrl, sizeof(URL_RECORD));
TopSlowURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopSlowURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlCopyMemory(&TopSlowURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
}
}
}
}
if (tmpUrl1.Requests != 0) {
free(tmpUrl1.URL);
}
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl2, sizeof(URL_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpUrl1.Requests != 0) {
RtlCopyMemory(&tmpUrl2, &TopConsumingURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopConsumingURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlCopyMemory(&tmpUrl1, &tmpUrl2, sizeof(URL_RECORD));
}
else if (!Found) {
if (TopConsumingURLs[i].Requests == 0) {
RtlCopyMemory(&TopConsumingURLs[i], pUrl, sizeof(URL_RECORD));
TopConsumingURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopConsumingURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlZeroMemory(&TopConsumingURLs[i], sizeof(URL_RECORD));
}
}
else if ((pUrl->KCPUTime + pUrl->UCPUTime) > (TopConsumingURLs[i].KCPUTime + TopConsumingURLs[i].UCPUTime)) {
RtlCopyMemory(&tmpUrl1, &TopConsumingURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopConsumingURLs[i], pUrl, sizeof(URL_RECORD));
TopConsumingURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopConsumingURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlCopyMemory(&TopConsumingURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
}
}
}
}
if (tmpUrl1.Requests != 0) {
free(tmpUrl1.URL);
}
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl2, sizeof(URL_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpUrl1.Requests != 0) {
RtlCopyMemory(&tmpUrl2, &TopBytesURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopBytesURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlCopyMemory(&tmpUrl1, &tmpUrl2, sizeof(URL_RECORD));
}
else if (!Found) {
if (TopBytesURLs[i].Requests == 0) {
RtlCopyMemory(&TopBytesURLs[i], pUrl, sizeof(URL_RECORD));
TopBytesURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopBytesURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlZeroMemory(&TopBytesURLs[i], sizeof(URL_RECORD));
}
}
else if (pUrl->BytesSent > TopBytesURLs[i].BytesSent) {
RtlCopyMemory(&tmpUrl1, &TopBytesURLs[i], sizeof(URL_RECORD));
RtlCopyMemory(&TopBytesURLs[i], pUrl, sizeof(URL_RECORD));
TopBytesURLs[i].URL = CopyUrlStr(pUrl->URL);
if (TopBytesURLs[i].URL != NULL) {
Found = TRUE;
}
else {
RtlCopyMemory(&TopBytesURLs[i], &tmpUrl1, sizeof(URL_RECORD));
RtlZeroMemory(&tmpUrl1, sizeof(URL_RECORD));
}
}
}
}
if (tmpUrl1.Requests != 0) {
free(tmpUrl1.URL);
}
}
if (pUrl->URL) {
free(pUrl->URL);
}
free(pUrl);
pUrl = GetHeadUrlRecord(k);
}
}
fprintf( procFile, "<table title='Most Requested URLs' top='%d'>", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopHitURLs[i].Requests != 0; i++) {
ULONGLONG AverageURLResponseTime = 0;
CHAR ReducedUrl[URL_NAME_COLUMN_SIZE + 1];
char* strPrint;
if (TopHitURLs[i].Requests != 0) {
AverageURLResponseTime = TopHitURLs[i].TotalResponseTime / TopHitURLs[i].Requests;
}
RemoveCtrlCharA(TopHitURLs[i].URL, strlen(TopHitURLs[i].URL));
if (strlen(TopHitURLs[i].URL) > URL_NAME_COLUMN_SIZE) {
ReduceStringA(ReducedUrl, URL_NAME_COLUMN_SIZE + 1, TopHitURLs[i].URL);
strPrint = ReducedUrl;
}else{
strPrint = TopHitURLs[i].URL;
}
fprintf(procFile,
"<url name='%s'>"
"<site_id>%d</site_id>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<response_time>%1.1f</response_time>"
"</url>\n",
strPrint,
TopHitURLs[i].SiteId,
(Duration ? ((double)TopHitURLs[i].Requests / (double)Duration) : 0.0),
TopHitURLs[i].Requests ? ((double)TopHitURLs[i].Hits / (double)TopHitURLs[i].Requests * 100.0) : 0.0,
((double)AverageURLResponseTime) / 10000.0);
}
fprintf( procFile, "</table>\n" );
fprintf(procFile, "<table title='Most Requested Static URLs' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopHitStaticURLs[i].Requests != 0; i++) {
char* strPrint;
ULONGLONG AverageURLResponseTime = 0;
CHAR ReducedUrl[URL_NAME_COLUMN_SIZE + 1];
if (TopHitStaticURLs[i].Requests != 0) {
AverageURLResponseTime = TopHitStaticURLs[i].TotalResponseTime / TopHitStaticURLs[i].Requests;
}
RemoveCtrlCharA(TopHitStaticURLs[i].URL, strlen(TopHitStaticURLs[i].URL));
if (strlen(TopHitStaticURLs[i].URL) > URL_NAME_COLUMN_SIZE) {
ReduceStringA(ReducedUrl, URL_NAME_COLUMN_SIZE + 1, TopHitStaticURLs[i].URL);
strPrint = ReducedUrl;
}else{
strPrint = TopHitStaticURLs[i].URL;
}
fprintf(procFile,
"<url name='%s'>"
"<site_id>%d</site_id>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<response_time>%1.1f</response_time>"
"</url>\n",
strPrint,
TopHitStaticURLs[i].SiteId,
(Duration ? ((double)TopHitStaticURLs[i].Requests / (double)Duration) : 0.0),
TopHitStaticURLs[i].Requests ? ((double)TopHitStaticURLs[i].Hits / (double)TopHitStaticURLs[i].Requests * 100.0) : 0.0,
((double)AverageURLResponseTime) / 10000.0);
}
fprintf(procFile, "</table>\n");
fprintf(procFile, "<table title='Slowest URLs' top='%d'>\n", DISPLAY_SIZE );
for (i = 0; i < DISPLAY_SIZE && TopSlowURLs[i].Requests != 0; i++) {
char* strPrint;
ULONGLONG AverageURLResponseTime = 0;
CHAR ReducedUrl[URL_NAME_COLUMN_SIZE + 1];
if (TopSlowURLs[i].Requests != 0) {
AverageURLResponseTime = TopSlowURLs[i].TotalResponseTime / TopSlowURLs[i].Requests;
}
RemoveCtrlCharA(TopSlowURLs[i].URL, strlen(TopSlowURLs[i].URL));
if (strlen(TopSlowURLs[i].URL) > URL_NAME_COLUMN_SIZE) {
ReduceStringA(ReducedUrl, URL_NAME_COLUMN_SIZE + 1, TopSlowURLs[i].URL);
strPrint = ReducedUrl;
}else{
strPrint = TopSlowURLs[i].URL;
}
fprintf(procFile,
"<url name='%s'>"
"<site_id>%d</site_id>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<response_time>%1.1f</response_time>"
"</url>\n",
strPrint,
TopSlowURLs[i].SiteId,
(Duration ? ((double)TopSlowURLs[i].Requests / (double)Duration) : 0.0),
TopSlowURLs[i].Requests ? ((double)TopSlowURLs[i].Hits / (double)TopSlowURLs[i].Requests * 100.0) : 0.0,
((double)AverageURLResponseTime) / 10000.0);
}
fprintf(procFile, "</table>\n" );
if (PrintCPUUsage) {
fprintf(procFile, "<table title='URLs with the Most CPU Usage' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopConsumingURLs[i].Requests != 0; i++) {
char* strPrint;
CHAR ReducedUrl[URL_NAME_COLUMN_SIZE + 1];
RemoveCtrlCharA(TopConsumingURLs[i].URL, strlen(TopConsumingURLs[i].URL));
if (strlen(TopConsumingURLs[i].URL) > URL_NAME_COLUMN_SIZE) {
ReduceStringA(ReducedUrl, URL_NAME_COLUMN_SIZE + 1, TopConsumingURLs[i].URL);
strPrint = ReducedUrl;
}else{
strPrint = TopConsumingURLs[i].URL;
}
fprintf(procFile,
"<url name='%s'>"
"<site_id>%d</site_id>"
"<rate>%1.3f</rate>"
"<cpu>%1.2f</cpu>"
"</url>\n",
strPrint,
TopConsumingURLs[i].SiteId,
(Duration ? ((double)TopConsumingURLs[i].Requests / (double)Duration) : 0.0),
MilDuration ? ((((double)TopConsumingURLs[i].KCPUTime + (double)TopConsumingURLs[i].UCPUTime) / (double)MilDuration) * 100.0) : 0.0);
}
fprintf(procFile, "</table>\n" );
}
fprintf(procFile, "<table title='URLs with the Most Bytes Sent' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopBytesURLs[i].Requests != 0; i++) {
char* strPrint;
CHAR ReducedUrl[URL_NAME_COLUMN_SIZE + 1];
RemoveCtrlCharA(TopBytesURLs[i].URL, strlen(TopBytesURLs[i].URL));
if (strlen(TopBytesURLs[i].URL) > URL_NAME_COLUMN_SIZE) {
ReduceStringA(ReducedUrl, URL_NAME_COLUMN_SIZE + 1, TopBytesURLs[i].URL);
strPrint = ReducedUrl;
}else{
strPrint = TopBytesURLs[i].URL;
}
fprintf(procFile,
"<url name='%s'>"
"<site_id>%d</site_id>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<bytes_sent_per_sec>%d</bytes_sent_per_sec>"
"</url>\n",
strPrint,
TopBytesURLs[i].SiteId,
(Duration ? ((double)TopBytesURLs[i].Requests / (double)Duration) : 0.0),
TopBytesURLs[i].Requests ? ((double)TopBytesURLs[i].Hits / (double)TopBytesURLs[i].Requests * 100.0) : 0.0,
Duration ? TopBytesURLs[i].BytesSent/ Duration : 0);
}
fprintf(procFile, "</table>\n" );
for (i = 0; i < DISPLAY_SIZE; i++) {
if (TopHitURLs[i].Requests != 0 && TopHitURLs[i].URL != NULL) {
free(TopHitURLs[i].URL);
}
if (TopHitStaticURLs[i].Requests != 0 && TopHitStaticURLs[i].URL != NULL) {
free(TopHitStaticURLs[i].URL);
}
if (TopSlowURLs[i].Requests != 0 && TopSlowURLs[i].URL != NULL) {
free(TopSlowURLs[i].URL);
}
if (TopConsumingURLs[i].Requests != 0 && TopConsumingURLs[i].URL != NULL) {
free(TopConsumingURLs[i].URL);
}
if (TopBytesURLs[i].Requests != 0 && TopBytesURLs[i].URL != NULL) {
free(TopBytesURLs[i].URL);
}
}
RtlZeroMemory(TopHitClients, sizeof(CLIENT_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopSlowClients, sizeof(CLIENT_RECORD) * DISPLAY_SIZE);
pClient = GetHeadClientRecord();
while (pClient != NULL) {
ULONGLONG AverageClientResponseTime = 0;
ULONG Found;
CLIENT_RECORD tmpClient1, tmpClient2;
if (pClient->Requests != 0) {
RtlZeroMemory(&tmpClient1, sizeof(CLIENT_RECORD));
RtlZeroMemory(&tmpClient2, sizeof(CLIENT_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpClient1.Requests != 0) {
RtlCopyMemory(&tmpClient2, &TopHitClients[i], sizeof(CLIENT_RECORD));
RtlCopyMemory(&TopHitClients[i], &tmpClient1, sizeof(CLIENT_RECORD));
RtlCopyMemory(&tmpClient1, &tmpClient2, sizeof(CLIENT_RECORD));
}
else if (!Found) {
if (TopHitClients[i].Requests == 0) {
RtlCopyMemory(&TopHitClients[i], pClient, sizeof(CLIENT_RECORD));
Found = TRUE;
}
else if (pClient->Requests > TopHitClients[i].Requests) {
RtlCopyMemory(&tmpClient1, &TopHitClients[i], sizeof(CLIENT_RECORD));
RtlCopyMemory(&TopHitClients[i], pClient, sizeof(CLIENT_RECORD));
Found = TRUE;
}
}
}
AverageClientResponseTime = pClient->TotalResponseTime / pClient->Requests;
RtlZeroMemory(&tmpClient1, sizeof(CLIENT_RECORD));
RtlZeroMemory(&tmpClient2, sizeof(CLIENT_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpClient1.Requests != 0) {
RtlCopyMemory(&tmpClient2, &TopSlowClients[i], sizeof(CLIENT_RECORD));
RtlCopyMemory(&TopSlowClients[i], &tmpClient1, sizeof(CLIENT_RECORD));
RtlCopyMemory(&tmpClient1, &tmpClient2, sizeof(CLIENT_RECORD));
}
else if (!Found) {
if (TopSlowClients[i].Requests == 0) {
RtlCopyMemory(&TopSlowClients[i], pClient, sizeof(CLIENT_RECORD));
Found = TRUE;
}
else if (AverageClientResponseTime > (TopSlowClients[i].TotalResponseTime / TopSlowClients[i].Requests)) {
RtlCopyMemory(&tmpClient1, &TopSlowClients[i], sizeof(CLIENT_RECORD));
RtlCopyMemory(&TopSlowClients[i], pClient, sizeof(CLIENT_RECORD));
Found = TRUE;
}
}
}
}
free(pClient);
pClient = GetHeadClientRecord();
}
fprintf(procFile, "<table title='Clients with the Most Requests' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopHitClients[i].Requests != 0; i++) {
ULONGLONG AverageClientResponseTime = 0;
CHAR ipAddrBuffer[MAX_ADDRESS_LENGTH];
PCHAR pszA = &ipAddrBuffer[0];
if (TopHitClients[i].Requests != 0) {
AverageClientResponseTime = TopHitClients[i].TotalResponseTime / TopHitClients[i].Requests;
}
DecodeIpAddressA(TopHitClients[i].IpAddrType,
&TopHitClients[i].IpAddrV4,
&TopHitClients[i].IpAddrV6[0],
pszA);
fprintf(procFile,
"<client ip='%s'>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<response_time>%I64u</response_time>"
"</client>\n",
ipAddrBuffer,
(Duration ? ((double)TopHitClients[i].Requests / (double)Duration) : 0.0),
TopHitClients[i].Requests ? ((double)TopHitClients[i].Hits / (double)TopHitClients[i].Requests * 100.0) : 0.0,
AverageClientResponseTime / 10000);
}
fprintf(procFile, "</table>\n" );
fprintf(procFile, "<table title='Clients with the Slowest Responses' top='%d'>", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopSlowClients[i].Requests != 0; i++) {
ULONGLONG AverageClientResponseTime = 0;
CHAR ipAddrBuffer[MAX_ADDRESS_LENGTH];
PCHAR pszA = &ipAddrBuffer[0];
if (TopSlowClients[i].Requests != 0) {
AverageClientResponseTime = TopSlowClients[i].TotalResponseTime / TopSlowClients[i].Requests;
}
DecodeIpAddressA(TopHitClients[i].IpAddrType,
&TopHitClients[i].IpAddrV4,
&TopHitClients[i].IpAddrV6[0],
pszA);
fprintf(procFile,
"<client ip='%s'>"
"<rate>%1.3f</rate>"
"<cache_hit>%1.1f</cache_hit>"
"<response_time>%I64u</response_time>"
"</client>\n",
ipAddrBuffer,
(Duration ? ((double)TopSlowClients[i].Requests / (double)Duration) : 0.0),
TopSlowClients[i].Requests ? ((double)TopSlowClients[i].Hits / (double)TopSlowClients[i].Requests * 100.0) : 0.0,
AverageClientResponseTime / 10000);
}
fprintf(procFile, "</table>" );
RtlZeroMemory(TopHitSites, sizeof(SITE_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopSlowSites, sizeof(SITE_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopConsumingSites, sizeof(SITE_RECORD) * DISPLAY_SIZE);
RtlZeroMemory(TopBytesSites, sizeof(SITE_RECORD) * DISPLAY_SIZE);
pSite = GetHeadSiteRecord();
while (pSite != NULL) {
ULONGLONG AverageSiteResponseTime = 0;
ULONG Found;
SITE_RECORD tmpSite1, tmpSite2;
if (pSite->Requests != 0) {
RtlZeroMemory(&tmpSite1, sizeof(SITE_RECORD));
RtlZeroMemory(&tmpSite2, sizeof(SITE_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpSite1.Requests != 0) {
RtlCopyMemory(&tmpSite2, &TopHitSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopHitSites[i], &tmpSite1, sizeof(SITE_RECORD));
RtlCopyMemory(&tmpSite1, &tmpSite2, sizeof(SITE_RECORD));
}
else if (!Found) {
if (TopHitSites[i].Requests == 0) {
RtlCopyMemory(&TopHitSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
else if (pSite->Requests > TopHitSites[i].Requests) {
RtlCopyMemory(&tmpSite1, &TopHitSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopHitSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
}
}
AverageSiteResponseTime = pSite->TotalResponseTime / pSite->Requests;
RtlZeroMemory(&tmpSite1, sizeof(SITE_RECORD));
RtlZeroMemory(&tmpSite2, sizeof(SITE_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpSite1.Requests != 0) {
RtlCopyMemory(&tmpSite2, &TopSlowSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopSlowSites[i], &tmpSite1, sizeof(SITE_RECORD));
RtlCopyMemory(&tmpSite1, &tmpSite2, sizeof(SITE_RECORD));
}
else if (!Found) {
if (TopSlowSites[i].Requests == 0) {
RtlCopyMemory(&TopSlowSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
else if (AverageSiteResponseTime > (TopSlowSites[i].TotalResponseTime / TopSlowSites[i].Requests)) {
RtlCopyMemory(&tmpSite1, &TopSlowSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopSlowSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
}
}
RtlZeroMemory(&tmpSite1, sizeof(SITE_RECORD));
RtlZeroMemory(&tmpSite2, sizeof(SITE_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpSite1.Requests != 0) {
RtlCopyMemory(&tmpSite2, &TopConsumingSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopConsumingSites[i], &tmpSite1, sizeof(SITE_RECORD));
RtlCopyMemory(&tmpSite1, &tmpSite2, sizeof(SITE_RECORD));
}
else if (!Found) {
if (TopConsumingSites[i].Requests == 0) {
RtlCopyMemory(&TopConsumingSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
else if ((pSite->KCPUTime + pSite->UCPUTime) > (TopConsumingSites[i].KCPUTime + TopConsumingSites[i].UCPUTime)) {
RtlCopyMemory(&tmpSite1, &TopConsumingSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopConsumingSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
}
}
RtlZeroMemory(&tmpSite1, sizeof(SITE_RECORD));
RtlZeroMemory(&tmpSite2, sizeof(SITE_RECORD));
Found = FALSE;
for (i = 0; i < DISPLAY_SIZE; i++) {
if (Found && tmpSite1.Requests != 0) {
RtlCopyMemory(&tmpSite2, &TopBytesSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopBytesSites[i], &tmpSite1, sizeof(SITE_RECORD));
RtlCopyMemory(&tmpSite1, &tmpSite2, sizeof(SITE_RECORD));
}
else if (!Found) {
if (TopBytesSites[i].Requests == 0) {
RtlCopyMemory(&TopBytesSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
else if (pSite->BytesSent > TopBytesSites[i].BytesSent) {
RtlCopyMemory(&tmpSite1, &TopBytesSites[i], sizeof(SITE_RECORD));
RtlCopyMemory(&TopBytesSites[i], pSite, sizeof(SITE_RECORD));
Found = TRUE;
}
}
}
}
free(pSite);
pSite = GetHeadSiteRecord();
}
fprintf(procFile, "<table title='Sites with the Most Requests' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopHitSites[i].Requests != 0; i++) {
ULONGLONG AverageSiteResponseTime = 0;
if (TopHitSites[i].Requests != 0) {
AverageSiteResponseTime = TopHitSites[i].TotalResponseTime / TopHitSites[i].Requests;
}
fprintf(procFile,
"<site id='%d'>"
"<rate>%1.3f</rate>"
"<response_time>%I64u</response_time>"
"<cache_hits>%1.1f</cache_hits>"
"<static>%1.1f</static>"
"<cgi>%1.1f</cgi>"
"<asp>%1.1f</asp>"
"</site>\n",
TopHitSites[i].SiteId,
(Duration ? ((double)TopHitSites[i].Requests / (double)Duration) : 0.0),
AverageSiteResponseTime / 10000,
TopHitSites[i].Requests ? ((double)TopHitSites[i].Hits / (double)TopHitSites[i].Requests * 100.0) : 0.0,
TopHitSites[i].Requests ? ((double)TopHitSites[i].FileRequests / (double)TopHitSites[i].Requests * 100.0) : 0.0,
TopHitSites[i].Requests ? ((double)TopHitSites[i].CGIRequests / (double)TopHitSites[i].Requests * 100.0) : 0.0,
TopHitSites[i].Requests ? ((double)TopHitSites[i].ISAPIRequests / (double)TopHitSites[i].Requests * 100.0) : 0.0);
}
fprintf(procFile, "</table>\n" );
fprintf(procFile, "<table title='Sites with the Slowest Responses' top='%d'>\n", DISPLAY_SIZE);
for (i = 0; i < DISPLAY_SIZE && TopSlowSites[i].Requests != 0; i++) {
ULONGLONG AverageSiteResponseTime = 0;
if (TopSlowSites[i].Requests != 0) {
AverageSiteResponseTime = TopSlowSites[i].TotalResponseTime / TopSlowSites[i].Requests;
}
fprintf(procFile,
"<site id='%d'>"
"<rate>%1.3f</rate>"
"<response_time>%I64u</response_time>"
"<cache_hits>%1.1f</cache_hits>"
"<static>%1.1f</static>"
"<cgi>%1.1f</cgi>"
"<asp>%1.1f</asp>"
"</site>\n",
TopSlowSites[i].SiteId,
(Duration ? ((double)TopSlowSites[i].Requests / (double)Duration) : 0.0),
AverageSiteResponseTime / 10000,
TopSlowSites[i].Requests ? ((double)TopSlowSites[i].Hits / (double)TopSlowSites[i].Requests * 100.0) : 0.0,
TopSlowSites[i].Requests ? ((double)TopSlowSites[i].FileRequests / (double)TopSlowSites[i].Requests * 100.0) : 0.0,
TopSlowSites[i].Requests ? ((double)TopSlowSites[i].CGIRequests / (double)TopSlowSites[i].Requests * 100.0) : 0.0,
TopSlowSites[i].Requests ? ((double)TopSlowSites[i].ISAPIRequests / (double)TopSlowSites[i].Requests * 100.0) : 0.0);
}
fprintf(procFile, "</table>\n" );
if (PrintCPUUsage) {
fprintf(procFile, "<table title='Sites with the Most CPU Time Usage' top='%d'>\n", DISPLAY_SIZE );
for (i = 0; i < DISPLAY_SIZE && TopConsumingSites[i].Requests != 0; i++) {
ULONGLONG AverageSiteResponseTime = 0;
if (TopConsumingSites[i].Requests != 0) {
AverageSiteResponseTime = TopConsumingSites[i].TotalResponseTime / TopConsumingSites[i].Requests;
}
fprintf(procFile,
"<site id='%d'>"
"<rate>%1.3f</rate>"
"<cache_hits>%1.1f</cache_hits>"
"<response_time>%I64u</response_time>"
"<cpu_time units='ms'>%d</cpu_time>"
"<cpu>%1.1f</cpu>"
"</site>\n",
TopConsumingSites[i].SiteId,
(Duration ? ((double)TopConsumingSites[i].Requests / (double)Duration) : 0.0),
TopConsumingSites[i].Requests ? ((double)TopConsumingSites[i].Hits / (double)TopConsumingSites[i].Requests * 100.0) : 0.0,
AverageSiteResponseTime / 10000,
TopConsumingSites[i].Requests ? (TopConsumingSites[i].KCPUTime + TopConsumingSites[i].UCPUTime) / TopConsumingSites[i].Requests : 0,
MilDuration ? ((((double)TopConsumingSites[i].KCPUTime + (double)TopConsumingSites[i].UCPUTime) / (double)MilDuration) * 100.0) : 0.0);
}
fprintf(procFile, "</table>\n" );
}
fprintf(procFile, "<table title='Sites with the Most Bytes Sent' top='%d'>\n", DISPLAY_SIZE );
for (i = 0; i < DISPLAY_SIZE && TopBytesSites[i].Requests != 0; i++) {
fprintf(procFile,
"<site id='%d'>"
"<rate>%1.3f</rate>"
"<bytes>%d</bytes>"
"<cache_hits>%1.1f</cache_hits>"
"<static>%1.1f</static>"
"<cgi>%1.1f</cgi>"
"<asp>%1.1f</asp>"
"</site>\n",
TopBytesSites[i].SiteId,
(Duration ? ((double)TopBytesSites[i].Requests / (double)Duration) : 0.0),
Duration ? TopBytesSites[i].BytesSent / Duration : 0,
TopBytesSites[i].Requests ? ((double)TopBytesSites[i].Hits / (double)TopBytesSites[i].Requests * 100.0) : 0.0,
TopBytesSites[i].Requests ? ((double)TopBytesSites[i].FileRequests / (double)TopBytesSites[i].Requests * 100.0) : 0.0,
TopBytesSites[i].Requests ? ((double)TopBytesSites[i].CGIRequests / (double)TopBytesSites[i].Requests * 100.0) : 0.0,
TopBytesSites[i].Requests ? ((double)TopBytesSites[i].ISAPIRequests / (double)TopBytesSites[i].Requests * 100.0) : 0.0);
}
fprintf(procFile, "</table>\n" );
free(IIS);
}
void
WriteSummary()
{
FILE* SummaryFile;
PLIST_ENTRY Head, Next;
PMOF_INFO pMofInfo;
ULONG i;
WCHAR buffer[MAXSTR];
FILETIME StTm, StlTm;
LARGE_INTEGER LargeTmp;
SYSTEMTIME tmf, emf;
BOOL bResult;
if( NULL == TraceContext->SummaryFileName ){
return;
}
if( 0 == TotalEventCount ){
return;
}
SummaryFile = _wfopen( TraceContext->SummaryFileName, L"w" );
if (SummaryFile == NULL){
return;
}
fwprintf(SummaryFile,L"Files Processed:\n");
for (i=0; i<TraceContext->LogFileCount; i++) {
fwprintf(SummaryFile,L"\t%s\n",TraceContext->LogFileName[i]);
}
LargeTmp.QuadPart = CurrentSystem.StartTime;
StTm.dwHighDateTime = LargeTmp.HighPart;
StTm.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&StTm, &StlTm);
bResult = FileTimeToSystemTime (
&StlTm,
&tmf
);
if( ! bResult || tmf.wMonth > 12 ){
ZeroMemory( &tmf, sizeof(SYSTEMTIME) );
buffer[0] = '\0';
}else{
GetDateFormatW( LOCALE_USER_DEFAULT, DATE_LONGDATE, &tmf, NULL, buffer, MAXSTR );
}
fwprintf(SummaryFile,
L"Total Buffers Processed %d\n"
L"Total Events Processed %d\n"
L"Total Events Lost %d\n"
L"Start Time %ws\n",
TotalBuffersRead,
TotalEventCount,
TotalEventsLost,
buffer );
LargeTmp.QuadPart = CurrentSystem.EndTime;
StTm.dwHighDateTime = LargeTmp.HighPart;
StTm.dwLowDateTime = LargeTmp.LowPart;
FileTimeToLocalFileTime(&StTm, &StlTm);
bResult = FileTimeToSystemTime (
&StlTm,
&emf
);
if( ! bResult || tmf.wMonth > 12 ){
ZeroMemory( &tmf, sizeof(SYSTEMTIME) );
buffer[0] = '\0';
}else{
GetDateFormatW( LOCALE_USER_DEFAULT, DATE_LONGDATE, &emf, NULL, buffer, MAXSTR );
}
ElapseTime = CurrentSystem.EndTime - CurrentSystem.StartTime;
fwprintf(SummaryFile,
L"End Time %ws\n"
L"Elapsed Time %I64d sec\n",
buffer,
(ElapseTime / 10000000) );
fwprintf(SummaryFile,
L"+-------------------------------------------------------------------------------------+\n"
L"|%10s %-20s %-10s %-38s|\n"
L"+-------------------------------------------------------------------------------------+\n",
L"Event Count",
L"Event Name",
L"Event Type",
L"Guid"
);
Head = &CurrentSystem.EventListHead;
Next = Head->Flink;
while (Head != Next) {
WCHAR wstr[MAXSTR];
PWCHAR str;
WCHAR s[MAX_GUID_STRING_SIZE];
PLIST_ENTRY vHead, vNext;
PMOF_VERSION pMofVersion;
RtlZeroMemory(&wstr[0], MAXSTR*sizeof(WCHAR));
pMofInfo = CONTAINING_RECORD(Next, MOF_INFO, Entry);
Next = Next->Flink;
if (pMofInfo->EventCount > 0) {
str = CpdiGuidToString(&s[0], MAX_GUID_STRING_SIZE, (LPGUID)&pMofInfo->Guid);
if( pMofInfo->strDescription != NULL ){
StringCchCopyW( wstr, MAXSTR, pMofInfo->strDescription );
}
//
// Get event count by type from MOF_VERSION structure
//
vHead = &pMofInfo->VersionHeader;
vNext = vHead->Flink;
while (vHead != vNext) {
pMofVersion = CONTAINING_RECORD(vNext, MOF_VERSION, Entry);
vNext = vNext->Flink;
if (pMofVersion->EventCountByType != 0) {
fwprintf(SummaryFile,L"| %10d %-20s %-10s %38s|\n",
pMofVersion->EventCountByType,
wstr,
pMofVersion->strType ? pMofVersion->strType : GUID_TYPE_DEFAULT,
str);
}
}
}
}
fwprintf(SummaryFile,
L"+-------------------------------------------------------------------------------------+\n"
);
fclose( SummaryFile );
}