windows-nt/Source/XPSP1/NT/com/oleutest/accctrl/server/oactestp.cxx
2020-09-26 16:20:57 +08:00

1050 lines
25 KiB
C++

/****************************************************************************
FILE: actestp.c
PURPOSE:
FUNCTIONS:
COMMENTS:
****************************************************************************/
#include <windows.h>
#include <ole2.h>
#include <oleext.h> // IAccessControl interface definition
#include <wchar.h>
#include <tchar.h>
#include <stdlib.h>
#include <stdio.h>
#include "oactest.h" // header file generated by MIDL compiler
void DecrementLockCount();
void IncrementLockCount();
void ObjectCreated();
void ObjectDestroyed();
const CLSID CLSID_COAccessControl_DCOM
= {0x52c0e9e1,0xc0c0,0x11cf,{0xae,0xec,0x00,0xaa,0x00,0x44,0xfb,0x89}};
extern "C" const CLSID CLSID_COAccessControlTest;
extern ULONG g_ulFrequency;
DWORD g_dwRegister;
long g_fClassRegistered = FALSE;
// IAccessControlTest implementation class
class CAccessControlTest : public IAccessControlTest
{
private:
IUnknown *m_pIUnknown;
IPersistStream *m_pIPersistStream;
IAccessControl *m_pIAccessControl;
ULONG m_cRef;
BOOL m_bInitialized;
// destructor
~CAccessControlTest()
{
DecrementLockCount();
ObjectDestroyed();
}
public:
// contructor
CAccessControlTest()
{
m_cRef = 0;
m_bInitialized = FALSE;
ObjectCreated();
IncrementLockCount();
}
STDMETHODIMP_(HRESULT) QueryInterface
(
REFIID iid,
void **ppv
);
STDMETHODIMP_(ULONG) AddRef();
STDMETHODIMP_(ULONG) Release();
STDMETHODIMP_(HRESULT) TestServer
(
LPSTR pszTestString
);
STDMETHODIMP_(HRESULT) GetClassID
(
CLSID *pClassID,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) InitNewACL
(
DOUBLE*pdMillisec
);
STDMETHODIMP_(HRESULT) LoadACL
(
LPCSTR pszFilename,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) SaveACL
(
LPCSTR pszFilename,
BOOL fClearDirty,
ULONG *pulBytesWritten,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) GetSizeMax
(
ULONG *pcdSize,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) IsDirty
(
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) GrantAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST *pAccessRequestList,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) DenyAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST pAccessRequestList[],
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) SetAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST pAccessRequestList[],
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) ReplaceAllAccessRights
(
ULONG cCount,
E_EXPLICIT_ACCESS pExplicitAccessList[],
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) RevokeExplicitAccessRights
(
ULONG cCount,
E_TRUSTEE pTrustee[],
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) IsAccessPermitted
(
E_TRUSTEE *pTrustee,
DWORD grfAccessPermissions,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) GetEffectiveAccessRights
(
E_TRUSTEE *pTrustee,
DWORD *pdwRights,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) GetExplicitAccessRights
(
ULONG *pcCount,
PE_EXPLICIT_ACCESS *ppExplicitAccessList,
DOUBLE *pdMillisec
);
STDMETHODIMP_(HRESULT) RevertAccessRights
(
);
STDMETHODIMP_(HRESULT) CommitAccessRights
(
DWORD grfCommitFlags
);
}; // CAccessControlTest
//+-------------------------------------------------------------------------
//
// Method: CAccessControlTest::AddRef, public
//
// Synopsis: Increment reference count
//
// See Also: IUnknown::AddRef
//
//--------------------------------------------------------------------------
STDMETHODIMP_(ULONG)
CAccessControlTest::AddRef()
{
InterlockedIncrement((long *) &m_cRef);
return m_cRef;
}
//+-------------------------------------------------------------------------
//
// Method: CAccessControlTest::Release, public
//
// Synopsis: Decrement DLL reference count
//
// Notes: After the m_cRef is decremented, the object may be
// deleted by another thread. In order to make this code safe
// for multiple threads, we have to access the object state
// before decrementing m_cRef.
//
// See Also: IUnknown::Release.
//
//--------------------------------------------------------------------------
STDMETHODIMP_(ULONG)
CAccessControlTest::Release()
{
unsigned long count;
count = m_cRef - 1;
if(InterlockedDecrement((long *) &m_cRef) == 0)
{
count = 0;
if(m_bInitialized)
{
m_pIUnknown->Release();
m_pIPersistStream->Release();
m_pIAccessControl->Release();
}
delete this;
}
return count;
}
//+-------------------------------------------------------------------------
//
// Method: CAccessControlTest::QueryInterface, public
//
// Synopsis: Query for an interface on the class factory.
//
// See Also: IUnknown:QueryInterface
//
//--------------------------------------------------------------------------
STDMETHODIMP_(HRESULT)
CAccessControlTest::QueryInterface
(
REFIID iid,
void **ppv
)
{
HRESULT hr;
if (!m_bInitialized)
{
hr = CoCreateInstance( CLSID_COAccessControl_DCOM
, NULL
, CLSCTX_INPROC_SERVER
, IID_IUnknown
, (void **)&m_pIUnknown);
if(FAILED(hr))
{
printf("Failed to create an instance of COAccessControl\n.");
return hr;
}
hr = m_pIUnknown->QueryInterface(IID_IPersistStream, (void **)&m_pIPersistStream);
if(FAILED(hr))
{
printf("Failed to query for the IPersistStream Interface.\n");
return hr;
}
hr = m_pIUnknown->QueryInterface(IID_IAccessControl, (void **)&m_pIAccessControl);
if(FAILED(hr))
{
printf("Failed to query for the IAccessControl interface.\n");
return hr;
}
m_bInitialized = TRUE;
}
if ( IsEqualGUID( iid, IID_IUnknown ) )
{
*ppv = (IUnknown *) this;
((IUnknown *)(*ppv))->AddRef();
hr = S_OK;
}
else if (IsEqualGUID( iid, IID_IAccessControlTest))
{
*ppv = (IUnknown *)(IAccessControlTest *)this;
((IAccessControlTest *)(*ppv))->AddRef();
hr = S_OK;
}
else
{
*ppv = 0;
hr = E_NOINTERFACE;
}
return hr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::TestServer
(
LPSTR pszTestString
)
{
printf("The test string is: %s\n", pszTestString);
return S_OK;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::GetClassID
(
CLSID *pClassID,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
CLSID clsid;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->GetClassID(&clsid);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pClassID = clsid;
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::InitNewACL
(
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->Load(NULL);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::LoadACL
(
LPCSTR pszFilename,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
IStream *pIStream = NULL;
HANDLE FileHandle;
DWORD dwFileSize;
void *pvBuffer;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
// Open the file specified by the client
FileHandle = CreateFileA( pszFilename
, GENERIC_READ
, FILE_SHARE_READ | FILE_SHARE_WRITE
, NULL
, OPEN_EXISTING
, FILE_ATTRIBUTE_NORMAL
, NULL );
if( FileHandle == INVALID_HANDLE_VALUE )
{
printf("Cannot open file %s.\n", pszFilename);
*pdMillisec = 0;
return HRESULT_FROM_WIN32(GetLastError());
}
// Get the size of the file
dwFileSize = GetFileSize(FileHandle, NULL);
// Create a buffer to hold the data in the file
pvBuffer = CoTaskMemAlloc(dwFileSize);
DWORD dwBytesRead;
// Read the data of the opened file into a buffer
if(!ReadFile( FileHandle
, pvBuffer
, dwFileSize
, &dwBytesRead
, NULL ))
{
return HRESULT_FROM_WIN32(GetLastError());
}
// Once we have read the file data into a buffer, we can
// close the file handle
CloseHandle( FileHandle);
// Create a stream on the the buffer
localhr = CreateStreamOnHGlobal(NULL, TRUE, &pIStream);
if (FAILED(localhr))
{
printf("Cannot create stream object.\n");
CoTaskMemFree(pvBuffer);
*pdMillisec = 0;
return localhr;
}
// Load the data in the buffer into the IStream object
localhr = pIStream->Write(pvBuffer, dwFileSize, NULL);
// Release the local buffer
CoTaskMemFree(pvBuffer);
if (FAILED(localhr))
{
printf("Failed to load data into stream object.\n");
*pdMillisec = 0;
return localhr;
}
// Rewind the stream pointer the starting position
LARGE_INTEGER li;
li.LowPart = 0;
li.HighPart = 0;
localhr = pIStream->Seek(li, STREAM_SEEK_SET, NULL);
if (FAILED(localhr))
{
printf("Failed to set the stream pointer to the starting position.\n");
*pdMillisec = 0;
return localhr;
}
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->Load(pIStream);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
// Release the local stream pointer
pIStream->Release();
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::SaveACL
(
LPCSTR pszFilename,
BOOL fClearDirty,
ULONG *pulNumOfBytesWritten,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
IStream *pIStream = NULL;
HANDLE FileHandle;
DWORD dwFileSize;
void *pvBuffer;
STATSTG StreamInfo;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
// Create a stream on the the buffer
localhr = CreateStreamOnHGlobal(NULL, TRUE, &pIStream);
if (FAILED(localhr))
{
printf("Cannot create stream object.\n");
*pdMillisec = 0;
return localhr;
}
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->Save(pIStream, fClearDirty);
QueryPerformanceCounter(&liCount2);
if(FAILED(localhr))
{
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
// Get the number of bytes written to the stream
localhr = pIStream->Stat(&StreamInfo, STATFLAG_NONAME);
if (FAILED(localhr))
{
printf("Unable to get information about the local stream.\n");
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
dwFileSize = StreamInfo.cbSize.LowPart;
LARGE_INTEGER liOffset;
liOffset.QuadPart = Int32x32To64(dwFileSize, -1);
pIStream->Seek(liOffset,STREAM_SEEK_CUR, NULL);
// Allocate a buffer to store the data in the stream
pvBuffer = CoTaskMemAlloc(dwFileSize);
// Write the stream
localhr = pIStream->Read(pvBuffer, dwFileSize, NULL);
if(FAILED(localhr))
{
printf("Unable to write data from stream to buffer.\n");
CoTaskMemFree(pvBuffer);
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
// Open the file specified by the client
FileHandle = CreateFileA( pszFilename
, GENERIC_WRITE | GENERIC_READ
, FILE_SHARE_READ | FILE_SHARE_WRITE
, NULL
, CREATE_ALWAYS
, FILE_ATTRIBUTE_NORMAL
, NULL );
if( FileHandle == INVALID_HANDLE_VALUE )
{
printf("Cannot open file %s.\n", pszFilename);
CoTaskMemFree(pvBuffer);
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return HRESULT_FROM_WIN32(GetLastError());
}
// Read the data of the opened file into a buffer
if(!WriteFile( FileHandle
, pvBuffer
, dwFileSize
, pulNumOfBytesWritten
, NULL ))
{
DWORD dwLastError;
dwLastError = GetLastError();
printf("Write failed with error code %x.", dwLastError);
}
// Flush the file buffers
FlushFileBuffers(FileHandle);
// Close the file handle
CloseHandle(FileHandle);
CoTaskMemFree(pvBuffer);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
*pulNumOfBytesWritten = dwFileSize;
// Release the local stream pointer
pIStream->Release();
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::GetSizeMax
(
ULONG *pcdSize,
DOUBLE *pdMillisec
)
{
ULARGE_INTEGER uliSize;
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->GetSizeMax(&uliSize);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pcdSize = uliSize.LowPart;
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::IsDirty
(
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIPersistStream->IsDirty();
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::GrantAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST pAccessRequestList[],
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->GrantAccessRights(cCount, (ACCESS_REQUEST_W *)pAccessRequestList);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::SetAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST pAccessRequestList[],
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->SetAccessRights(cCount, (ACCESS_REQUEST_W *)pAccessRequestList);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::DenyAccessRights
(
ULONG cCount,
E_ACCESS_REQUEST pAccessRequestList[],
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->DenyAccessRights(cCount, (ACCESS_REQUEST_W *)pAccessRequestList);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::ReplaceAllAccessRights
(
ULONG cCount,
E_EXPLICIT_ACCESS pExplicitAccessList[],
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->ReplaceAllAccessRights(cCount, (EXPLICIT_ACCESS_W *)pExplicitAccessList);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::RevokeExplicitAccessRights
(
ULONG cCount,
E_TRUSTEE pTrustee[],
DOUBLE *pdMillisec
)
{
char *pszClientName;
RPC_STATUS status;
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->RevokeExplicitAccessRights(cCount, (TRUSTEE_W *)pTrustee);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::IsAccessPermitted
(
E_TRUSTEE *pTrustee,
DWORD grfAccessPermissions,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->IsAccessPermitted((TRUSTEE_W *)pTrustee, grfAccessPermissions);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::GetEffectiveAccessRights
(
E_TRUSTEE *pTrustee,
DWORD *pdwRights,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->GetEffectiveAccessRights((TRUSTEE_W *)pTrustee, pdwRights);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::GetExplicitAccessRights
(
ULONG *pcCount,
PE_EXPLICIT_ACCESS *ppExplicitAccessList,
DOUBLE *pdMillisec
)
{
HRESULT localhr;
LARGE_INTEGER liCount1;
LARGE_INTEGER liCount2;
QueryPerformanceCounter(&liCount1);
localhr = m_pIAccessControl->GetExplicitAccessRights(pcCount, (PEXPLICIT_ACCESS_W *)ppExplicitAccessList);
QueryPerformanceCounter(&liCount2);
// Assign calues to the out parameters
*pdMillisec = ((DOUBLE)(liCount2.LowPart) - (DOUBLE)(liCount1.LowPart)) / (DOUBLE)(g_ulFrequency) * 1000.0;
return localhr;
}
STDMETHODIMP_(HRESULT) CAccessControlTest::RevertAccessRights
(
)
{
return m_pIAccessControl->RevertAccessRights();
}
STDMETHODIMP_(HRESULT) CAccessControlTest::CommitAccessRights
(
DWORD grfCommitFlags
)
{
return m_pIAccessControl->CommitAccessRights(grfCommitFlags);
}
class CACTestClassFactory : public IClassFactory
{
private:
unsigned long m_cRef;
//destructor
~CACTestClassFactory()
{
ObjectDestroyed();
}
public:
//constructor
CACTestClassFactory()
{
m_cRef = 0;
ObjectCreated();
}
HRESULT STDMETHODCALLTYPE QueryInterface(
REFIID iid,
void **ppv);
ULONG STDMETHODCALLTYPE AddRef();
ULONG STDMETHODCALLTYPE Release();
HRESULT STDMETHODCALLTYPE CreateInstance(
IUnknown *punkOuter,
REFIID riid,
void **ppv);
HRESULT STDMETHODCALLTYPE LockServer(
BOOL fLock );
};
//+-------------------------------------------------------------------------
//
// Method: CACTestClassFactory::AddRef, public
//
// Synopsis: Increment DLL reference counts
//
// See Also: IUnknown::AddRef
//
//--------------------------------------------------------------------------
ULONG STDMETHODCALLTYPE
CACTestClassFactory::AddRef()
{
InterlockedIncrement((long *) &m_cRef);
return m_cRef;
}
//+-------------------------------------------------------------------------
//
// Method: CACTestClassFactory::CreateInstance, public
//
// Synopsis: Create an instance of CAccessControlTest.
//
// See Also: IClassFactory::CreateInstance
//
//--------------------------------------------------------------------------
HRESULT STDMETHODCALLTYPE
CACTestClassFactory::CreateInstance
(
IUnknown *punkOuter,
REFIID riid,
void **ppv
)
{
HRESULT hr;
CAccessControlTest *pACTest;
if(punkOuter != 0)
{
//The CAccessControlTest class doesn't support aggregation.
return CLASS_E_NOAGGREGATION;
}
pACTest = new CAccessControlTest();
if(pACTest != 0)
{
hr = pACTest->QueryInterface(riid, ppv);
}
else
{
hr = E_OUTOFMEMORY;
*ppv = 0;
}
return hr;
}
//+-------------------------------------------------------------------------
//
// Method: CACTestClassFactory::LockServer, public
//
// Synopsis: Lock the server in memory (by adding an extra reference)
//
// Notes: The class factory will be revoked when the lock count
// is decremented to zero. LockServer(TRUE) will increment the
// lock count and ensure that the class factory will
// not be revoked.
//
// See Also: IClassFactory::LockServer
//
//--------------------------------------------------------------------------
HRESULT STDMETHODCALLTYPE
CACTestClassFactory::LockServer(
BOOL fLock )
{
if (fLock == TRUE)
IncrementLockCount();
else
DecrementLockCount();
return S_OK;
}
//+-------------------------------------------------------------------------
//
// Method: CACTestClassFactory::QueryInterface, public
//
// Synopsis: Query for an interface on the class factory.
//
// See Also: IUnknown::QueryInterface
//
//--------------------------------------------------------------------------
HRESULT STDMETHODCALLTYPE
CACTestClassFactory::QueryInterface (
REFIID iid,
void **ppv )
{
HRESULT hr;
if ( IsEqualGUID( iid, IID_IUnknown) ||
IsEqualGUID( iid, IID_IClassFactory ) )
{
*ppv = this;
((IUnknown *)(*ppv))->AddRef();
hr = S_OK;
}
else
{
*ppv = 0;
hr = E_NOINTERFACE;
}
return hr;
}
//+-------------------------------------------------------------------------
//
// Method: CACTestClassFactory::Release, public
//
// Synopsis: Decrement DLL reference count
//
// See Also: IUnknown::Release
//
//--------------------------------------------------------------------------
ULONG STDMETHODCALLTYPE
CACTestClassFactory::Release()
{
unsigned long count;
count = m_cRef - 1;
if(InterlockedDecrement((long *) &m_cRef) == 0)
{
count = 0;
delete this;
}
return count;
}
//+-------------------------------------------------------------------------
//
// Function: RegisterClassFactory.
//
// Synopsis: Register the class factory if it is not currently registered.
//
//--------------------------------------------------------------------------
HRESULT RegisterClassFactory()
{
HRESULT hr;
CACTestClassFactory *pClassFactory;
if(InterlockedExchange(&g_fClassRegistered, TRUE) == FALSE)
{
pClassFactory = new CACTestClassFactory;
if(pClassFactory != 0)
{
hr = CoRegisterClassObject(CLSID_COAccessControlTest,
(IUnknown *) pClassFactory,
CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE,
&g_dwRegister);
}
else
{
hr = E_OUTOFMEMORY;
}
}
else
{
hr = S_OK;
}
return hr;
}
//+-------------------------------------------------------------------------
//
// Function: RevokeClassFactory.
//
// Synopsis: Revoke the registered class factories if they have not
// already been revoked.
//
//--------------------------------------------------------------------------
HRESULT RevokeClassFactory()
{
HRESULT hr;
if(InterlockedExchange(&g_fClassRegistered, FALSE) == TRUE)
{
hr = CoRevokeClassObject(g_dwRegister);
}
else
{
hr = S_OK;
}
return hr;
}