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

876 lines
22 KiB
C++

#include "private.h"
#include "subsmgrp.h"
#include "subitem.h"
// Contains implementations of IEnumSubscription and ISubscriptionMgr2
HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE *pCookie, ISubscriptionItem **ppSubscriptionItem);
HRESULT DoGetItemFromURL(LPCTSTR pszURL, ISubscriptionItem **ppSubscriptionItem)
{
HRESULT hr;
SUBSCRIPTIONCOOKIE cookie;
if ((NULL == pszURL) || (NULL == ppSubscriptionItem))
{
return E_INVALIDARG;
}
hr = ReadCookieFromInetDB(pszURL, &cookie);
if (SUCCEEDED(hr))
{
hr = SubscriptionItemFromCookie(FALSE, &cookie, ppSubscriptionItem);
}
return hr;
}
HRESULT DoGetItemFromURLW(LPCWSTR pwszURL, ISubscriptionItem **ppSubscriptionItem)
{
TCHAR szURL[INTERNET_MAX_URL_LENGTH];
if ((NULL == pwszURL) || (NULL == ppSubscriptionItem))
{
return E_INVALIDARG;
}
#ifdef UNICODE
StrCpyN(szURL, pwszURL, ARRAYSIZE(szURL));
#else
MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL);
#endif
return DoGetItemFromURL(szURL, ppSubscriptionItem);
}
HRESULT DoAbortItems(
/* [in] */ DWORD dwNumCookies,
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
{
HRESULT hr;
if ((0 == dwNumCookies) || (NULL == pCookies))
{
return E_INVALIDARG;
}
ISubscriptionThrottler *pst;
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
if (SUCCEEDED(hr))
{
hr = pst->AbortItems(dwNumCookies, pCookies);
pst->Release();
}
else
{
hr = S_FALSE;
}
return hr;
}
HRESULT DoCreateSubscriptionItem(
/* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo,
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
HRESULT hr;
ISubscriptionItem *psi;
if ((NULL == pSubscriptionItemInfo) || (NULL == pNewCookie) || (NULL == ppSubscriptionItem))
{
return E_INVALIDARG;
}
*ppSubscriptionItem = NULL;
CreateCookie(pNewCookie);
hr = SubscriptionItemFromCookie(TRUE, pNewCookie, &psi);
if (SUCCEEDED(hr))
{
ASSERT(NULL != psi);
hr = psi->SetSubscriptionItemInfo(pSubscriptionItemInfo);
if (SUCCEEDED(hr))
{
*ppSubscriptionItem = psi;
}
else
{
// Don't leak or leave slop hanging around
psi->Release();
DoDeleteSubscriptionItem(pNewCookie, FALSE);
}
}
return hr;
}
HRESULT DoCloneSubscriptionItem(
/* [in] */ ISubscriptionItem *pSubscriptionItem,
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
HRESULT hr;
SUBSCRIPTIONCOOKIE NewCookie;
if ((NULL == pSubscriptionItem) || (NULL == ppSubscriptionItem))
{
return E_INVALIDARG;
}
IEnumItemProperties *peip;
SUBSCRIPTIONITEMINFO sii;
*ppSubscriptionItem = NULL;
// First get existing subscription details
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
hr = pSubscriptionItem->GetSubscriptionItemInfo(&sii);
if (SUCCEEDED(hr))
{
ISubscriptionItem *psi;
// Mark as temp and create a new subscription item
sii.dwFlags |= SI_TEMPORARY;
hr = DoCreateSubscriptionItem(&sii, &NewCookie, &psi);
if (SUCCEEDED(hr))
{
if (pNewCookie)
*pNewCookie = NewCookie;
// Get properties from existing item
hr = pSubscriptionItem->EnumProperties(&peip);
if (SUCCEEDED(hr))
{
ULONG count;
ASSERT(NULL != peip);
hr = peip->GetCount(&count);
if (SUCCEEDED(hr))
{
ITEMPROP *pProps = new ITEMPROP[count];
LPWSTR *pNames = new LPWSTR[count];
VARIANT *pVars = new VARIANT[count];
if ((NULL != pProps) && (NULL != pNames) && (NULL != pVars))
{
hr = peip->Next(count, pProps, &count);
if (SUCCEEDED(hr))
{
ULONG i;
for (i = 0; i < count; i++)
{
pNames[i] = pProps[i].pwszName;
pVars[i] = pProps[i].variantValue;
}
hr = psi->WriteProperties(count, pNames, pVars);
// clean up from enum
for (i = 0; i < count; i++)
{
if (pProps[i].pwszName)
{
CoTaskMemFree(pProps[i].pwszName);
}
VariantClear(&pProps[i].variantValue);
}
}
}
else
{
hr = E_OUTOFMEMORY;
}
SAFEDELETE(pProps);
SAFEDELETE(pNames);
SAFEDELETE(pVars);
}
peip->Release();
}
if (SUCCEEDED(hr))
{
*ppSubscriptionItem = psi;
}
else
{
psi->Release();
}
}
}
return hr;
}
HRESULT DoDeleteSubscriptionItem(
/* [in] */ const SUBSCRIPTIONCOOKIE *pCookie,
/* [in] */ BOOL fAbortItem)
{
HRESULT hr;
TCHAR szKey[MAX_PATH];
if (NULL == pCookie)
{
return E_INVALIDARG;
}
if (fAbortItem)
{
DoAbortItems(1, pCookie);
}
if (ItemKeyNameFromCookie(pCookie, szKey, ARRAYSIZE(szKey)))
{
// Notify the agent that it is about to get deleted.
ISubscriptionItem *pItem=NULL;
if (SUCCEEDED(SubscriptionItemFromCookie(FALSE, pCookie, &pItem)))
{
SUBSCRIPTIONITEMINFO sii = { sizeof(SUBSCRIPTIONITEMINFO) };
if (SUCCEEDED(pItem->GetSubscriptionItemInfo(&sii)))
{
ASSERT(!(sii.dwFlags & SI_TEMPORARY));
ISubscriptionAgentControl *psac=NULL;
if (SUCCEEDED(CoCreateInstance(sii.clsidAgent, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionAgentControl, (void**)&psac)))
{
psac->SubscriptionControl(pItem, SUBSCRIPTION_AGENT_DELETE);
psac->Release();
}
FireSubscriptionEvent(SUBSNOTF_DELETE, pCookie);
if (GUID_NULL != sii.ScheduleGroup)
{
ISyncScheduleMgr *pSyncScheduleMgr;
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
if (SUCCEEDED(hr))
{
pSyncScheduleMgr->RemoveSchedule(&sii.ScheduleGroup);
pSyncScheduleMgr->Release();
}
}
}
pItem->Release();
}
hr = (SHDeleteKey(HKEY_CURRENT_USER, szKey) == ERROR_SUCCESS) ? S_OK : E_FAIL;
}
else
{
TraceMsg(TF_ALWAYS, "Failed to delete subscription item.");
hr = E_FAIL;
}
return hr;
}
HRESULT AddUpdateSubscription(SUBSCRIPTIONCOOKIE *pCookie,
SUBSCRIPTIONITEMINFO *psii,
LPCWSTR pwszURL,
ULONG nProps,
const LPWSTR rgwszName[],
VARIANT rgValue[])
{
HRESULT hr = S_OK;
ASSERT((0 == nProps) || ((NULL != rgwszName) && (NULL != rgValue)));
TCHAR szURL[INTERNET_MAX_URL_LENGTH];
ISubscriptionItem *psi = NULL;
SUBSCRIPTIONCOOKIE cookie;
#ifdef UNICODE
StrCpyNW(szURL, pwszURL, ARRAYSIZE(szURL));
#else
MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL);
#endif
// Try and get the cookie from the inet db otherwise
// create a new one.
if (*pCookie == CLSID_NULL)
{
CreateCookie(&cookie);
}
else
{
cookie = *pCookie;
}
// Update the inet db
WriteCookieToInetDB(szURL, &cookie, FALSE);
hr = SubscriptionItemFromCookie(TRUE, &cookie, &psi);
if (SUCCEEDED(hr))
{
hr = psi->SetSubscriptionItemInfo(psii);
if (SUCCEEDED(hr) && (nProps > 0))
{
ASSERT(NULL != psi);
hr = psi->WriteProperties(nProps, rgwszName, rgValue);
}
psi->Release();
if (FAILED(hr))
{
DoDeleteSubscriptionItem(&cookie, TRUE);
}
}
return hr;
}
HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE *pCookie, ISubscriptionItem **ppSubscriptionItem)
{
HRESULT hr;
ASSERT((NULL != pCookie) && (NULL != ppSubscriptionItem));
HKEY hkey;
if (OpenItemKey(pCookie, fCreateNew ? TRUE : FALSE, KEY_READ | KEY_WRITE, &hkey))
{
*ppSubscriptionItem = new CSubscriptionItem(pCookie, hkey);
if (NULL != *ppSubscriptionItem)
{
hr = S_OK;
}
else
{
hr = E_OUTOFMEMORY;
}
RegCloseKey(hkey);
}
else
{
*ppSubscriptionItem = NULL;
hr = E_FAIL;
}
return hr;
}
BOOL ItemKeyNameFromCookie(const SUBSCRIPTIONCOOKIE *pCookie, TCHAR *pszKeyName, DWORD cchKeyName)
{
WCHAR wszGuid[GUIDSTR_MAX];
ASSERT((NULL != pCookie) && (NULL != pszKeyName) && (cchKeyName >= ARRAYSIZE(WEBCHECK_REGKEY_STORE) + GUIDSTR_MAX));
if (StringFromGUID2(*pCookie, wszGuid, ARRAYSIZE(wszGuid)))
{
#ifdef UNICODE
wnsprintfW(pszKeyName, cchKeyName, L"%s\\%s", c_szRegKeyStore, wszGuid);
#else
wnsprintfA(pszKeyName, cchKeyName, "%s\\%S", c_szRegKeyStore, wszGuid);
#endif
return TRUE;
}
return FALSE;
}
BOOL OpenItemKey(const SUBSCRIPTIONCOOKIE *pCookie, BOOL fCreateNew, REGSAM samDesired, HKEY *phkey)
{
TCHAR szKeyName[MAX_PATH];
ASSERT((NULL != pCookie) && (NULL != phkey));
if (ItemKeyNameFromCookie(pCookie, szKeyName, ARRAYSIZE(szKeyName)))
{
if (fCreateNew)
{
DWORD dwDisposition;
return RegCreateKeyEx(HKEY_CURRENT_USER, szKeyName, 0, NULL, REG_OPTION_NON_VOLATILE, samDesired, NULL, phkey, &dwDisposition) == ERROR_SUCCESS;
}
else
{
return RegOpenKeyEx(HKEY_CURRENT_USER, szKeyName, 0, samDesired, phkey) == ERROR_SUCCESS;
}
}
return FALSE;
}
// ISubscriptionMgr2 members
STDMETHODIMP CSubscriptionMgr::GetItemFromURL(
/* [in] */ LPCWSTR pwszURL,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
return DoGetItemFromURLW(pwszURL, ppSubscriptionItem);
}
STDMETHODIMP CSubscriptionMgr::GetItemFromCookie(
/* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookie,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
if ((NULL == pSubscriptionCookie) || (NULL == ppSubscriptionItem))
{
return E_INVALIDARG;
}
return SubscriptionItemFromCookie(FALSE, pSubscriptionCookie, ppSubscriptionItem);
}
STDMETHODIMP CSubscriptionMgr::GetSubscriptionRunState(
/* [in] */ DWORD dwNumCookies,
/* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookies,
/* [out] */ DWORD *pdwRunState)
{
HRESULT hr;
if ((0 == dwNumCookies) || (NULL == pSubscriptionCookies) || (NULL == pdwRunState))
{
return E_INVALIDARG;
}
ISubscriptionThrottler *pst;
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
if (SUCCEEDED(hr))
{
hr = pst->GetSubscriptionRunState(dwNumCookies, pSubscriptionCookies, pdwRunState);
pst->Release();
}
else
{
// Couldn't connect to a running throttler so assume nothing
// is running.
for (DWORD i = 0; i < dwNumCookies; i++)
{
*pdwRunState++ = 0;
}
hr = S_FALSE;
}
return hr;
}
STDMETHODIMP CSubscriptionMgr::EnumSubscriptions(
/* [in] */ DWORD dwFlags,
/* [out] */ IEnumSubscription **ppEnumSubscriptions)
{
HRESULT hr;
if ((dwFlags & ~SUBSMGRENUM_MASK) || (NULL == ppEnumSubscriptions))
{
return E_INVALIDARG;
}
CEnumSubscription *pes = new CEnumSubscription;
*ppEnumSubscriptions = NULL;
if (NULL != pes)
{
hr = pes->Initialize(dwFlags);
if (SUCCEEDED(hr))
{
hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppEnumSubscriptions);
}
pes->Release();
}
else
{
hr = E_OUTOFMEMORY;
}
return hr;
}
STDMETHODIMP CSubscriptionMgr::UpdateItems(
/* [in] */ DWORD dwFlags,
/* [in] */ DWORD dwNumCookies,
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
{
HRESULT hr;
if ((dwFlags & ~SUBSMGRUPDATE_MASK) || (0 == dwNumCookies) || (NULL == pCookies))
{
return E_INVALIDARG;
}
// Fail if restrictions are in place.
// BUGBUG: Should we have a flag parameter to override this?
if (SHRestricted2W(REST_NoManualUpdates, NULL, 0))
{
SGMessageBox(NULL, IDS_RESTRICTED, MB_OK);
return E_ACCESSDENIED;
}
ISyncMgrSynchronizeInvoke *pSyncMgrInvoke;
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncMgrSynchronizeInvoke, (void **)&pSyncMgrInvoke);
if (SUCCEEDED(hr))
{
DWORD dwInvokeFlags = SYNCMGRINVOKE_STARTSYNC;
if (dwFlags & SUBSMGRUPDATE_MINIMIZE)
{
dwInvokeFlags |= SYNCMGRINVOKE_MINIMIZED;
}
hr = pSyncMgrInvoke->UpdateItems(dwInvokeFlags, CLSID_WebCheckOfflineSync, dwNumCookies * sizeof(SUBSCRIPTIONCOOKIE), (const BYTE *)pCookies);
pSyncMgrInvoke->Release();
}
return hr;
}
STDMETHODIMP CSubscriptionMgr::AbortItems(
/* [in] */ DWORD dwNumCookies,
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
{
return DoAbortItems(dwNumCookies, pCookies);
}
STDMETHODIMP CSubscriptionMgr::AbortAll()
{
HRESULT hr;
ISubscriptionThrottler *pst;
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
if (SUCCEEDED(hr))
{
hr = pst->AbortAll();
pst->Release();
}
else
{
hr = S_FALSE;
}
return hr;
}
// ISubscriptionMgrPriv
STDMETHODIMP CSubscriptionMgr::CreateSubscriptionItem(
/* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo,
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
return DoCreateSubscriptionItem(pSubscriptionItemInfo, pNewCookie, ppSubscriptionItem);
}
STDMETHODIMP CSubscriptionMgr::CloneSubscriptionItem(
/* [in] */ ISubscriptionItem *pSubscriptionItem,
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
{
return DoCloneSubscriptionItem(pSubscriptionItem, pNewCookie, ppSubscriptionItem);
}
STDMETHODIMP CSubscriptionMgr::DeleteSubscriptionItem(
/* [in] */ const SUBSCRIPTIONCOOKIE *pCookie)
{
return DoDeleteSubscriptionItem(pCookie, TRUE);
}
// ** CEnumSubscription **
CEnumSubscription::CEnumSubscription()
{
ASSERT(NULL == m_pCookies);
ASSERT(0 == m_nCurrent);
ASSERT(0 == m_nCount);
m_cRef = 1;
DllAddRef();
}
CEnumSubscription::~CEnumSubscription()
{
if (NULL != m_pCookies)
{
delete [] m_pCookies;
}
DllRelease();
}
HRESULT CEnumSubscription::Initialize(DWORD dwFlags)
{
HRESULT hr = E_FAIL;
HKEY hkey = NULL;
DWORD dwDisposition;
ASSERT(0 == m_nCount);
if (RegCreateKeyEx(HKEY_CURRENT_USER, c_szRegKeyStore, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE, NULL, &hkey, &dwDisposition) == ERROR_SUCCESS)
{
DWORD nCount;
if (RegQueryInfoKey(hkey,
NULL, // address of buffer for class string
NULL, // address of size of class string buffer
NULL, // reserved
&nCount, // address of buffer for number of subkeys
NULL, // address of buffer for longest subkey name length
NULL, // address of buffer for longest class string length
NULL, // address of buffer for number of value entries
NULL, // address of buffer for longest value name length
NULL, // address of buffer for longest value data length
NULL, // address of buffer for security descriptor length
NULL // address of buffer for last write time
) == ERROR_SUCCESS)
{
SUBSCRIPTIONCOOKIE Cookie;
m_pCookies = new SUBSCRIPTIONCOOKIE[nCount];
if (NULL != m_pCookies)
{
TCHAR szKeyName[GUIDSTR_MAX];
hr = S_OK;
for (ULONG i = 0; (i < nCount) && (S_OK == hr); i++)
{
if (RegEnumKey(hkey, i, szKeyName, ARRAYSIZE(szKeyName)) == ERROR_SUCCESS)
{
HRESULT hrConvert;
#ifdef UNICODE
hrConvert = CLSIDFromString(szKeyName, &Cookie);
#else
WCHAR wszKeyName[GUIDSTR_MAX];
MultiByteToWideChar(CP_ACP, 0, szKeyName, -1,
wszKeyName, ARRAYSIZE(wszKeyName));
hrConvert = CLSIDFromString(wszKeyName, &Cookie);
#endif
if (SUCCEEDED(hrConvert))
{
ISubscriptionItem *psi;
m_pCookies[m_nCount] = Cookie;
hr = SubscriptionItemFromCookie(FALSE, &Cookie, &psi);
if (SUCCEEDED(hr))
{
SUBSCRIPTIONITEMINFO sii;
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
if (SUCCEEDED(psi->GetSubscriptionItemInfo(&sii)))
{
// Only count this if it's not a temporary
// or the caller asked for temporary items.
if ((!(sii.dwFlags & SI_TEMPORARY)) || (dwFlags & SUBSMGRENUM_TEMP))
{
m_nCount++;
}
}
psi->Release();
}
}
}
}
}
else
{
hr = E_OUTOFMEMORY;
}
}
RegCloseKey(hkey);
}
return hr;
}
// IUnknown members
STDMETHODIMP CEnumSubscription::QueryInterface(REFIID riid, void **ppv)
{
HRESULT hr;
if (NULL == ppv)
{
return E_INVALIDARG;
}
if ((IID_IUnknown == riid) || (IID_IEnumSubscription == riid))
{
*ppv = (IEnumSubscription *)this;
AddRef();
hr = S_OK;
}
else
{
*ppv = NULL;
hr = E_NOINTERFACE;
}
return hr;
}
STDMETHODIMP_(ULONG) CEnumSubscription::AddRef()
{
return ++m_cRef;
}
STDMETHODIMP_(ULONG) CEnumSubscription::Release()
{
if (--m_cRef == 0)
{
delete this;
return 0;
}
return m_cRef;
}
HRESULT CEnumSubscription::CopyRange(ULONG nStart, ULONG nCount, SUBSCRIPTIONCOOKIE *pCookies, ULONG *pnCopied)
{
ULONG nCopied = 0;
ASSERT((NULL != pCookies) && (NULL != pnCopied));
if (m_nCurrent < m_nCount)
{
ULONG nRemaining = m_nCount - m_nCurrent;
nCopied = min(nRemaining, nCount);
memcpy(pCookies, m_pCookies + m_nCurrent, nCopied * sizeof(SUBSCRIPTIONCOOKIE));
}
*pnCopied = nCopied;
return (nCopied == nCount) ? S_OK : S_FALSE;
}
// IEnumSubscription
STDMETHODIMP CEnumSubscription::Next(
/* [in] */ ULONG celt,
/* [length_is][size_is][out] */ SUBSCRIPTIONCOOKIE *rgelt,
/* [out] */ ULONG *pceltFetched)
{
HRESULT hr;
if ((0 == celt) || ((celt > 1) && (NULL == pceltFetched)) || (NULL == rgelt))
{
return E_INVALIDARG;
}
DWORD nFetched;
hr = CopyRange(m_nCurrent, celt, rgelt, &nFetched);
m_nCurrent += nFetched;
if (pceltFetched)
{
*pceltFetched = nFetched;
}
return hr;
}
STDMETHODIMP CEnumSubscription::Skip(
/* [in] */ ULONG celt)
{
HRESULT hr;
m_nCurrent += celt;
if (m_nCurrent > (m_nCount - 1))
{
m_nCurrent = m_nCount; // Passed the last one
hr = S_FALSE;
}
else
{
hr = S_OK;
}
return hr;
}
STDMETHODIMP CEnumSubscription::Reset()
{
m_nCurrent = 0;
return S_OK;
}
STDMETHODIMP CEnumSubscription::Clone(
/* [out] */ IEnumSubscription **ppenum)
{
HRESULT hr = E_OUTOFMEMORY;
*ppenum = NULL;
CEnumSubscription *pes = new CEnumSubscription;
if (NULL != pes)
{
pes->m_pCookies = new SUBSCRIPTIONCOOKIE[m_nCount];
if (NULL != pes->m_pCookies)
{
ULONG nFetched;
hr = E_FAIL;
pes->m_nCount = m_nCount;
CopyRange(0, m_nCount, pes->m_pCookies, &nFetched);
ASSERT(m_nCount == nFetched);
if (m_nCount == nFetched)
{
hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppenum);
}
}
pes->Release();
}
return hr;
}
STDMETHODIMP CEnumSubscription::GetCount(
/* [out] */ ULONG *pnCount)
{
if (NULL == pnCount)
{
return E_INVALIDARG;
}
*pnCount = m_nCount;
return S_OK;
}