2 * IDxDiagContainer Implementation
4 * Copyright 2004 Raphael Junqueira
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "dxdiag_private.h"
26 #include "wine/debug.h"
27 #include "wine/unicode.h"
29 WINE_DEFAULT_DEBUG_CHANNEL(dxdiag
);
31 /* IDxDiagContainer IUnknown parts follow: */
32 HRESULT WINAPI
IDxDiagContainerImpl_QueryInterface(PDXDIAGCONTAINER iface
, REFIID riid
, LPVOID
*ppobj
)
34 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
36 if (IsEqualGUID(riid
, &IID_IUnknown
)
37 || IsEqualGUID(riid
, &IID_IDxDiagContainer
)) {
38 IUnknown_AddRef(iface
);
43 WARN("(%p)->(%s,%p),not found\n",This
,debugstr_guid(riid
),ppobj
);
47 static ULONG WINAPI
IDxDiagContainerImpl_AddRef(PDXDIAGCONTAINER iface
) {
48 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
49 ULONG refCount
= InterlockedIncrement(&This
->ref
);
51 TRACE("(%p)->(ref before=%u)\n", This
, refCount
- 1);
58 static ULONG WINAPI
IDxDiagContainerImpl_Release(PDXDIAGCONTAINER iface
) {
59 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
60 ULONG refCount
= InterlockedDecrement(&This
->ref
);
62 TRACE("(%p)->(ref before=%u)\n", This
, refCount
+ 1);
65 HeapFree(GetProcessHeap(), 0, This
);
68 DXDIAGN_UnlockModule();
73 /* IDxDiagContainer Interface follow: */
74 static HRESULT WINAPI
IDxDiagContainerImpl_GetNumberOfChildContainers(PDXDIAGCONTAINER iface
, DWORD
* pdwCount
) {
75 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
76 TRACE("(%p)\n", iface
);
77 if (NULL
== pdwCount
) {
80 *pdwCount
= This
->nSubContainers
;
84 static HRESULT WINAPI
IDxDiagContainerImpl_EnumChildContainerNames(PDXDIAGCONTAINER iface
, DWORD dwIndex
, LPWSTR pwszContainer
, DWORD cchContainer
) {
85 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
86 IDxDiagContainerImpl_SubContainer
* p
= NULL
;
89 TRACE("(%p, %u, %s, %u)\n", iface
, dwIndex
, debugstr_w(pwszContainer
), cchContainer
);
91 if (NULL
== pwszContainer
) {
94 if (256 > cchContainer
) {
95 return DXDIAG_E_INSUFFICIENT_BUFFER
;
98 p
= This
->subContainers
;
101 if (cchContainer
<= strlenW(p
->contName
)) {
102 return DXDIAG_E_INSUFFICIENT_BUFFER
;
104 lstrcpynW(pwszContainer
, p
->contName
, cchContainer
);
113 static HRESULT WINAPI
IDxDiagContainerImpl_GetChildContainerInternal(PDXDIAGCONTAINER iface
, LPCWSTR pwszContainer
, IDxDiagContainer
** ppInstance
) {
114 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
115 IDxDiagContainerImpl_SubContainer
* p
= NULL
;
117 p
= This
->subContainers
;
119 if (0 == lstrcmpW(p
->contName
, pwszContainer
)) {
120 *ppInstance
= p
->pCont
;
128 static HRESULT WINAPI
IDxDiagContainerImpl_GetChildContainer(PDXDIAGCONTAINER iface
, LPCWSTR pwszContainer
, IDxDiagContainer
** ppInstance
) {
129 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
130 IDxDiagContainer
* pContainer
= NULL
;
131 LPWSTR tmp
, orig_tmp
;
134 HRESULT hr
= E_INVALIDARG
;
136 TRACE("(%p, %s, %p)\n", iface
, debugstr_w(pwszContainer
), ppInstance
);
138 if (NULL
== ppInstance
|| NULL
== pwszContainer
) {
142 pContainer
= (PDXDIAGCONTAINER
) This
;
144 tmp_len
= strlenW(pwszContainer
) + 1;
145 orig_tmp
= tmp
= HeapAlloc(GetProcessHeap(), 0, tmp_len
* sizeof(WCHAR
));
146 if (NULL
== tmp
) return E_FAIL
;
147 lstrcpynW(tmp
, pwszContainer
, tmp_len
);
149 cur
= strchrW(tmp
, '.');
150 while (NULL
!= cur
) {
151 *cur
= '\0'; /* cut tmp string to '.' */
152 hr
= IDxDiagContainerImpl_GetChildContainerInternal(pContainer
, tmp
, &pContainer
);
153 if (!SUCCEEDED(hr
) || NULL
== pContainer
)
155 cur
++; /* go after '.' (just replaced by \0) */
157 cur
= strchrW(tmp
, '.');
160 hr
= IDxDiagContainerImpl_GetChildContainerInternal(pContainer
, tmp
, ppInstance
);
162 IDxDiagContainerImpl_AddRef(*ppInstance
);
166 HeapFree(GetProcessHeap(), 0, orig_tmp
);
170 static HRESULT WINAPI
IDxDiagContainerImpl_GetNumberOfProps(PDXDIAGCONTAINER iface
, DWORD
* pdwCount
) {
171 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
172 TRACE("(%p)\n", iface
);
173 if (NULL
== pdwCount
) {
176 *pdwCount
= This
->nProperties
;
180 static HRESULT WINAPI
IDxDiagContainerImpl_EnumPropNames(PDXDIAGCONTAINER iface
, DWORD dwIndex
, LPWSTR pwszPropName
, DWORD cchPropName
) {
181 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
182 IDxDiagContainerImpl_Property
* p
= NULL
;
185 TRACE("(%p, %u, %s, %u)\n", iface
, dwIndex
, debugstr_w(pwszPropName
), cchPropName
);
187 if (NULL
== pwszPropName
) {
190 if (256 > cchPropName
) {
191 return DXDIAG_E_INSUFFICIENT_BUFFER
;
194 p
= This
->properties
;
197 if (cchPropName
<= strlenW(p
->vName
)) {
198 return DXDIAG_E_INSUFFICIENT_BUFFER
;
200 lstrcpynW(pwszPropName
, p
->vName
, cchPropName
);
209 static HRESULT WINAPI
IDxDiagContainerImpl_GetProp(PDXDIAGCONTAINER iface
, LPCWSTR pwszPropName
, VARIANT
* pvarProp
) {
210 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
211 IDxDiagContainerImpl_Property
* p
= NULL
;
213 TRACE("(%p, %s, %p)\n", iface
, debugstr_w(pwszPropName
), pvarProp
);
215 if (NULL
== pvarProp
|| NULL
== pwszPropName
) {
219 p
= This
->properties
;
221 if (0 == lstrcmpW(p
->vName
, pwszPropName
)) {
222 VariantCopy(pvarProp
, &p
->v
);
230 HRESULT WINAPI
IDxDiagContainerImpl_AddProp(PDXDIAGCONTAINER iface
, LPCWSTR pwszPropName
, VARIANT
* pVarProp
) {
231 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
232 IDxDiagContainerImpl_Property
* p
= NULL
;
233 IDxDiagContainerImpl_Property
* pNew
= NULL
;
235 TRACE("(%p, %s, %p)\n", iface
, debugstr_w(pwszPropName
), pVarProp
);
237 if (NULL
== pVarProp
|| NULL
== pwszPropName
) {
241 pNew
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(IDxDiagContainerImpl_Property
));
243 return E_OUTOFMEMORY
;
245 VariantInit(&pNew
->v
);
246 VariantCopy(&pNew
->v
, pVarProp
);
247 pNew
->vName
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, (lstrlenW(pwszPropName
) + 1) * sizeof(WCHAR
));
248 lstrcpyW(pNew
->vName
, pwszPropName
);
251 p
= This
->properties
;
253 This
->properties
= pNew
;
255 while (NULL
!= p
->next
) {
264 HRESULT WINAPI
IDxDiagContainerImpl_AddChildContainer(PDXDIAGCONTAINER iface
, LPCWSTR pszContName
, PDXDIAGCONTAINER pSubCont
) {
265 IDxDiagContainerImpl
*This
= (IDxDiagContainerImpl
*)iface
;
266 IDxDiagContainerImpl_SubContainer
* p
= NULL
;
267 IDxDiagContainerImpl_SubContainer
* pNew
= NULL
;
269 TRACE("(%p, %s, %p)\n", iface
, debugstr_w(pszContName
), pSubCont
);
271 if (NULL
== pSubCont
|| NULL
== pszContName
) {
275 pNew
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(IDxDiagContainerImpl_SubContainer
));
277 return E_OUTOFMEMORY
;
279 pNew
->pCont
= pSubCont
;
280 pNew
->contName
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, (lstrlenW(pszContName
) + 1) * sizeof(WCHAR
));
281 lstrcpyW(pNew
->contName
, pszContName
);
284 p
= This
->subContainers
;
286 This
->subContainers
= pNew
;
288 while (NULL
!= p
->next
) {
293 ++This
->nSubContainers
;
297 static const IDxDiagContainerVtbl DxDiagContainer_Vtbl
=
299 IDxDiagContainerImpl_QueryInterface
,
300 IDxDiagContainerImpl_AddRef
,
301 IDxDiagContainerImpl_Release
,
302 IDxDiagContainerImpl_GetNumberOfChildContainers
,
303 IDxDiagContainerImpl_EnumChildContainerNames
,
304 IDxDiagContainerImpl_GetChildContainer
,
305 IDxDiagContainerImpl_GetNumberOfProps
,
306 IDxDiagContainerImpl_EnumPropNames
,
307 IDxDiagContainerImpl_GetProp
311 HRESULT
DXDiag_CreateDXDiagContainer(REFIID riid
, LPVOID
*ppobj
) {
312 IDxDiagContainerImpl
* container
;
314 TRACE("(%p, %p)\n", debugstr_guid(riid
), ppobj
);
316 container
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(IDxDiagContainerImpl
));
317 if (NULL
== container
) {
319 return E_OUTOFMEMORY
;
321 container
->lpVtbl
= &DxDiagContainer_Vtbl
;
322 container
->ref
= 0; /* will be inited with QueryInterface */
323 return IDxDiagContainerImpl_QueryInterface((PDXDIAGCONTAINER
)container
, riid
, ppobj
);