Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1383 lines
42 KiB

/*++
Copyright (C) 1996-1999 Microsoft Corporation
Module Name:
perfdata.c
Abstract:
<abstract>
--*/
#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include <winperf.h>
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <tchar.h>
#include <mbctype.h>
#include "pdhitype.h"
#include "pdhidef.h"
#include "perftype.h"
#include "perfdata.h"
#include "pdhmsg.h"
#include "strings.h"
// the following strings are for getting texts from perflib
#define OLD_VERSION 0x010000
#define tohexdigit(x) ((CHAR) (((x) < 10) ? ((x) + L'0') : ((x) + L'a' - 10)))
DWORD
PdhiMakePerfPrimaryLangId(
IN LANGID lID,
OUT LPWSTR szBuffer
)
{
WCHAR LangId;
WCHAR nDigit;
LangId = (WCHAR) PRIMARYLANGID(lID);
nDigit = (WCHAR) (LangId >> 8);
szBuffer[0] = tohexdigit(nDigit);
nDigit = (WCHAR) (LangId & 0XF0) >> 4;
szBuffer[1] = tohexdigit(nDigit);
nDigit = (WCHAR) (LangId & 0xF);
szBuffer[2] = tohexdigit(nDigit);
szBuffer[3] = L'\0';
return ERROR_SUCCESS;
}
BOOL IsMatchingInstance (
PERF_INSTANCE_DEFINITION *pInstanceDef,
DWORD dwCodePage,
LPWSTR szInstanceNameToMatch,
DWORD dwInstanceNameLength
)
// compares pInstanceName to the name in the instance
{
DWORD dwThisInstanceNameLength;
LPWSTR szThisInstanceName = NULL;
WCHAR szBufferForANSINames[MAX_PATH];
ZeroMemory(szBufferForANSINames, sizeof(WCHAR) * MAX_PATH);
if (dwInstanceNameLength == 0) {
// get the length to compare
dwInstanceNameLength = lstrlenW (szInstanceNameToMatch);
}
if (dwCodePage == 0) {
// try to take a shortcut here if it's a unicode string
// compare to the length of the shortest string
// get the pointer to this string
szThisInstanceName = GetInstanceName(pInstanceDef);
// convert instance Name from bytes to chars
dwThisInstanceNameLength = pInstanceDef->NameLength / sizeof(WCHAR);
// see if this length includes the term. null. If so shorten it
if (szThisInstanceName[dwThisInstanceNameLength-1] == 0) {
dwThisInstanceNameLength--;
}
} else {
// go the long way and read/translate/convert the string
dwThisInstanceNameLength =GetInstanceNameStr (pInstanceDef,
szBufferForANSINames,
dwCodePage);
if (dwThisInstanceNameLength > 0) {
szThisInstanceName = &szBufferForANSINames[0];
}
}
// if the lengths are not equal then the names can't be either
if (dwInstanceNameLength != dwThisInstanceNameLength) {
return FALSE;
} else {
if (szThisInstanceName != NULL) {
if (lstrcmpiW(szInstanceNameToMatch, szThisInstanceName) == 0) {
// this is a match
return TRUE;
} else {
// this is not a match
return FALSE;
}
} else {
// this is not a match
return FALSE;
}
}
}
LPWSTR
*BuildNameTable(
LPWSTR szComputerName, // computer to query names from
LANGID LangId, // language ID
PPERF_MACHINE pMachine // update member fields
)
/*++
BuildNameTable
Arguments:
hKeyRegistry
Handle to an open registry (this can be local or remote.) and
is the value returned by RegConnectRegistry or a default key.
lpszLangId
The unicode id of the language to look up. (default is 409)
Return Value:
pointer to an allocated table. (the caller must free it when finished!)
the table is an array of pointers to zero terminated strings. NULL is
returned if an error occured.
--*/
{
LPWSTR *lpCounterId;
LPWSTR lpCounterNames;
LPWSTR lpHelpText;
LPWSTR lpThisName;
LONG lWin32Status = ERROR_SUCCESS;
DWORD dwLastError;
DWORD dwValueType;
DWORD dwArraySize;
DWORD dwBufferSize;
DWORD dwCounterSize = 0;
DWORD dwHelpSize = 0;
DWORD dw009CounterSize = 0;
DWORD dw009HelpSize = 0;
DWORD dwThisCounter;
DWORD dwLastCounter;
DWORD dwSystemVersion;
DWORD dwLastId;
DWORD dwLastHelpId;
HKEY hKeyRegistry = NULL;
HKEY hKeyValue = NULL;
HKEY hKeyNames = NULL;
HKEY hKey009Names = NULL;
LPWSTR lpValueNameString;
LPWSTR lp009ValueNameString;
WCHAR CounterNameBuffer[50];
WCHAR HelpNameBuffer[50];
WCHAR Counter009NameBuffer[50];
WCHAR Help009NameBuffer[50];
WCHAR lpszLangId[16];
BOOL bUse009Locale = FALSE;
BOOL bUsePerfTextKey = TRUE;
lpValueNameString = NULL; //initialize to NULL
lp009ValueNameString = NULL;
pMachine->szPerfStrings = NULL;
pMachine->sz009PerfStrings = NULL;
pMachine->typePerfStrings = NULL;
if (szComputerName == NULL) {
// use local machine
hKeyRegistry = HKEY_LOCAL_MACHINE;
} else {
if ((lWin32Status = RegConnectRegistryW(szComputerName,
HKEY_LOCAL_MACHINE, & hKeyRegistry)) != ERROR_SUCCESS) {
// unable to connect to registry
goto BNT_BAILOUT;
}
}
// check for null arguments and insert defaults if necessary
if ( (LangId == MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US))
|| (PRIMARYLANGID(LangId) == LANG_ENGLISH)) {
bUse009Locale = TRUE;
}
PdhiMakePerfPrimaryLangId(LangId, lpszLangId);
// open registry to get number of items for computing array size
lWin32Status = RegOpenKeyExW (
hKeyRegistry,
cszNamesKey,
RESERVED,
KEY_READ,
&hKeyValue);
if (lWin32Status != ERROR_SUCCESS) {
goto BNT_BAILOUT;
}
// get last update time of registry key
lWin32Status = RegQueryInfoKey (
hKeyValue, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, NULL, & pMachine->LastStringUpdateTime);
// get number of items
dwBufferSize = sizeof (dwLastHelpId);
lWin32Status = RegQueryValueExW (
hKeyValue,
cszLastHelp,
RESERVED,
&dwValueType,
(LPBYTE)&dwLastHelpId,
&dwBufferSize);
if ((lWin32Status != ERROR_SUCCESS) || (dwValueType != REG_DWORD)) {
goto BNT_BAILOUT;
}
// get number of items
dwBufferSize = sizeof (dwLastId);
lWin32Status = RegQueryValueExW (
hKeyValue,
cszLastCounter,
RESERVED,
&dwValueType,
(LPBYTE)&dwLastId,
&dwBufferSize);
if ((lWin32Status != ERROR_SUCCESS) || (dwValueType != REG_DWORD)) {
goto BNT_BAILOUT;
}
if (dwLastId < dwLastHelpId)
dwLastId = dwLastHelpId;
dwArraySize = (dwLastId + 1) * sizeof(LPWSTR);
// get Perflib system version
dwBufferSize = sizeof (dwSystemVersion);
lWin32Status = RegQueryValueExW (
hKeyValue,
cszVersionName,
RESERVED,
&dwValueType,
(LPBYTE)&dwSystemVersion,
&dwBufferSize);
if ((lWin32Status != ERROR_SUCCESS) || (dwValueType != REG_DWORD)) {
dwSystemVersion = OLD_VERSION;
}
if (dwSystemVersion == OLD_VERSION) {
// get names from registry
lpValueNameString = G_ALLOC (
lstrlenW(cszNamesKey) * sizeof (WCHAR) +
lstrlenW(cszBackSlash) * sizeof (WCHAR) +
lstrlenW(lpszLangId) * sizeof (WCHAR) +
sizeof (UNICODE_NULL));
if (!lpValueNameString) goto BNT_BAILOUT;
lstrcpyW (lpValueNameString, cszNamesKey);
lstrcatW (lpValueNameString, cszBackSlash);
lstrcatW (lpValueNameString, lpszLangId);
lWin32Status = RegOpenKeyExW (
hKeyRegistry,
lpValueNameString,
RESERVED,
KEY_READ,
&hKeyNames);
if (! bUse009Locale && lWin32Status == ERROR_SUCCESS) {
lp009ValueNameString = G_ALLOC(sizeof(UNICODE_NULL)
+ lstrlenW(cszNamesKey) * sizeof (WCHAR)
+ lstrlenW(cszBackSlash) * sizeof (WCHAR)
+ lstrlenW(cszDefaultLangId) * sizeof (WCHAR));
if (!lpValueNameString) goto BNT_BAILOUT;
lstrcpyW (lpValueNameString, cszNamesKey);
lstrcatW (lpValueNameString, cszBackSlash);
lstrcatW (lpValueNameString, cszDefaultLangId);
lWin32Status = RegOpenKeyExW(hKeyRegistry,
lp009ValueNameString,
RESERVED,
KEY_READ,
& hKey009Names);
}
} else {
__try {
if (bUse009Locale == FALSE) {
if ((lWin32Status = RegConnectRegistryW(szComputerName,
HKEY_PERFORMANCE_NLSTEXT,
& hKeyNames)) == ERROR_SUCCESS) {
if ((lWin32Status = RegConnectRegistryW(szComputerName,
HKEY_PERFORMANCE_TEXT,
& hKey009Names)) != ERROR_SUCCESS) {
bUsePerfTextKey = FALSE;
RegCloseKey(hKeyNames);
}
}
else {
bUsePerfTextKey = FALSE;
}
}
else {
if ((lWin32Status = RegConnectRegistryW(szComputerName,
HKEY_PERFORMANCE_TEXT,
& hKeyNames)) != ERROR_SUCCESS) {
bUsePerfTextKey = FALSE;
}
else {
hKey009Names = hKeyNames;
}
}
}
__except (EXCEPTION_EXECUTE_HANDLER) {
bUsePerfTextKey = FALSE;
}
}
if (! bUsePerfTextKey) {
lstrcpyW(CounterNameBuffer, cszCounterName);
lstrcatW(CounterNameBuffer, lpszLangId);
lstrcpyW(HelpNameBuffer, cszHelpName);
lstrcatW(HelpNameBuffer, lpszLangId);
lstrcpyW(Counter009NameBuffer, cszCounterName);
lstrcatW(Counter009NameBuffer, cszDefaultLangId);
lstrcpyW(Help009NameBuffer, cszHelpName);
lstrcatW(Help009NameBuffer, cszDefaultLangId);
// cannot open HKEY_PERFORMANCE_TEXT, try the old way
//
if (szComputerName == NULL) {
hKeyNames = HKEY_PERFORMANCE_DATA;
}
else if ((lWin32Status = RegConnectRegistryW(szComputerName,
HKEY_PERFORMANCE_DATA,
& hKeyNames)) != ERROR_SUCCESS) {
goto BNT_BAILOUT;
}
hKey009Names = hKeyNames;
}
else {
lstrcpyW(CounterNameBuffer, cszCounters);
lstrcpyW(HelpNameBuffer, cszHelp);
lstrcpyW(Counter009NameBuffer, cszCounters);
lstrcpyW(Help009NameBuffer, cszHelp);
}
// get size of counter names and add that to the arrays
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwBufferSize = 0;
lWin32Status = RegQueryValueExW(
hKeyNames,
CounterNameBuffer,
RESERVED,
& dwValueType,
NULL,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwCounterSize = dwBufferSize;
if (! bUse009Locale) {
dwBufferSize = 0;
lWin32Status = RegQueryValueExW(hKey009Names,
Counter009NameBuffer,
RESERVED,
& dwValueType,
NULL,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dw009CounterSize = dwBufferSize;
}
else {
dw009CounterSize = dwCounterSize;
}
// get size of counter names and add that to the arrays
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwBufferSize = 0;
lWin32Status = RegQueryValueExW(
hKeyNames,
HelpNameBuffer,
RESERVED,
& dwValueType,
NULL,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwHelpSize = dwBufferSize;
if (! bUse009Locale) {
dwBufferSize = 0;
lWin32Status = RegQueryValueExW(hKey009Names,
Help009NameBuffer,
RESERVED,
& dwValueType,
NULL,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dw009HelpSize = dwBufferSize;
}
else {
dw009HelpSize = dwHelpSize;
}
pMachine->szPerfStrings = G_ALLOC(dwArraySize + dwCounterSize + dwHelpSize);
if (! pMachine->szPerfStrings) goto BNT_BAILOUT;
if (bUse009Locale) {
pMachine->sz009PerfStrings = pMachine->szPerfStrings;
}
else {
pMachine->sz009PerfStrings =
G_ALLOC(dwArraySize + dw009CounterSize + dw009HelpSize);
if (! pMachine->sz009PerfStrings) goto BNT_BAILOUT;
}
pMachine->typePerfStrings = G_ALLOC(dwLastId + 1);
if (! pMachine->typePerfStrings) goto BNT_BAILOUT;
// initialize pointers into buffer
lpCounterId = pMachine->szPerfStrings;
lpCounterNames = (LPWSTR)((LPBYTE)lpCounterId + dwArraySize);
lpHelpText = (LPWSTR)((LPBYTE)lpCounterNames + dwCounterSize);
// read counters into memory
dwBufferSize = dwCounterSize;
lWin32Status = RegQueryValueExW (
hKeyNames,
CounterNameBuffer,
RESERVED,
& dwValueType,
(LPVOID)lpCounterNames,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwBufferSize = dwHelpSize;
lWin32Status = RegQueryValueExW (
hKeyNames,
HelpNameBuffer,
RESERVED,
& dwValueType,
(LPVOID)lpHelpText,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
// load counter array items
dwLastCounter = 0;
for (lpThisName = lpCounterNames;
*lpThisName;
lpThisName += (lstrlenW(lpThisName)+1) ) {
// first string should be an integer (in decimal unicode digits)
dwThisCounter = wcstoul (lpThisName, NULL, 10);
// check for registry corruption. This shouldn't occur under
// normal conditions
assert (dwThisCounter > dwLastCounter);
// point to corresponding counter name
lpThisName += (lstrlenW(lpThisName)+1);
// and load array element;
if ((dwThisCounter > 0) && (dwThisCounter <= dwLastId)) {
lpCounterId[dwThisCounter] = lpThisName;
pMachine->typePerfStrings[dwThisCounter] = STR_COUNTER;
dwLastCounter = dwThisCounter;
}
}
dwLastCounter = 0;
for (lpThisName = lpHelpText;
*lpThisName;
lpThisName += (lstrlenW(lpThisName)+1) ) {
// first string should be an integer (in decimal unicode digits)
dwThisCounter = wcstoul (lpThisName, NULL, 10);
// check for registry corruption. This shouldn't occur under
// normal conditions
assert (dwThisCounter > dwLastCounter);
// point to corresponding counter name
lpThisName += (lstrlenW(lpThisName)+1);
// and load array element;
if ((dwThisCounter > 0) && (dwThisCounter <= dwLastId)) {
lpCounterId[dwThisCounter] = lpThisName;
pMachine->typePerfStrings[dwThisCounter] = STR_HELP;
dwLastCounter = dwThisCounter;
}
}
lpCounterId = pMachine->sz009PerfStrings;
lpCounterNames = (LPWSTR) ((LPBYTE) lpCounterId + dwArraySize);
lpHelpText = (LPWSTR) ((LPBYTE) lpCounterNames + dw009CounterSize);
// read counters into memory
dwBufferSize = dw009CounterSize;
lWin32Status = RegQueryValueExW(hKey009Names,
Counter009NameBuffer,
RESERVED,
& dwValueType,
(LPVOID) lpCounterNames,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
dwBufferSize = dw009HelpSize;
lWin32Status = RegQueryValueExW(hKey009Names,
Help009NameBuffer,
RESERVED,
& dwValueType,
(LPVOID) lpHelpText,
& dwBufferSize);
if (lWin32Status != ERROR_SUCCESS) goto BNT_BAILOUT;
for ( lpThisName = lpCounterNames;
* lpThisName;
lpThisName += (lstrlenW(lpThisName) + 1)) {
dwThisCounter = wcstoul(lpThisName, NULL, 10);
lpThisName += (lstrlenW(lpThisName) + 1);
if ((dwThisCounter > 0) && (dwThisCounter <= dwLastId)) {
lpCounterId[dwThisCounter] = lpThisName;
}
}
for ( lpThisName = lpHelpText;
* lpThisName;
lpThisName += (lstrlenW(lpThisName) + 1) ) {
dwThisCounter = wcstoul (lpThisName, NULL, 10);
lpThisName += (lstrlenW(lpThisName) + 1);
if ((dwThisCounter > 0) && (dwThisCounter <= dwLastId)) {
lpCounterId[dwThisCounter] = lpThisName;
}
}
pMachine->dwLastPerfString = dwLastId;
if (lpValueNameString) G_FREE ((LPVOID) lpValueNameString);
if (lp009ValueNameString) G_FREE ((LPVOID) lp009ValueNameString);
RegCloseKey(hKeyValue);
if (hKey009Names && hKey009Names != hKeyNames) RegCloseKey(hKey009Names);
RegCloseKey(hKeyNames);
if (hKeyRegistry && hKeyRegistry != HKEY_LOCAL_MACHINE)
RegCloseKey(hKeyRegistry);
return pMachine->szPerfStrings;
BNT_BAILOUT:
if (lWin32Status != ERROR_SUCCESS) {
dwLastError = GetLastError();
}
if (lpValueNameString) {
G_FREE ((LPVOID)lpValueNameString);
}
if ( pMachine->sz009PerfStrings
&& pMachine->sz009PerfStrings != pMachine->szPerfStrings) {
G_FREE(pMachine->sz009PerfStrings);
}
if (pMachine->szPerfStrings) {
G_FREE(pMachine->szPerfStrings);
}
if (pMachine->typePerfStrings) {
G_FREE(pMachine->typePerfStrings);
}
if (hKeyValue) RegCloseKey(hKeyValue);
if (hKey009Names && hKey009Names != hKeyNames) RegCloseKey(hKey009Names);
if (hKeyNames) RegCloseKey(hKeyNames);
if (hKeyRegistry) RegCloseKey(hKeyRegistry);
return NULL;
}
#pragma warning ( disable : 4127 )
PERF_OBJECT_TYPE *
GetObjectDefByTitleIndex(
IN PERF_DATA_BLOCK *pDataBlock,
IN DWORD ObjectTypeTitleIndex
)
{
DWORD NumTypeDef;
PERF_OBJECT_TYPE *pObjectDef = NULL;
PERF_OBJECT_TYPE *pReturnObject = NULL;
PERF_OBJECT_TYPE *pEndOfBuffer = NULL;
__try {
pObjectDef = FirstObject(pDataBlock);
pEndOfBuffer = (PPERF_OBJECT_TYPE)
((DWORD_PTR)pDataBlock +
pDataBlock->TotalByteLength);
assert (pObjectDef != NULL);
NumTypeDef = 0;
while (1) {
if ( pObjectDef->ObjectNameTitleIndex == ObjectTypeTitleIndex ) {
pReturnObject = pObjectDef;
break;
} else {
NumTypeDef++;
if (NumTypeDef < pDataBlock->NumObjectTypes) {
pObjectDef = NextObject(pObjectDef);
//make sure next object is legit
if (pObjectDef != NULL) {
if (pObjectDef->TotalByteLength > 0) {
if (pObjectDef >= pEndOfBuffer) {
// looks like we ran off the end of the data buffer
assert (pObjectDef < pEndOfBuffer);
break;
}
} else {
// 0-length object buffer returned
assert (pObjectDef->TotalByteLength > 0);
break;
}
} else {
// and continue
assert (pObjectDef != NULL);
break;
}
} else {
// no more data objects in this data block
break;
}
}
}
} __except (EXCEPTION_EXECUTE_HANDLER) {
pReturnObject = NULL;
}
return pReturnObject;
}
PERF_OBJECT_TYPE *
GetObjectDefByName (
IN PERF_DATA_BLOCK *pDataBlock,
IN DWORD dwLastNameIndex,
IN LPCWSTR *NameArray,
IN LPCWSTR szObjectName
)
{
DWORD NumTypeDef;
PERF_OBJECT_TYPE *pReturnObject = NULL;
PERF_OBJECT_TYPE *pObjectDef = NULL;
PERF_OBJECT_TYPE *pEndOfBuffer = NULL;
__try {
pObjectDef = FirstObject(pDataBlock);
pEndOfBuffer = (PPERF_OBJECT_TYPE)
((DWORD_PTR)pDataBlock +
pDataBlock->TotalByteLength);
assert (pObjectDef != NULL);
NumTypeDef = 0;
while (1) {
if ( pObjectDef->ObjectNameTitleIndex < dwLastNameIndex ) {
// look up name of object & compare
if (lstrcmpiW(NameArray[pObjectDef->ObjectNameTitleIndex],
szObjectName) == 0) {
pReturnObject = pObjectDef;
break;
}
}
NumTypeDef++;
if (NumTypeDef < pDataBlock->NumObjectTypes) {
pObjectDef = NextObject(pObjectDef); // get next
//make sure next object is legit
if (pObjectDef != NULL) {
if (pObjectDef->TotalByteLength > 0) {
if (pObjectDef >= pEndOfBuffer) {
// looks like we ran off the end of the data buffer
assert (pObjectDef < pEndOfBuffer);
break;
}
} else {
// 0-length object buffer returned
assert (pObjectDef->TotalByteLength > 0);
break;
}
} else {
// null pointer
assert (pObjectDef != NULL);
break;
}
} else {
// end of data block
break;
}
}
} __except (EXCEPTION_EXECUTE_HANDLER) {
pReturnObject = NULL;
}
return pReturnObject;
}
#pragma warning ( default : 4127 )
PERF_INSTANCE_DEFINITION *
GetInstance(
IN PERF_OBJECT_TYPE *pObjectDef,
IN LONG InstanceNumber
)
{
PERF_INSTANCE_DEFINITION *pInstanceDef;
LONG NumInstance;
if (!pObjectDef) {
return 0;
}
pInstanceDef = FirstInstance(pObjectDef);
for ( NumInstance = 0;
NumInstance < pObjectDef->NumInstances;
NumInstance++ ) {
if ( InstanceNumber == NumInstance ) {
return pInstanceDef;
}
pInstanceDef = NextInstance(pInstanceDef);
}
return NULL;
}
PERF_INSTANCE_DEFINITION *
GetInstanceByUniqueId(
IN PERF_OBJECT_TYPE *pObjectDef,
IN LONG InstanceUniqueId
)
{
PERF_INSTANCE_DEFINITION *pInstanceDef;
LONG NumInstance;
if (!pObjectDef) {
return 0;
}
pInstanceDef = FirstInstance(pObjectDef);
for ( NumInstance = 0;
NumInstance < pObjectDef->NumInstances;
NumInstance++ ) {
if ( InstanceUniqueId == pInstanceDef->UniqueID ) {
return pInstanceDef;
}
pInstanceDef = NextInstance(pInstanceDef);
}
return NULL;
}
DWORD
GetAnsiInstanceName (PPERF_INSTANCE_DEFINITION pInstance,
LPWSTR lpszInstance,
DWORD dwCodePage)
{
LPSTR szSource;
DWORD dwLength;
szSource = (LPSTR)GetInstanceName(pInstance);
// the locale should be set here
DBG_UNREFERENCED_PARAMETER(dwCodePage);
// pInstance->NameLength == the number of bytes (chars) in the string
dwLength = (DWORD) MultiByteToWideChar(_getmbcp(),
0,
szSource,
lstrlenA(szSource),
(LPWSTR) lpszInstance,
pInstance->NameLength);
lpszInstance[dwLength] = 0; // null terminate string buffer
return dwLength;
}
DWORD
GetUnicodeInstanceName (PPERF_INSTANCE_DEFINITION pInstance,
LPWSTR lpszInstance)
{
LPWSTR wszSource;
DWORD dwLength;
wszSource = GetInstanceName(pInstance) ;
// pInstance->NameLength == length of string in BYTES so adjust to
// number of wide characters here
dwLength = pInstance->NameLength / sizeof(WCHAR);
wcsncpy (lpszInstance,
(LPWSTR)wszSource,
dwLength);
// add null termination if string length does not include the null
if ((dwLength > 0) && (lpszInstance[dwLength-1] != 0)) { // i.e. it's the last character of the string
lpszInstance[dwLength] = 0; // then add a terminating null char to the string
} else {
// assume that the length value includes the terminating NULL
// so adjust value to indicate chars only
dwLength--;
}
return (dwLength); // just incase there's null's in the string
}
DWORD
GetInstanceNameStr (PPERF_INSTANCE_DEFINITION pInstance,
LPWSTR lpszInstance,
DWORD dwCodePage)
{
DWORD dwCharSize;
DWORD dwLength = 0;
if (pInstance != NULL) {
if (lpszInstance != NULL) {
if (dwCodePage > 0) {
dwCharSize = sizeof(CHAR);
dwLength = GetAnsiInstanceName (pInstance, lpszInstance, dwCodePage);
} else { // it's a UNICODE name
dwCharSize = sizeof(WCHAR);
dwLength = GetUnicodeInstanceName (pInstance, lpszInstance);
}
// sanity check here...
// the returned string length (in characters) plus the terminating NULL
// should be the same as the specified length in bytes divided by the
// character size. If not then the codepage and instance data type
// don't line up so test that here
if ((dwLength + 1) != (pInstance->NameLength / dwCharSize)) {
// something isn't quite right so try the "other" type of string type
if (dwCharSize == sizeof(CHAR)) {
// then we tried to read it as an ASCII string and that didn't work
// so try it as a UNICODE (if that doesn't work give up and return
// it any way.
dwLength = GetUnicodeInstanceName (pInstance, lpszInstance);
} else if (dwCharSize == sizeof(WCHAR)) {
// then we tried to read it as a UNICODE string and that didn't work
// so try it as an ASCII string (if that doesn't work give up and return
// it any way.
dwLength = GetAnsiInstanceName (pInstance, lpszInstance, dwCodePage);
}
}
} // else return buffer is null
} else {
// no instance def object is specified so return an empty string
*lpszInstance = 0;
}
return dwLength;
}
PERF_INSTANCE_DEFINITION *
GetInstanceByNameUsingParentTitleIndex(
PERF_DATA_BLOCK *pDataBlock,
PERF_OBJECT_TYPE *pObjectDef,
LPWSTR pInstanceName,
LPWSTR pParentName,
DWORD dwIndex
)
{
BOOL fHaveParent;
PERF_OBJECT_TYPE *pParentObj;
PERF_INSTANCE_DEFINITION *pParentInst,
*pInstanceDef;
LONG NumInstance;
DWORD dwLocalIndex;
DWORD dwInstanceNameLength;
fHaveParent = FALSE;
pInstanceDef = FirstInstance(pObjectDef);
if (pInstanceDef == NULL) return NULL;
dwLocalIndex = dwIndex;
dwInstanceNameLength = lstrlenW(pInstanceName);
for ( NumInstance = 0;
NumInstance < pObjectDef->NumInstances;
NumInstance++ )
{
if (IsMatchingInstance (pInstanceDef,
pObjectDef->CodePage,
pInstanceName,
dwInstanceNameLength)) {
// Instance name matches
if ( pParentName == NULL )
{
// No parent, we're done if this is the right "copy"
if (dwLocalIndex == 0) {
return pInstanceDef;
} else {
--dwLocalIndex;
}
}
else
{
// Must match parent as well
pParentObj = GetObjectDefByTitleIndex(
pDataBlock,
pInstanceDef->ParentObjectTitleIndex);
if (!pParentObj)
{
// can't locate the parent, forget it
break ;
}
// Object type of parent found; now find parent
// instance
pParentInst = GetInstance(pParentObj,
pInstanceDef->ParentObjectInstance);
if (!pParentInst)
{
// can't locate the parent instance, forget it
break ;
}
if (IsMatchingInstance (pParentInst,
pParentObj->CodePage,
pParentName, 0)) {
// Parent Instance Name matches that passed in
if (dwLocalIndex == 0) {
return pInstanceDef;
} else {
--dwLocalIndex;
}
}
}
}
pInstanceDef = NextInstance(pInstanceDef);
if (pInstanceDef == NULL) return NULL;
}
return 0;
}
PERF_INSTANCE_DEFINITION *
GetInstanceByName(
PERF_DATA_BLOCK *pDataBlock,
PERF_OBJECT_TYPE *pObjectDef,
LPWSTR pInstanceName,
LPWSTR pParentName,
DWORD dwIndex
)
{
BOOL fHaveParent;
PERF_OBJECT_TYPE *pParentObj;
PERF_INSTANCE_DEFINITION *pParentInst,
*pInstanceDef;
LONG NumInstance;
DWORD dwLocalIndex;
DWORD dwInstanceNameLength;
fHaveParent = FALSE;
pInstanceDef = FirstInstance(pObjectDef);
if (pInstanceDef == NULL) return NULL;
dwLocalIndex = dwIndex;
assert (pInstanceDef != NULL);
dwInstanceNameLength = lstrlenW(pInstanceName);
for ( NumInstance = 0;
NumInstance < pObjectDef->NumInstances;
NumInstance++ ) {
if (IsMatchingInstance (pInstanceDef,
pObjectDef->CodePage,
pInstanceName,
dwInstanceNameLength)) {
// Instance name matches
if (( !pInstanceDef->ParentObjectTitleIndex ) || (pParentName == NULL)){
// No parent, we're done
if (dwLocalIndex == 0) {
return pInstanceDef;
} else {
--dwLocalIndex;
}
} else {
// Must match parent as well
pParentObj = GetObjectDefByTitleIndex(
pDataBlock,
pInstanceDef->ParentObjectTitleIndex);
// if parent object is not found,
// then exit and return NULL
if (pParentObj == NULL) return NULL;
// Object type of parent found; now find parent
// instance
pParentInst = GetInstance(pParentObj,
pInstanceDef->ParentObjectInstance);
if (pParentInst != NULL) {
if (IsMatchingInstance (pParentInst,
pParentObj->CodePage,
pParentName, 0)) {
// Parent Instance Name matches that passed in
if (dwLocalIndex == 0) {
return pInstanceDef;
} else {
--dwLocalIndex;
}
}
} else {
// continue
}
}
}
pInstanceDef = NextInstance(pInstanceDef);
if (pInstanceDef == NULL) return NULL;
}
return 0;
} // GetInstanceByName
PERF_COUNTER_DEFINITION *
GetCounterDefByName (
IN PERF_OBJECT_TYPE *pObject,
IN DWORD dwLastNameIndex,
IN LPWSTR *NameArray,
IN LPWSTR szCounterName
)
{
DWORD NumTypeDef;
PERF_COUNTER_DEFINITION *pThisCounter;
pThisCounter = FirstCounter(pObject);
// no counter found so bail out
if (pThisCounter == NULL) return NULL;
for ( NumTypeDef = 0;
NumTypeDef < pObject->NumCounters;
NumTypeDef++ ) {
if ((pThisCounter->CounterNameTitleIndex > 0) &&
(pThisCounter->CounterNameTitleIndex < dwLastNameIndex )) {
// look up name of counter & compare
if (lstrcmpiW(NameArray[pThisCounter->CounterNameTitleIndex],
szCounterName) == 0) {
return pThisCounter;
}
}
pThisCounter = NextCounter(pThisCounter); // get next
if (pThisCounter == NULL) return NULL;
}
return NULL;
}
PERF_COUNTER_DEFINITION *
GetCounterDefByTitleIndex(
IN PERF_OBJECT_TYPE *pObjectDef,
IN BOOL bBaseCounterDef,
IN DWORD CounterTitleIndex
)
{
DWORD NumCounters;
PERF_COUNTER_DEFINITION * pCounterDef;
pCounterDef = FirstCounter(pObjectDef);
if (pCounterDef == NULL) return NULL;
for ( NumCounters = 0;
NumCounters < pObjectDef->NumCounters;
NumCounters++ ) {
if ( pCounterDef->CounterNameTitleIndex == CounterTitleIndex ) {
if (bBaseCounterDef) {
// get next definition block
if (++NumCounters < pObjectDef->NumCounters) {
// then it should be in there
pCounterDef = NextCounter(pCounterDef);
if (pCounterDef) {
assert (pCounterDef->CounterType & PERF_COUNTER_BASE);
// make sure this is really a base counter
if (!(pCounterDef->CounterType & PERF_COUNTER_BASE)) {
// it's not and it should be so return NULL
pCounterDef = NULL;
}
}
}
} else {
// found so return it as is
}
return pCounterDef;
}
pCounterDef = NextCounter(pCounterDef);
if (pCounterDef == NULL) return NULL;
}
return NULL;
}
#pragma warning ( disable : 4127 )
LONG
GetSystemPerfData (
IN HKEY hKeySystem,
IN PPERF_DATA_BLOCK *ppPerfData,
IN LPWSTR szObjectList,
IN BOOL bCollectCostlyData
)
{ // GetSystemPerfData
LONG lError = ERROR_SUCCESS;
DWORD Size;
DWORD Type;
PPERF_DATA_BLOCK pCostlyPerfData;
DWORD CostlySize;
LPDWORD pdwSrc, pdwDest, pdwLast;
if (*ppPerfData == NULL) {
*ppPerfData = G_ALLOC (INITIAL_SIZE);
if (*ppPerfData == NULL) return PDH_MEMORY_ALLOCATION_FAILURE;
}
__try {
while (TRUE) {
Size = (DWORD)HeapSize (hPdhHeap, 0, *ppPerfData);
lError = RegQueryValueExW (
hKeySystem,
szObjectList,
RESERVED,
&Type,
(LPBYTE)*ppPerfData,
&Size);
if ((!lError) &&
(Size > 0) &&
((*ppPerfData)->Signature[0] == (WCHAR)'P') &&
((*ppPerfData)->Signature[1] == (WCHAR)'E') &&
((*ppPerfData)->Signature[2] == (WCHAR)'R') &&
((*ppPerfData)->Signature[3] == (WCHAR)'F')) {
if (bCollectCostlyData) {
// collect the costly counters now
// the size available is that not used by the above call
CostlySize = (DWORD)HeapSize (hPdhHeap, 0, *ppPerfData) - Size;
pCostlyPerfData =
(PPERF_DATA_BLOCK)((LPBYTE)(*ppPerfData) + Size);
lError = RegQueryValueExW (
hKeySystem,
cszCostly,
RESERVED,
&Type,
(LPBYTE)pCostlyPerfData,
&CostlySize);
if ((!lError) &&
(CostlySize > 0) &&
(pCostlyPerfData->Signature[0] == (WCHAR)'P') &&
(pCostlyPerfData->Signature[1] == (WCHAR)'E') &&
(pCostlyPerfData->Signature[2] == (WCHAR)'R') &&
(pCostlyPerfData->Signature[3] == (WCHAR)'F')) {
// update the header block
(*ppPerfData)->TotalByteLength +=
pCostlyPerfData->TotalByteLength -
pCostlyPerfData->HeaderLength;
(*ppPerfData)->NumObjectTypes +=
pCostlyPerfData->NumObjectTypes;
// move the costly data to the end of the global data
pdwSrc = (LPDWORD)((LPBYTE)pCostlyPerfData +
pCostlyPerfData->HeaderLength);
pdwDest = (LPDWORD)pCostlyPerfData ;
pdwLast = (LPDWORD)((LPBYTE)pCostlyPerfData +
pCostlyPerfData->TotalByteLength -
pCostlyPerfData->HeaderLength);
while (pdwSrc < pdwLast) {*pdwDest++ = *pdwSrc++;}
lError = ERROR_SUCCESS;
break;
}
} else {
lError = ERROR_SUCCESS;
break;
}
}
if (lError == ERROR_MORE_DATA) {
Size = (DWORD)HeapSize (hPdhHeap, 0, *ppPerfData);
G_FREE (*ppPerfData);
*ppPerfData = NULL;
*ppPerfData = G_ALLOC ((Size + EXTEND_SIZE));
if (*ppPerfData == NULL) {
lError = PDH_MEMORY_ALLOCATION_FAILURE;
break;
}
} else {
break;
}
}
} __except (EXCEPTION_EXECUTE_HANDLER) {
lError = GetExceptionCode();
}
return lError;
} // GetSystemPerfData
#pragma warning ( default : 4127 )
DWORD
GetFullInstanceNameStr (
PERF_DATA_BLOCK *pPerfData,
PERF_OBJECT_TYPE *pObjectDef,
PERF_INSTANCE_DEFINITION *pInstanceDef,
LPWSTR szInstanceName
)
{
WCHAR szInstanceNameString[1024];
WCHAR szParentNameString[1024];
// compile instance name.
// the instance name can either be just
// the instance name itself or it can be
// the concatenation of the parent instance,
// a delimiting char (backslash) followed by
// the instance name
DWORD dwLength = 0;
PERF_OBJECT_TYPE * pParentObjectDef;
PERF_INSTANCE_DEFINITION * pParentInstanceDef;
ZeroMemory(szInstanceNameString, sizeof(WCHAR) * 1024);
ZeroMemory(szParentNameString, sizeof(WCHAR) * 1024);
if (pInstanceDef->UniqueID == PERF_NO_UNIQUE_ID) {
dwLength = GetInstanceNameStr (pInstanceDef,
szInstanceNameString,
pObjectDef->CodePage);
} else {
// make a string out of the unique ID
_ltow (pInstanceDef->UniqueID, szInstanceNameString, 10);
dwLength = lstrlenW (szInstanceNameString);
}
if (dwLength > 0) {
if (pInstanceDef->ParentObjectTitleIndex > 0) {
// then add in parent instance name
pParentObjectDef = GetObjectDefByTitleIndex (
pPerfData,
pInstanceDef->ParentObjectTitleIndex);
if (pParentObjectDef != NULL) {
pParentInstanceDef = GetInstance (
pParentObjectDef,
pInstanceDef->ParentObjectInstance);
assert ((ULONG_PTR)pParentObjectDef != (DWORD)0xFFFFFFFF);
if (pParentInstanceDef != NULL) {
if (pParentInstanceDef->UniqueID == PERF_NO_UNIQUE_ID) {
dwLength += GetInstanceNameStr (pParentInstanceDef,
szParentNameString,
pParentObjectDef->CodePage);
} else {
// make a string out of the unique ID
_ltow (pParentInstanceDef->UniqueID, szParentNameString, 10);
dwLength += lstrlenW (szParentNameString);
}
lstrcatW (szParentNameString, cszSlash);
dwLength += 1;
lstrcatW (szParentNameString, szInstanceNameString);
lstrcpyW (szInstanceName, szParentNameString);
} else {
lstrcpyW (szInstanceName, szInstanceNameString);
}
} else {
lstrcpyW (szInstanceName, szInstanceNameString);
}
} else {
lstrcpyW (szInstanceName, szInstanceNameString);
}
}
return dwLength;
}
#if DBG
#define DEBUG_BUFFER_LENGTH 1024
UCHAR PdhDebugBuffer[DEBUG_BUFFER_LENGTH];
// debug level:
// 5 = memory allocs (if _VALIDATE_PDH_MEM_ALLOCS defined) and all 4's
// 4 = function entry and exits (w/ status codes) and all 3's
// 3 = Not impl
// 2 = Not impl
// 1 = Not impl
// 0 = No messages
ULONG pdhDebugLevel = 0;
VOID
__cdecl
PdhDebugPrint(
ULONG DebugPrintLevel,
char * DebugMessage,
...
)
{
va_list ap;
if ((DebugPrintLevel <= (pdhDebugLevel & 0x0000ffff)) ||
((1 << (DebugPrintLevel + 15)) & pdhDebugLevel)) {
DbgPrint("%d:PDH!", GetCurrentThreadId());
}
else return;
va_start(ap, DebugMessage);
_vsnprintf((PCHAR)PdhDebugBuffer, DEBUG_BUFFER_LENGTH, DebugMessage, ap);
DbgPrint((PCHAR)PdhDebugBuffer);
va_end(ap);
}
#endif // DBG