Windows2003-3790/sdktools/pdh/pdhdll/cutils.c
2020-09-30 16:53:55 +02:00

1873 lines
84 KiB
C

/*++
Copyright (C) 1995-1999 Microsoft Corporation
Module Name:
cutils.c
Abstract:
Counter management utility functions
--*/
#include <windows.h>
#include <math.h>
#include "strsafe.h"
#include <pdh.h>
#include "pdhitype.h"
#include "pdhidef.h"
#include "perftype.h"
#include "perfdata.h"
#include "pdhmsg.h"
#include "strings.h"
BOOL
IsValidCounter(
HCOUNTER hCounter
)
/*++
Routine Description:
examines the counter handle to verify it is a valid counter. For now
the test amounts to:
the Handle is NOT NULL
the memory is accessible (i.e. it doesn't AV)
the signature array is valid
the size field is correct
if any tests fail, the handle is presumed to be invalid
Arguments:
IN HCOUNTER hCounter
the handle of the counter to test
Return Value:
TRUE the handle passes all the tests
FALSE one of the test's failed and the handle is not a valid counter
--*/
{
BOOL bReturn = FALSE; // assume it's not a valid query
PPDHI_COUNTER pCounter;
LONG lStatus = ERROR_SUCCESS;
__try {
if (hCounter != NULL) {
// see if a valid signature
pCounter = (PPDHI_COUNTER) hCounter;
if ((* (DWORD *) & pCounter->signature[0] == SigCounter) &&
(pCounter->dwLength == sizeof (PDHI_COUNTER))) {
bReturn = TRUE;
}
}
}
__except (EXCEPTION_EXECUTE_HANDLER) {
// something failed miserably so we can assume this is invalid
lStatus = GetExceptionCode();
}
return bReturn;
}
BOOL
InitCounter(
PPDHI_COUNTER pCounter
)
/*++
Routine Description:
Initialized the counter data structure by:
Allocating the memory block to contain the counter structure
and all the associated data fields. If this allocation
is successful, then the fields are initialized by
verifying the counter is valid.
Arguments:
IN PPDHI_COUNTER pCounter
pointer of the counter to initialize using the system data
Return Value:
TRUE if the counter was successfully initialized
FALSE if a problem was encountered
In either case, the CStatus field of the structure is updated to
indicate the status of the operation.
--*/
{
PPERF_MACHINE pMachine = NULL;
DWORD dwBufferSize = MEDIUM_BUFFER_SIZE;
DWORD dwOldSize;
BOOL bInstances = FALSE;
LPVOID pLocalCounterPath = NULL;
BOOL bReturn = TRUE;
LONG lOffset;
// reset the last error value
pCounter->ThisValue.CStatus = ERROR_SUCCESS;
SetLastError(ERROR_SUCCESS);
if (pCounter->szFullName != NULL) {
// allocate counter path buffer
if (pCounter->pCounterPath != NULL) {
__try {
G_FREE(pCounter->pCounterPath);
}
__except (EXCEPTION_EXECUTE_HANDLER) {
// no need to do anything
}
pCounter->pCounterPath = NULL;
}
pLocalCounterPath = G_ALLOC(dwBufferSize);
if (pLocalCounterPath == NULL) {
// could not allocate string buffer
pCounter->ThisValue.CStatus = PDH_MEMORY_ALLOCATION_FAILURE;
bReturn = FALSE;
}
else {
dwOldSize = dwBufferSize;
if (ParseFullPathNameW(pCounter->szFullName, & dwBufferSize, pLocalCounterPath, FALSE)) {
// resize to only the space required
if (dwOldSize < dwBufferSize) {
pCounter->pCounterPath = G_REALLOC(pLocalCounterPath, dwBufferSize);
}
else {
pCounter->pCounterPath = pLocalCounterPath;
}
if (pCounter->pCounterPath != NULL) {
if (pLocalCounterPath != pCounter->pCounterPath) { // ???
// the memory block moved so
// correct addresses inside structure
lOffset = (LONG) ((ULONG_PTR) pCounter->pCounterPath - (ULONG_PTR) pLocalCounterPath);
if (lOffset != 0 && pCounter->pCounterPath->szMachineName != NULL) {
pCounter->pCounterPath->szMachineName = (LPWSTR) (
(LPBYTE)pCounter->pCounterPath->szMachineName + lOffset);
}
if (lOffset != 0 && pCounter->pCounterPath->szObjectName != NULL) {
pCounter->pCounterPath->szObjectName = (LPWSTR) (
(LPBYTE)pCounter->pCounterPath->szObjectName + lOffset);
}
if (lOffset != 0 && pCounter->pCounterPath->szInstanceName != NULL) {
pCounter->pCounterPath->szInstanceName = (LPWSTR) (
(LPBYTE)pCounter->pCounterPath->szInstanceName + lOffset);
}
if (lOffset != 0 && pCounter->pCounterPath->szParentName != NULL) {
pCounter->pCounterPath->szParentName = (LPWSTR) (
(LPBYTE)pCounter->pCounterPath->szParentName + lOffset);
}
if (lOffset != 0 && pCounter->pCounterPath->szCounterName != NULL) {
pCounter->pCounterPath->szCounterName = (LPWSTR) (
(LPBYTE)pCounter->pCounterPath->szCounterName + lOffset);
}
}
if (pCounter->pOwner->hLog == NULL) {
// validate realtime counter
// try to connect to machine and get machine pointer
pMachine = GetMachine(pCounter->pCounterPath->szMachineName, 0, PDH_GM_UPDATE_PERFNAME_ONLY);
if (pMachine == NULL) {
// unable to find machine
pCounter->ThisValue.CStatus = PDH_CSTATUS_NO_MACHINE;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
else if (pMachine->szPerfStrings == NULL || pMachine->typePerfStrings == NULL) {
// a machine entry was found, but the machine is not available
pMachine->dwRefCount --;
RELEASE_MUTEX(pMachine->hMutex);
pCounter->ThisValue.CStatus = pMachine->dwStatus;
if (pMachine->dwStatus == PDH_ACCESS_DENIED) {
// then don't add this counter since the machine
// won't let us in
bReturn = FALSE;
}
}
else {
// init raw counter value
ZeroMemory(& pCounter->ThisValue, sizeof(PDH_RAW_COUNTER));
ZeroMemory(& pCounter->LastValue, sizeof(PDH_RAW_COUNTER));
// look up object name
pCounter->plCounterInfo.dwObjectId = GetObjectId(pMachine,
pCounter->pCounterPath->szObjectName,
& bInstances);
if (pCounter->plCounterInfo.dwObjectId == (DWORD) -1) {
// unable to lookup object on this machine
pCounter->plCounterInfo.dwObjectId = (DWORD) -1;
pCounter->ThisValue.CStatus = PDH_CSTATUS_NO_OBJECT;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
else {
// update instanceName look up instances if necessary
if (bInstances) {
if (pCounter->pCounterPath->szInstanceName != NULL) {
if (* pCounter->pCounterPath->szInstanceName != SPLAT_L) {
if (! GetInstanceByNameMatch(pMachine, pCounter)) {
// unable to lookup instance
pCounter->ThisValue.CStatus = PDH_CSTATUS_NO_INSTANCE;
// keep the counter since the instance may return
}
}
else {
// this is a wild card query so don't look
// for any instances yet
pCounter->dwFlags |= PDHIC_MULTI_INSTANCE;
}
}
else {
// the path for this object should include an instance name
// and doesn't so return an error
// this is an unrecoverable error so indicate that it's finished
//
pCounter->ThisValue.CStatus = PDH_CSTATUS_BAD_COUNTERNAME;
pCounter->dwFlags &= ~PDHIC_COUNTER_NOT_INIT;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
}
pCounter->dwFlags &= ~PDHIC_COUNTER_NOT_INIT;
}
pMachine->dwRefCount --;
RELEASE_MUTEX(pMachine->hMutex);
if (bReturn) {
// look up counter
if (*pCounter->pCounterPath->szCounterName != SPLAT_L) {
pCounter->plCounterInfo.dwCounterId = GetCounterId(
pMachine,
pCounter->plCounterInfo.dwObjectId,
pCounter->pCounterPath->szCounterName);
if (pCounter->plCounterInfo.dwCounterId != (DWORD) -1) {
// load and initialize remaining counter values
if (AddMachineToQueryLists(pMachine, pCounter)) {
if (InitPerflibCounterInfo(pCounter)) {
// assign the appropriate calculation function
bReturn = AssignCalcFunction(
pCounter->plCounterInfo.dwCounterType,
& pCounter->CalcFunc,
& pCounter->StatFunc);
TRACE((PDH_DBG_TRACE_INFO),
(__LINE__,
PDH_CUTILS,
ARG_DEF(ARG_TYPE_WSTR, 1),
ERROR_SUCCESS,
TRACE_WSTR(pCounter->szFullName),
TRACE_DWORD(pCounter->plCounterInfo.dwCounterType),
NULL));
if (! bReturn) {
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
}
}
else {
// unable to initialize this counter
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
}
else {
// machine could not be added, error is already
// in "LastError" so free string buffer and leave
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
}
else {
// unable to lookup counter
pCounter->ThisValue.CStatus = PDH_CSTATUS_NO_COUNTER;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
}
else {
if (AddMachineToQueryLists(pMachine, pCounter)) {
pCounter->dwFlags |= PDHIC_COUNTER_OBJECT;
pCounter->pThisObject = NULL;
pCounter->pLastObject = NULL;
}
else {
// machine could not be added, error is already
// in "LastError" so free string buffer and leave
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
}
}
}
}
else {
PDH_STATUS pdhStatus;
// validate counter from log file
pdhStatus = PdhiGetLogCounterInfo(pCounter->pOwner->hLog, pCounter);
if (pdhStatus == ERROR_SUCCESS) {
// finish initializing the counter
//
pCounter->ThisValue.TimeStamp.dwLowDateTime = 0;
pCounter->ThisValue.TimeStamp.dwHighDateTime = 0;
pCounter->ThisValue.MultiCount = 1;
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
//
pCounter->LastValue.TimeStamp.dwLowDateTime = 0;
pCounter->LastValue.TimeStamp.dwHighDateTime = 0;
pCounter->LastValue.MultiCount = 1;
pCounter->LastValue.FirstValue = 0;
pCounter->LastValue.SecondValue = 0;
//
// lastly update status
//
pCounter->ThisValue.CStatus = PDH_CSTATUS_VALID_DATA;
pCounter->LastValue.CStatus = PDH_CSTATUS_VALID_DATA;
// assign the appropriate calculation function
bReturn = AssignCalcFunction(pCounter->plCounterInfo.dwCounterType,
& pCounter->CalcFunc,
& pCounter->StatFunc);
TRACE((PDH_DBG_TRACE_INFO),
(__LINE__,
PDH_CUTILS,
ARG_DEF(ARG_TYPE_WSTR, 1),
ERROR_SUCCESS,
TRACE_WSTR(pCounter->szFullName),
TRACE_DWORD(pCounter->plCounterInfo.dwCounterType),
NULL));
}
else {
// set the counter status to the error returned
pCounter->ThisValue.CStatus = pdhStatus;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
pCounter->dwFlags &= ~PDHIC_COUNTER_NOT_INIT;
}
if (! bReturn) {
//free string buffer
G_FREE(pCounter->pCounterPath);
pCounter->pCounterPath = NULL;
}
}
else {
G_FREE(pLocalCounterPath);
// unable to realloc
pCounter->ThisValue.CStatus = PDH_MEMORY_ALLOCATION_FAILURE;
bReturn = FALSE;
}
}
else {
// unable to parse counter name
pCounter->ThisValue.CStatus = PDH_CSTATUS_BAD_COUNTERNAME;
pCounter->dwFlags &= ~PDHIC_COUNTER_NOT_INIT;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
G_FREE(pLocalCounterPath);
bReturn = FALSE;
}
}
}
else {
// no counter name
pCounter->ThisValue.CStatus = PDH_CSTATUS_NO_COUNTERNAME;
pCounter->dwFlags &= ~PDHIC_COUNTER_NOT_INIT;
pCounter->dwFlags |= PDHIC_COUNTER_INVALID;
bReturn = FALSE;
}
if (! bReturn && pCounter->ThisValue.CStatus != ERROR_SUCCESS) {
SetLastError(pCounter->ThisValue.CStatus);
}
return bReturn;
}
BOOL
ParseInstanceName(
LPCWSTR szInstanceString,
LPWSTR szInstanceName,
LPWSTR szParentName,
DWORD dwName,
LPDWORD lpIndex
)
/*
parses the instance name formatted as follows
[parent/]instance[#index]
parent is optional and if present, is delimited by a forward slash
index is optional and if present, is delimited by a colon
parent and instance may be any legal file name character except a
delimeter character "/#\()" Index must be a string composed of
decimal digit characters (0-9), less than 10 characters in length, and
equate to a value between 0 and 2**32-1 (inclusive).
This function assumes that the instance name and parent name buffers
are of sufficient size.
NOTE: szInstanceName and szInstanceString can be the same buffer
*/
{
LPWSTR szSrcChar = (LPWSTR) szInstanceString;
LPWSTR szDestChar = (LPWSTR) szInstanceName;
LPWSTR szLastPound = NULL;
BOOL bReturn = FALSE;
DWORD dwIndex = 0;
DWORD dwInstCount = 0;
szDestChar = (LPWSTR) szInstanceName;
szSrcChar = (LPWSTR) szInstanceString;
__try {
do {
* szDestChar = * szSrcChar;
if (* szDestChar == POUNDSIGN_L) szLastPound = szDestChar;
szDestChar ++;
szSrcChar ++;
dwInstCount ++;
}
while (dwInstCount <= dwName && (* szSrcChar != L'\0') && (* szSrcChar != SLASH_L));
if (dwInstCount <= dwName) {
// see if that was really the parent or not
if (* szSrcChar == SLASH_L) {
// terminate destination after test in case they are the same buffer
* szDestChar = L'\0';
szSrcChar ++; // and move source pointer past delimter
// it was the parent name so copy it to the parent
StringCchCopyW(szParentName, dwName, szInstanceName);
// and copy the rest of the string after the "/" to the
// instance name field
szDestChar = szInstanceName;
dwInstCount = 0;
do {
* szDestChar = * szSrcChar;
if (* szDestChar == POUNDSIGN_L) szLastPound = szDestChar;
szDestChar ++;
szSrcChar ++;
dwInstCount ++;
}
while (dwInstCount <= dwName && (* szSrcChar != L'\0'));
}
else {
// that was the only element so load an empty string for the parent
* szParentName = L'\0';
}
if (dwInstCount <= dwName) {
// if szLastPound is NOT null and is inside the instance string, then
// see if it points to a decimal number. If it does, then it's an index
// otherwise it's part of the instance name
* szDestChar = L'\0'; // terminate the destination string
dwIndex = 0;
if (szLastPound != NULL) {
if (szLastPound > szInstanceName) {
// there's a pound sign in the instance name
// see if it's preceded by a non-space char
szLastPound --;
if (* szLastPound > SPACE_L) {
szLastPound ++;
// see if it's followed by a digit
szLastPound ++;
if ((* szLastPound >= L'0') && (*szLastPound <= L'9')) {
dwIndex = wcstoul(szLastPound, NULL, 10);
szLastPound --;
* szLastPound = L'\0'; // terminate the name at the pound sign
}
}
}
}
* lpIndex = dwIndex;
bReturn = TRUE;
}
}
}
__except (EXCEPTION_EXECUTE_HANDLER) {
// unable to move strings
bReturn = FALSE;
}
return bReturn;
}
BOOL
ParseFullPathNameW(
LPCWSTR szFullCounterPath,
PDWORD pdwBufferSize,
PPDHI_COUNTER_PATH pCounter,
BOOL bWbemSyntax
)
/*
interprets counter path string as either a
\\machine\object(instance)\counter
or if bWbemSyntax == TRUE
\\machine\namespace:ClassName(InstanceName)\CounterName
and returns the component in the counter path structure
\\machine or \\machine\namespace may be omitted on the local machine
(instance) may be omitted on counters with no instance structures
if object or counter is missing, then FALSE is returned, otherwise
TRUE is returned if the parsing was successful
*/
{
LPWSTR szWorkMachine = NULL;
LPWSTR szWorkObject = NULL;
LPWSTR szWorkInstance = NULL;
LPWSTR szWorkParent = NULL;
LPWSTR szWorkCounter = NULL;
DWORD dwBufferSize = lstrlenW(szFullCounterPath) + 1;
BOOL bReturn = FALSE;
LPWSTR szSrcChar, szDestChar;
DWORD dwBufferLength = 0;
DWORD dwWorkMachineLength = 0;
DWORD dwWorkObjectLength = 0;
DWORD dwWorkInstanceLength = 0;
DWORD dwWorkParentLength = 0;
DWORD dwWorkCounterLength = 0;
DWORD dwWorkIndex = 0;
DWORD dwParenDepth = 0;
WCHAR wDelimiter = 0;
LPWSTR pszBsDelim[2] = {0,0};
LPWSTR szThisChar;
DWORD dwParenCount = 0;
LPWSTR szLastParen = NULL;
if (dwBufferSize < MAX_PATH) dwBufferSize = MAX_PATH;
szWorkMachine = G_ALLOC(dwBufferSize * sizeof(WCHAR));
szWorkObject = G_ALLOC(dwBufferSize * sizeof(WCHAR));
szWorkInstance = G_ALLOC(dwBufferSize * sizeof(WCHAR));
szWorkParent = G_ALLOC(dwBufferSize * sizeof(WCHAR));
szWorkCounter = G_ALLOC(dwBufferSize * sizeof(WCHAR));
if (szWorkMachine != NULL && szWorkObject != NULL && szWorkInstance != NULL
&& szWorkParent != NULL && szWorkCounter != NULL) {
// get machine name from counter path
szSrcChar = (LPWSTR) szFullCounterPath;
//define the delimiter char between the machine and the object
// or in WBEM parlance, the server & namespace and the Class name
if (bWbemSyntax) {
wDelimiter = COLON_L;
}
else {
wDelimiter = BACKSLASH_L;
// if this is backslash delimited string, then find the
// backslash the denotes the end of the machine and start of the
// object by walking down the string and finding the 2nd to the last
// backslash.
// this is necessary since a WBEM machine\namespace path can have
// multiple backslashes in it while there will ALWAYS be two at
// the end, one at the start of the object name and one at the start
// of the counter name
dwParenDepth = 0;
for (szThisChar = szSrcChar; * szThisChar != L'\0'; szThisChar++) {
if (* szThisChar == LEFTPAREN_L) {
if (dwParenDepth == 0) dwParenCount ++;
dwParenDepth ++;
}
else if (* szThisChar == RIGHTPAREN_L) {
if (dwParenDepth > 0) dwParenDepth --;
}
else {
if (dwParenDepth == 0) {
// ignore delimiters inside parenthesis
if (* szThisChar == wDelimiter) {
pszBsDelim[0] = pszBsDelim[1];
pszBsDelim[1] = szThisChar;
}
// ignore it and go to the next character
}
}
}
if ((dwParenCount > 0) && (pszBsDelim[0] != NULL) && (pszBsDelim[1] != NULL)) {
dwParenDepth = 0;
for (szThisChar = pszBsDelim[0]; ((* szThisChar != L'\0') && (szThisChar < pszBsDelim[1])); szThisChar ++) {
if (* szThisChar == LEFTPAREN_L) {
if (dwParenDepth == 0) {
// see if the preceeding char is whitespace
-- szThisChar;
if (* szThisChar > SPACE_L) {
// then this could be an instance delim
szLastParen = ++ szThisChar;
}
else {
// else it's probably part of the instance name
++ szThisChar;
}
}
dwParenDepth ++;
}
else if (* szThisChar == RIGHTPAREN_L) {
if (dwParenDepth > 0) dwParenDepth --;
}
}
}
}
// see if this is really a machine name by looking for leading "\\"
if ((szSrcChar[0] == BACKSLASH_L) && (szSrcChar[1] == BACKSLASH_L)) {
szDestChar = szWorkMachine;
* szDestChar ++ = * szSrcChar ++;
* szDestChar ++ = * szSrcChar ++;
dwWorkMachineLength = 2;
// must be a machine name so find the object delimiter and zero terminate
// it there
while (* szSrcChar != L'\0') {
if (pszBsDelim[0] != NULL) {
// then go to this pointer
if (szSrcChar == pszBsDelim[0]) break;
}
else {
// go to the next delimiter
if (* szSrcChar != wDelimiter) break;
}
* szDestChar ++ = * szSrcChar ++;
dwWorkMachineLength ++;
}
if (* szSrcChar == L'\0') {
// no other required fields
goto Cleanup;
}
else {
// null terminate and continue
* szDestChar ++ = L'\0';
}
}
else {
// no machine name, so they must have skipped that field
// which is OK. We'll insert the local machine name here
StringCchCopyW(szWorkMachine, dwBufferSize, szStaticLocalMachineName);
dwWorkMachineLength = lstrlenW(szWorkMachine);
}
// szSrcChar should be pointing to the backslash preceeding the
// object name now.
if (szSrcChar[0] == wDelimiter) {
szSrcChar ++; // to move past backslash
szDestChar = szWorkObject;
// copy until:
// a) the end of the source string is reached
// b) the instance delimiter is found "("
// c) the counter delimiter is found "\"
// d) a non-printable, non-space char is found
while ((* szSrcChar != L'\0') && (szSrcChar != szLastParen)
&& (* szSrcChar != BACKSLASH_L) && (* szSrcChar >= SPACE_L)) {
dwWorkObjectLength ++;
* szDestChar ++ = * szSrcChar ++;
}
// see why it ended:
if (* szSrcChar < SPACE_L) {
// ran of source string
goto Cleanup;
}
else if (szSrcChar == szLastParen) {
dwParenDepth = 1;
// there's an instance so copy that to the instance field
* szDestChar = L'\0'; // terminate destination string
szDestChar = szWorkInstance;
// skip past open paren
++ szSrcChar;
// copy until:
// a) the end of the source string is reached
// b) the instance delimiter is found "("
while ((* szSrcChar != L'\0') && (dwParenDepth > 0)) {
if (* szSrcChar == RIGHTPAREN_L) {
dwParenDepth --;
}
else if (* szSrcChar == LEFTPAREN_L) {
dwParenDepth ++;
}
if (dwParenDepth > 0) {
// copy all parenthesis except the last one
dwWorkInstanceLength ++;
* szDestChar ++ = * szSrcChar ++;
}
}
// see why it ended:
if (* szSrcChar == L'\0') {
// ran of source string
goto Cleanup;
}
else {
// move source to object delimiter
if (* ++ szSrcChar != BACKSLASH_L) {
// bad format
goto Cleanup;
}
else {
* szDestChar = L'\0';
// check instance string for a parent
if (ParseInstanceName(
szWorkInstance, szWorkInstance, szWorkParent, dwBufferSize, & dwWorkIndex)) {
dwWorkInstanceLength = lstrlenW(szWorkInstance);
dwWorkParentLength = lstrlenW(szWorkParent);
}
else {
// instance string not formatted correctly
goto Cleanup;
}
}
}
}
else {
// terminate the destination string
* szDestChar = L'\0';
}
// finally copy the counter name
szSrcChar ++; // to move past backslash
szDestChar = szWorkCounter;
// copy until:
// a) the end of the source string is reached
while (* szSrcChar != L'\0') {
dwWorkCounterLength ++;
* szDestChar ++ = * szSrcChar ++;
}
* szDestChar = L'\0';
// now to see if all this will fit in the users's buffer
dwBufferLength = sizeof(PDHI_COUNTER_PATH) - sizeof(BYTE);
dwBufferLength += DWORD_MULTIPLE((dwWorkMachineLength + 1) * sizeof(WCHAR));
dwBufferLength += DWORD_MULTIPLE((dwWorkObjectLength + 1) * sizeof(WCHAR));
if (dwWorkInstanceLength > 0) {
dwBufferLength += DWORD_MULTIPLE((dwWorkInstanceLength + 1) * sizeof(WCHAR));
}
if (dwWorkParentLength > 0) {
dwBufferLength += DWORD_MULTIPLE((dwWorkParentLength + 1) * sizeof(WCHAR));
}
dwBufferLength += DWORD_MULTIPLE((dwWorkCounterLength + 1) * sizeof(WCHAR));
TRACE((PDH_DBG_TRACE_INFO),
(__LINE__,
PDH_CUTILS,
ARG_DEF(ARG_TYPE_WSTR, 1) | ARG_DEF(ARG_TYPE_WSTR, 2)
| ARG_DEF(ARG_TYPE_WSTR, 3)
| ARG_DEF(ARG_TYPE_WSTR, 4)
| ARG_DEF(ARG_TYPE_WSTR, 5)
| ARG_DEF(ARG_TYPE_WSTR, 6),
ERROR_SUCCESS,
TRACE_WSTR(szFullCounterPath),
TRACE_WSTR(szWorkMachine),
TRACE_WSTR(szWorkObject),
TRACE_WSTR(szWorkCounter),
TRACE_WSTR(szWorkInstance),
TRACE_WSTR(szWorkParent),
TRACE_DWORD(dwWorkIndex),
TRACE_DWORD(dwBufferLength),
NULL));
if (dwBufferLength < * pdwBufferSize) {
// it looks like it'll fit so start filling things in
szDestChar = (LPWSTR) & pCounter->pBuffer[0];
if (dwWorkMachineLength != 0) {
pCounter->szMachineName = szDestChar;
StringCchCopyW(szDestChar, dwWorkMachineLength + 1, szWorkMachine);
szDestChar += dwWorkMachineLength + 1;
szDestChar = ALIGN_ON_DWORD(szDestChar);
}
else {
pCounter->szMachineName = NULL;
}
pCounter->szObjectName = szDestChar;
StringCchCopyW(szDestChar, dwWorkObjectLength + 1, szWorkObject);
szDestChar += dwWorkObjectLength + 1;
szDestChar = ALIGN_ON_DWORD(szDestChar);
if (dwWorkInstanceLength != 0) {
pCounter->szInstanceName = szDestChar;
StringCchCopyW(szDestChar, dwWorkInstanceLength + 1, szWorkInstance);
szDestChar += dwWorkInstanceLength + 1;
szDestChar = ALIGN_ON_DWORD(szDestChar);
}
else {
pCounter->szInstanceName = NULL;
}
if (dwWorkParentLength != 0) {
pCounter->szParentName = szDestChar;
StringCchCopyW(szDestChar, dwWorkParentLength + 1, szWorkParent);
szDestChar += dwWorkParentLength + 1;
szDestChar = ALIGN_ON_DWORD(szDestChar);
}
else {
pCounter->szParentName = NULL;
}
pCounter->dwIndex = dwWorkIndex;
pCounter->szCounterName = szDestChar;
StringCchCopyW(szDestChar, dwWorkCounterLength + 1, szWorkCounter);
szDestChar += dwWorkCounterLength + 1;
szDestChar = ALIGN_ON_DWORD(szDestChar);
* pdwBufferSize = dwBufferLength;
bReturn = TRUE;
}
else {
//insufficient buffer
}
}
else {
// no object found so return
}
}
else {
// incoming string is too long
}
Cleanup:
G_FREE(szWorkMachine);
G_FREE(szWorkObject);
G_FREE(szWorkInstance);
G_FREE(szWorkParent);
G_FREE(szWorkCounter);
return bReturn;
}
BOOL
FreeCounter(
PPDHI_COUNTER pThisCounter
)
{
// NOTE:
// This function assumes the query containing
// this counter has already been locked by the calling
// function.
PPDHI_COUNTER pPrevCounter;
PPDHI_COUNTER pNextCounter;
PPDHI_QUERY pParentQuery;
// define pointers
pPrevCounter = pThisCounter->next.blink;
pNextCounter = pThisCounter->next.flink;
pParentQuery = pThisCounter->pOwner;
// decrement machine reference counter if a machine has been assigned
if (pThisCounter->pQMachine != NULL) {
if (pThisCounter->pQMachine->pMachine != NULL) {
if (--pThisCounter->pQMachine->pMachine->dwRefCount == 0) {
// then this is the last counter so remove machine
// freeing the machine in this call causes all kinds of
// multi-threading problems so we'll keep it around until
// the DLL unloads.
// FreeMachine (pThisCounter->pQMachine->pMachine, FALSE);
pThisCounter->pQMachine->pMachine = NULL;
}
else {
// the ref count is non-zero so leave the pointer alone
}
}
else {
// the pointer has already been cleared
}
}
else {
// there's no machine
}
// free allocated memory in the counter
G_FREE(pThisCounter->pCounterPath);
pThisCounter->pCounterPath = NULL;
G_FREE(pThisCounter->szFullName);
pThisCounter->szFullName = NULL;
if (pParentQuery != NULL) {
if (pParentQuery->hLog == NULL) {
G_FREE(pThisCounter->pThisObject);
pThisCounter->pThisObject = NULL;
G_FREE(pThisCounter->pLastObject);
pThisCounter->pLastObject = NULL;
G_FREE(pThisCounter->pThisRawItemList);
pThisCounter->pThisRawItemList = NULL;
G_FREE(pThisCounter->pLastRawItemList);
pThisCounter->pLastRawItemList = NULL;
}
}
// check for WBEM items
if ((pThisCounter->dwFlags & PDHIC_WBEM_COUNTER) && (pThisCounter->pOwner != NULL)) {
PdhiCloseWbemCounter(pThisCounter);
}
// update pointers if they've been assigned
if ((pPrevCounter != NULL) && (pNextCounter != NULL)) {
if ((pPrevCounter != pThisCounter) && (pNextCounter != pThisCounter)) {
// update query list pointers
pPrevCounter->next.flink = pNextCounter;
pNextCounter->next.blink = pPrevCounter;
}
else {
// this is the only counter entry in the list
// so the caller must deal with updating the head pointer
}
}
// delete this counter
G_FREE(pThisCounter);
return TRUE;
}
BOOL
UpdateCounterValue(
PPDHI_COUNTER pCounter,
PPERF_DATA_BLOCK pPerfData
)
{
DWORD LocalCStatus = 0;
DWORD LocalCType = 0;
LPVOID pData = NULL;
PDWORD pdwData;
UNALIGNED LONGLONG * pllData;
PPERF_OBJECT_TYPE pPerfObject = NULL;
BOOL bReturn = FALSE;
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
0,
& pPerfObject,
& LocalCStatus);
pCounter->ThisValue.CStatus = LocalCStatus;
if (IsSuccessSeverity(LocalCStatus)) {
// assume success
bReturn = TRUE;
// load counter value based on counter type
LocalCType = pCounter->plCounterInfo.dwCounterType;
switch (LocalCType) {
//
// these counter types are loaded as:
// Numerator = Counter data from perf data block
// Denominator = Perf Time from perf data block
// (the time base is the PerfFreq)
//
case PERF_COUNTER_COUNTER:
case PERF_COUNTER_QUEUELEN_TYPE:
case PERF_SAMPLE_COUNTER:
pCounter->ThisValue.FirstValue = (LONGLONG) (* (DWORD *) pData);
pCounter->ThisValue.SecondValue = pPerfData->PerfTime.QuadPart;
break;
case PERF_OBJ_TIME_TIMER:
pCounter->ThisValue.FirstValue = (LONGLONG) (* (DWORD *) pData);
pCounter->ThisValue.SecondValue = pPerfObject->PerfTime.QuadPart;
break;
case PERF_COUNTER_100NS_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pCounter->ThisValue.SecondValue = pPerfData->PerfTime100nSec.QuadPart;
break;
case PERF_COUNTER_OBJ_TIME_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pCounter->ThisValue.SecondValue = pPerfObject->PerfTime.QuadPart;
break;
case PERF_COUNTER_TIMER:
case PERF_COUNTER_TIMER_INV:
case PERF_COUNTER_BULK_COUNT:
case PERF_COUNTER_LARGE_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pCounter->ThisValue.SecondValue = pPerfData->PerfTime.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
pCounter->ThisValue.MultiCount = (DWORD) * ++ pllData;
}
break;
//
// this is a hack to make the PDH work like PERFMON for
// this counter type
//
case PERF_COUNTER_MULTI_TIMER:
case PERF_COUNTER_MULTI_TIMER_INV:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
// begin hack code
pCounter->ThisValue.FirstValue *= (DWORD) pPerfData->PerfFreq.QuadPart;
// end hack code
pCounter->ThisValue.SecondValue = pPerfData->PerfTime.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
pCounter->ThisValue.MultiCount = (DWORD) * ++ pllData;
}
break;
//
// These counters do not use any time reference
//
case PERF_COUNTER_RAWCOUNT:
case PERF_COUNTER_RAWCOUNT_HEX:
case PERF_COUNTER_DELTA:
pCounter->ThisValue.FirstValue = (LONGLONG) (* (DWORD *) pData);
pCounter->ThisValue.SecondValue = 0;
break;
case PERF_COUNTER_LARGE_RAWCOUNT:
case PERF_COUNTER_LARGE_RAWCOUNT_HEX:
case PERF_COUNTER_LARGE_DELTA:
pCounter->ThisValue.FirstValue = * (LONGLONG *) pData;
pCounter->ThisValue.SecondValue = 0;
break;
//
// These counters use the 100 Ns time base in thier calculation
//
case PERF_100NSEC_TIMER:
case PERF_100NSEC_TIMER_INV:
case PERF_100NSEC_MULTI_TIMER:
case PERF_100NSEC_MULTI_TIMER_INV:
pllData = (UNALIGNED LONGLONG *)pData;
pCounter->ThisValue.FirstValue = * pllData;
pCounter->ThisValue.SecondValue = pPerfData->PerfTime100nSec.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
++ pllData;
pCounter->ThisValue.MultiCount = * (DWORD *) pllData;
}
break;
//
// These counters use two data points, the one pointed to by
// pData and the one immediately after
//
case PERF_SAMPLE_FRACTION:
case PERF_RAW_FRACTION:
pdwData = (DWORD *) pData;
pCounter->ThisValue.FirstValue = (LONGLONG) (* pdwData);
// find the pointer to the base value in the structure
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
GPCDP_GET_BASE_DATA,
NULL,
& LocalCStatus);
if (IsSuccessSeverity(LocalCStatus)) {
pdwData = (DWORD *) pData;
pCounter->ThisValue.SecondValue = (LONGLONG) (* pdwData);
}
else {
// unable to locate base value
pCounter->ThisValue.SecondValue = 0;
pCounter->ThisValue.CStatus = LocalCStatus;
bReturn = FALSE;
}
break;
case PERF_LARGE_RAW_FRACTION:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
GPCDP_GET_BASE_DATA,
NULL,
& LocalCStatus);
if (IsSuccessSeverity(LocalCStatus)) {
pllData = (LONGLONG *) pData;
pCounter->ThisValue.SecondValue = * pllData;
}
else {
pCounter->ThisValue.SecondValue = 0;
pCounter->ThisValue.CStatus = LocalCStatus;
bReturn = FALSE;
}
break;
case PERF_PRECISION_SYSTEM_TIMER:
case PERF_PRECISION_100NS_TIMER:
case PERF_PRECISION_OBJECT_TIMER:
pllData = (LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
// find the pointer to the base value in the structure
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
GPCDP_GET_BASE_DATA,
NULL,
& LocalCStatus);
if (IsSuccessSeverity(LocalCStatus)) {
pllData = (LONGLONG *) pData;
pCounter->ThisValue.SecondValue = * pllData;
}
else {
// unable to locate base value
pCounter->ThisValue.SecondValue = 0;
pCounter->ThisValue.CStatus = LocalCStatus;
bReturn = FALSE;
}
break;
case PERF_AVERAGE_TIMER:
case PERF_AVERAGE_BULK:
// counter (numerator) is a LONGLONG, while the
// denominator is just a DWORD
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
GPCDP_GET_BASE_DATA,
NULL,
& LocalCStatus);
if (IsSuccessSeverity(LocalCStatus)) {
pdwData = (DWORD *) pData;
pCounter->ThisValue.SecondValue = * pdwData;
} else {
// unable to locate base value
pCounter->ThisValue.SecondValue = 0;
pCounter->ThisValue.CStatus = LocalCStatus;
bReturn = FALSE;
}
break;
//
// These counters are used as the part of another counter
// and as such should not be used, but in case they are
// they'll be handled here.
//
case PERF_SAMPLE_BASE:
case PERF_AVERAGE_BASE:
case PERF_COUNTER_MULTI_BASE:
case PERF_RAW_BASE:
case PERF_LARGE_RAW_BASE:
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
break;
case PERF_ELAPSED_TIME:
// this counter type needs the object perf data as well
if (GetObjectPerfInfo(pPerfData,
pCounter->plCounterInfo.dwObjectId,
& pCounter->ThisValue.SecondValue,
& pCounter->TimeBase)) {
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
}
else {
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
}
break;
//
// These counters are not supported by this function (yet)
//
case PERF_COUNTER_TEXT:
case PERF_COUNTER_NODATA:
case PERF_COUNTER_HISTOGRAM_TYPE:
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
break;
default:
// an unidentified counter was returned so
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
bReturn = FALSE;
break;
}
}
else {
// else this counter is not valid so this value == 0
pCounter->ThisValue.FirstValue = pCounter->LastValue.FirstValue;
pCounter->ThisValue.SecondValue = pCounter->LastValue.SecondValue;
pCounter->ThisValue.CStatus = LocalCStatus;
bReturn = FALSE;
}
return bReturn;
}
BOOL
UpdateRealTimeCounterValue(
PPDHI_COUNTER pCounter
)
{
BOOL bResult = FALSE;
DWORD LocalCStatus = 0;
FILETIME GmtFileTime;
// move current value to last value buffer
pCounter->LastValue = pCounter->ThisValue;
// and clear the old value
pCounter->ThisValue.MultiCount = 1;
pCounter->ThisValue.FirstValue = 0;
pCounter->ThisValue.SecondValue = 0;
// don't process if the counter has not been initialized
if (!(pCounter->dwFlags & PDHIC_COUNTER_UNUSABLE)) {
// get the counter's machine status first. There's no point in
// contuning if the machine is offline
LocalCStatus = pCounter->pQMachine->lQueryStatus;
if (IsSuccessSeverity(LocalCStatus) && pCounter->pQMachine->pPerfData != NULL) {
// update timestamp
SystemTimeToFileTime(& pCounter->pQMachine->pPerfData->SystemTime, & GmtFileTime);
FileTimeToLocalFileTime(& GmtFileTime, & pCounter->ThisValue.TimeStamp);
bResult = UpdateCounterValue(pCounter, pCounter->pQMachine->pPerfData);
}
else {
// unable to read data from this counter's machine so use the
// query's timestamp
//
pCounter->ThisValue.TimeStamp.dwLowDateTime = LODWORD(pCounter->pQMachine->llQueryTime);
pCounter->ThisValue.TimeStamp.dwHighDateTime = HIDWORD(pCounter->pQMachine->llQueryTime);
// all other data fields remain un-changed
pCounter->ThisValue.CStatus = LocalCStatus; // save counter status
}
}
else {
if (pCounter->dwFlags & PDHIC_COUNTER_NOT_INIT) {
// try to init it
InitCounter (pCounter);
}
}
return bResult;
}
BOOL
UpdateMultiInstanceCounterValue(
PPDHI_COUNTER pCounter,
PPERF_DATA_BLOCK pPerfData,
LONGLONG TimeStamp
)
{
PPERF_OBJECT_TYPE pPerfObject = NULL;
PPERF_INSTANCE_DEFINITION pPerfInstance = NULL;
PPERF_OBJECT_TYPE pParentObject = NULL;
PPERF_INSTANCE_DEFINITION pThisParentInstance = NULL;
PPERF_COUNTER_DEFINITION pNumPerfCounter = NULL;
PPERF_COUNTER_DEFINITION pDenPerfCounter = NULL;
DWORD LocalCStatus = 0;
DWORD LocalCType = 0;
LPVOID pData = NULL;
PDWORD pdwData;
UNALIGNED LONGLONG * pllData;
FILETIME GmtFileTime;
DWORD dwSize;
DWORD dwFinalSize;
LONG nThisInstanceIndex;
LONG nParentInstanceIndex;
LPWSTR szNextNameString;
DWORD dwStrSize;
PPDHI_RAW_COUNTER_ITEM pThisItem;
BOOL bReturn = FALSE;
pPerfObject = GetObjectDefByTitleIndex(pPerfData, pCounter->plCounterInfo.dwObjectId);
if (pPerfObject != NULL) {
// this should be caught during the AddCounter operation
//
// allocate a new buffer for the current data
// this should be large enough to handle the header,
// all instances and thier name strings
//
dwSize = sizeof(PDHI_RAW_COUNTER_ITEM_BLOCK) - sizeof(PDHI_RAW_COUNTER_ITEM);
dwStrSize = 0;
pPerfInstance = FirstInstance(pPerfObject);
// make sure pointer is still within the same instance
for (nThisInstanceIndex = 0;
pPerfInstance != NULL && nThisInstanceIndex < pPerfObject->NumInstances;
nThisInstanceIndex ++) {
// this should only fail in dire cases
if (pPerfInstance == NULL) break;
// for this instance add the size of the data item
dwSize += sizeof(PDHI_RAW_COUNTER_ITEM);
// and the size of the name string
dwSize += pPerfInstance->NameLength + sizeof(WCHAR);
dwStrSize += pPerfInstance->NameLength / sizeof(WCHAR) + 1;
// to the required buffer size
// if this instance has a parent, see how long it's string
// is
// first see if we've already got the pointer to the parent
if (pPerfInstance->ParentObjectTitleIndex != 0) {
// then include the parent instance name
if (pParentObject == NULL) {
// get parent object
pParentObject = GetObjectDefByTitleIndex(pPerfData, pPerfInstance->ParentObjectTitleIndex);
}
else {
if (pParentObject->ObjectNameTitleIndex != pPerfInstance->ParentObjectTitleIndex) {
pParentObject = GetObjectDefByTitleIndex(pPerfData, pPerfInstance->ParentObjectTitleIndex);
}
}
if (pParentObject == NULL) break;
// now go to the corresponding instance entry
pThisParentInstance = FirstInstance(pParentObject);
// make sure pointer is still within the same instance
if (pThisParentInstance != NULL) {
if (pPerfInstance->ParentObjectInstance < (DWORD) pParentObject->NumInstances) {
for (nParentInstanceIndex = 0;
(DWORD) nParentInstanceIndex != pPerfInstance->ParentObjectInstance;
nParentInstanceIndex ++) {
pThisParentInstance = NextInstance(pParentObject, pThisParentInstance);
if (pThisParentInstance == NULL) break;
}
if (pThisParentInstance != NULL) {
// found it so add in it's string length
dwSize += pThisParentInstance->NameLength + sizeof(WCHAR);
dwStrSize += pThisParentInstance->NameLength / sizeof(WCHAR) + 1;
}
}
else {
// the index is not in the parent
pThisParentInstance = NULL;
// so don't change the size required field
}
}
}
// round up to the next DWORD address
dwSize = DWORD_MULTIPLE(dwSize);
// and go to the next instance
pPerfInstance = NextInstance(pPerfObject, pPerfInstance);
}
//
//
pCounter->pThisRawItemList = G_ALLOC(dwSize);
if (pCounter->pThisRawItemList != NULL) {
pCounter->pThisRawItemList->dwLength = dwSize;
pNumPerfCounter = GetCounterDefByTitleIndex(pPerfObject, 0, pCounter->plCounterInfo.dwCounterId);
// just in case we need it later
pDenPerfCounter = pNumPerfCounter + 1;
// fill in the counter data
pCounter->pThisRawItemList->dwItemCount = pPerfObject->NumInstances;
pCounter->pThisRawItemList->CStatus = LocalCStatus;
// update timestamp
SystemTimeToFileTime(& pPerfData->SystemTime, & GmtFileTime);
FileTimeToLocalFileTime(& GmtFileTime, & pCounter->pThisRawItemList->TimeStamp);
pThisItem = & pCounter->pThisRawItemList->pItemArray[0];
szNextNameString = (LPWSTR) & (pCounter->pThisRawItemList->pItemArray[pPerfObject->NumInstances]);
pPerfInstance = FirstInstance(pPerfObject);
if (pPerfInstance != NULL) {
// make sure pointer is still within the same instance
// for each instance log the raw data values for this counter
for (nThisInstanceIndex = 0;
pPerfInstance != NULL && nThisInstanceIndex < pPerfObject->NumInstances;
nThisInstanceIndex ++) {
// make sure pointe is still within the same instance
// make a new instance entry
// get the name of this instance
pThisItem->szName = (DWORD) (((LPBYTE) szNextNameString) - ((LPBYTE) pCounter->pThisRawItemList));
if (dwStrSize == 0) {
SetLastError(ERROR_OUTOFMEMORY);
bReturn = FALSE;
break;
}
dwSize = GetFullInstanceNameStr(pPerfData, pPerfObject, pPerfInstance, szNextNameString, dwStrSize);
if (dwSize == 0) {
// unable to read instance name
// so make one up (and assert in DBG builds)
_ltow(nThisInstanceIndex, szNextNameString, 10);
dwSize = lstrlenW(szNextNameString);
}
if (dwSize + 1 > dwStrSize) {
SetLastError(ERROR_OUTOFMEMORY);
bReturn = FALSE;
break;
}
szNextNameString += dwSize + 1;
dwStrSize -= (dwSize + 1);
// get the pointer to the counter data
pData = GetPerfCounterDataPtr(pPerfData,
pCounter->pCounterPath,
& pCounter->plCounterInfo,
0,
NULL,
& LocalCStatus);
if (pNumPerfCounter != NULL) {
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pNumPerfCounter);
}
if (pData == NULL) {
SetLastError(PDH_CSTATUS_NO_INSTANCE);
bReturn = FALSE;
break;
}
bReturn = TRUE; // assume success
// load counter value based on counter type
LocalCType = pCounter->plCounterInfo.dwCounterType;
switch (LocalCType) {
//
// these counter types are loaded as:
// Numerator = Counter data from perf data block
// Denominator = Perf Time from perf data block
// (the time base is the PerfFreq)
//
case PERF_COUNTER_COUNTER:
case PERF_COUNTER_QUEUELEN_TYPE:
case PERF_SAMPLE_COUNTER:
pThisItem->FirstValue = (LONGLONG) (* (DWORD *) pData);
pThisItem->SecondValue = pPerfData->PerfTime.QuadPart;
break;
case PERF_OBJ_TIME_TIMER:
pThisItem->FirstValue = (LONGLONG) (* (DWORD *) pData);
pThisItem->SecondValue = pPerfObject->PerfTime.QuadPart;
break;
case PERF_COUNTER_100NS_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
pThisItem->SecondValue = pPerfData->PerfTime100nSec.QuadPart;
break;
case PERF_COUNTER_OBJ_TIME_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
pThisItem->SecondValue = pPerfObject->PerfTime.QuadPart;
break;
case PERF_COUNTER_TIMER:
case PERF_COUNTER_TIMER_INV:
case PERF_COUNTER_BULK_COUNT:
case PERF_COUNTER_LARGE_QUEUELEN_TYPE:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
pThisItem->SecondValue = pPerfData->PerfTime.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
pThisItem->MultiCount = (DWORD) * ++pllData;
}
break;
case PERF_COUNTER_MULTI_TIMER:
case PERF_COUNTER_MULTI_TIMER_INV:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
// begin hack code
pThisItem->FirstValue *= (DWORD) pPerfData->PerfFreq.QuadPart;
// end hack code
pThisItem->SecondValue = pPerfData->PerfTime.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
pThisItem->MultiCount = (DWORD) * ++ pllData;
}
break;
//
// These counters do not use any time reference
//
case PERF_COUNTER_RAWCOUNT:
case PERF_COUNTER_RAWCOUNT_HEX:
case PERF_COUNTER_DELTA:
pThisItem->FirstValue = (LONGLONG) (* (DWORD *) pData);
pThisItem->SecondValue = 0;
break;
case PERF_COUNTER_LARGE_RAWCOUNT:
case PERF_COUNTER_LARGE_RAWCOUNT_HEX:
case PERF_COUNTER_LARGE_DELTA:
pThisItem->FirstValue = * (LONGLONG *) pData;
pThisItem->SecondValue = 0;
break;
//
// These counters use the 100 Ns time base in thier calculation
//
case PERF_100NSEC_TIMER:
case PERF_100NSEC_TIMER_INV:
case PERF_100NSEC_MULTI_TIMER:
case PERF_100NSEC_MULTI_TIMER_INV:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
pThisItem->SecondValue = pPerfData->PerfTime100nSec.QuadPart;
if ((LocalCType & PERF_MULTI_COUNTER) == PERF_MULTI_COUNTER) {
++ pllData;
pThisItem->MultiCount = * (DWORD *) pllData;
}
break;
//
// These counters use two data points, the one pointed to by
// pData and the one pointed by the definition following
// immediately after
//
case PERF_SAMPLE_FRACTION:
case PERF_RAW_FRACTION:
pdwData = (DWORD *) pData;
pThisItem->FirstValue = (LONGLONG)(* pdwData);
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pDenPerfCounter);
pdwData = (DWORD *) pData;
pThisItem->SecondValue = (LONGLONG) (* pdwData);
break;
case PERF_LARGE_RAW_FRACTION:
pllData = (UNALIGNED LONGLONG *) pData;
pCounter->ThisValue.FirstValue = * pllData;
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pDenPerfCounter);
if (pData) {
pllData = (LONGLONG *) pData;
pCounter->ThisValue.SecondValue = * pllData;
}
else {
pCounter->ThisValue.SecondValue = 0;
bReturn = FALSE;
}
break;
case PERF_PRECISION_SYSTEM_TIMER:
case PERF_PRECISION_100NS_TIMER:
case PERF_PRECISION_OBJECT_TIMER:
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
// find the pointer to the base value in the structure
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pDenPerfCounter);
pllData = (LONGLONG *) pData;
pThisItem->SecondValue = * pllData;
break;
case PERF_AVERAGE_TIMER:
case PERF_AVERAGE_BULK:
// counter (numerator) is a LONGLONG, while the
// denominator is just a DWORD
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pDenPerfCounter);
pdwData = (DWORD *) pData;
pThisItem->SecondValue = (LONGLONG) * pdwData;
break;
//
// These counters are used as the part of another counter
// and as such should not be used, but in case they are
// they'll be handled here.
//
case PERF_SAMPLE_BASE:
case PERF_AVERAGE_BASE:
case PERF_COUNTER_MULTI_BASE:
case PERF_RAW_BASE:
case PERF_LARGE_RAW_BASE:
pThisItem->FirstValue = 0;
pThisItem->SecondValue = 0;
break;
case PERF_ELAPSED_TIME:
// this counter type needs the object perf data as well
if (GetObjectPerfInfo(pPerfData,
pCounter->plCounterInfo.dwObjectId,
& pThisItem->SecondValue,
& pCounter->TimeBase)) {
pllData = (UNALIGNED LONGLONG *) pData;
pThisItem->FirstValue = * pllData;
}
else {
pThisItem->FirstValue = 0;
pThisItem->SecondValue = 0;
}
break;
//
// These counters are not supported by this function (yet)
//
case PERF_COUNTER_TEXT:
case PERF_COUNTER_NODATA:
case PERF_COUNTER_HISTOGRAM_TYPE:
pThisItem->FirstValue = 0;
pThisItem->SecondValue = 0;
break;
default:
// an unrecognized counter type was returned
pThisItem->FirstValue = 0;
pThisItem->SecondValue = 0;
bReturn = FALSE;
break;
}
pThisItem ++; // go to the next entry
// go to the next instance data block
pPerfInstance = NextInstance(pPerfObject, pPerfInstance);
} // end for each instance
}
else {
// no instance found so ignore
}
// measure the memory block used
dwFinalSize = (DWORD)((LPBYTE)szNextNameString - (LPBYTE) pCounter->pThisRawItemList);
}
else {
// unable to allocate a new buffer so return error
SetLastError(ERROR_OUTOFMEMORY);
bReturn = FALSE;
}
}
else {
pCounter->pThisRawItemList = G_ALLOC(sizeof(PDHI_RAW_COUNTER_ITEM_BLOCK));
if (pCounter->pThisRawItemList != NULL) {
pCounter->pThisRawItemList->dwLength = sizeof(PDHI_RAW_COUNTER_ITEM_BLOCK);
pCounter->pThisRawItemList->dwItemCount = 0;
pCounter->pThisRawItemList->CStatus = LocalCStatus;
pCounter->pThisRawItemList->TimeStamp.dwLowDateTime = LODWORD(TimeStamp);
pCounter->pThisRawItemList->TimeStamp.dwHighDateTime = HIDWORD(TimeStamp);
}
else {
SetLastError(ERROR_OUTOFMEMORY);
bReturn = FALSE;
}
}
return bReturn;
}
BOOL
UpdateRealTimeMultiInstanceCounterValue(
PPDHI_COUNTER pCounter
)
{
BOOL bResult = TRUE;
DWORD LocalCStatus = 0;
if (pCounter->pThisRawItemList != NULL) {
// free old counter buffer list
if (pCounter->pLastRawItemList && pCounter->pLastRawItemList != pCounter->pThisRawItemList) {
G_FREE(pCounter->pLastRawItemList);
}
pCounter->pLastRawItemList = pCounter->pThisRawItemList;
pCounter->pThisRawItemList = NULL;
}
// don't process if the counter has not been initialized
if (!(pCounter->dwFlags & PDHIC_COUNTER_UNUSABLE)) {
// get the counter's machine status first. There's no point in
// contuning if the machine is offline
LocalCStatus = pCounter->pQMachine->lQueryStatus;
if (IsSuccessSeverity(LocalCStatus)) {
bResult = UpdateMultiInstanceCounterValue(pCounter,
pCounter->pQMachine->pPerfData,
pCounter->pQMachine->llQueryTime);
}
else {
// unable to read data from this counter's machine so use the
// query's timestamp
pCounter->pThisRawItemList = G_ALLOC(sizeof(PDHI_RAW_COUNTER_ITEM_BLOCK));
if (pCounter->pThisRawItemList != NULL) {
pCounter->pThisRawItemList->dwLength = sizeof(PDHI_RAW_COUNTER_ITEM_BLOCK);
pCounter->pThisRawItemList->dwItemCount = 0;
pCounter->pThisRawItemList->CStatus = LocalCStatus;
pCounter->pThisRawItemList->TimeStamp.dwLowDateTime = LODWORD(pCounter->pQMachine->llQueryTime);
pCounter->pThisRawItemList->TimeStamp.dwHighDateTime = HIDWORD(pCounter->pQMachine->llQueryTime);
}
else {
SetLastError(ERROR_OUTOFMEMORY);
bResult = FALSE;
}
}
}
else {
if (pCounter->dwFlags & PDHIC_COUNTER_NOT_INIT) {
// try to init is
InitCounter(pCounter);
}
}
return bResult;
}
BOOL
UpdateCounterObject(
PPDHI_COUNTER pCounter
)
{
BOOL bReturn = TRUE;
PPERF_OBJECT_TYPE pPerfObject = NULL;
PPERF_OBJECT_TYPE pLogPerfObj;
DWORD dwBufferSize = sizeof(PERF_DATA_BLOCK);
FILETIME ftGmtTime;
FILETIME ftLocTime;
if (pCounter == NULL) {
SetLastError(PDH_INVALID_ARGUMENT);
bReturn = FALSE;
}
else {
if (pCounter->pThisObject != NULL) {
if (pCounter->pLastObject && pCounter->pThisObject != pCounter->pLastObject) {
G_FREE(pCounter->pLastObject);
}
pCounter->pLastObject = pCounter->pThisObject;
pCounter->pThisObject = NULL;
}
// don't process if the counter has not been initialized
if (!(pCounter->dwFlags & PDHIC_COUNTER_UNUSABLE)) {
if (IsSuccessSeverity(pCounter->pQMachine->lQueryStatus)) {
pPerfObject = GetObjectDefByTitleIndex(pCounter->pQMachine->pPerfData,
pCounter->plCounterInfo.dwObjectId);
dwBufferSize = pCounter->pQMachine->pPerfData->HeaderLength;
dwBufferSize += ((pPerfObject == NULL) ? sizeof(PERF_OBJECT_TYPE) : pPerfObject->TotalByteLength);
pCounter->pThisObject = G_ALLOC(dwBufferSize);
if (pCounter->pThisObject != NULL) {
RtlCopyMemory(pCounter->pThisObject,
pCounter->pQMachine->pPerfData,
pCounter->pQMachine->pPerfData->HeaderLength);
pCounter->pThisObject->TotalByteLength = dwBufferSize;
pCounter->pThisObject->NumObjectTypes = 1;
SystemTimeToFileTime(& pCounter->pThisObject->SystemTime, & ftGmtTime);
FileTimeToLocalFileTime(& ftGmtTime, & ftLocTime);
FileTimeToSystemTime(& ftLocTime, & pCounter->pThisObject->SystemTime);
pLogPerfObj = (PPERF_OBJECT_TYPE)
((LPBYTE) pCounter->pThisObject + pCounter->pQMachine->pPerfData->HeaderLength);
if (pPerfObject != NULL) {
RtlCopyMemory(pLogPerfObj, pPerfObject, pPerfObject->TotalByteLength);
}
else {
ZeroMemory(pLogPerfObj, sizeof(PERF_OBJECT_TYPE));
pLogPerfObj->TotalByteLength = sizeof(PERF_OBJECT_TYPE);
pLogPerfObj->DefinitionLength = sizeof(PERF_OBJECT_TYPE);
pLogPerfObj->HeaderLength = sizeof(PERF_OBJECT_TYPE);
pLogPerfObj->ObjectNameTitleIndex = pCounter->plCounterInfo.dwObjectId;
pLogPerfObj->ObjectHelpTitleIndex = pCounter->plCounterInfo.dwObjectId + 1;
}
}
else {
SetLastError(ERROR_OUTOFMEMORY);
bReturn = FALSE;
}
}
else {
pCounter->pThisObject = G_ALLOC(sizeof(PERF_DATA_BLOCK));
if (pCounter->pThisObject == NULL) {
pCounter->pThisObject = pCounter->pLastObject;
}
else {
pCounter->pThisObject->Signature[0] = L'P';
pCounter->pThisObject->Signature[1] = L'E';
pCounter->pThisObject->Signature[2] = L'R';
pCounter->pThisObject->Signature[3] = L'F';
pCounter->pThisObject->LittleEndian = TRUE;
pCounter->pThisObject->Version = PERF_DATA_VERSION;
pCounter->pThisObject->Revision = PERF_DATA_REVISION;
pCounter->pThisObject->TotalByteLength = sizeof(PERF_DATA_BLOCK);
pCounter->pThisObject->NumObjectTypes = 1;
pCounter->pThisObject->DefaultObject = pCounter->plCounterInfo.dwObjectId;
pCounter->pThisObject->SystemNameLength = 0;
pCounter->pThisObject->SystemNameOffset = 0;
pCounter->pThisObject->HeaderLength = sizeof(PERF_DATA_BLOCK);
pCounter->pThisObject->PerfTime.QuadPart = 0;
pCounter->pThisObject->PerfFreq.QuadPart = 0;
pCounter->pThisObject->PerfTime100nSec.QuadPart = 0;
GetLocalTime(& pCounter->pThisObject->SystemTime);
}
SetLastError(PDH_CSTATUS_INVALID_DATA);
bReturn = FALSE;
}
}
else {
if (pCounter->dwFlags & PDHIC_COUNTER_NOT_INIT) {
InitCounter(pCounter);
}
pCounter->pThisObject = pCounter->pLastObject;
SetLastError(PDH_CSTATUS_INVALID_DATA);
bReturn = FALSE;
}
}
return bReturn;
}
PVOID
GetPerfCounterDataPtr(
PPERF_DATA_BLOCK pPerfData,
PPDHI_COUNTER_PATH pPath,
PPERFLIB_COUNTER pplCtr ,
DWORD dwFlags,
PPERF_OBJECT_TYPE *pPerfObjectArg,
PDWORD pStatus
)
{
PPERF_OBJECT_TYPE pPerfObject = NULL;
PPERF_INSTANCE_DEFINITION pPerfInstance = NULL;
PPERF_COUNTER_DEFINITION pPerfCounter = NULL;
DWORD dwTestValue = 0;
PVOID pData = NULL;
DWORD dwCStatus = PDH_CSTATUS_INVALID_DATA;
pPerfObject = GetObjectDefByTitleIndex(pPerfData, pplCtr->dwObjectId);
if (pPerfObject != NULL) {
if (pPerfObjectArg != NULL) * pPerfObjectArg = pPerfObject;
if (pPerfObject->NumInstances == PERF_NO_INSTANCES) {
// then just look up the counter
pPerfCounter = GetCounterDefByTitleIndex(pPerfObject,
((dwFlags & GPCDP_GET_BASE_DATA) ? TRUE : FALSE),
pplCtr->dwCounterId);
if (pPerfCounter != NULL) {
// get data and return it
pData = GetCounterDataPtr(pPerfObject, pPerfCounter);
if (pData != NULL) {
// test the pointer to see if it fails
__try {
dwTestValue = * (DWORD *) pData;
dwCStatus = PDH_CSTATUS_VALID_DATA;
}
__except (EXCEPTION_EXECUTE_HANDLER) {
pData = NULL;
dwCStatus = PDH_CSTATUS_INVALID_DATA;
}
}
else {
dwCStatus = PDH_CSTATUS_INVALID_DATA;
}
}
else {
// unable to find counter
dwCStatus = PDH_CSTATUS_NO_COUNTER;
}
}
else {
// find instance
if (pplCtr->lInstanceId == PERF_NO_UNIQUE_ID && pPath->szInstanceName != NULL) {
pPerfInstance = GetInstanceByName(pPerfData,
pPerfObject,
pPath->szInstanceName,
pPath->szParentName,
pPath->dwIndex);
if (pPerfInstance == NULL && pPath->szInstanceName[0] >= L'0' && pPath->szInstanceName[0] <= L'9') {
LONG lInstanceId = (LONG) _wtoi(pPath->szInstanceName);
pPerfInstance = GetInstanceByUniqueId(pPerfObject, lInstanceId);
}
}
else {
pPerfInstance = GetInstanceByUniqueId(pPerfObject, pplCtr->lInstanceId);
}
if (pPerfInstance != NULL) {
// instance found so find pointer to counter data
pPerfCounter = GetCounterDefByTitleIndex(pPerfObject,
((dwFlags & GPCDP_GET_BASE_DATA) ? TRUE : FALSE),
pplCtr->dwCounterId);
if (pPerfCounter != NULL) {
// counter found so get data pointer
pData = GetInstanceCounterDataPtr(pPerfObject, pPerfInstance, pPerfCounter);
if (pData != NULL) {
// test the pointer to see if it's valid
__try {
dwTestValue = * (DWORD *) pData;
dwCStatus = PDH_CSTATUS_VALID_DATA;
}
__except (EXCEPTION_EXECUTE_HANDLER) {
pData = NULL;
dwCStatus = PDH_CSTATUS_INVALID_DATA;
}
}
else {
dwCStatus = PDH_CSTATUS_INVALID_DATA;
}
}
else {
// counter not found
dwCStatus = PDH_CSTATUS_NO_COUNTER;
}
}
else {
// instance not found
dwCStatus = PDH_CSTATUS_NO_INSTANCE;
}
}
}
else {
// unable to find object
dwCStatus = PDH_CSTATUS_NO_OBJECT;
}
if (pStatus != NULL) {
__try {
* pStatus = dwCStatus;
}
__except (EXCEPTION_EXECUTE_HANDLER) {
// ?
}
}
return pData;
}