2020-09-30 17:12:32 +02:00

369 lines
7.8 KiB
C++

// StandardEnum.Cpp
// Copyright 1995 Microsoft Corporation. All Rights Reserved.
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
// PARTICULAR PURPOSE.
// implementation of a generic enumerator object.
#include "stdafx.h"
#pragma hdrstop
#include "stdenum.h"
// Used by creators of CStandardEnum
void WINAPI CopyAndAddRefObject
(
void *pDest, // dest
const void *pSource, // src
DWORD dwSize // size, ignored, since it's always 4
)
{
IUnknown *pUnk = *((IUnknown **)pSource);
*((IUnknown **)pDest) = pUnk;
pUnk->AddRef();
}
void* CStandardEnum_CreateInstance(REFIID riid, BOOL fMembersAreInterfaces, int cElement, int cbElement, void *rgElements,
void (WINAPI * pfnCopyElement)(void *, const void *, DWORD))
{
return (LPVOID)new CStandardEnum(riid, fMembersAreInterfaces, cElement, cbElement, rgElements, pfnCopyElement);
}
// CStandardEnum::CStandardEnum
// create the object and initialize the refcount
// Parameters:
// REFCLSID - [in] type of enumerator that we are
// int - [in] number of elements in the enumeration
// int - [in] size of each element
// void * - [in] pointer to element data
// void (WINAPI *pfnCopyElement)(void *, const void *, DWORD)
// - [in] copying function
// Notes:
#pragma warning(disable:4355) // using 'this' in constructor
CStandardEnum::CStandardEnum
(
REFCLSID rclsid,
BOOL fMembersAreInterfaces,
int cElements,
int cbElementSize,
void *rgElements,
void (WINAPI *pfnCopyElement)(void *, const void *, DWORD)
)
: m_cRef(1),
m_iid(rclsid),
m_cElements(cElements),
m_cbElementSize(cbElementSize),
m_iCurrent(0),
m_rgElements(rgElements),
m_pfnCopyElement(pfnCopyElement),
m_fMembersAreInterfaces(fMembersAreInterfaces)
{
m_pEnumClonedFrom = NULL;
if(m_fMembersAreInterfaces)
{
if(m_rgElements)
{
int i;
for(i=0; i<m_cElements; i++)
{
LPUNKNOWN *ppunk = (IUnknown **)GetNthElement(i);
(*ppunk)->AddRef();
}
}
}
}
#pragma warning(default:4355) // using 'this' in constructor
// CStandardEnum::CStandardEnum
// "it is not death, but dying, which is terrible."
// - Henry Fielding (1707-54)
// Notes:
CStandardEnum::~CStandardEnum ()
{
// if we're a cloned object, then just release our parent object and
// we're done. otherwise, free up the allocated memory we were given
if (m_pEnumClonedFrom)
{
m_pEnumClonedFrom->Release();
}
else
{
if (m_rgElements)
{
if(m_fMembersAreInterfaces)
{
int i;
for(i=0; i<m_cElements; i++)
{
LPUNKNOWN *ppunk = (IUnknown **)GetNthElement(i);
(*ppunk)->Release();
}
}
GlobalFree(m_rgElements);
}
}
}
// CStandardEnum::InternalQueryInterface
// we support our internal iid, and that's all
// Parameters:
// REFIID - [in] interface they want
// void ** - [out] where they want to put the resulting object ptr.
// Output:
// HRESULT - S_OK, E_NOINTERFACE
// Notes:
HRESULT CStandardEnum::QueryInterface
(
REFIID riid,
void **ppvObjOut
)
{
*ppvObjOut = NULL;
if (IsEqualIID(riid, m_iid))
{
*ppvObjOut = (IEnumGeneric *)this;
}
else if (IsEqualIID(riid, IID_IUnknown))
{
*ppvObjOut = (IUnknown *)this;
}
if (*ppvObjOut)
{
AddRef();
return S_OK;
}
else
{
return E_NOINTERFACE;
}
}
ULONG CStandardEnum::AddRef(void)
{
return ++m_cRef;
}
ULONG CStandardEnum::Release(void)
{
int n = --m_cRef;
if (n == 0)
delete this;
return(n);
}
// CStandardEnum::Next
// returns the next dude in our iteration
// Parameters:
// unsigned long - [in] count of elements requested
// void * - [out] array of slots to put values in.
// unsigned long * - [out] actual number fetched
// Output:
// HRESULT - S_OK, E_INVALIDARG, S_FALSE
// Notes:
STDMETHODIMP CStandardEnum::Next
(
unsigned long cElm,
void *rgDest,
unsigned long *pcElmOut
)
{
unsigned long cElementsFetched = 0;
void *pElementDest = rgDest;
const void *pElementSrc = GetNthElement(m_iCurrent);
while (cElementsFetched < cElm) {
// if we hit EOF, break out
if (m_iCurrent >= m_cElements)
break;
// copy the element out for them
m_pfnCopyElement(pElementDest, pElementSrc, m_cbElementSize);
// increase the counters
pElementDest = (LPBYTE)pElementDest + m_cbElementSize;
pElementSrc = (const BYTE *)pElementSrc + m_cbElementSize;
m_iCurrent++;
cElementsFetched++;
}
if (pcElmOut)
*pcElmOut = cElementsFetched;
return (cElementsFetched < cElm)? S_FALSE : S_OK;
}
// CStandardEnum::Skip
// skips the requested number of rows.
// Parameters:
// unsigned long - [in] number to skip
// Output:
// HRESULT - S_OK, S_FALSE
// Notes:
STDMETHODIMP CStandardEnum::Skip
(
unsigned long cSkip
)
{
// handle running off the end
if (m_iCurrent + (int)cSkip > m_cElements) {
m_iCurrent = m_cElements;
return S_FALSE;
}
m_iCurrent += cSkip;
return S_OK;
}
// CStandardEnum::Reset
// reset the counter.
// Output:
// HRESULT - S_OK
// Notes:
STDMETHODIMP CStandardEnum::Reset
(
void
)
{
m_iCurrent = 0;
return S_OK;
}
// CStandardEnum::Clone
// clones the object and gives the new one the same position
// Parameters:
// IEnumVARIANT ** - [out] where to put the new object.
// Output;
// HRESULT - S_OK, E_OUTOFMEMORY
// Notes:
STDMETHODIMP CStandardEnum::Clone
(
IEnumGeneric **ppEnumClone
)
{
CStandardEnum *pNewEnum = new CStandardEnum(m_iid, m_fMembersAreInterfaces, m_cElements,
m_cbElementSize, m_rgElements, m_pfnCopyElement);
RETURN_ON_NULLALLOC(pNewEnum);
// The clone has the same current position as we do
pNewEnum->m_iCurrent = m_iCurrent;
// hold on to who we were cloned from so m_rgElements stays alive, and we don't
// have to copy it.
pNewEnum->m_pEnumClonedFrom = this;
// AddRef() ourselves on their behalf.
AddRef();
return S_OK;
}
// Helper function for creating IConnectionPoint enumerators
HRESULT CreateInstance_IEnumConnectionPoints(LPENUMCONNECTIONPOINTS * ppEnum, DWORD count, ...)
{
DWORD countTemp;
IConnectionPoint **rgCPs;
if (NULL == ppEnum)
return E_POINTER;
ASSERT(count > 0);
// GlobalAlloc an array of connection points [since our standard enum
// assumes this and GlobalFree's it later]
rgCPs = (LPCONNECTIONPOINT*)GlobalAlloc(GPTR, SIZEOF(LPCONNECTIONPOINT) * count);
if (NULL == rgCPs)
return E_OUTOFMEMORY;
va_list ArgList;
va_start(ArgList, count);
IConnectionPoint **prgCPs = rgCPs;
countTemp = count;
while (countTemp)
{
IConnectionPoint *pArg = va_arg(ArgList, IConnectionPoint*);
*prgCPs = pArg;
prgCPs++;
countTemp--;
}
va_end(ArgList);
*ppEnum = (IEnumConnectionPoints *)(IEnumGeneric *) new CStandardEnum(IID_IEnumConnectionPoints,
TRUE, count, SIZEOF(LPCONNECTIONPOINT), (LPVOID)rgCPs,
CopyAndAddRefObject);
if (!*ppEnum)
{
GlobalFree(rgCPs);
return E_OUTOFMEMORY;
}
return S_OK;
}