6e068e572d55c44ee8137d546b39105f8c3b936f
[reactos.git] / dll / directx / wine / dxdiagn / container.c
1 /*
2 * IDxDiagContainer Implementation
3 *
4 * Copyright 2004 Raphael Junqueira
5 *
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.
10 *
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.
15 *
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
19 *
20 */
21
22 #include "config.h"
23
24 #define COBJMACROS
25 #include "dxdiag_private.h"
26 #include "wine/debug.h"
27 #include "wine/unicode.h"
28
29 WINE_DEFAULT_DEBUG_CHANNEL(dxdiag);
30
31 static inline IDxDiagContainerImpl *impl_from_IDxDiagContainer(IDxDiagContainer *iface)
32 {
33 return CONTAINING_RECORD(iface, IDxDiagContainerImpl, IDxDiagContainer_iface);
34 }
35
36 /* IDxDiagContainer IUnknown parts follow: */
37 static HRESULT WINAPI IDxDiagContainerImpl_QueryInterface(IDxDiagContainer *iface, REFIID riid,
38 void **ppobj)
39 {
40 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
41
42 if (!ppobj) return E_INVALIDARG;
43
44 if (IsEqualGUID(riid, &IID_IUnknown)
45 || IsEqualGUID(riid, &IID_IDxDiagContainer)) {
46 IUnknown_AddRef(iface);
47 *ppobj = This;
48 return S_OK;
49 }
50
51 WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppobj);
52 *ppobj = NULL;
53 return E_NOINTERFACE;
54 }
55
56 static ULONG WINAPI IDxDiagContainerImpl_AddRef(IDxDiagContainer *iface)
57 {
58 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
59 ULONG refCount = InterlockedIncrement(&This->ref);
60
61 TRACE("(%p)->(ref before=%u)\n", This, refCount - 1);
62
63 DXDIAGN_LockModule();
64
65 return refCount;
66 }
67
68 static ULONG WINAPI IDxDiagContainerImpl_Release(IDxDiagContainer *iface)
69 {
70 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
71 ULONG refCount = InterlockedDecrement(&This->ref);
72
73 TRACE("(%p)->(ref before=%u)\n", This, refCount + 1);
74
75 if (!refCount) {
76 IDxDiagProvider_Release(This->pProv);
77 HeapFree(GetProcessHeap(), 0, This);
78 }
79
80 DXDIAGN_UnlockModule();
81
82 return refCount;
83 }
84
85 /* IDxDiagContainer Interface follow: */
86 static HRESULT WINAPI IDxDiagContainerImpl_GetNumberOfChildContainers(IDxDiagContainer *iface,
87 DWORD *pdwCount)
88 {
89 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
90
91 TRACE("(%p)\n", iface);
92 if (NULL == pdwCount) {
93 return E_INVALIDARG;
94 }
95 *pdwCount = This->cont->nSubContainers;
96 return S_OK;
97 }
98
99 static HRESULT WINAPI IDxDiagContainerImpl_EnumChildContainerNames(IDxDiagContainer *iface,
100 DWORD dwIndex, LPWSTR pwszContainer, DWORD cchContainer)
101 {
102 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
103 IDxDiagContainerImpl_Container *p;
104 DWORD i = 0;
105
106 TRACE("(%p, %u, %p, %u)\n", iface, dwIndex, pwszContainer, cchContainer);
107
108 if (NULL == pwszContainer || 0 == cchContainer) {
109 return E_INVALIDARG;
110 }
111
112 LIST_FOR_EACH_ENTRY(p, &This->cont->subContainers, IDxDiagContainerImpl_Container, entry)
113 {
114 if (dwIndex == i) {
115 TRACE("Found container name %s, copying string\n", debugstr_w(p->contName));
116 lstrcpynW(pwszContainer, p->contName, cchContainer);
117 return (cchContainer <= strlenW(p->contName)) ?
118 DXDIAG_E_INSUFFICIENT_BUFFER : S_OK;
119 }
120 ++i;
121 }
122
123 TRACE("Failed to find container name at specified index\n");
124 *pwszContainer = '\0';
125 return E_INVALIDARG;
126 }
127
128 static HRESULT IDxDiagContainerImpl_GetChildContainerInternal(IDxDiagContainerImpl_Container *cont, LPCWSTR pwszContainer, IDxDiagContainerImpl_Container **subcont) {
129 IDxDiagContainerImpl_Container *p;
130
131 LIST_FOR_EACH_ENTRY(p, &cont->subContainers, IDxDiagContainerImpl_Container, entry)
132 {
133 if (0 == lstrcmpW(p->contName, pwszContainer)) {
134 *subcont = p;
135 return S_OK;
136 }
137 }
138
139 return E_INVALIDARG;
140 }
141
142 static HRESULT WINAPI IDxDiagContainerImpl_GetChildContainer(IDxDiagContainer *iface,
143 LPCWSTR pwszContainer, IDxDiagContainer **ppInstance)
144 {
145 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
146 IDxDiagContainerImpl_Container *pContainer = This->cont;
147 LPWSTR tmp, orig_tmp;
148 INT tmp_len;
149 WCHAR* cur;
150 HRESULT hr = E_INVALIDARG;
151
152 TRACE("(%p, %s, %p)\n", iface, debugstr_w(pwszContainer), ppInstance);
153
154 if (NULL == ppInstance || NULL == pwszContainer) {
155 return E_INVALIDARG;
156 }
157
158 *ppInstance = NULL;
159
160 tmp_len = strlenW(pwszContainer) + 1;
161 orig_tmp = tmp = HeapAlloc(GetProcessHeap(), 0, tmp_len * sizeof(WCHAR));
162 if (NULL == tmp) return E_FAIL;
163 lstrcpynW(tmp, pwszContainer, tmp_len);
164
165 /* special handling for an empty string and leaf container */
166 if (!tmp[0] && list_empty(&pContainer->subContainers)) {
167 hr = DXDiag_CreateDXDiagContainer(&IID_IDxDiagContainer, pContainer, This->pProv, (void **)ppInstance);
168 if (SUCCEEDED(hr))
169 TRACE("Succeeded in getting the container instance\n");
170 goto out;
171 }
172
173 cur = strchrW(tmp, '.');
174 while (NULL != cur) {
175 *cur = '\0'; /* cut tmp string to '.' */
176 if (!*(cur + 1)) break; /* Account for a lone terminating period, as in "cont1.cont2.". */
177 TRACE("Trying to get parent container %s\n", debugstr_w(tmp));
178 hr = IDxDiagContainerImpl_GetChildContainerInternal(pContainer, tmp, &pContainer);
179 if (FAILED(hr))
180 goto out;
181 cur++; /* go after '.' (just replaced by \0) */
182 tmp = cur;
183 cur = strchrW(tmp, '.');
184 }
185
186 TRACE("Trying to get container %s\n", debugstr_w(tmp));
187 hr = IDxDiagContainerImpl_GetChildContainerInternal(pContainer, tmp, &pContainer);
188 if (SUCCEEDED(hr)) {
189 hr = DXDiag_CreateDXDiagContainer(&IID_IDxDiagContainer, pContainer, This->pProv, (void **)ppInstance);
190 if (SUCCEEDED(hr))
191 TRACE("Succeeded in getting the container instance\n");
192 }
193
194 out:
195 HeapFree(GetProcessHeap(), 0, orig_tmp);
196 return hr;
197 }
198
199 static HRESULT WINAPI IDxDiagContainerImpl_GetNumberOfProps(IDxDiagContainer *iface,
200 DWORD *pdwCount)
201 {
202 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
203
204 TRACE("(%p)\n", iface);
205 if (NULL == pdwCount) {
206 return E_INVALIDARG;
207 }
208 *pdwCount = This->cont->nProperties;
209 return S_OK;
210 }
211
212 static HRESULT WINAPI IDxDiagContainerImpl_EnumPropNames(IDxDiagContainer *iface, DWORD dwIndex,
213 LPWSTR pwszPropName, DWORD cchPropName)
214 {
215 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
216 IDxDiagContainerImpl_Property *p;
217 DWORD i = 0;
218
219 TRACE("(%p, %u, %p, %u)\n", iface, dwIndex, pwszPropName, cchPropName);
220
221 if (NULL == pwszPropName || 0 == cchPropName) {
222 return E_INVALIDARG;
223 }
224
225 LIST_FOR_EACH_ENTRY(p, &This->cont->properties, IDxDiagContainerImpl_Property, entry)
226 {
227 if (dwIndex == i) {
228 TRACE("Found property name %s, copying string\n", debugstr_w(p->propName));
229 lstrcpynW(pwszPropName, p->propName, cchPropName);
230 return (cchPropName <= strlenW(p->propName)) ?
231 DXDIAG_E_INSUFFICIENT_BUFFER : S_OK;
232 }
233 ++i;
234 }
235
236 TRACE("Failed to find property name at specified index\n");
237 return E_INVALIDARG;
238 }
239
240 static HRESULT WINAPI IDxDiagContainerImpl_GetProp(IDxDiagContainer *iface, LPCWSTR pwszPropName,
241 VARIANT *pvarProp)
242 {
243 IDxDiagContainerImpl *This = impl_from_IDxDiagContainer(iface);
244 IDxDiagContainerImpl_Property *p;
245
246 TRACE("(%p, %s, %p)\n", iface, debugstr_w(pwszPropName), pvarProp);
247
248 if (NULL == pvarProp || NULL == pwszPropName) {
249 return E_INVALIDARG;
250 }
251
252 LIST_FOR_EACH_ENTRY(p, &This->cont->properties, IDxDiagContainerImpl_Property, entry)
253 {
254 if (0 == lstrcmpW(p->propName, pwszPropName)) {
255 VariantInit(pvarProp);
256 return VariantCopy(pvarProp, &p->vProp);
257 }
258 }
259
260 return E_INVALIDARG;
261 }
262
263 static const IDxDiagContainerVtbl DxDiagContainer_Vtbl =
264 {
265 IDxDiagContainerImpl_QueryInterface,
266 IDxDiagContainerImpl_AddRef,
267 IDxDiagContainerImpl_Release,
268 IDxDiagContainerImpl_GetNumberOfChildContainers,
269 IDxDiagContainerImpl_EnumChildContainerNames,
270 IDxDiagContainerImpl_GetChildContainer,
271 IDxDiagContainerImpl_GetNumberOfProps,
272 IDxDiagContainerImpl_EnumPropNames,
273 IDxDiagContainerImpl_GetProp
274 };
275
276
277 HRESULT DXDiag_CreateDXDiagContainer(REFIID riid, IDxDiagContainerImpl_Container *cont, IDxDiagProvider *pProv, LPVOID *ppobj) {
278 IDxDiagContainerImpl* container;
279
280 TRACE("(%s, %p)\n", debugstr_guid(riid), ppobj);
281
282 container = HeapAlloc(GetProcessHeap(), 0, sizeof(IDxDiagContainerImpl));
283 if (NULL == container) {
284 *ppobj = NULL;
285 return E_OUTOFMEMORY;
286 }
287 container->IDxDiagContainer_iface.lpVtbl = &DxDiagContainer_Vtbl;
288 container->ref = 0; /* will be inited with QueryInterface */
289 container->cont = cont;
290 container->pProv = pProv;
291 IDxDiagProvider_AddRef(pProv);
292 return IDxDiagContainerImpl_QueryInterface(&container->IDxDiagContainer_iface, riid, ppobj);
293 }