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.
424 lines
10 KiB
424 lines
10 KiB
/*++
|
|
|
|
Copyright (c) 1996 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
perfobj.c
|
|
|
|
Abstract:
|
|
|
|
This file implements an Performance Object that presents
|
|
System Object Performance Counters
|
|
|
|
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>
|
|
#include <perfutil.h>
|
|
#include <stdio.h>
|
|
#include "perfos.h"
|
|
#include "perfosmc.h"
|
|
#include "dataobj.h"
|
|
|
|
DWORD dwObjOpenCount = 0; // count of "Open" threads
|
|
|
|
// variables local to this module.
|
|
|
|
HANDLE hEvent = NULL;
|
|
HANDLE hMutex = NULL;
|
|
HANDLE hSemaphore = NULL;
|
|
HANDLE hSection = NULL;
|
|
|
|
|
|
DWORD APIENTRY
|
|
OpenObjectsObject (
|
|
LPWSTR lpDeviceNames
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will initialize the data structures used to pass
|
|
data back to the registry
|
|
|
|
Arguments:
|
|
|
|
Pointer to object ID of each device to be opened (PerfGen)
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
DWORD status = ERROR_SUCCESS;
|
|
LONG_PTR TempHandle = -1;
|
|
//
|
|
// Since WINLOGON is multi-threaded and will call this routine in
|
|
// order to service remote performance queries, this library
|
|
// must keep track of how many times it has been opened (i.e.
|
|
// how many threads have accessed it). the registry routines will
|
|
// limit access to the initialization routine to only one thread
|
|
// at a time so synchronization (i.e. reentrancy) should not be
|
|
// a problem
|
|
//
|
|
|
|
UNREFERENCED_PARAMETER (lpDeviceNames);
|
|
|
|
if (dwObjOpenCount == 0) {
|
|
// open Eventlog interface
|
|
|
|
hEvent = CreateEvent(NULL,TRUE,TRUE,NULL);
|
|
hSemaphore = CreateSemaphore(NULL,1,256,NULL);
|
|
hMutex = CreateMutex(NULL,FALSE,NULL);
|
|
hSection = CreateFileMapping((HANDLE)TempHandle,NULL,PAGE_READWRITE,0,8192,NULL);
|
|
}
|
|
|
|
dwObjOpenCount++; // increment OPEN counter
|
|
|
|
status = ERROR_SUCCESS; // for successful exit
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
DWORD APIENTRY
|
|
CollectObjectsObjectData (
|
|
IN OUT LPVOID *lppData,
|
|
IN OUT LPDWORD lpcbTotalBytes,
|
|
IN OUT LPDWORD lpNumObjectTypes
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine will return the data for the system objects 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
|
|
|
|
--*/
|
|
{
|
|
DWORD TotalLen; // Length of the total return block
|
|
|
|
NTSTATUS status;
|
|
|
|
POBJECTS_DATA_DEFINITION pObjectsDataDefinition;
|
|
POBJECTS_COUNTER_DATA pOCD;
|
|
|
|
POBJECT_TYPE_INFORMATION ObjectInfo;
|
|
WCHAR Buffer[ 256 ];
|
|
|
|
#ifdef DBG
|
|
STARTTIMING;
|
|
#endif
|
|
|
|
//
|
|
// Check for sufficient space for objects data
|
|
//
|
|
|
|
pObjectsDataDefinition = (OBJECTS_DATA_DEFINITION *) *lppData;
|
|
|
|
TotalLen = sizeof(OBJECTS_DATA_DEFINITION) +
|
|
sizeof (OBJECTS_COUNTER_DATA);
|
|
|
|
TotalLen = QWORD_MULTIPLE (TotalLen);
|
|
|
|
if ( *lpcbTotalBytes < TotalLen ) {
|
|
*lpcbTotalBytes = (DWORD) 0;
|
|
*lpNumObjectTypes = (DWORD) 0;
|
|
return ERROR_MORE_DATA;
|
|
}
|
|
|
|
//
|
|
// Define objects data block
|
|
//
|
|
|
|
memcpy (pObjectsDataDefinition,
|
|
&ObjectsDataDefinition,
|
|
sizeof(OBJECTS_DATA_DEFINITION));
|
|
|
|
//
|
|
// Format and collect objects data
|
|
//
|
|
|
|
pOCD = (POBJECTS_COUNTER_DATA)&pObjectsDataDefinition[1];
|
|
|
|
pOCD->CounterBlock.ByteLength = QWORD_MULTIPLE(sizeof (OBJECTS_COUNTER_DATA));
|
|
|
|
ObjectInfo = (POBJECT_TYPE_INFORMATION)Buffer;
|
|
status = NtQueryObject( NtCurrentProcess(),
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Processes = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_PROCESS_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Processes = 0;
|
|
}
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
|
|
status = NtQueryObject( NtCurrentThread(),
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Threads = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_THREAD_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Threads = 0;
|
|
}
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
|
|
status = NtQueryObject( hEvent,
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Events = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_EVENT_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Events = 0;
|
|
}
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
|
|
status = NtQueryObject( hSemaphore,
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Semaphores = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_SEMAPHORE_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Semaphores = 0;
|
|
}
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
|
|
status = NtQueryObject( hMutex,
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Mutexes = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_MUTEX_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Mutexes = 0;
|
|
}
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
|
|
status = NtQueryObject( hSection,
|
|
ObjectTypeInformation,
|
|
ObjectInfo,
|
|
sizeof( Buffer ),
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
pOCD->Sections = ObjectInfo->TotalNumberOfObjects;
|
|
} else {
|
|
if (hEventLog != NULL) {
|
|
ReportEvent (hEventLog,
|
|
EVENTLOG_WARNING_TYPE,
|
|
0,
|
|
PERFOS_UNABLE_QUERY_SECTION_OBJECT_INFO,
|
|
NULL,
|
|
0,
|
|
sizeof(DWORD),
|
|
NULL,
|
|
(LPVOID)&status);
|
|
}
|
|
|
|
pOCD->Sections = 0;
|
|
}
|
|
|
|
*lpcbTotalBytes =
|
|
pObjectsDataDefinition->ObjectsObjectType.TotalByteLength =
|
|
(DWORD) QWORD_MULTIPLE(((LPBYTE) (& pOCD[1])) -
|
|
(LPBYTE) pObjectsDataDefinition);
|
|
* lppData = (LPVOID) (((LPBYTE) pObjectsDataDefinition) + * lpcbTotalBytes);
|
|
|
|
*lpNumObjectTypes = 1;
|
|
|
|
#ifdef DBG
|
|
ENDTIMING (("PERFOBJ: %d takes %I64u ms\n", __LINE__, diff));
|
|
#endif
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
DWORD APIENTRY
|
|
CloseObjectsObject (
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine closes the open handles to the Signal Gen counters.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS
|
|
|
|
--*/
|
|
|
|
{
|
|
if (dwObjOpenCount > 0) {
|
|
dwObjOpenCount--;
|
|
if (dwObjOpenCount == 0) { // when this is the last thread...
|
|
// close stuff here
|
|
if (hEvent != NULL) {
|
|
CloseHandle(hEvent);
|
|
hEvent = NULL;
|
|
}
|
|
|
|
if (hMutex != NULL) {
|
|
CloseHandle(hMutex);
|
|
hMutex = NULL;
|
|
}
|
|
|
|
if (hSemaphore != NULL) {
|
|
CloseHandle(hSemaphore);
|
|
hSemaphore = NULL;
|
|
}
|
|
|
|
if (hSection != NULL) {
|
|
CloseHandle(hSection);
|
|
hSection = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|