/* * IDxDiagContainer Implementation * * Copyright 2004 Raphael Junqueira * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA * */ #include "config.h" #define COBJMACROS #include "dxdiag_private.h" #include "wine/debug.h" #include "wine/unicode.h" WINE_DEFAULT_DEBUG_CHANNEL(dxdiag); /* IDxDiagContainer IUnknown parts follow: */ HRESULT WINAPI IDxDiagContainerImpl_QueryInterface(PDXDIAGCONTAINER iface, REFIID riid, LPVOID *ppobj) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; if (!ppobj) return E_INVALIDARG; if (IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IDxDiagContainer)) { IUnknown_AddRef(iface); *ppobj = This; return S_OK; } WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppobj); *ppobj = NULL; return E_NOINTERFACE; } static ULONG WINAPI IDxDiagContainerImpl_AddRef(PDXDIAGCONTAINER iface) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; ULONG refCount = InterlockedIncrement(&This->ref); TRACE("(%p)->(ref before=%u)\n", This, refCount - 1); DXDIAGN_LockModule(); return refCount; } static ULONG WINAPI IDxDiagContainerImpl_Release(PDXDIAGCONTAINER iface) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; ULONG refCount = InterlockedDecrement(&This->ref); TRACE("(%p)->(ref before=%u)\n", This, refCount + 1); if (!refCount) { HeapFree(GetProcessHeap(), 0, This); } DXDIAGN_UnlockModule(); return refCount; } /* IDxDiagContainer Interface follow: */ static HRESULT WINAPI IDxDiagContainerImpl_GetNumberOfChildContainers(PDXDIAGCONTAINER iface, DWORD* pdwCount) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; TRACE("(%p)\n", iface); if (NULL == pdwCount) { return E_INVALIDARG; } *pdwCount = This->nSubContainers; return S_OK; } static HRESULT WINAPI IDxDiagContainerImpl_EnumChildContainerNames(PDXDIAGCONTAINER iface, DWORD dwIndex, LPWSTR pwszContainer, DWORD cchContainer) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_SubContainer *p; DWORD i = 0; TRACE("(%p, %u, %p, %u)\n", iface, dwIndex, pwszContainer, cchContainer); if (NULL == pwszContainer || 0 == cchContainer) { return E_INVALIDARG; } LIST_FOR_EACH_ENTRY(p, &This->subContainers, IDxDiagContainerImpl_SubContainer, entry) { if (dwIndex == i) { TRACE("Found container name %s, copying string\n", debugstr_w(p->contName)); lstrcpynW(pwszContainer, p->contName, cchContainer); return (cchContainer <= strlenW(p->contName)) ? DXDIAG_E_INSUFFICIENT_BUFFER : S_OK; } ++i; } TRACE("Failed to find container name at specified index\n"); *pwszContainer = '\0'; return E_INVALIDARG; } static HRESULT IDxDiagContainerImpl_GetChildContainerInternal(PDXDIAGCONTAINER iface, LPCWSTR pwszContainer, IDxDiagContainer** ppInstance) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_SubContainer *p; LIST_FOR_EACH_ENTRY(p, &This->subContainers, IDxDiagContainerImpl_SubContainer, entry) { if (0 == lstrcmpW(p->contName, pwszContainer)) { *ppInstance = p->pCont; return S_OK; } } return E_INVALIDARG; } static HRESULT WINAPI IDxDiagContainerImpl_GetChildContainer(PDXDIAGCONTAINER iface, LPCWSTR pwszContainer, IDxDiagContainer** ppInstance) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainer* pContainer = (PDXDIAGCONTAINER)This; LPWSTR tmp, orig_tmp; INT tmp_len; WCHAR* cur; HRESULT hr = E_INVALIDARG; TRACE("(%p, %s, %p)\n", iface, debugstr_w(pwszContainer), ppInstance); if (NULL == ppInstance || NULL == pwszContainer) { return E_INVALIDARG; } *ppInstance = NULL; tmp_len = strlenW(pwszContainer) + 1; orig_tmp = tmp = HeapAlloc(GetProcessHeap(), 0, tmp_len * sizeof(WCHAR)); if (NULL == tmp) return E_FAIL; lstrcpynW(tmp, pwszContainer, tmp_len); cur = strchrW(tmp, '.'); while (NULL != cur) { *cur = '\0'; /* cut tmp string to '.' */ if (!*(cur + 1)) break; /* Account for a lone terminating period, as in "cont1.cont2.". */ TRACE("Trying to get parent container %s\n", debugstr_w(tmp)); hr = IDxDiagContainerImpl_GetChildContainerInternal(pContainer, tmp, &pContainer); if (FAILED(hr) || NULL == pContainer) goto on_error; cur++; /* go after '.' (just replaced by \0) */ tmp = cur; cur = strchrW(tmp, '.'); } TRACE("Trying to get container %s\n", debugstr_w(tmp)); hr = IDxDiagContainerImpl_GetChildContainerInternal(pContainer, tmp, ppInstance); if (SUCCEEDED(hr)) { TRACE("Succeeded in getting the container instance\n"); IDxDiagContainerImpl_AddRef(*ppInstance); } on_error: HeapFree(GetProcessHeap(), 0, orig_tmp); return hr; } static HRESULT WINAPI IDxDiagContainerImpl_GetNumberOfProps(PDXDIAGCONTAINER iface, DWORD* pdwCount) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; TRACE("(%p)\n", iface); if (NULL == pdwCount) { return E_INVALIDARG; } *pdwCount = This->nProperties; return S_OK; } static HRESULT WINAPI IDxDiagContainerImpl_EnumPropNames(PDXDIAGCONTAINER iface, DWORD dwIndex, LPWSTR pwszPropName, DWORD cchPropName) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_Property *p; DWORD i = 0; TRACE("(%p, %u, %p, %u)\n", iface, dwIndex, pwszPropName, cchPropName); if (NULL == pwszPropName || 0 == cchPropName) { return E_INVALIDARG; } LIST_FOR_EACH_ENTRY(p, &This->properties, IDxDiagContainerImpl_Property, entry) { if (dwIndex == i) { TRACE("Found property name %s, copying string\n", debugstr_w(p->propName)); lstrcpynW(pwszPropName, p->propName, cchPropName); return (cchPropName <= strlenW(p->propName)) ? DXDIAG_E_INSUFFICIENT_BUFFER : S_OK; } ++i; } TRACE("Failed to find property name at specified index\n"); return E_INVALIDARG; } static HRESULT WINAPI IDxDiagContainerImpl_GetProp(PDXDIAGCONTAINER iface, LPCWSTR pwszPropName, VARIANT* pvarProp) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_Property *p; TRACE("(%p, %s, %p)\n", iface, debugstr_w(pwszPropName), pvarProp); if (NULL == pvarProp || NULL == pwszPropName) { return E_INVALIDARG; } LIST_FOR_EACH_ENTRY(p, &This->properties, IDxDiagContainerImpl_Property, entry) { if (0 == lstrcmpW(p->propName, pwszPropName)) { HRESULT hr = VariantClear(pvarProp); if (hr == DISP_E_ARRAYISLOCKED || hr == DISP_E_BADVARTYPE) VariantInit(pvarProp); return VariantCopy(pvarProp, &p->vProp); } } return E_INVALIDARG; } HRESULT WINAPI IDxDiagContainerImpl_AddProp(PDXDIAGCONTAINER iface, LPCWSTR pwszPropName, VARIANT* pVarProp) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_Property *pNew; TRACE("(%p, %s, %p)\n", iface, debugstr_w(pwszPropName), pVarProp); if (NULL == pVarProp || NULL == pwszPropName) { return E_INVALIDARG; } pNew = HeapAlloc(GetProcessHeap(), 0, sizeof(IDxDiagContainerImpl_Property)); if (NULL == pNew) { return E_OUTOFMEMORY; } pNew->propName = HeapAlloc(GetProcessHeap(), 0, (lstrlenW(pwszPropName) + 1) * sizeof(WCHAR)); if (NULL == pNew->propName) { HeapFree(GetProcessHeap(), 0, pNew); return E_OUTOFMEMORY; } lstrcpyW(pNew->propName, pwszPropName); VariantInit(&pNew->vProp); VariantCopy(&pNew->vProp, pVarProp); list_add_tail(&This->properties, &pNew->entry); ++This->nProperties; return S_OK; } HRESULT WINAPI IDxDiagContainerImpl_AddChildContainer(PDXDIAGCONTAINER iface, LPCWSTR pszContName, PDXDIAGCONTAINER pSubCont) { IDxDiagContainerImpl *This = (IDxDiagContainerImpl *)iface; IDxDiagContainerImpl_SubContainer *pNew; TRACE("(%p, %s, %p)\n", iface, debugstr_w(pszContName), pSubCont); if (NULL == pSubCont || NULL == pszContName) { return E_INVALIDARG; } pNew = HeapAlloc(GetProcessHeap(), 0, sizeof(IDxDiagContainerImpl_SubContainer)); if (NULL == pNew) { return E_OUTOFMEMORY; } pNew->contName = HeapAlloc(GetProcessHeap(), 0, (lstrlenW(pszContName) + 1) * sizeof(WCHAR)); if (NULL == pNew->contName) { HeapFree(GetProcessHeap(), 0, pNew); return E_OUTOFMEMORY; } lstrcpyW(pNew->contName, pszContName); pNew->pCont = pSubCont; list_add_tail(&This->subContainers, &pNew->entry); ++This->nSubContainers; return S_OK; } static const IDxDiagContainerVtbl DxDiagContainer_Vtbl = { IDxDiagContainerImpl_QueryInterface, IDxDiagContainerImpl_AddRef, IDxDiagContainerImpl_Release, IDxDiagContainerImpl_GetNumberOfChildContainers, IDxDiagContainerImpl_EnumChildContainerNames, IDxDiagContainerImpl_GetChildContainer, IDxDiagContainerImpl_GetNumberOfProps, IDxDiagContainerImpl_EnumPropNames, IDxDiagContainerImpl_GetProp }; HRESULT DXDiag_CreateDXDiagContainer(REFIID riid, LPVOID *ppobj) { IDxDiagContainerImpl* container; TRACE("(%p, %p)\n", debugstr_guid(riid), ppobj); container = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDxDiagContainerImpl)); if (NULL == container) { *ppobj = NULL; return E_OUTOFMEMORY; } container->lpVtbl = &DxDiagContainer_Vtbl; container->ref = 0; /* will be inited with QueryInterface */ list_init(&container->properties); list_init(&container->subContainers); return IDxDiagContainerImpl_QueryInterface((PDXDIAGCONTAINER)container, riid, ppobj); }