/*++ 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 #include #include #include #include #include #include #include #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; }