dmloader: Don't claim partial success when loading fails.
[wine] / dlls / d3drm / tests / d3drm.c
1 /*
2  * Copyright 2010 Christian Costa
3  * Copyright 2012 AndrĂ© Hentschel
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  */
19
20 #define COBJMACROS
21 #include <d3drm.h>
22 #include <initguid.h>
23
24 #include "wine/test.h"
25
26 static HMODULE d3drm_handle = 0;
27
28 static HRESULT (WINAPI * pDirect3DRMCreate)(LPDIRECT3DRM* ppDirect3DRM);
29
30 #define CHECK_REFCOUNT(obj,rc) \
31     { \
32         int rc_new = rc; \
33         int count = get_refcount( (IUnknown *)obj ); \
34         ok(count == rc_new, "Invalid refcount. Expected %d got %d\n", rc_new, count); \
35     }
36
37 #define D3DRM_GET_PROC(func) \
38     p ## func = (void*)GetProcAddress(d3drm_handle, #func); \
39     if(!p ## func) { \
40       trace("GetProcAddress(%s) failed\n", #func); \
41       FreeLibrary(d3drm_handle); \
42       return FALSE; \
43     }
44
45 static BOOL InitFunctionPtrs(void)
46 {
47     d3drm_handle = LoadLibraryA("d3drm.dll");
48
49     if(!d3drm_handle)
50     {
51         skip("Could not load d3drm.dll\n");
52         return FALSE;
53     }
54
55     D3DRM_GET_PROC(Direct3DRMCreate)
56
57     return TRUE;
58 }
59
60 static int get_refcount(IUnknown *object)
61 {
62     IUnknown_AddRef( object );
63     return IUnknown_Release( object );
64 }
65
66 static char data_bad_version[] =
67 "xof 0302txt 0064\n"
68 "Header Object\n"
69 "{\n"
70 "1; 2; 3;\n"
71 "}\n";
72
73 static char data_no_mesh[] =
74 "xof 0302txt 0064\n"
75 "Header Object\n"
76 "{\n"
77 "1; 0; 1;\n"
78 "}\n";
79
80 static char data_ok[] =
81 "xof 0302txt 0064\n"
82 "Header Object\n"
83 "{\n"
84 "1; 0; 1;\n"
85 "}\n"
86 "Mesh Object\n"
87 "{\n"
88 "4;\n"
89 "1.0; 0.0; 0.0;,\n"
90 "0.0; 1.0; 0.0;,\n"
91 "0.0; 0.0; 1.0;,\n"
92 "1.0; 1.0; 1.0;;\n"
93 "3;\n"
94 "3; 0, 1, 2;,\n"
95 "3; 1, 2, 3;,\n"
96 "3; 3, 1, 2;;\n"
97 "}\n";
98
99 static char data_full[] =
100 "xof 0302txt 0064\n"
101 "Header { 1; 0; 1; }\n"
102 "Mesh {\n"
103 " 3;\n"
104 " 0.1; 0.2; 0.3;,\n"
105 " 0.4; 0.5; 0.6;,\n"
106 " 0.7; 0.8; 0.9;;\n"
107 " 1;\n"
108 " 3; 0, 1, 2;;\n"
109 " MeshMaterialList {\n"
110 "  1; 1; 0;\n"
111 "  Material {\n"
112 "   0.0; 1.0; 0.0; 1.0;;\n"
113 "   30.0;\n"
114 "   1.0; 0.0; 0.0;;\n"
115 "   0.5; 0.5; 0.5;;\n"
116 "   TextureFileName {\n"
117 "    \"Texture.bmp\";\n"
118 "   }\n"
119 "  }\n"
120 " }\n"
121 " MeshNormals {\n"
122 "  3;\n"
123 "  1.1; 1.2; 1.3;,\n"
124 "  1.4; 1.5; 1.6;,\n"
125 "  1.7; 1.8; 1.9;;\n"
126 "  1;"
127 "  3; 0, 1, 2;;\n"
128 " }\n"
129 " MeshTextureCoords {\n"
130 "  3;\n"
131 "  0.13; 0.17;,\n"
132 "  0.23; 0.27;,\n"
133 "  0.33; 0.37;;\n"
134 " }\n"
135 "}\n";
136
137 static void test_MeshBuilder(void)
138 {
139     HRESULT hr;
140     LPDIRECT3DRM pD3DRM;
141     LPDIRECT3DRMMESHBUILDER pMeshBuilder;
142     D3DRMLOADMEMORY info;
143     int val;
144     DWORD val1, val2, val3;
145     D3DVALUE valu, valv;
146     D3DVECTOR v[3];
147     D3DVECTOR n[3];
148     DWORD f[8];
149
150     hr = pDirect3DRMCreate(&pD3DRM);
151     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
152
153     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
154     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
155
156     info.lpMemory = data_bad_version;
157     info.dSize = strlen(data_bad_version);
158     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
159     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
160
161     info.lpMemory = data_no_mesh;
162     info.dSize = strlen(data_no_mesh);
163     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
164     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
165
166     info.lpMemory = data_ok;
167     info.dSize = strlen(data_ok);
168     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
169     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
170
171     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
172     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
173
174     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
175     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
176
177     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, NULL, &val2, NULL, &val3, NULL);
178     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
179     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
180     todo_wine ok(val2 == 4, "Wrong number of normals %d (must be 4)\n", val2);
181     ok(val3 == 22, "Wrong number of face data bytes %d (must be 22)\n", val3);
182
183     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
184     valu = 1.23f;
185     valv = 3.21f;
186     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
187     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
188     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
189     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
190     valu = 1.23f;
191     valv = 3.21f;
192     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 1, &valu, &valv);
193     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
194     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
195     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
196     valu = 1.23f;
197     valv = 3.21f;
198     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 2, &valu, &valv);
199     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
200     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
201     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
202     valu = 1.23f;
203     valv = 3.21f;
204     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 3, &valu, &valv);
205     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
206     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
207     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
208     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 4, &valu, &valv);
209     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
210
211     valu = 1.23f;
212     valv = 3.21f;
213     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 0, valu, valv);
214     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
215     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 4, valu, valv);
216     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
217
218     valu = 0.0f;
219     valv = 0.0f;
220     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
221     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
222     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
223     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
224
225     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
226
227     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
228     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
229
230     info.lpMemory = data_full;
231     info.dSize = strlen(data_full);
232     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
233     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
234
235     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
236     ok(val == 3, "Wrong number of vertices %d (must be 3)\n", val);
237
238     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
239     ok(val == 1, "Wrong number of faces %d (must be 1)\n", val);
240
241     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, v, &val2, n, &val3, f);
242     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
243     ok(val1 == 3, "Wrong number of vertices %d (must be 3)\n", val1);
244     ok(val2 == 3, "Wrong number of normals %d (must be 3)\n", val2);
245     ok(val3 == 8, "Wrong number of face data bytes %d (must be 8)\n", val3);
246     ok(U1(v[0]).x == 0.1f, "Wrong component v[0].x = %f (expected 0.1)\n", U1(v[0]).x);
247     ok(U2(v[0]).y == 0.2f, "Wrong component v[0].y = %f (expected 0.2)\n", U2(v[0]).y);
248     ok(U3(v[0]).z == 0.3f, "Wrong component v[0].z = %f (expected 0.3)\n", U3(v[0]).z);
249     ok(U1(v[1]).x == 0.4f, "Wrong component v[1].x = %f (expected 0.4)\n", U1(v[1]).x);
250     ok(U2(v[1]).y == 0.5f, "Wrong component v[1].y = %f (expected 0.5)\n", U2(v[1]).y);
251     ok(U3(v[1]).z == 0.6f, "Wrong component v[1].z = %f (expected 0.6)\n", U3(v[1]).z);
252     ok(U1(v[2]).x == 0.7f, "Wrong component v[2].x = %f (expected 0.7)\n", U1(v[2]).x);
253     ok(U2(v[2]).y == 0.8f, "Wrong component v[2].y = %f (expected 0.8)\n", U2(v[2]).y);
254     ok(U3(v[2]).z == 0.9f, "Wrong component v[2].z = %f (expected 0.9)\n", U3(v[2]).z);
255     ok(U1(n[0]).x == 1.1f, "Wrong component n[0].x = %f (expected 1.1)\n", U1(n[0]).x);
256     ok(U2(n[0]).y == 1.2f, "Wrong component n[0].y = %f (expected 1.2)\n", U2(n[0]).y);
257     ok(U3(n[0]).z == 1.3f, "Wrong component n[0].z = %f (expected 1.3)\n", U3(n[0]).z);
258     ok(U1(n[1]).x == 1.4f, "Wrong component n[1].x = %f (expected 1.4)\n", U1(n[1]).x);
259     ok(U2(n[1]).y == 1.5f, "Wrong component n[1].y = %f (expected 1.5)\n", U2(n[1]).y);
260     ok(U3(n[1]).z == 1.6f, "Wrong component n[1].z = %f (expected 1.6)\n", U3(n[1]).z);
261     ok(U1(n[2]).x == 1.7f, "Wrong component n[2].x = %f (expected 1.7)\n", U1(n[2]).x);
262     ok(U2(n[2]).y == 1.8f, "Wrong component n[2].y = %f (expected 1.8)\n", U2(n[2]).y);
263     ok(U3(n[2]).z == 1.9f, "Wrong component n[2].z = %f (expected 1.9)\n", U3(n[2]).z);
264     ok(f[0] == 3 , "Wrong component f[0] = %d (expected 3)\n", f[0]);
265     ok(f[1] == 0 , "Wrong component f[1] = %d (expected 0)\n", f[1]);
266     ok(f[2] == 0 , "Wrong component f[2] = %d (expected 0)\n", f[2]);
267     ok(f[3] == 1 , "Wrong component f[3] = %d (expected 1)\n", f[3]);
268     ok(f[4] == 1 , "Wrong component f[4] = %d (expected 1)\n", f[4]);
269     ok(f[5] == 2 , "Wrong component f[5] = %d (expected 2)\n", f[5]);
270     ok(f[6] == 2 , "Wrong component f[6] = %d (expected 2)\n", f[6]);
271     ok(f[7] == 0 , "Wrong component f[7] = %d (expected 0)\n", f[7]);
272
273     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
274
275     IDirect3DRM_Release(pD3DRM);
276 }
277
278 static void test_MeshBuilder3(void)
279 {
280     HRESULT hr;
281     LPDIRECT3DRM pD3DRM;
282     LPDIRECT3DRM3 pD3DRM3;
283     LPDIRECT3DRMMESHBUILDER3 pMeshBuilder3;
284     D3DRMLOADMEMORY info;
285     int val;
286     DWORD val1;
287     D3DVALUE valu, valv;
288
289     hr = pDirect3DRMCreate(&pD3DRM);
290     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
291
292     hr = IDirect3DRM_QueryInterface(pD3DRM, &IID_IDirect3DRM3, (LPVOID*)&pD3DRM3);
293     if (FAILED(hr))
294     {
295         win_skip("Cannot get IDirect3DRM3 interface (hr = %x), skipping tests\n", hr);
296         IDirect3DRM_Release(pD3DRM);
297         return;
298     }
299
300     hr = IDirect3DRM3_CreateMeshBuilder(pD3DRM3, &pMeshBuilder3);
301     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder3 interface (hr = %x)\n", hr);
302
303     info.lpMemory = data_bad_version;
304     info.dSize = strlen(data_bad_version);
305     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
306     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
307
308     info.lpMemory = data_no_mesh;
309     info.dSize = strlen(data_no_mesh);
310     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
311     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
312
313     info.lpMemory = data_ok;
314     info.dSize = strlen(data_ok);
315     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
316     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
317
318     val = IDirect3DRMMeshBuilder3_GetVertexCount(pMeshBuilder3);
319     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
320
321     val = IDirect3DRMMeshBuilder3_GetFaceCount(pMeshBuilder3);
322     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
323
324     hr = IDirect3DRMMeshBuilder3_GetVertices(pMeshBuilder3, 0, &val1, NULL);
325     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
326     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
327
328     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
329     valu = 1.23f;
330     valv = 3.21f;
331     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
332     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
333     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
334     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
335     valu = 1.23f;
336     valv = 3.21f;
337     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 1, &valu, &valv);
338     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
339     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
340     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
341     valu = 1.23f;
342     valv = 3.21f;
343     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 2, &valu, &valv);
344     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
345     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
346     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
347     valu = 1.23f;
348     valv = 3.21f;
349     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 3, &valu, &valv);
350     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
351     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
352     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
353     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 4, &valu, &valv);
354     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
355
356     valu = 1.23f;
357     valv = 3.21f;
358     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 0, valu, valv);
359     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
360     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 4, valu, valv);
361     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
362
363     valu = 0.0f;
364     valv = 0.0f;
365     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
366     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
367     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
368     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
369
370     IDirect3DRMMeshBuilder3_Release(pMeshBuilder3);
371     IDirect3DRM3_Release(pD3DRM3);
372     IDirect3DRM_Release(pD3DRM);
373 }
374
375 static void test_Frame(void)
376 {
377     HRESULT hr;
378     LPDIRECT3DRM pD3DRM;
379     LPDIRECT3DRMFRAME pFrameC;
380     LPDIRECT3DRMFRAME pFrameP1;
381     LPDIRECT3DRMFRAME pFrameP2;
382     LPDIRECT3DRMFRAME pFrameTmp;
383     LPDIRECT3DRMFRAMEARRAY pArray;
384     DWORD count;
385
386     hr = pDirect3DRMCreate(&pD3DRM);
387     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
388
389     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameC);
390     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
391     CHECK_REFCOUNT(pFrameC, 1);
392
393     pFrameTmp = (void*)0xdeadbeef;
394     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
395     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
396     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
397     CHECK_REFCOUNT(pFrameC, 1);
398
399     pArray = NULL;
400     hr = IDirect3DRMFrame_GetChildren(pFrameC, &pArray);
401     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
402     todo_wine ok(pArray != NULL, "pArray = %p\n", pArray);
403     if (pArray)
404     {
405         count = IDirect3DRMFrameArray_GetSize(pArray);
406         ok(count == 0, "count = %u\n", count);
407         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
408         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
409         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
410         IDirect3DRMFrameArray_Release(pArray);
411     }
412
413     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP1);
414     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
415
416     /* GetParent with NULL pointer */
417     hr = IDirect3DRMFrame_GetParent(pFrameP1, NULL);
418     todo_wine ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
419     CHECK_REFCOUNT(pFrameP1, 1);
420
421     /* [Add/Delete]Child with NULL pointer */
422     hr = IDirect3DRMFrame_AddChild(pFrameP1, NULL);
423     todo_wine ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
424     CHECK_REFCOUNT(pFrameP1, 1);
425
426     hr = IDirect3DRMFrame_DeleteChild(pFrameP1, NULL);
427     todo_wine ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
428     CHECK_REFCOUNT(pFrameP1, 1);
429
430     /* Add child to first parent */
431     pFrameTmp = (void*)0xdeadbeef;
432     hr = IDirect3DRMFrame_GetParent(pFrameP1, &pFrameTmp);
433     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
434     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
435
436     hr = IDirect3DRMFrame_AddChild(pFrameP1, pFrameC);
437     todo_wine ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
438     CHECK_REFCOUNT(pFrameP1, 1);
439     todo_wine CHECK_REFCOUNT(pFrameC, 2);
440
441     pArray = NULL;
442     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
443     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
444     if (pArray)
445     {
446         count = IDirect3DRMFrameArray_GetSize(pArray);
447         ok(count == 1, "count = %u\n", count);
448         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
449         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
450         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
451         IDirect3DRMFrame_Release(pFrameTmp);
452         IDirect3DRMFrameArray_Release(pArray);
453     }
454
455     pFrameTmp = (void*)0xdeadbeef;
456     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
457     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
458     todo_wine ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
459     todo_wine CHECK_REFCOUNT(pFrameP1, 2);
460
461     /* Add child to second parent */
462     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP2);
463     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
464
465     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
466     todo_wine ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
467     todo_wine CHECK_REFCOUNT(pFrameC, 2);
468
469     pArray = NULL;
470     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
471     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
472     if (pArray)
473     {
474         count = IDirect3DRMFrameArray_GetSize(pArray);
475         ok(count == 1, "count = %u\n", count);
476         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
477         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
478         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
479         IDirect3DRMFrame_Release(pFrameTmp);
480         IDirect3DRMFrameArray_Release(pArray);
481     }
482
483     pArray = NULL;
484     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
485     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
486     if (pArray)
487     {
488         count = IDirect3DRMFrameArray_GetSize(pArray);
489         ok(count == 0, "count = %u\n", count);
490         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
491         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
492         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
493         IDirect3DRMFrameArray_Release(pArray);
494     }
495
496     pFrameTmp = (void*)0xdeadbeef;
497     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
498     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
499     todo_wine ok(pFrameTmp == pFrameP2, "pFrameTmp = %p\n", pFrameTmp);
500     todo_wine CHECK_REFCOUNT(pFrameP2, 2);
501     todo_wine CHECK_REFCOUNT(pFrameC, 2);
502
503     /* Add child again */
504     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
505     todo_wine ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
506     todo_wine CHECK_REFCOUNT(pFrameC, 2);
507
508     pArray = NULL;
509     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
510     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
511     if (pArray)
512     {
513         count = IDirect3DRMFrameArray_GetSize(pArray);
514         ok(count == 1, "count = %u\n", count);
515         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
516         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
517         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
518         IDirect3DRMFrame_Release(pFrameTmp);
519         IDirect3DRMFrameArray_Release(pArray);
520     }
521
522     /* Delete child */
523     hr = IDirect3DRMFrame_DeleteChild(pFrameP2, pFrameC);
524     todo_wine ok(hr == D3DRM_OK, "Cannot delete child frame (hr = %x)\n", hr);
525     CHECK_REFCOUNT(pFrameC, 1);
526
527     pArray = NULL;
528     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
529     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
530     if (pArray)
531     {
532         count = IDirect3DRMFrameArray_GetSize(pArray);
533         ok(count == 0, "count = %u\n", count);
534         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
535         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
536         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
537         IDirect3DRMFrameArray_Release(pArray);
538     }
539
540     pFrameTmp = (void*)0xdeadbeef;
541     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
542     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
543     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
544
545     /* Add two children */
546     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
547     todo_wine ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
548     todo_wine CHECK_REFCOUNT(pFrameC, 2);
549
550     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameP1);
551     todo_wine ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
552     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
553
554     pArray = NULL;
555     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
556     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
557     if (pArray)
558     {
559         count = IDirect3DRMFrameArray_GetSize(pArray);
560         ok(count == 2, "count = %u\n", count);
561         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
562         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
563         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
564         IDirect3DRMFrame_Release(pFrameTmp);
565         hr = IDirect3DRMFrameArray_GetElement(pArray, 1, &pFrameTmp);
566         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
567         ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
568         IDirect3DRMFrame_Release(pFrameTmp);
569         IDirect3DRMFrameArray_Release(pArray);
570     }
571
572     IDirect3DRMMeshBuilder_Release(pFrameP2);
573     todo_wine CHECK_REFCOUNT(pFrameC, 2);
574     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
575
576     IDirect3DRMMeshBuilder_Release(pFrameC);
577     IDirect3DRMMeshBuilder_Release(pFrameP1);
578
579     IDirect3DRM_Release(pD3DRM);
580 }
581
582 START_TEST(d3drm)
583 {
584     if (!InitFunctionPtrs())
585         return;
586
587     test_MeshBuilder();
588     test_MeshBuilder3();
589     test_Frame();
590
591     FreeLibrary(d3drm_handle);
592 }