|
|
/*++
Copyright (C) 1999-2001 Microsoft Corporation
Module Name:
RESYNC2.CPP
Abstract:
implements the listener for WDM events and events for sync-up with IdleTask
History:
ivanbrug 01-Oct-2000 changed for svchost migration
--*/
#include "precomp.h"
#include <winntsec.h>
#include <malloc.h>
#include <initguid.h>
#include "WinMgmt.h"
#include <Wmistr.h>
#include <wmium.h>
#include <wmicom.h>
#include <wmimof.h>
#include "resync2.h"
#include "wbemdelta.h" // for DeltaDredge
#include "arrtempl.h"
//
//
// this is because WDMLib is __BADLY__ DESIGNED
//
/////////////////////////////////////////////////////////////
void WINAPI EventCallbackRoutine(PWNODE_HEADER WnodeHeader, ULONG_PTR Context) { return; }
//
//
// This class listens on WDM events
//
/////////////////////////////////////////////////////////////
CWDMListener::CWDMListener(): m_dwSignature(SIG_WDMEVENTS_FREE), m_hEventAdd(NULL), m_hEventRem(NULL), m_hWaitAdd(NULL), m_hWaitRem(NULL), m_UnInited(TRUE), m_GuidAdd(GUID_MOF_RESOURCE_ADDED_NOTIFICATION), m_GuidRem(GUID_MOF_RESOURCE_REMOVED_NOTIFICATION) { }
DWORD CWDMListener::OpenAdd() { DWORD dwErr; dwErr = WmiOpenBlock(&m_GuidAdd, WMIGUID_NOTIFICATION | SYNCHRONIZE, &m_hEventAdd); if (ERROR_SUCCESS == dwErr) {
if (RegisterWaitForSingleObject(&m_hWaitAdd, m_hEventAdd, CWDMListener::EvtCallBackAdd, this, INFINITE, WT_EXECUTEONLYONCE)) { return ERROR_SUCCESS; } else { dwErr = GetLastError(); } } else { dwErr = GetLastError(); }
// if here, some errors
CloseAdd(); return dwErr; }
DWORD CWDMListener::OpenRemove() { DWORD dwRet;
dwRet = WmiOpenBlock(&m_GuidRem, WMIGUID_NOTIFICATION | SYNCHRONIZE, &m_hEventRem); if (ERROR_SUCCESS == dwRet) { if (RegisterWaitForSingleObject(&m_hWaitRem, m_hEventRem, CWDMListener::EvtCallBackRem, this, INFINITE, WT_EXECUTEONLYONCE)) { return ERROR_SUCCESS; } else { dwRet = GetLastError(); } } else { dwRet = GetLastError(); }
CloseRemove(); return dwRet;
}
DWORD CWDMListener::CloseAdd() { if (m_hWaitAdd) { UnregisterWaitEx(m_hWaitAdd,NULL); m_hWaitAdd = NULL; } if (m_hEventAdd) { WmiCloseBlock(m_hEventAdd); m_hEventAdd = NULL; } return 0; }
DWORD CWDMListener::CloseRemove() { if (m_hWaitRem) { UnregisterWaitEx(m_hWaitRem,NULL); m_hWaitRem = NULL; } if (m_hEventRem) { WmiCloseBlock(m_hEventRem); m_hEventRem = NULL; } return 0; }
VOID CWDMListener::Unregister() { CInCritSec ics(&m_cs); if (m_UnInited) return;
CloseAdd(); CloseRemove(); m_dwSignature = SIG_WDMEVENTS_FREE; m_UnInited = TRUE; }
CWDMListener::~CWDMListener() { Unregister(); }
DWORD CWDMListener::Register() { CInCritSec ics(&m_cs);
if (!m_UnInited) // prevent multiple calls
return 0; if (ERROR_SUCCESS == OpenAdd() && ERROR_SUCCESS == OpenRemove()) { m_dwSignature = SIG_WDMEVENTS_BUSY; m_UnInited = FALSE; }
return GetLastError(); }
VOID NTAPI CWDMListener::EvtCallBackAdd(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; CWDMListener * pThis = (CWDMListener *)pContext; if (NULL == pThis) return; if (SIG_WDMEVENTS_BUSY != pThis->m_dwSignature) return;
pThis->EvtCallThis(bTimerFired,Type_Added);
//
// we have process the WDM event
// since we are in the RtlpWorkerThread and
// we are registred with WT_EXECUTEONLYONCE
// REDO FROM START
//
{ CInCritSec ics(&pThis->m_cs); if (ERROR_SUCCESS == pThis->CloseAdd()) { pThis->OpenAdd(); } } }
VOID NTAPI CWDMListener::EvtCallBackRem(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; CWDMListener * pThis = (CWDMListener *)pContext; if (NULL == pThis) return; if (SIG_WDMEVENTS_BUSY != pThis->m_dwSignature) return; pThis->EvtCallThis(bTimerFired,Type_Removed); //
// we have process the WDM event
// since we are in the RtlpWorkerThread and
// we are registred with WT_EXECUTEONLYONCE
// REDO FROM START
//
{ CInCritSec ics(&pThis->m_cs); if (ERROR_SUCCESS == pThis->CloseRemove()) { pThis->OpenRemove(); } } }
VOID CWDMListener::EvtCallThis(BOOLEAN bTimerFired, int Type) { if (bTimerFired) { _DBG_ASSERT(FALSE); } else { if (m_UnInited) return; DWORD dwRet; if (Type_Added == Type) { dwRet = WmiReceiveNotifications(1,&m_hEventAdd,CWDMListener::WmiCallBack,(ULONG_PTR)this); } else if (Type_Removed == Type) { dwRet = WmiReceiveNotifications(1,&m_hEventRem,CWDMListener::WmiCallBack,(ULONG_PTR)this); } } }
VOID WINAPI CWDMListener::WmiCallBack(PWNODE_HEADER Wnode, UINT_PTR NotificationContext) { // pThis is checked in the EventCallBack
CWDMListener * pThis = (CWDMListener *)NotificationContext; #ifdef DEBUG_ADAP
WCHAR pszClsID[40]; StringFromGUID2(Wnode->Guid,pszClsID,40); DBG_PRINTFA((pBuff,"Flag %08x ProvId %08x %p GUID %S\n", Wnode->Flags,Wnode->ProviderId,(ULONG_PTR)Wnode->ClientContext,pszClsID)); if (WNODE_FLAG_ALL_DATA & Wnode->Flags) { WNODE_ALL_DATA * pAllData = (WNODE_ALL_DATA *)Wnode; DWORD i; for (i=0;i<pAllData->InstanceCount;i++) { WCHAR pTmpBuff[MAX_PATH+1]; pTmpBuff[MAX_PATH] = 0; DWORD dwSize = (pAllData->OffsetInstanceDataAndLength[i].LengthInstanceData>MAX_PATH)?MAX_PATH:pAllData->OffsetInstanceDataAndLength[i].LengthInstanceData; memcpy(pTmpBuff,(BYTE*)pAllData+pAllData->OffsetInstanceDataAndLength[i].OffsetInstanceData,dwSize); DBG_PRINTFA((pBuff,"%d - %S\n",i,pTmpBuff)); } };
#endif
#ifdef DBG
if (!HeapValidate(GetProcessHeap(),0,NULL)) { DebugBreak(); } if (!HeapValidate(CWin32DefaultArena::GetArenaHeap(),0,NULL)) { DebugBreak(); } #endif
CWMIBinMof WMIBinMof; //=============================================================================
// Note: this combo will always succeed, as all the initialize is doing is
// setting a flag to FALSE and returning S_OK
//=============================================================================
if( SUCCEEDED( WMIBinMof.Initialize(NULL,FALSE)) ) { if (WMIBinMof.BinaryMofEventChanged(Wnode)) { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"---- WMIBinMof.BinaryMofEventChanged == CHANGED ----\n")); #endif
DEBUGTRACE((LOG_WINMGMT,"WDM event && WMIBinMof.BinaryMofEventChanged == TRUE\n"));
ResyncPerf(RESYNC_TYPE_WDMEVENT); #ifdef DBG
if (!HeapValidate(GetProcessHeap(),0,NULL)) { DebugBreak(); } if (!HeapValidate(CWin32DefaultArena::GetArenaHeap(),0,NULL)) { DebugBreak(); } #endif
} else { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"---- WMIBinMof.BinaryMofEventChanged == NOT CHANGED ----\n")); #endif
} } return; }
CCounterEvts::CCounterEvts(): m_dwSignature(SIG_COUNTEEVENTS_BUSY), m_LoadCtrEvent(NULL), m_UnloadCtrEvent(NULL), m_Uninited(TRUE), m_WaitLoadCtr(NULL), m_WaitUnloadCtr(NULL), m_hWmiReverseAdapSetLodCtr(NULL), m_hWmiReverseAdapLodCtrDone(NULL), m_hPendingTasksStart(NULL), m_hPendingTasksComplete(NULL) { }
//
// allows localsystem to use the event
//
//
// SDDL: L"O:SYG:SYD:(A;;0x1f0003;;;SY)"
//
DWORD g_LocalSystemSD[] = { 0x80040001, 0x00000014, 0x00000020, 0x00000000, 0x0000002c, 0x00000101, 0x05000000, 0x00000012, 0x00000101, 0x05000000, 0x00000012, 0x00300002, 0x00000001, 0x00140000, 0x001f0003, 0x00000101, 0x05000000, 0x00000012, 0x00000000, 0x00000000 };
//
// allow administrators and localsystem to use the event
//
//
// SDDL: L"O:SYG:SYD:(A;;0x1f0003;;;SY)(A;;0x1f0003;;;BA)"
//
DWORD g_LocalSystemAdminsSD[] = { 0x80040001, 0x00000014, 0x00000020, 0x00000000, 0x0000002c, 0x00000101, 0x05000000, 0x00000012, 0x00000101, 0x05000000, 0x00000012, 0x00340002, 0x00000002, 0x00140000, 0x001f0003, 0x00000101, 0x05000000, 0x00000012, 0x00180000, 0x001f0003, 0x00000201, 0x05000000, 0x00000020, 0x00000220 };
DWORD CCounterEvts::Init() { if (!m_Uninited) return 0;
SECURITY_ATTRIBUTES sa; sa.nLength = sizeof(SECURITY_ATTRIBUTES); sa.lpSecurityDescriptor = (LPVOID)g_LocalSystemAdminsSD; sa.bInheritHandle = FALSE; m_LoadCtrEvent = CreateEvent(&sa, FALSE, FALSE,LOAD_CTR_EVENT_NAME); if (NULL == m_LoadCtrEvent) goto end_fail;
m_UnloadCtrEvent = CreateEvent(&sa, FALSE, FALSE, UNLOAD_CTR_EVENT_NAME); if (NULL == m_UnloadCtrEvent) goto end_fail;
m_hWmiReverseAdapSetLodCtr = CreateEvent(&sa,FALSE,FALSE,REVERSE_DREDGE_EVENT_NAME_SET); if (NULL == m_hWmiReverseAdapSetLodCtr) goto end_fail;
m_hWmiReverseAdapLodCtrDone = CreateEvent(&sa,FALSE,FALSE,REVERSE_DREDGE_EVENT_NAME_ACK); if (NULL == m_hWmiReverseAdapLodCtrDone) goto end_fail;
sa.lpSecurityDescriptor = (LPVOID)g_LocalSystemSD;
m_hPendingTasksStart = CreateEvent(&sa,FALSE,FALSE,PENDING_TASK_START); if (!m_hPendingTasksStart) goto end_fail;
m_hPendingTasksComplete = CreateEvent(&sa,TRUE,TRUE,PENDING_TASK_COMPLETE); if (!m_hPendingTasksComplete) goto end_fail;
m_Uninited = FALSE; return NO_ERROR; end_fail: return GetLastError(); }
VOID NTAPI CCounterEvts::EvtCallBackLoad(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; CCounterEvts * pCounter = (CCounterEvts *)pContext; if (NULL == pCounter) return; if (SIG_COUNTEEVENTS_BUSY != pCounter->m_dwSignature) return; pCounter->CallBack(bTimerFired,Type_Load); }
VOID NTAPI CCounterEvts::EvtCallBackUnload(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; CCounterEvts * pCounter = (CCounterEvts *)pContext; if (NULL == pCounter) return; if (SIG_COUNTEEVENTS_BUSY != pCounter->m_dwSignature) return; pCounter->CallBack(bTimerFired,Type_Unload); }
VOID NTAPI CCounterEvts::EvtCallBackPendingTask(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; CCounterEvts * pCounter = (CCounterEvts *)pContext; if (NULL == pCounter) return; if (SIG_COUNTEEVENTS_BUSY != pCounter->m_dwSignature) return; pCounter->CallBackPending(bTimerFired); }
VOID CCounterEvts::CallBack(BOOLEAN bTimerFired,int Type) { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"CallBack with type %d called\n",Type)); #endif
if (GLOB_IsResyncAllowed()) { DWORD dwRet = WaitForSingleObject(m_hWmiReverseAdapSetLodCtr,0); if (WAIT_OBJECT_0 == dwRet) { // this is the hack not to spawn a Delta Dredge when there is before a Reverese Dredge
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff," - SetEvent(m_hWmiReverseAdapLodCtrDone);\n")); #endif
SetEvent(m_hWmiReverseAdapLodCtrDone); } else { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff," - ResyncPerf(RESYNC_TYPE_LODCTR);\n")); #endif
ResyncPerf(RESYNC_TYPE_LODCTR); } } }
VOID CCounterEvts::CallBackPending(BOOLEAN bTimerFired) { if (GLOB_IsResyncAllowed()) { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff," - PendingTask Start set\n")); #endif
ResyncPerf(RESYNC_TYPE_PENDING_TASKS); } }
VOID RevertRegister_(HANDLE & hWaitHandle) { UnregisterWaitEx(hWaitHandle,NULL); hWaitHandle = NULL; }
DWORD CCounterEvts::Register() {
// automatic reset
if (!RegisterWaitForSingleObject(&m_WaitLoadCtr, m_LoadCtrEvent, CCounterEvts::EvtCallBackLoad, this, INFINITE, WT_EXECUTEDEFAULT)) return GetLastError(); OnDeleteIf<HANDLE &,VOID(*)(HANDLE &),RevertRegister_> UnReg1(m_WaitLoadCtr); // automatic reset
if(!RegisterWaitForSingleObject(&m_WaitUnloadCtr, m_UnloadCtrEvent, CCounterEvts::EvtCallBackUnload, this, INFINITE, WT_EXECUTEDEFAULT)) return GetLastError(); OnDeleteIf<HANDLE &,VOID(*)(HANDLE &),RevertRegister_> UnReg2(m_WaitUnloadCtr);
if (!RegisterWaitForSingleObject(&m_hWaitPendingTasksStart, m_hPendingTasksStart, CCounterEvts::EvtCallBackPendingTask, this, INFINITE, WT_EXECUTEDEFAULT)) return GetLastError(); OnDeleteIf<HANDLE &,VOID(*)(HANDLE &),RevertRegister_> UnReg3(m_hWaitPendingTasksStart);
// here everything is OK
UnReg1.dismiss(); UnReg2.dismiss(); UnReg3.dismiss(); m_dwSignature = SIG_COUNTEEVENTS_BUSY; return ERROR_SUCCESS; }
DWORD CCounterEvts::Unregister() { if (m_WaitLoadCtr) { UnregisterWaitEx(m_WaitLoadCtr,NULL); m_WaitLoadCtr = NULL; } if (m_WaitUnloadCtr) { UnregisterWaitEx(m_WaitUnloadCtr,NULL); m_WaitUnloadCtr = NULL; } if (m_hWaitPendingTasksStart) { UnregisterWaitEx(m_hWaitPendingTasksStart,NULL); m_hWaitPendingTasksStart = NULL; } m_dwSignature = SIG_COUNTEEVENTS_FREE; return 0; }
VOID CCounterEvts::UnInit() { if (!m_Uninited) return; if(m_LoadCtrEvent) { CloseHandle(m_LoadCtrEvent); m_LoadCtrEvent = NULL; } if(m_UnloadCtrEvent) { CloseHandle(m_UnloadCtrEvent); m_UnloadCtrEvent = NULL; } if (m_hWmiReverseAdapSetLodCtr) { CloseHandle(m_hWmiReverseAdapSetLodCtr); m_hWmiReverseAdapSetLodCtr = NULL; } if (m_hWmiReverseAdapLodCtrDone) { CloseHandle(m_hWmiReverseAdapLodCtrDone); m_hWmiReverseAdapLodCtrDone = NULL; } if (m_hPendingTasksStart) { CloseHandle(m_hPendingTasksStart); m_hPendingTasksStart = NULL; } if (m_hPendingTasksComplete) { CloseHandle(m_hPendingTasksComplete); m_hPendingTasksComplete = NULL; } m_Uninited = TRUE; }
CCounterEvts::~CCounterEvts() { if (!m_Uninited) UnInit(); m_dwSignature = SIG_COUNTEEVENTS_FREE; }
//
// this is the main abstraction
// the child classes will call the ResyncPerf function,
// as long as the CWbemServices write hook.
// The ResyncPerf function will grab the global monitor
// and register a Timer Callback
// the gate will be implemented in the GetAvailable function
//
//
/////////////////////////////////////////////////////////////////////
CMonitorEvents::CMonitorEvents(): m_bInit(FALSE), m_bRegistred(FALSE) { };
CMonitorEvents::~CMonitorEvents() { }
//
// We install a console control handler because we want to unregister
// the WDM event monitor before the GIUD_REMOVE events gets in
// the console applications get notified of shutdown before services and drivers
//
////////////////////////////////////////////////////////
BOOL WINAPI CMonitorEvents::MonitorCtrlHandler( DWORD dwCtrlType ) { BOOL bRet = FALSE; switch(dwCtrlType) { case CTRL_SHUTDOWN_EVENT: GLOB_GetMonitor()->m_WDMListener.Unregister(); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"WDM Handles closed\n")); #endif
bRet = TRUE; break; default: bRet = FALSE; }; return bRet; };
BOOL CMonitorEvents::Init() { if (m_bInit) return TRUE;
CInCritSec ics(&m_cs); if (m_bInit) return TRUE;
m_dwSig = 'VEOM'; m_CntsEvts.Init(); m_dwADAPDelaySec = WMIADAP_DEFAULT_DELAY; m_dwLodCtrDelaySec = WMIADAP_DEFAULT_DELAY_LODCTR; m_dwTimeToFull = WMIADAP_DEFAULT_TIMETOFULL; m_dwTimeToKillAdap = MAX_PROCESS_WAIT;
memset(&m_FileTime,0,sizeof(m_FileTime)); RegRead(); for (DWORD i=0;i<RESYNC_TYPE_MAX;i++) { m_ResyncTasks[i].dwSig = SIG_RESYNC_PERF; m_ResyncTasks[i].bFree = TRUE; m_ResyncTasks[i].pMonitor = this; m_ResyncTasks[i].hTimer = NULL; m_ResyncTasks[i].hWaitHandle = NULL; m_ResyncTasks[i].hProcess = NULL; m_ResyncTasks[i].Enabled = TRUE; }
//m_ResyncTasks[RESYNC_TYPE_LODCTR].CmdType // to be decided by DeltaDredge
m_ResyncTasks[RESYNC_TYPE_INITIAL].dwTimeDue = (m_dwADAPDelaySec)*1000; m_ResyncTasks[RESYNC_TYPE_LODCTR].CmdType = RESYNC_DELTA_THROTTLE; m_ResyncTasks[RESYNC_TYPE_LODCTR].dwTimeDue = (m_dwLodCtrDelaySec)*1000;
// //RESYNC_TYPE_CLASSCREATION is the same
m_ResyncTasks[RESYNC_TYPE_WDMEVENT].CmdType = RESYNC_RADAPD_THROTTLE; m_ResyncTasks[RESYNC_TYPE_WDMEVENT].dwTimeDue = (m_dwLodCtrDelaySec)*1000;
m_ResyncTasks[RESYNC_TYPE_PENDING_TASKS].CmdType = RESYNC_FULL_RADAPD_NOTHROTTLE; m_ResyncTasks[RESYNC_TYPE_PENDING_TASKS].dwTimeDue = 500; // hard coded
//
// set up the console handler
//
SetConsoleCtrlHandler( MonitorCtrlHandler, TRUE );
//
// let's asses some initial state for the IdleTask business
//
m_OutStandingProcesses = 0; m_bFullReverseNeeded = FALSE;
m_bInit = TRUE;
return TRUE; };
BOOL CMonitorEvents::Uninit() { if (!m_bInit) return TRUE; CInCritSec ics(&m_cs); if (!m_bInit) return TRUE;
for (DWORD i=0;i<RESYNC_TYPE_MAX;i++) { if (m_ResyncTasks[i].hTimer) { DeleteTimerQueueTimer(NULL,m_ResyncTasks[i].hTimer,NULL); m_ResyncTasks[i].hTimer = NULL; } if (m_ResyncTasks[i].hWaitHandle) { UnregisterWaitEx(m_ResyncTasks[i].hWaitHandle,NULL); m_ResyncTasks[i].hWaitHandle = NULL; } if (m_ResyncTasks[i].hProcess) { CloseHandle(m_ResyncTasks[i].hProcess); m_ResyncTasks[i].hProcess = NULL; } m_ResyncTasks[i].dwSig = (DWORD)'eerf'; }
m_CntsEvts.UnInit();
//
// tear-down the console handler
//
SetConsoleCtrlHandler( MonitorCtrlHandler, FALSE );
m_bInit = FALSE; m_dwSig = 'veom';
return TRUE; };
//
//
// called in the running/continue
//
/////////////
DWORD CMonitorEvents::Register() { m_CntsEvts.Register(); m_WDMListener.Register();
m_bRegistred = TRUE; return 0; };
//
//
// called in the pause/stop
//
//////////////////////////////////////////////////////////
DWORD CMonitorEvents::Unregister(BOOL bIsSystemShutDown) {
m_bRegistred = FALSE;
if (!bIsSystemShutDown) { m_CntsEvts.Unregister(); m_WDMListener.Unregister(); } return 0; };
//
//
//
/////////////////////////////////////////////////////////
VOID CMonitorEvents::RegRead() { // Read the initialization information
LONG lRet; HKEY hKey; DWORD dwTemp;
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("Software\\Microsoft\\WBEM\\CIMOM"), NULL, KEY_READ|KEY_WRITE, &hKey); if (ERROR_SUCCESS == lRet) { OnDelete<HKEY,LONG(*)(HKEY),RegCloseKey> cm(hKey); DWORD dwType; DWORD dwSize = sizeof(DWORD); lRet = RegQueryValueEx(hKey, TEXT("ADAPDelay"), NULL, &dwType, (BYTE *)&m_dwADAPDelaySec, &dwSize);
if (ERROR_SUCCESS == lRet && REG_DWORD == dwType) { //This is what we want
} else if ( ERROR_FILE_NOT_FOUND == lRet ) { dwTemp = WMIADAP_DEFAULT_DELAY; RegSetValueEx(hKey, TEXT("ADAPDelay"), NULL, REG_DWORD, (BYTE *)&dwTemp, sizeof(DWORD)); } else { // Error
ERRORTRACE( ( LOG_WINMGMT, "ResyncPerf experienced an error while attempting to read the WMIADAPDelay value in the CIMOM subkey. Continuing using a default value.\n" ) ); }
dwSize = sizeof(DWORD); lRet = RegQueryValueEx(hKey, TEXT("LodCtrDelay"), NULL, &dwType, (BYTE *)&m_dwLodCtrDelaySec, &dwSize);
if (ERROR_SUCCESS == lRet && REG_DWORD == dwType) { //This is what we want
} else if ( ERROR_FILE_NOT_FOUND == lRet ) { dwTemp = WMIADAP_DEFAULT_DELAY_LODCTR; RegSetValueEx(hKey, TEXT("LodCtrDelay"), NULL, REG_DWORD, (BYTE *)&dwTemp, sizeof(DWORD)); } else { // Error
ERRORTRACE( ( LOG_WINMGMT, "ResyncPerf experienced an error while attempting to read the WMIADAPDelay value in the CIMOM subkey. Continuing using a default value.\n" ) ); }
dwSize = sizeof(DWORD); lRet = RegQueryValueEx(hKey, ADAP_TIME_TO_FULL, NULL, &dwType, (BYTE *)&m_dwTimeToFull, &dwSize);
if (ERROR_SUCCESS == lRet && REG_DWORD == dwType) { //This is what we want
} else if ( ERROR_FILE_NOT_FOUND == lRet ) { dwTemp = WMIADAP_DEFAULT_TIMETOFULL; RegSetValueEx(hKey, ADAP_TIME_TO_FULL, NULL, REG_DWORD, (BYTE *)&dwTemp, sizeof(DWORD)); } else { // Error
ERRORTRACE( ( LOG_WINMGMT, "ResyncPerf experienced an error while attempting to read the WMIADAPDelay value in the CIMOM subkey. Continuing using a default value.\n" ) ); }
dwSize = sizeof(DWORD); lRet = RegQueryValueEx(hKey, ADAP_TIME_TO_KILL_ADAP, NULL, &dwType, (BYTE *)&m_dwTimeToKillAdap, &dwSize);
if (ERROR_SUCCESS == lRet && REG_DWORD == dwType) { //This is what we want
} else if ( ERROR_FILE_NOT_FOUND == lRet ) { dwTemp = MAX_PROCESS_WAIT; RegSetValueEx(hKey, ADAP_TIME_TO_KILL_ADAP, NULL, REG_DWORD, (BYTE *)&dwTemp, sizeof(DWORD)); } else { // Error
ERRORTRACE( ( LOG_WINMGMT, "ResyncPerf experienced an error while attempting to read the %S value in the CIMOM subkey. Continuing using a default value.\n",ADAP_TIME_TO_KILL_ADAP)); }
//ADAP_TIMESTAMP_FULL
dwSize = sizeof(FILETIME); lRet = RegQueryValueEx(hKey, ADAP_TIMESTAMP_FULL, NULL, &dwType, (BYTE *)&m_FileTime, &dwSize);
} else { // Error
ERRORTRACE( ( LOG_WINMGMT, "ResyncPerf could not open the CIMOM subkey to read initialization data. Continuing using a default value.\n" ) );
}
}
//
//
//
////////////////////////////////////////////////////////
ResyncPerfTask * CMonitorEvents::GetAvailable(DWORD dwReason) { ResyncPerfTask * pPerf = NULL; CInCritSec ics(&m_cs);
if (m_ResyncTasks[dwReason].bFree) { m_ResyncTasks[dwReason].bFree = FALSE; m_ResyncTasks[dwReason].Type = dwReason; pPerf = &m_ResyncTasks[dwReason]; } return pPerf; }
TCHAR * g_Strings[] = { TEXT("/F /T"), // FULL Throttle
TEXT("/D /T"), // DELTA Throttle
TEXT("/R /T"), // REVERSE_ADAPTER Throttle
TEXT("/F /R /T"), // FULL REVERSE_ADAPTER Throttle
TEXT("/D /R /T"), // DELTA REVERSE_ADAPTER Throttle
TEXT("/F /R") // FULL REVERSE no Throttle
};
void inline DoUnThrottleDredges() { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"DoUnThrottleDredges\n")); #endif
RegSetDWORD(HKEY_LOCAL_MACHINE,HOME_REG_PATH,DO_THROTTLE,0); return; }
void inline DoThrottleDredges() { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"DoThrottleDredges\n")); #endif
RegSetDWORD(HKEY_LOCAL_MACHINE,HOME_REG_PATH,DO_THROTTLE,1); return; }
BOOL CMonitorEvents::CreateProcess_(TCHAR * pCmdLine, CMonitorEvents * pMonitor, ResyncPerfTask * pPerf) { BOOL bRes = FALSE; STARTUPINFO si; PROCESS_INFORMATION ProcInfo; memset(&si,0,sizeof(STARTUPINFO)); si.cb = sizeof(STARTUPINFO); si.dwFlags = STARTF_FORCEOFFFEEDBACK;
// Get the appropriate cmdline and attach the proper command line switches
LPTSTR pWriteableBuff = GetWMIADAPCmdLine( 64 ); CVectorDeleteMe<TCHAR> vdm( pWriteableBuff );
if ( NULL == pWriteableBuff ) { ERRORTRACE((LOG_WINMGMT,"Memory Allocation error spawning dredger!\n")); pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock(); return bRes; }
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"Creating process: %S\n",pCmdLine)); #endif
DEBUGTRACE((LOG_WINMGMT,"Creating process: %S\n",pCmdLine));
bRes = CreateProcess(pWriteableBuff, pCmdLine, NULL, NULL, FALSE, CREATE_NO_WINDOW, NULL, NULL, &si, &ProcInfo); if (bRes) { CloseHandle(ProcInfo.hThread);
pPerf->hProcess = ProcInfo.hProcess;
if (RegisterWaitForSingleObject(&pPerf->hWaitHandle, pPerf->hProcess, CMonitorEvents::EventCallBack, pPerf, pMonitor->m_dwTimeToKillAdap, WT_EXECUTEONLYONCE|WT_EXECUTEINWAITTHREAD)) { //
// we don't need to free the slot,
// because the event callback will do that
//
} else { DEBUGTRACE((LOG_WINMGMT,"Unable to schedule WmiADAP process termination handler: err %d\n",GetLastError())); CloseHandle(pPerf->hProcess); pPerf->hProcess = NULL; pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock(); } } else { ERRORTRACE((LOG_WINMGMT,"CreatProcess %S err: %d\n",pWriteableBuff,GetLastError())); pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock(); } return bRes; }
VOID NTAPI CMonitorEvents::EventCallBack(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) return; ResyncPerfTask * pPerf = (ResyncPerfTask *)pContext;
if (!pPerf || (SIG_RESYNC_PERF != pPerf->dwSig)) return; CMonitorEvents * pMonitor = pPerf->pMonitor; HANDLE hProcess = pPerf->hProcess; if(bTimerFired) { //
// The LONG time-out for our process has expired
// Kill The Process
//
TerminateProcess(pPerf->hProcess,0); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"WmiADAP did not finish within %d msec\n",pMonitor->m_dwTimeToKillAdap)); #endif
ERRORTRACE((LOG_WINMGMT,"the ResyncTask of type %d timed-out and has been killed\n",pPerf->Type)); } else { //
// the handle has been signaled, meaning that
// the process exited normally
//
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"ResyncPerf for task %d completed\n",pPerf->Type)); #endif
}
CloseHandle(pPerf->hProcess); //
// if there was a call to ProcessIdleTasks
// if we were forced to unthrottle the running tasks
// revert back
//
if (RESYNC_TYPE_PENDING_TASKS == pPerf->Type) { pMonitor->m_bFullReverseNeeded = FALSE; DoThrottleDredges(); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"Setting the WMI_ProcessIdleTasksComplete\n")); #endif
if (GLOB_GetMonitor()->IsRegistred()) SetEvent(GLOB_GetMonitor()->GetTaskCompleteEvent()); } else // a process has exited or it has been terminated
{ LONG nProc = InterlockedDecrement(&pMonitor->m_OutStandingProcesses); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"(-) Outstanding Tasks %d\n",pMonitor->m_OutStandingProcesses)); #endif
if (0 == nProc && pMonitor->m_bFullReverseNeeded) { // Create Here the process
CMonitorEvents * pMonitor = GLOB_GetMonitor(); ResyncPerfTask * pPerfTask = pMonitor->GetAvailable(RESYNC_TYPE_PENDING_TASKS); if (pPerfTask) { TCHAR pCmdLine[64]; StringCchCopy(pCmdLine,64,TEXT("wmiadap.exe ")); StringCchCat(pCmdLine,64,g_Strings[pPerfTask->CmdType]); CMonitorEvents::CreateProcess_(pCmdLine,pMonitor,pPerfTask); } else { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"GetAvailable(RESYNC_TYPE_PENDING_TASKS) returned NULL\n")); #endif
} } } pPerf->hProcess = NULL; pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock();
UnregisterWaitEx(pPerf->hWaitHandle,NULL); pPerf->hWaitHandle = NULL;
}
//
// This is the main function executed when
// -1- the LoadCtr/UnlaodCtr events are set
// -2- the Unthrottle events is set by Schedule via the API exposed in Advapi32
// -3- it runs the 4-minutes-after-boot dredge
// This function also take care of Reverse-Adapter dredges, if needed
//
////////////////////////////////////////////////////////////////
VOID NTAPI CMonitorEvents::TimerCallBack(VOID * pContext,BOOLEAN bTimerFired) { if (!GLOB_Monitor_IsRegistred()) { return; }
if(bTimerFired) { ResyncPerfTask * pPerf = (ResyncPerfTask *)pContext; CMonitorEvents * pMonitor = pPerf->pMonitor; BOOL bFreeSlot = FALSE;
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"TIMER: Command Type %x\n",pPerf->Type)); #endif
// check if the Delta Task was disabled on the Fly
if (!pPerf->Enabled) { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"Task %d was disabled on the fly\n",pPerf->Type)); #endif
bFreeSlot = TRUE; goto unregister_timer; }
BOOL bDoSomething = TRUE; BOOL RunDeltaLogic = TRUE; BOOL AddReverseAdapter = FALSE; BOOL WDMTriggeredReverseAdapter = FALSE; BOOL bDoFullSystemReverseHere = FALSE;
if (RESYNC_TYPE_PENDING_TASKS == pPerf->Type) { pMonitor->Lock(); // here disable tasks that are on the wait list
for (DWORD i=0;i<RESYNC_TYPE_MAX;i++) { if (RESYNC_TYPE_PENDING_TASKS != i) { if (pMonitor->m_ResyncTasks[i].hTimer) { #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"Disabling the pending task %d\n",i)); #endif
pMonitor->m_ResyncTasks[i].Enabled = FALSE; } } } pMonitor->Unlock(); // now check if the are processes running
DoUnThrottleDredges(); if (pMonitor->m_OutStandingProcesses) { pMonitor->m_bFullReverseNeeded = TRUE; // no need to CreateProcess, the last outstanding process will do that
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"OutStandingProcess, no CreateProcessHere\n")); #endif
bFreeSlot = TRUE; goto unregister_timer; } else // no processes outstanding, create the process now
{ bDoFullSystemReverseHere = TRUE; #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"GOTO CreateProcess\n")); #endif
goto createprocess_label; } }
if (RESYNC_TYPE_INITIAL == pPerf->Type ) { // check if the Reverse Adapters need a Delta
LONG lRet; HKEY hKey; DWORD dwTemp;
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, WBEM_REG_REVERSE_KEY, NULL, KEY_READ, &hKey); if (ERROR_SUCCESS == lRet) { DWORD dwType; DWORD dwSize = sizeof(DWORD); DWORD dwVal; lRet = RegQueryValueEx(hKey, WBEM_REG_REVERSE_VALUE, NULL, &dwType, (BYTE *)&dwVal, &dwSize); // if the key is there, is NULL and it is of the right type
// OR if the key is not there
if( ERROR_SUCCESS == lRet && REG_DWORD == dwType && dwVal ) { AddReverseAdapter = TRUE; #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"\"Performance Refresh\" key set to %d\n",dwVal)); #endif
DEBUGTRACE((LOG_WINMGMT,"\"Performance Refresh\" key set to %d\n",dwVal)); } RegCloseKey(hKey); }
// check the WDM stuff
if (!AddReverseAdapter) { #ifdef DBG
if (!HeapValidate(GetProcessHeap(),0,NULL)) { DebugBreak(); } if (!HeapValidate(CWin32DefaultArena::GetArenaHeap(),0,NULL)) { DebugBreak(); } #endif
CWMIBinMof BinMof; //=============================================================================
// Note: this combo will always succeed, as all the initialize is doing is
// setting a flag to FALSE and returning S_OK
//=============================================================================
if( SUCCEEDED( BinMof.Initialize(NULL,FALSE) ) ) { WDMTriggeredReverseAdapter = BinMof.BinaryMofsHaveChanged(); if (WDMTriggeredReverseAdapter) { // override the previous decition
AddReverseAdapter = TRUE; #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"BinaryMofs DO HAVE changed\n")); #endif
DEBUGTRACE((LOG_WINMGMT,"CWMIBinMof.BinaryMofsHaveChanged == TRUE\n")); }
#ifdef DBG
if (!HeapValidate(GetProcessHeap(),0,NULL)) { DebugBreak(); } if (!HeapValidate(CWin32DefaultArena::GetArenaHeap(),0,NULL)) { DebugBreak(); } #endif
} } // overrides delta with full, if the case
if (WMIADAP_DEFAULT_TIMETOFULL == pMonitor->GetFullTime()) { // no override
} else // read timestamp and decide
{ ULARGE_INTEGER li; li.LowPart = pMonitor->GetTimeStamp().dwLowDateTime; li.HighPart = pMonitor->GetTimeStamp().dwHighDateTime; __int64 Seconds = pMonitor->GetFullTime(); Seconds *= 10000000; // number of 100ns units in 1 second
ULARGE_INTEGER liNow; GetSystemTimeAsFileTime((FILETIME *)&liNow); if ((li.QuadPart + Seconds) < liNow.QuadPart) { pPerf->CmdType = RESYNC_FULL_THROTTLE; RunDeltaLogic = FALSE; } } } // end if command type initial
if ((RESYNC_TYPE_INITIAL == pPerf->Type) && RunDeltaLogic) { DWORD ret = DeltaDredge2(0,NULL); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"DeltaDredge2 ret %d\n",ret)); #endif
switch(ret) { case FULL_DREDGE: pPerf->CmdType = RESYNC_FULL_THROTTLE; break; case PARTIAL_DREDGE: pPerf->CmdType = RESYNC_DELTA_THROTTLE; break; case NO_DREDGE: //
// this is the case where we do nothing
DEBUGTRACE((LOG_WINMGMT,"No Dredge to run\n")); //
bDoSomething = FALSE; break; default: //
// never here
//
break; }
#ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"DeltaDredge2() ret = %d, bDoSomething = %d \n",ret,bDoSomething)); #endif
DEBUGTRACE((LOG_WINMGMT,"DeltaDredge2() ret = %d, bDoSomething = %d \n",ret,bDoSomething)); }
if (bDoSomething || AddReverseAdapter) { createprocess_label: TCHAR pCmdLine[64]; StringCchCopy(pCmdLine,64,TEXT("wmiadap.exe ")); if (bDoFullSystemReverseHere) { StringCchCat(pCmdLine,64,g_Strings[pPerf->CmdType]); } else { if (bDoSomething && AddReverseAdapter) { StringCchCat(pCmdLine,64,g_Strings[pPerf->CmdType]); StringCchCat(pCmdLine,64,TEXT(" /R")); } if (bDoSomething && !AddReverseAdapter) { StringCchCat(pCmdLine,64,g_Strings[pPerf->CmdType]); } if (!bDoSomething && AddReverseAdapter) { StringCchCat(pCmdLine,64,g_Strings[RESYNC_RADAPD_THROTTLE]); } } CMonitorEvents::CreateProcess_(pCmdLine,pMonitor,pPerf);
if (GLOB_GetMonitor()->IsRegistred()) { if (!bDoFullSystemReverseHere) { InterlockedIncrement(&(GLOB_GetMonitor()->m_OutStandingProcesses)); #ifdef DEBUG_ADAP
DBG_PRINTFA((pBuff,"(+) Outstanding Tasks %d\n",GLOB_GetMonitor()->m_OutStandingProcesses)); #endif
} } } else { pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock(); }
unregister_timer: if (bFreeSlot) { pMonitor->Lock(); pPerf->bFree = TRUE; pMonitor->Unlock(); } DeleteTimerQueueTimer(NULL,pPerf->hTimer,NULL); pPerf->hTimer = NULL; pPerf->Enabled = TRUE; } else { // never here
_DBG_ASSERT(FALSE); } }
//
//
//
///////////////////////////////////////////////
DWORD ResyncPerf(DWORD dwReason) {
if(!GLOB_IsResyncAllowed()) { ERRORTRACE((LOG_WINMGMT,"ResyncPerf disable g_fSetup or g_fDoResync\n")); return 0; } ResyncPerfTask * pPerfTask = GLOB_GetMonitor()->GetAvailable(dwReason);
if (pPerfTask) { // here you have the slot for execution
// tell Reverse_Adapter that it's scheduled
if (RESYNC_TYPE_WDMEVENT == dwReason || RESYNC_TYPE_CLASSCREATION == dwReason) {
LONG lRet; HKEY hKey; DWORD dwTemp;
lRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, WBEM_REG_REVERSE_KEY, NULL, KEY_READ|KEY_WRITE, &hKey); if (ERROR_SUCCESS == lRet) { DWORD dwType; DWORD dwSize = sizeof(DWORD); DWORD dwVal; lRet = RegQueryValueEx(hKey, WBEM_REG_REVERSE_VALUE, NULL, &dwType, (BYTE *)&dwVal, &dwSize); // if the key is there, is NULL and it is of the right type
// OR if the key is not there
if((ERROR_SUCCESS == lRet && REG_DWORD == dwType && 0 == dwVal) || (ERROR_FILE_NOT_FOUND == lRet)) { dwVal = 1; RegSetValueEx(hKey, WBEM_REG_REVERSE_VALUE, 0, REG_DWORD, (BYTE *)&dwVal, sizeof(DWORD)); } RegCloseKey(hKey); } }; if (CreateTimerQueueTimer(&pPerfTask->hTimer, NULL, CMonitorEvents::TimerCallBack, pPerfTask, pPerfTask->dwTimeDue, 0, WT_EXECUTEONLYONCE|WT_EXECUTELONGFUNCTION)) { return 0; } else { // ERRORTRACE
return GetLastError(); } } else { // no slot availables
return ERROR_BUSY; } }
//
//
// This function is called by the Hook installed in wbemcore
// that monitors class creation
//
///////////////////////////////////////////
DWORD __stdcall DredgeRA(VOID * pReserved) { //DBG_PRINTFA((pBuff,"Classes\n"));
return ResyncPerf(RESYNC_TYPE_CLASSCREATION); };
|