windows-nt/Source/XPSP1/NT/admin/wmi/wbem/winmgmt/ess3/evtools.cpp

390 lines
8.4 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
//******************************************************************************
//
// EVTOOLS.CPP
//
// Copyright (C) 1996-1999 Microsoft Corporation
//
//******************************************************************************
#include "precomp.h"
#include <stdio.h>
#include <wbemcomn.h>
#include <evtools.h>
#include <cominit.h>
//*****************************************************************************
//
// Implementation:
//
// This class contains a queue of event handles. The top() one is always
// signalled --- it corresponds to the turn that is currently allowed to
// execute. Handles are added to the queue in GetInLine. Handles are removed
// from the queue in EndTurn, at which time the next handle in line is
// signalled.
//
// m_pCurrentTurn contains the pointer to the turn that has returned from from
// WaitForTurn, but not from EndTurn. Note, that m_pCurrentTurn may be empty
// even if a turn is scheduled to execute --- there is a gap between the time
// when a turn's handle is signalled, and its WaitForTurn succeeds.
//
// However, it is guranteed that by the time a legitimate call to EndTurn is
// made, m_pCurrentTurn contains the pointer to the turn in question, at which
// time it is reset to NULL.
//
// An additional optimization is that if the same thread calls GetInLine
// multiple times concurrently, we simply
//
//*****************************************************************************
CExecLine::CExecLine() : m_pCurrentTurn(NULL), m_pLastIssuedTurn(NULL),
m_dwLastIssuedTurnThreadId(0)
{
}
CExecLine::~CExecLine()
{
// No need to do anything --- handles are closed by tokens
// =======================================================
}
CExecLine::CTurn* CExecLine::GetInLine()
{
CInCritSec ics(&m_cs);
//
// First, check if this thread was the guy who got the last turn
//
if(m_pLastIssuedTurn && m_dwLastIssuedTurnThreadId == GetCurrentThreadId())
{
//
// It is us --- just reuse that turn and be done with it
//
m_pLastIssuedTurn->AddRef();
return m_pLastIssuedTurn;
}
//
// Allocate a new turn
//
CTurn* pTurn = new CTurn;
if(pTurn == NULL)
return NULL;
if(!pTurn->Init())
{
ERRORTRACE((LOG_ESS, "Unable to initialize turn: %d\n",
GetLastError()));
delete pTurn;
return NULL;
}
//
// Add its event to the queue
//
try
{
m_qTurns.push_back(pTurn);
}
catch( CX_MemoryException )
{
return NULL;
}
//
// Check if we are currently executing
//
if(m_qTurns.size() == 1)
{
//
// Release first in line
//
if(!ReleaseFirst())
{
//
// Something went horribly wrong
//
ERRORTRACE((LOG_ESS, "Unable to release first turn: %d\n",
GetLastError()));
m_qTurns.pop_front();
delete pTurn;
return NULL;
}
}
//
// Mark ourselves as the last issued turn
//
m_pLastIssuedTurn = pTurn;
m_dwLastIssuedTurnThreadId = GetCurrentThreadId();
return pTurn;
}
// assumes in m_cs and m_qTurns is not empty
BOOL CExecLine::ReleaseFirst()
{
return SetEvent(m_qTurns.front()->GetEvent());
}
DWORD CExecLine::WaitForTurn(CTurn* pTurn, DWORD dwTimeout)
{
// Wait for the turn event to be signaled
// ======================================
DWORD dwRes = WbemWaitForSingleObject(pTurn->GetEvent(), dwTimeout);
{
CInCritSec ics(&m_cs);
if(dwRes == WAIT_OBJECT_0)
{
// Got it --- record this turn as executing
// ========================================
m_pCurrentTurn = pTurn;
}
}
return dwRes;
}
BOOL CExecLine::EndTurn(CTurn* pTurn)
{
CInCritSec ics(&m_cs);
// Check that this is the running turn
// ===================================
if(pTurn != m_pCurrentTurn)
return FALSE;
m_pCurrentTurn = NULL;
// Delete the turn object
// ======================
if(pTurn->Release() > 0)
{
//
// This is not the last incarnation of this turn. No further action is
// required, as the same thread will call Wait and End again
//
return TRUE;
}
//
// Remove the last issued turn if this is it
//
if(m_pLastIssuedTurn == pTurn)
{
m_pLastIssuedTurn = NULL;
m_dwLastIssuedTurnThreadId = 0;
}
//
// Pop its handle off the queue
//
m_qTurns.pop_front();
//
// Signal the next one
//
if(!m_qTurns.empty())
return ReleaseFirst();
else
return TRUE;
}
BOOL CExecLine::DiscardTurn(ACQUIRE CTurn* pTurn)
{
CInCritSec ics(&m_cs);
if(pTurn->Release() > 0)
{
//
// This is not the last incarnation of this turn. No further action is
// required
//
return TRUE;
}
else
{
// if pTurn is going away, we'd better make sure that we don't try to reuse it...
// HMH 4/12/99, RAID 48420
if (pTurn == m_pLastIssuedTurn)
m_pLastIssuedTurn = NULL;
}
//
// Search for it in the queue
//
BOOL bFound = FALSE;
for(TTurnIterator it = m_qTurns.begin(); it != m_qTurns.end();)
{
if((*it) == pTurn)
{
//
// erase it and continue
//
it = m_qTurns.erase(it);
bFound = TRUE;
break;
}
else
it++;
}
if(!bFound)
return FALSE;
if(it == m_qTurns.begin() && it != m_qTurns.end())
{
//
// Discarded turn was actually active --- signal the next one
//
ReleaseFirst();
}
return TRUE;
}
CExecLine::CTurn::CTurn() : m_hEvent(NULL), m_lRef(1)
{
}
BOOL CExecLine::CTurn::Init()
{
m_dwOwningThreadId = GetCurrentThreadId();
m_hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
return (m_hEvent != NULL);
}
long CExecLine::CTurn::AddRef()
{
return InterlockedIncrement(&m_lRef);
}
long CExecLine::CTurn::Release()
{
long l = InterlockedDecrement(&m_lRef);
if(l == 0)
delete this;
return l;
}
CExecLine::CTurn::~CTurn()
{
if(m_hEvent)
CloseHandle(m_hEvent);
}
void* CExecLine::CTurn::operator new(size_t nSize)
{
return CTemporaryHeap::Alloc(nSize);
}
void CExecLine::CTurn::operator delete(void* p)
{
CTemporaryHeap::Free(p, sizeof(CExecLine::CTurn));
}
INTERNAL const SECURITY_DESCRIPTOR* GetSD( IWbemEvent* pEvent,ULONG* pcEvent )
{
static long mstatic_lSdHandle = -1;
HRESULT hres;
//
// Get the SD from the event
//
_IWmiObject* pEventEx = NULL;
pEvent->QueryInterface(IID__IWmiObject, (void**)&pEventEx);
CReleaseMe rm1(pEventEx);
if(mstatic_lSdHandle == -1)
{
pEventEx->GetPropertyHandleEx(SECURITY_DESCRIPTOR_PROPNAME, 0, NULL,
&mstatic_lSdHandle);
}
const SECURITY_DESCRIPTOR* pSD = NULL;
hres = pEventEx->GetArrayPropAddrByHandle(mstatic_lSdHandle, 0, pcEvent,
(void**)&pSD);
if(FAILED(hres) || pSD == NULL)
return NULL;
else
return pSD;
}
HRESULT SetSD(IWbemEvent* pEvent, const SECURITY_DESCRIPTOR* pSD)
{
HRESULT hres;
VARIANT vSd;
VariantInit(&vSd);
CClearMe cm1(&vSd);
long lLength = GetSecurityDescriptorLength((SECURITY_DESCRIPTOR*)pSD);
V_VT(&vSd) = VT_ARRAY | VT_UI1;
SAFEARRAYBOUND sab;
sab.cElements = lLength;
sab.lLbound = 0;
V_ARRAY(&vSd) = SafeArrayCreate(VT_UI1, 1, &sab);
if(V_ARRAY(&vSd) == NULL)
return WBEM_E_OUT_OF_MEMORY;
BYTE* abSd = NULL;
hres = SafeArrayAccessData(V_ARRAY(&vSd), (void**)&abSd);
if(FAILED(hres))
return WBEM_E_OUT_OF_MEMORY;
CUnaccessMe uam(V_ARRAY(&vSd));
memcpy(abSd, pSD, lLength);
// Put it into the consumer
// ========================
hres = pEvent->Put(SECURITY_DESCRIPTOR_PROPNAME, 0, &vSd, 0);
return hres;
}
CTempMemoryManager CTemporaryHeap::mstatic_Manager;
/*
CTemporaryHeap::CHeapHandle CTemporaryHeap::mstatic_HeapHandle;
CTemporaryHeap::CHeapHandle::CHeapHandle()
{
m_hHeap = HeapCreate(0, 0, 0);
}
CTemporaryHeap::CHeapHandle::~CHeapHandle()
{
HeapDestroy(m_hHeap);
}
*/