Windows2003-3790/admin/snapin/dsadmin/dataobj.cpp
2020-09-30 16:53:55 +02:00

849 lines
25 KiB
C++

//+----------------------------------------------------------------------------
//
// DS Administration MMC snapin.
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1999
//
// File: DataObj.cpp
//
// Contents: Data Object implementation.
//
// Classes: CDSDataObject
//
// History: 02-Oct-96 WayneSc Created
// 06-Feb-97 EricB - added Property Page Data support
//-----------------------------------------------------------------------------
#include "stdafx.h"
#include "resource.h"
#include "DSdirect.h"
#include "DataObj.h"
#include "dssnap.h"
#include <lm.h>
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
//+----------------------------------------------------------------------------
// MMC's clipboard formats:
//-----------------------------------------------------------------------------
// Snap-in NodeType in both GUID format and string format
CLIPFORMAT CDSDataObject::m_cfNodeType =
(CLIPFORMAT)RegisterClipboardFormat(CCF_NODETYPE);
CLIPFORMAT CDSDataObject::m_cfNodeTypeString =
(CLIPFORMAT)RegisterClipboardFormat(CCF_SZNODETYPE);
CLIPFORMAT CDSDataObject::m_cfDisplayName =
(CLIPFORMAT)RegisterClipboardFormat(CCF_DISPLAY_NAME);
CLIPFORMAT CDSDataObject::m_cfCoClass =
(CLIPFORMAT)RegisterClipboardFormat(CCF_SNAPIN_CLASSID);
CLIPFORMAT CDSDataObject::m_cfpMultiSelDataObj =
(CLIPFORMAT)RegisterClipboardFormat(CCF_MMC_MULTISELECT_DATAOBJECT);
CLIPFORMAT CDSDataObject::m_cfMultiObjTypes =
(CLIPFORMAT)RegisterClipboardFormat(CCF_OBJECT_TYPES_IN_MULTI_SELECT);
CLIPFORMAT CDSDataObject::m_cfMultiSelDataObjs =
(CLIPFORMAT)RegisterClipboardFormat(CCF_MULTI_SELECT_SNAPINS);
CLIPFORMAT CDSDataObject::m_cfPreload =
(CLIPFORMAT)RegisterClipboardFormat(CCF_SNAPIN_PRELOADS);
//+----------------------------------------------------------------------------
// Our clipboard formats:
//-----------------------------------------------------------------------------
CLIPFORMAT CDSDataObject::m_cfInternal =
(CLIPFORMAT)RegisterClipboardFormat(SNAPIN_INTERNAL);
CLIPFORMAT CDSDataObject::m_cfDsObjectNames =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DSOBJECTNAMES);
CLIPFORMAT CDSDataObject::m_cfDsDisplaySpecOptions =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_DISPLAY_SPEC_OPTIONS);
CLIPFORMAT CDSDataObject::m_cfDsSchemaPath =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_SCHEMA_PATH);
CLIPFORMAT CDSDataObject::m_cfPropSheetCfg =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_PROPSHEETCONFIG);
CLIPFORMAT CDSDataObject::m_cfParentHwnd =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_PARENTHWND);
CLIPFORMAT CDSDataObject::m_cfComponentData =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_COMPDATA);
CLIPFORMAT CDSDataObject::m_cfColumnID =
(CLIPFORMAT)RegisterClipboardFormat(CCF_COLUMN_SET_ID);
CLIPFORMAT CDSDataObject::m_cfMultiSelectProppage =
(CLIPFORMAT)RegisterClipboardFormat(CFSTR_DS_MULTISELECTPROPPAGE);
//+----------------------------------------------------------------------------
// CDSDataObject implementation
//-----------------------------------------------------------------------------
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::IDataObject::GetData
//
// Synopsis: Returns data, in this case the Prop Page format data.
//
//-----------------------------------------------------------------------------
STDMETHODIMP
CDSDataObject::GetData(FORMATETC * pFormatEtc, STGMEDIUM * pMedium)
{
//TRACE(_T("xx.%03x> CDSDataObject(0x%x)::GetData\n"),
// GetCurrentThreadId(), this);
HRESULT hr = S_OK;
if (IsBadWritePtr(pMedium, sizeof(STGMEDIUM)))
{
return E_INVALIDARG;
}
if (!(pFormatEtc->tymed & TYMED_HGLOBAL))
{
return DV_E_TYMED;
}
pMedium->pUnkForRelease = NULL;
pMedium->tymed = TYMED_HGLOBAL;
if (pFormatEtc->cfFormat == m_cfDsObjectNames)
{
DWORD dwCachedBytes = 0;
LPDSOBJECTNAMES pObjectNames = GetDsObjectNames(dwCachedBytes);
if (pObjectNames)
{
// make a deep copy of the cached data
pMedium->hGlobal = GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
m_nDSObjCachedBytes);
if (pMedium->hGlobal == NULL)
{
return E_OUTOFMEMORY;
}
memcpy(pMedium->hGlobal, pObjectNames, dwCachedBytes);
}
else
{
return E_OUTOFMEMORY;
}
}
else if (pFormatEtc->cfFormat == m_cfDsDisplaySpecOptions)
{
//
// Get the DSDISPLAYSPECOPTIONS structure.
// Use the value cached in the component data.
//
if (m_pDsComponentData != NULL)
{
PDSDISPLAYSPECOPTIONS pDsDisplaySpecOptions = m_pDsComponentData->GetDsDisplaySpecOptions();
pMedium->hGlobal = (HGLOBAL)pDsDisplaySpecOptions;
if (pDsDisplaySpecOptions == NULL)
return E_OUTOFMEMORY;
}
else
{
return E_FAIL;
}
}
else if (pFormatEtc->cfFormat == m_cfDsSchemaPath)
{
ASSERT(m_pDsComponentData);
LPCWSTR lpszSchemaNamingCtx = m_pDsComponentData->GetBasePathsInfo()->GetSchemaNamingContext();
size_t nSchemaNamingCtxLen = wcslen(lpszSchemaNamingCtx);
if (nSchemaNamingCtxLen == 0)
{
return E_FAIL;
}
PWSTR pwzSchemaPath = (PWSTR)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
(nSchemaNamingCtxLen +1) * sizeof(WCHAR));
if (pwzSchemaPath == NULL)
{
return STG_E_MEDIUMFULL;
}
wcscpy(pwzSchemaPath, lpszSchemaNamingCtx);
pMedium->hGlobal = pwzSchemaPath;
}
else if (pFormatEtc->cfFormat == m_cfPropSheetCfg)
{
// Return the property sheet notification handle.
//
PPROPSHEETCFG pSheetCfg;
pSheetCfg = (PPROPSHEETCFG)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
sizeof(PROPSHEETCFG));
if (pSheetCfg == NULL)
{
return STG_E_MEDIUMFULL;
}
pSheetCfg->lNotifyHandle = m_lNotifyHandle;
pSheetCfg->hwndParentSheet = m_hwndParentSheet;
pSheetCfg->hwndHidden = m_pDsComponentData->GetHiddenWindow();
pSheetCfg->wParamSheetClose = reinterpret_cast<WPARAM>(m_internal.m_cookie);
pMedium->hGlobal = (HGLOBAL)pSheetCfg;
}
else if (pFormatEtc->cfFormat == m_cfParentHwnd)
{
// return the HWND of the MMC frame window
HWND* pHWndMain = (HWND*)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
sizeof(HWND));
m_pDsComponentData->m_pFrame->GetMainWindow(pHWndMain);
pMedium->hGlobal = (HGLOBAL)pHWndMain;
}
else if (pFormatEtc->cfFormat == m_cfComponentData)
{
// return the m_pDsComponentData the data object is bound to
CDSComponentData** ppCD = (CDSComponentData**)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
sizeof(CDSComponentData*));
if (ppCD != NULL)
{
*ppCD = m_pDsComponentData;
pMedium->hGlobal = (HGLOBAL)ppCD;
}
else
{
return STG_E_MEDIUMFULL;
}
}
else if (pFormatEtc->cfFormat == m_cfMultiSelectProppage)
{
if (m_szUniqueID == _T(""))
{
return E_FAIL;
}
UINT nLength = m_szUniqueID.GetLength();
PWSTR pszGuidString = (PWSTR)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
(nLength +1) * sizeof(WCHAR));
if (pszGuidString == NULL)
{
return STG_E_MEDIUMFULL;
}
wcscpy(pszGuidString, m_szUniqueID);
pMedium->hGlobal = pszGuidString;
}
else if (pFormatEtc->cfFormat == m_cfMultiObjTypes)
{
hr = CreateMultiSelectObject(pMedium);
}
else if (pFormatEtc->cfFormat == m_cfInternal)
{
hr = CreateInternal(pMedium);
}
else if (pFormatEtc->cfFormat == m_cfColumnID)
{
hr = CreateColumnID(pMedium);
}
else
{
return DV_E_FORMATETC;
}
pMedium->tymed = TYMED_HGLOBAL;
pMedium->pUnkForRelease = NULL;
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::IDataObject::GetDataHere
//
// Synopsis: Returns data in callers storage medium
//
//-----------------------------------------------------------------------------
STDMETHODIMP
CDSDataObject::GetDataHere(LPFORMATETC lpFormatetc, LPSTGMEDIUM lpMedium)
{
HRESULT hr = DV_E_CLIPFORMAT;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// Based on the CLIPFORMAT write data to the stream
const CLIPFORMAT cf = lpFormatetc->cfFormat;
if(cf == m_cfNodeType)
{
hr = CreateNodeTypeData(lpMedium);
}
else if(cf == m_cfNodeTypeString)
{
hr = CreateNodeTypeStringData(lpMedium);
}
else if (cf == m_cfDisplayName)
{
hr = CreateDisplayName(lpMedium);
}
else if (cf == m_cfCoClass)
{
hr = CreateCoClassID(lpMedium);
}
else if (cf == m_cfPreload)
{
// MMC notify the snapin before loading with the MMCN_PRELOAD notify message
BOOL bPreload = TRUE;
hr = Create(&bPreload, sizeof(BOOL), lpMedium);
}
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::IDataObject::SetData
//
// Synopsis: Allows the caller to set data object values.
//
//-----------------------------------------------------------------------------
STDMETHODIMP CDSDataObject::SetData(FORMATETC * pFormatEtc, STGMEDIUM * pMedium,
BOOL fRelease)
{
HRESULT hr = S_OK;
if (pFormatEtc->cfFormat == m_cfPropSheetCfg)
{
if (!(pFormatEtc->tymed & TYMED_HGLOBAL))
{
return DV_E_TYMED;
}
PPROPSHEETCFG pSheetCfg = (PPROPSHEETCFG)pMedium->hGlobal;
if ( NULL != pSheetCfg->lNotifyHandle)
{
m_lNotifyHandle = pSheetCfg->lNotifyHandle;
}
if (NULL != pSheetCfg->hwndParentSheet)
{
m_hwndParentSheet = pSheetCfg->hwndParentSheet;
}
if (fRelease)
{
GlobalFree(pMedium->hGlobal);
}
return S_OK;
}
else if (pFormatEtc->cfFormat == m_cfMultiSelectProppage)
{
if (!(pFormatEtc->tymed & TYMED_HGLOBAL))
{
return DV_E_TYMED;
}
PWSTR pszGuidString = (PWSTR)pMedium->hGlobal;
if (pszGuidString == NULL)
{
ASSERT(FALSE);
return E_FAIL;
}
m_szUniqueID = pszGuidString;
}
else
{
return DV_E_FORMATETC;
}
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::IDataObject::EnumFormatEtc
//
// Synopsis:
//
//-----------------------------------------------------------------------------
STDMETHODIMP CDSDataObject::EnumFormatEtc(DWORD, LPENUMFORMATETC*)
{
return E_NOTIMPL;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::Create
//
// Synopsis:
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::Create(const void * pBuffer, int len, LPSTGMEDIUM pMedium)
{
HRESULT hr = DV_E_TYMED;
// Do some simple validation
if (pBuffer == NULL || pMedium == NULL)
return E_POINTER;
// Make sure the type medium is HGLOBAL
if (pMedium->tymed == TYMED_HGLOBAL)
{
// Create the stream on the hGlobal passed in
LPSTREAM lpStream;
hr = CreateStreamOnHGlobal(pMedium->hGlobal, FALSE, &lpStream);
if (SUCCEEDED(hr))
{
// Write to the stream the number of bytes
unsigned long written;
hr = lpStream->Write(pBuffer, len, &written);
// Because we told CreateStreamOnHGlobal with 'FALSE',
// only the stream is released here.
// Note - the caller (i.e. snap-in, object) will free the HGLOBAL
// at the correct time. This is according to the IDataObject specification.
lpStream->Release();
}
}
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateNodeTypeData
//
// Synopsis: Create the node type object in GUID format
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateNodeTypeData(LPSTGMEDIUM lpMedium)
{
TRACE(_T("xx.%03x> GetDataHere on Node Type\n"), GetCurrentThreadId());
GUID* pGuid = m_internal.m_cookie->GetGUID();
return Create(pGuid, sizeof(GUID), lpMedium);
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateNodeTypeStringData
//
// Synopsis: Create the node type object in GUID string format
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateNodeTypeStringData(LPSTGMEDIUM lpMedium)
{
TRACE(_T("xx.%03x> GetDataHere on Node Type String\n"), GetCurrentThreadId());
GUID* pGuid = m_internal.m_cookie->GetGUID();
CString strGUID;
WCHAR * szGUID;
StringFromCLSID(*pGuid, &szGUID);
strGUID = szGUID;
return Create (strGUID, strGUID.GetLength()+ sizeof(wchar_t),
lpMedium);
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateDisplayName
//
// Synopsis: This is the display named used in the scope pane and snap-in
// manager.
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateDisplayName(LPSTGMEDIUM lpMedium)
{
TRACE(_T("xx.%03x> GetDataHere on Display Name\n"), GetCurrentThreadId());
// Load the name from resource
// Note - if this is not provided, the console will used the snap-in name
CString szDispName;
szDispName.LoadString( ResourceIDForSnapinType[ m_internal.m_snapintype ]);
return Create(szDispName, ((szDispName.GetLength()+1) * sizeof(wchar_t)), lpMedium);
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateMultiSelectObject
//
// Synopsis: this is to create the list of types selected
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateMultiSelectObject(LPSTGMEDIUM lpMedium)
{
TRACE(_T("xx.%03x> GetDataHere on MultiSelectObject\n"), GetCurrentThreadId());
CUINode** cookieArray = NULL;
cookieArray = (CUINode**) GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
m_internal.m_cookie_count*sizeof(CUINode*));
if (!cookieArray) {
return E_OUTOFMEMORY;
}
for (UINT k=0; k<m_internal.m_cookie_count; k++)
{
if (k==0)
cookieArray[k] = m_internal.m_cookie;
else
cookieArray[k] = m_internal.m_p_cookies[k-1];
}
BOOL* bDuplicateArr = NULL;
bDuplicateArr = (BOOL*)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
m_internal.m_cookie_count*sizeof(BOOL));
if (!bDuplicateArr) {
if (cookieArray)
GlobalFree (cookieArray);
return E_OUTOFMEMORY;
}
//ZeroMemory(bDuplicateArr, m_internal.m_cookie_count*sizeof(BOOL));
UINT cCount = 0;
for (UINT index=0; index<m_internal.m_cookie_count; index++)
{
for (UINT j=0; j<index;j++)
{
GUID Guid1 = *(cookieArray[index]->GetGUID());
GUID Guid2 = *(cookieArray[j]->GetGUID());
if (IsEqualGUID (Guid1, Guid2))
{
bDuplicateArr[index] = TRUE;
break; //repeated GUID
}
}
if (!bDuplicateArr[index])
cCount++;
}
UINT size = sizeof(SMMCObjectTypes) + (cCount - 1) *
sizeof(GUID);
void * pTmp = ::GlobalAlloc(GPTR, size);
if (!pTmp) {
if (cookieArray) {
GlobalFree (cookieArray);
}
if (bDuplicateArr) {
GlobalFree (bDuplicateArr);
}
return E_OUTOFMEMORY;
}
SMMCObjectTypes* pdata = reinterpret_cast<SMMCObjectTypes*>(pTmp);
pdata->count = cCount;
UINT i = 0;
for (index=0; index<m_internal.m_cookie_count; index++)
{
if (!bDuplicateArr[index])
pdata->guid[i++] = *(cookieArray[index]->GetGUID());
}
ASSERT(i == cCount);
lpMedium->hGlobal = pTmp;
GlobalFree (cookieArray);
GlobalFree (bDuplicateArr);
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateInternal
//
// Synopsis:
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateInternal(LPSTGMEDIUM lpMedium)
{
HRESULT hr = S_OK;
INTERNAL * pInt = NULL;
void * pBuf = NULL;
UINT size = sizeof(INTERNAL);
size += sizeof(CUINode *) * (m_internal.m_cookie_count - 1);
pBuf = GlobalAlloc (GPTR, size);
pInt = (INTERNAL *) pBuf;
lpMedium->hGlobal = pBuf;
if (pInt != NULL &&
m_internal.m_cookie_count > 1)
{
// copy the data
pInt->m_type = m_internal.m_type;
pInt->m_cookie = m_internal.m_cookie;
pInt->m_snapintype = m_internal.m_snapintype;
pInt->m_cookie_count = m_internal.m_cookie_count;
pInt->m_p_cookies = (CUINode **) ((BYTE *)pInt + sizeof(INTERNAL));
memcpy (pInt->m_p_cookies, m_internal.m_p_cookies,
sizeof(CUINode *) * (m_internal.m_cookie_count - 1));
hr = Create(pBuf, size, lpMedium);
}
else
{
hr = Create(&m_internal, sizeof(INTERNAL), lpMedium);
}
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateCoClassID
//
// Synopsis:
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateCoClassID(LPSTGMEDIUM lpMedium)
{
TRACE(_T("xx.%03x> GetDataHere on CoClass\n"), GetCurrentThreadId());
CLSID CoClassID;
switch (m_internal.m_snapintype) {
case SNAPINTYPE_DS:
CoClassID = CLSID_DSSnapin;
break;
case SNAPINTYPE_SITE:
CoClassID = CLSID_SiteSnapin;
break;
default:
memset (&CoClassID,0,sizeof(CLSID));
}
return Create(&CoClassID, sizeof(CLSID), lpMedium);
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateColumnID
//
// Synopsis:
//
//-----------------------------------------------------------------------------
HRESULT
CDSDataObject::CreateColumnID(LPSTGMEDIUM lpMedium)
{
// build the column id
CDSColumnSet* pColumnSet = (m_internal.m_cookie)->GetColumnSet(m_pDsComponentData);
if (pColumnSet == NULL)
return DV_E_TYMED;
LPCWSTR lpszID = pColumnSet->GetColumnID();
size_t iLen = wcslen(lpszID);
// allocate enough memory for the struct and the string for the column id
SColumnSetID* pColumnID = (SColumnSetID*)malloc(sizeof(SColumnSetID) + (iLen * sizeof(WCHAR)));
if (pColumnID != NULL)
{
memset(pColumnID, 0, sizeof(SColumnSetID) + (iLen * sizeof(WCHAR)));
pColumnID->cBytes = static_cast<ULONG>(iLen * sizeof(WCHAR));
memcpy(pColumnID->id, lpszID, (iLen * sizeof(WCHAR)));
// copy the column id to global memory
size_t cb = sizeof(SColumnSetID) + (iLen * sizeof(WCHAR));
lpMedium->tymed = TYMED_HGLOBAL;
lpMedium->hGlobal = ::GlobalAlloc(GMEM_SHARE|GMEM_MOVEABLE, cb);
if (lpMedium->hGlobal == NULL)
return STG_E_MEDIUMFULL;
BYTE* pb = reinterpret_cast<BYTE*>(::GlobalLock(lpMedium->hGlobal));
memcpy(pb, pColumnID, cb);
::GlobalUnlock(lpMedium->hGlobal);
free(pColumnID);
}
return S_OK;
}
LPDSOBJECTNAMES CDSDataObject::GetDsObjectNames(DWORD& dwCachedBytes)
{
if (!m_pDSObjCached)
{
HRESULT hr = CreateDsObjectNamesCached();
if (FAILED(hr))
{
TRACE(L"Failed to create the cached DSOBJECTNAMES: hr = 0x%x\n", hr);
return 0;
}
}
dwCachedBytes = m_nDSObjCachedBytes;
return m_pDSObjCached;
}
HRESULT CDSDataObject::CreateDsObjectNamesCached()
{
if (m_pDSObjCached != NULL)
{
::free(m_pDSObjCached);
m_pDSObjCached = NULL;
m_nDSObjCachedBytes = 0;
}
// figure out how much storage we need
DWORD cbStorage = 0;
INT cbStruct = sizeof(DSOBJECTNAMES) +
((m_internal.m_cookie_count - 1) * sizeof(DSOBJECT));
CString strPath;
CString strClass;
CUINode* pNode = 0;
CDSCookie * pCookie = 0;
//
// this loop is to calc how much storage we need.
//
for (UINT index = 0; index < m_internal.m_cookie_count; index++)
{
if (index == 0)
{
pNode = m_internal.m_cookie;
}
else
{
pNode = m_internal.m_p_cookies[index - 1];
}
pCookie = NULL;
if (IS_CLASS(pNode, DS_UI_NODE))
{
pCookie = GetDSCookieFromUINode(pNode);
}
//
// All the nodes must be of type CDSUINode or else we fail
//
if (pCookie == NULL)
{
return E_FAIL;
}
m_pDsComponentData->GetBasePathsInfo()->ComposeADsIPath(strPath, pCookie->GetPath());
strClass = pCookie->GetClass();
if (_wcsicmp(strClass, L"Unknown") == 0)
{
strClass = L"";
}
cbStorage += (strPath.GetLength() + 1 + strClass.GetLength() + 1) * sizeof(TCHAR);
}
//
// Allocate the needed storage
//
m_pDSObjCached = (LPDSOBJECTNAMES)malloc(cbStruct + cbStorage);
if (m_pDSObjCached == NULL)
{
return STG_E_MEDIUMFULL;
}
m_nDSObjCachedBytes = cbStruct + cbStorage;
switch (m_internal.m_snapintype)
{
case SNAPINTYPE_DS:
m_pDSObjCached->clsidNamespace = CLSID_DSSnapin;
break;
case SNAPINTYPE_SITE:
m_pDSObjCached->clsidNamespace = CLSID_SiteSnapin;
break;
default:
memset (&m_pDSObjCached->clsidNamespace, 0, sizeof(CLSID));
}
m_pDSObjCached->cItems = m_internal.m_cookie_count;
DWORD NextOffset = cbStruct;
for (index = 0; index < m_internal.m_cookie_count; index++)
{
if (index == 0)
{
pNode = m_internal.m_cookie;
}
else
{
pNode = m_internal.m_p_cookies[index - 1];
}
pCookie = NULL;
if (IS_CLASS(pNode, DS_UI_NODE))
{
pCookie = GetDSCookieFromUINode(pNode);
}
//
// All nodes must be of type CDSUINode or else we fail
//
if (pCookie == NULL)
{
return E_FAIL;
}
//
// Set the data from the node and node data
//
m_pDSObjCached->aObjects[index].dwFlags = pNode->IsContainer() ? DSOBJECT_ISCONTAINER : 0;
m_pDSObjCached->aObjects[index].dwProviderFlags = (m_pDsComponentData->IsAdvancedView()) ?
DSPROVIDER_ADVANCED : 0;
m_pDsComponentData->GetBasePathsInfo()->ComposeADsIPath(strPath, pCookie->GetPath());
strClass = pCookie->GetClass();
if (_wcsicmp(strClass, L"Unknown") == 0)
{
strClass = L"";
}
m_pDSObjCached->aObjects[index].offsetName = NextOffset;
m_pDSObjCached->aObjects[index].offsetClass = NextOffset +
(strPath.GetLength() + 1) * sizeof(TCHAR);
_tcscpy((LPTSTR)((BYTE *)m_pDSObjCached + NextOffset), strPath);
NextOffset += (strPath.GetLength() + 1) * sizeof(TCHAR);
_tcscpy((LPTSTR)((BYTE *)m_pDSObjCached + NextOffset), strClass);
NextOffset += (strClass.GetLength() + 1) * sizeof(TCHAR);
}
return S_OK;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::AddCookie
//
// Synopsis: adds a cookie to the data object. if this is
// the first cookie it goes in m_cookie, else it
// goes into the cookie list m_p_cookies.
//
//-----------------------------------------------------------------------------
void
CDSDataObject::AddCookie(CUINode* pUINode)
{
const UINT MEM_CHUNK_SIZE = 10;
void * pTMP = NULL;
if (m_internal.m_cookie) { // already have a cookie
if ((m_internal.m_cookie_count - 1) % MEM_CHUNK_SIZE == 0) {
if (m_internal.m_p_cookies) {
pTMP = realloc (m_internal.m_p_cookies,
(m_internal.m_cookie_count - 1 +
MEM_CHUNK_SIZE) * sizeof (CUINode *));
} else {
pTMP = malloc (MEM_CHUNK_SIZE * sizeof (CUINode *));
}
if (pTMP == NULL) {
TRACE(_T("CDataObject::AddCookie - malloc/realloc failed.."));
ASSERT (pTMP != NULL);
}
m_internal.m_p_cookies = (CUINode **)pTMP;
}
(*(m_internal.m_p_cookies + m_internal.m_cookie_count - 1)) = pUINode;
m_internal.m_cookie_count++;
} else {
m_internal.m_cookie = pUINode;
m_internal.m_cookie_count = 1;
}
}