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