/*++ Copyright (C) 1996-2001 Microsoft Corporation Module Name: ProvResv.cpp Abstract: History: --*/ #include "PreComp.h" #include #include "Globals.h" #include "CGlobals.h" #include "ProvInSk.h" #include "guids.h" #include enum { CALLED = 0, NOTCALLED = -1}; /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ CServerObject_ProviderInitSink :: CServerObject_ProviderInitSink ( SECURITY_DESCRIPTOR *a_SecurityDescriptor ) : m_ReferenceCount ( 0 ) , m_Event ( NULL ) , m_StatusCalled ( FALSE ) , m_Result ( S_OK ) , m_SecurityDescriptor ( NULL ) { } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ CServerObject_ProviderInitSink :: ~CServerObject_ProviderInitSink () { if ( m_Event ) { CloseHandle ( m_Event ) ; } if ( m_SecurityDescriptor ) { delete [] ( BYTE * ) m_SecurityDescriptor ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ HRESULT CServerObject_ProviderInitSink :: SinkInitialize ( SECURITY_DESCRIPTOR *a_SecurityDescriptor ) { HRESULT t_Result = S_OK ; m_Event = OS::CreateEvent ( NULL , FALSE , FALSE , NULL ) ; if ( m_Event ) { if ( a_SecurityDescriptor ) { m_SecurityDescriptor = ( SECURITY_DESCRIPTOR * ) new BYTE [ GetSecurityDescriptorLength ( a_SecurityDescriptor ) ] ; if ( m_SecurityDescriptor ) { CopyMemory ( m_SecurityDescriptor , a_SecurityDescriptor , GetSecurityDescriptorLength ( a_SecurityDescriptor ) ) ; } else { t_Result = WBEM_E_OUT_OF_MEMORY ; } } } else { t_Result = WBEM_E_OUT_OF_MEMORY ; } return t_Result ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP CServerObject_ProviderInitSink :: QueryInterface ( REFIID iid , LPVOID FAR *iplpv ) { *iplpv = NULL ; if ( iid == IID_IUnknown ) { *iplpv = ( LPVOID ) this ; } else if ( iid == IID_IWbemProviderInitSink ) { *iplpv = ( LPVOID ) this ; } if ( *iplpv ) { ( ( LPUNKNOWN ) *iplpv )->AddRef () ; return ResultFromScode ( S_OK ) ; } else { return ResultFromScode ( E_NOINTERFACE ) ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP_( ULONG ) CServerObject_ProviderInitSink :: AddRef () { return InterlockedIncrement ( & m_ReferenceCount ) ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP_(ULONG) CServerObject_ProviderInitSink :: Release () { LONG t_ReferenceCount = InterlockedDecrement ( & m_ReferenceCount ) ; if ( t_ReferenceCount == 0 ) { delete this ; return 0 ; } else { return t_ReferenceCount ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ HRESULT CServerObject_ProviderInitSink :: SetStatus ( LONG a_Status, LONG a_Flags ) { HRESULT t_Result = S_OK ; if ( m_SecurityDescriptor ) { t_Result = OS::CoImpersonateClient () ; if ( SUCCEEDED ( t_Result ) ) { t_Result = DecoupledProviderSubSystem_Globals :: Check_SecurityDescriptor_CallIdentity ( m_SecurityDescriptor , MASK_PROVIDER_BINDING_BIND , & g_ProviderBindingMapping ) ; CoRevertToSelf () ; } } if ( SUCCEEDED ( t_Result ) ) { m_Result = a_Status ; } else { m_Result = t_Result ; } SetEvent ( m_Event ) ; return t_Result ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ CInterceptor_IWbemProviderInitSink :: CInterceptor_IWbemProviderInitSink ( IWbemProviderInitSink *a_InterceptedSink ) : m_ReferenceCount ( 0 ) , m_InterceptedSink ( a_InterceptedSink ) , m_GateClosed ( FALSE ) , m_InProgress ( 0 ) , m_StatusCalled ( NOTCALLED ) { InterlockedIncrement ( & DecoupledProviderSubSystem_Globals :: s_CInterceptor_IWbemProviderInitSink_ObjectsInProgress ) ; InterlockedIncrement (&DecoupledProviderSubSystem_Globals::s_ObjectsInProgress); if ( m_InterceptedSink ) { m_InterceptedSink->AddRef () ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ CInterceptor_IWbemProviderInitSink::~CInterceptor_IWbemProviderInitSink () { if ( m_StatusCalled == NOTCALLED ) { m_InterceptedSink->SetStatus ( WBEM_E_UNEXPECTED , 0 ) ; } if ( m_InterceptedSink ) { m_InterceptedSink->Release () ; } InterlockedDecrement ( & DecoupledProviderSubSystem_Globals :: s_CInterceptor_IWbemProviderInitSink_ObjectsInProgress ) ; InterlockedDecrement (&DecoupledProviderSubSystem_Globals :: s_ObjectsInProgress) ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP CInterceptor_IWbemProviderInitSink::QueryInterface ( REFIID iid , LPVOID FAR *iplpv ) { *iplpv = NULL ; if ( iid == IID_IUnknown ) { *iplpv = ( LPVOID ) this ; } else if ( iid == IID_IWbemProviderInitSink ) { *iplpv = ( LPVOID ) ( IWbemProviderInitSink * ) this ; } if ( *iplpv ) { ( ( LPUNKNOWN ) *iplpv )->AddRef () ; return ResultFromScode ( S_OK ) ; } else { return ResultFromScode ( E_NOINTERFACE ) ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP_( ULONG ) CInterceptor_IWbemProviderInitSink :: AddRef () { return InterlockedIncrement (&m_ReferenceCount) ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ STDMETHODIMP_(ULONG) CInterceptor_IWbemProviderInitSink :: Release () { if ( ( InterlockedDecrement (&m_ReferenceCount)) == 0 ) { delete this ; return 0 ; } else { return m_ReferenceCount ; } } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ HRESULT CInterceptor_IWbemProviderInitSink :: SetStatus ( LONG a_Status, LONG a_Flags ) { HRESULT t_Result = S_OK ; m_StatusCalled = CALLED ; InterlockedIncrement (&m_InProgress) ; if ( m_GateClosed == 1 ) { t_Result = WBEM_E_SHUTTING_DOWN ; } else { t_Result = m_InterceptedSink->SetStatus ( a_Status, a_Flags ) ; } InterlockedDecrement (&m_InProgress); return t_Result ; } /****************************************************************************** * * Name: * * * Description: * * *****************************************************************************/ HRESULT CInterceptor_IWbemProviderInitSink :: Shutdown () { HRESULT t_Result = S_OK ; InterlockedIncrement (&m_GateClosed) ; bool t_Acquired = false ; while ( ! t_Acquired ) { if ( m_InProgress == 0 ) { t_Acquired = true ; break ; } ::Sleep(0); } return t_Result ; }