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.
318 lines
10 KiB
318 lines
10 KiB
/*++
|
|
|
|
Copyright (c) 1996 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
perfimag.c
|
|
|
|
Abstract:
|
|
|
|
This file implements an Performance Object that presents
|
|
Image details performance object data
|
|
|
|
Created:
|
|
|
|
Bob Watson 22-Oct-1996
|
|
|
|
Revision History
|
|
|
|
|
|
--*/
|
|
//
|
|
// Include Files
|
|
//
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#include <windows.h>
|
|
#include <winperf.h>
|
|
#include <ntprfctr.h>
|
|
#define PERF_HEAP hLibHeap
|
|
#include <perfutil.h>
|
|
#include "perfsprc.h"
|
|
#include "perfmsg.h"
|
|
#include "dataimag.h"
|
|
|
|
DWORD APIENTRY
|
|
BuildImageObject (
|
|
IN OUT LPVOID *lppData,
|
|
IN OUT LPDWORD lpcbTotalBytes,
|
|
IN OUT LPDWORD lpNumObjectTypes,
|
|
IN BOOL bLongImageName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will return the data for the processor object
|
|
|
|
Arguments:
|
|
|
|
IN OUT LPVOID *lppData
|
|
IN: pointer to the address of the buffer to receive the completed
|
|
PerfDataBlock and subordinate structures. This routine will
|
|
append its data to the buffer starting at the point referenced
|
|
by *lppData.
|
|
OUT: points to the first byte after the data structure added by this
|
|
routine. This routine updated the value at lppdata after appending
|
|
its data.
|
|
|
|
IN OUT LPDWORD lpcbTotalBytes
|
|
IN: the address of the DWORD that tells the size in bytes of the
|
|
buffer referenced by the lppData argument
|
|
OUT: the number of bytes added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
IN OUT LPDWORD NumObjectTypes
|
|
IN: the address of the DWORD to receive the number of objects added
|
|
by this routine
|
|
OUT: the number of objects added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
IN BOOL bLongImageName
|
|
TRUE -- use the full path of the library file name in the instance
|
|
FALSE - use only the file name in the instance
|
|
|
|
Returns:
|
|
|
|
0 if successful, else Win 32 error code of failure
|
|
|
|
--*/
|
|
{
|
|
DWORD TotalLen; // Length of the total return block
|
|
|
|
PIMAGE_DATA_DEFINITION pImageDataDefinition;
|
|
PPERF_INSTANCE_DEFINITION pPerfInstanceDefinition;
|
|
PIMAGE_COUNTER_DATA pICD;
|
|
DWORD dwNumInstances;
|
|
|
|
DWORD dwImageNameLength;
|
|
|
|
DWORD dwProcessIndex;
|
|
|
|
PPROCESS_VA_INFO pThisProcess;
|
|
PMODINFO pThisImage;
|
|
|
|
dwNumInstances = 0;
|
|
|
|
pImageDataDefinition = (IMAGE_DATA_DEFINITION *) *lppData;
|
|
|
|
//
|
|
// Check for sufficient space for Image object type definition
|
|
//
|
|
|
|
TotalLen = sizeof(IMAGE_DATA_DEFINITION) +
|
|
sizeof(PERF_INSTANCE_DEFINITION) +
|
|
MAX_PROCESS_NAME_LENGTH +
|
|
sizeof(IMAGE_COUNTER_DATA);
|
|
|
|
if ( *lpcbTotalBytes < TotalLen ) {
|
|
*lpcbTotalBytes = (DWORD) 0;
|
|
*lpNumObjectTypes = (DWORD) 0;
|
|
return ERROR_MORE_DATA;
|
|
}
|
|
|
|
//
|
|
// Define Page File data block
|
|
//
|
|
|
|
memcpy(pImageDataDefinition,
|
|
&ImageDataDefinition,
|
|
sizeof(IMAGE_DATA_DEFINITION));
|
|
|
|
// update object title index if this is a Long Image object
|
|
|
|
if (bLongImageName) {
|
|
pImageDataDefinition->ImageObjectType.ObjectNameTitleIndex =
|
|
LONG_IMAGE_OBJECT_TITLE_INDEX;
|
|
pImageDataDefinition->ImageObjectType.ObjectHelpTitleIndex =
|
|
LONG_IMAGE_OBJECT_TITLE_INDEX + 1;
|
|
}
|
|
|
|
pPerfInstanceDefinition = (PERF_INSTANCE_DEFINITION *)
|
|
&pImageDataDefinition[1];
|
|
|
|
// Now load data for each Image
|
|
|
|
pThisProcess = pProcessVaInfo;
|
|
dwProcessIndex = 0;
|
|
TotalLen = sizeof(IMAGE_DATA_DEFINITION);
|
|
|
|
while (pThisProcess) {
|
|
|
|
pThisImage = pThisProcess->pMemBlockInfo;
|
|
|
|
while (pThisImage) {
|
|
|
|
dwImageNameLength = (bLongImageName ? pThisImage->LongInstanceName->Length :
|
|
pThisImage->InstanceName->Length);
|
|
dwImageNameLength += sizeof(WCHAR);
|
|
dwImageNameLength = QWORD_MULTIPLE(dwImageNameLength);
|
|
|
|
// see if this instance will fit
|
|
|
|
TotalLen += sizeof (PERF_INSTANCE_DEFINITION) +
|
|
dwImageNameLength +
|
|
// (MAX_PROCESS_NAME_LENGTH + 1) * sizeof (WCHAR) +
|
|
sizeof (DWORD) +
|
|
sizeof (IMAGE_COUNTER_DATA);
|
|
|
|
if ( *lpcbTotalBytes < TotalLen ) {
|
|
*lpcbTotalBytes = (DWORD) 0;
|
|
*lpNumObjectTypes = (DWORD) 0;
|
|
return ERROR_MORE_DATA;
|
|
}
|
|
|
|
MonBuildInstanceDefinition (pPerfInstanceDefinition,
|
|
(PVOID *) &pICD,
|
|
EXPROCESS_OBJECT_TITLE_INDEX,
|
|
dwProcessIndex,
|
|
(DWORD)-1,
|
|
(bLongImageName ? pThisImage->LongInstanceName->Buffer :
|
|
pThisImage->InstanceName->Buffer));
|
|
|
|
pICD->CounterBlock.ByteLength = QWORD_MULTIPLE(sizeof(IMAGE_COUNTER_DATA));
|
|
|
|
pICD->ImageAddrNoAccess = pThisImage->CommitVector[NOACCESS];
|
|
pICD->ImageAddrReadOnly = pThisImage->CommitVector[READONLY];
|
|
pICD->ImageAddrReadWrite = pThisImage->CommitVector[READWRITE];
|
|
pICD->ImageAddrWriteCopy = pThisImage->CommitVector[WRITECOPY];
|
|
pICD->ImageAddrExecute = pThisImage->CommitVector[EXECUTE];
|
|
pICD->ImageAddrExecuteReadOnly = pThisImage->CommitVector[EXECUTEREAD];
|
|
pICD->ImageAddrExecuteReadWrite = pThisImage->CommitVector[EXECUTEREADWRITE];
|
|
pICD->ImageAddrExecuteWriteCopy = pThisImage->CommitVector[EXECUTEWRITECOPY];
|
|
|
|
pPerfInstanceDefinition = (PERF_INSTANCE_DEFINITION *)&pICD[1];
|
|
|
|
// adjust Total Length value to reflect ACTUAL size used
|
|
|
|
TotalLen = (DWORD)((PCHAR) pPerfInstanceDefinition -
|
|
(PCHAR) pImageDataDefinition);
|
|
|
|
dwNumInstances++;
|
|
|
|
pThisImage = pThisImage->pNextModule;
|
|
}
|
|
pThisProcess = pThisProcess->pNextProcess;
|
|
dwProcessIndex++;
|
|
}
|
|
|
|
pImageDataDefinition->ImageObjectType.NumInstances += dwNumInstances;
|
|
|
|
*lpcbTotalBytes =
|
|
pImageDataDefinition->ImageObjectType.TotalByteLength =
|
|
QWORD_MULTIPLE(
|
|
(DWORD)((PCHAR) pPerfInstanceDefinition -
|
|
(PCHAR) pImageDataDefinition));
|
|
|
|
#if DBG
|
|
if (*lpcbTotalBytes > TotalLen ) {
|
|
DbgPrint ("\nPERFPROC: Image Perf Ctr. Instance Size Underestimated:");
|
|
DbgPrint ("\nPERFPROC: Estimated size: %d, Actual Size: %d", TotalLen, *lpcbTotalBytes);
|
|
}
|
|
#endif
|
|
|
|
*lppData = (LPVOID) ((PCHAR) pImageDataDefinition + *lpcbTotalBytes);
|
|
|
|
// increment number of objects in this data block
|
|
*lpNumObjectTypes = 1;
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
DWORD APIENTRY
|
|
CollectImageObjectData (
|
|
IN OUT LPVOID *lppData,
|
|
IN OUT LPDWORD lpcbTotalBytes,
|
|
IN OUT LPDWORD lpNumObjectTypes
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will return the data for the processor object
|
|
|
|
Arguments:
|
|
|
|
IN OUT LPVOID *lppData
|
|
IN: pointer to the address of the buffer to receive the completed
|
|
PerfDataBlock and subordinate structures. This routine will
|
|
append its data to the buffer starting at the point referenced
|
|
by *lppData.
|
|
OUT: points to the first byte after the data structure added by this
|
|
routine. This routine updated the value at lppdata after appending
|
|
its data.
|
|
|
|
IN OUT LPDWORD lpcbTotalBytes
|
|
IN: the address of the DWORD that tells the size in bytes of the
|
|
buffer referenced by the lppData argument
|
|
OUT: the number of bytes added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
IN OUT LPDWORD NumObjectTypes
|
|
IN: the address of the DWORD to receive the number of objects added
|
|
by this routine
|
|
OUT: the number of objects added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
Returns:
|
|
|
|
0 if successful, else Win 32 error code of failure
|
|
|
|
--*/
|
|
{
|
|
return BuildImageObject (
|
|
lppData,
|
|
lpcbTotalBytes,
|
|
lpNumObjectTypes,
|
|
FALSE); // use short names
|
|
}
|
|
|
|
DWORD APIENTRY
|
|
CollectLongImageObjectData (
|
|
IN OUT LPVOID *lppData,
|
|
IN OUT LPDWORD lpcbTotalBytes,
|
|
IN OUT LPDWORD lpNumObjectTypes
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will return the data for the processor object
|
|
|
|
Arguments:
|
|
|
|
IN OUT LPVOID *lppData
|
|
IN: pointer to the address of the buffer to receive the completed
|
|
PerfDataBlock and subordinate structures. This routine will
|
|
append its data to the buffer starting at the point referenced
|
|
by *lppData.
|
|
OUT: points to the first byte after the data structure added by this
|
|
routine. This routine updated the value at lppdata after appending
|
|
its data.
|
|
|
|
IN OUT LPDWORD lpcbTotalBytes
|
|
IN: the address of the DWORD that tells the size in bytes of the
|
|
buffer referenced by the lppData argument
|
|
OUT: the number of bytes added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
IN OUT LPDWORD NumObjectTypes
|
|
IN: the address of the DWORD to receive the number of objects added
|
|
by this routine
|
|
OUT: the number of objects added by this routine is writted to the
|
|
DWORD pointed to by this argument
|
|
|
|
Returns:
|
|
|
|
0 if successful, else Win 32 error code of failure
|
|
|
|
--*/
|
|
{
|
|
return BuildImageObject (
|
|
lppData,
|
|
lpcbTotalBytes,
|
|
lpNumObjectTypes,
|
|
TRUE); // use long names
|
|
}
|