dmusic: Pass creation parameters to DMUSIC_CreateDirectMusicBufferImpl then allocate...
[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 char data_d3drm_load[] =
138 "xof 0302txt 0064\n"
139 "Header Object\n"
140 "{\n"
141 "1; 0; 1;\n"
142 "}\n"
143 "Mesh Object1\n"
144 "{\n"
145 " 1;\n"
146 " 0.1; 0.2; 0.3;,\n"
147 " 1;\n"
148 " 3; 0, 1, 2;;\n"
149 "}\n"
150 "Mesh Object2\n"
151 "{\n"
152 " 1;\n"
153 " 0.1; 0.2; 0.3;,\n"
154 " 1;\n"
155 " 3; 0, 1, 2;;\n"
156 "}\n"
157 "Frame Scene\n"
158 "{\n"
159 " {Object1}\n"
160 " {Object2}\n"
161 "}\n";
162
163 static void test_MeshBuilder(void)
164 {
165     HRESULT hr;
166     LPDIRECT3DRM pD3DRM;
167     LPDIRECT3DRMMESHBUILDER pMeshBuilder;
168     D3DRMLOADMEMORY info;
169     int val;
170     DWORD val1, val2, val3;
171     D3DVALUE valu, valv;
172     D3DVECTOR v[3];
173     D3DVECTOR n[3];
174     DWORD f[8];
175
176     hr = pDirect3DRMCreate(&pD3DRM);
177     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
178
179     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
180     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
181
182     info.lpMemory = data_bad_version;
183     info.dSize = strlen(data_bad_version);
184     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
185     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
186
187     info.lpMemory = data_no_mesh;
188     info.dSize = strlen(data_no_mesh);
189     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
190     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
191
192     info.lpMemory = data_ok;
193     info.dSize = strlen(data_ok);
194     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
195     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
196
197     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
198     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
199
200     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
201     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
202
203     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, NULL, &val2, NULL, &val3, NULL);
204     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
205     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
206     todo_wine ok(val2 == 4, "Wrong number of normals %d (must be 4)\n", val2);
207     ok(val3 == 22, "Wrong number of face data bytes %d (must be 22)\n", val3);
208
209     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
210     valu = 1.23f;
211     valv = 3.21f;
212     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
213     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
214     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
215     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
216     valu = 1.23f;
217     valv = 3.21f;
218     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 1, &valu, &valv);
219     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
220     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
221     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
222     valu = 1.23f;
223     valv = 3.21f;
224     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 2, &valu, &valv);
225     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
226     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
227     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
228     valu = 1.23f;
229     valv = 3.21f;
230     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 3, &valu, &valv);
231     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
232     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
233     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
234     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 4, &valu, &valv);
235     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
236
237     valu = 1.23f;
238     valv = 3.21f;
239     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 0, valu, valv);
240     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
241     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 4, valu, valv);
242     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
243
244     valu = 0.0f;
245     valv = 0.0f;
246     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
247     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
248     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
249     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
250
251     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
252
253     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
254     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
255
256     info.lpMemory = data_full;
257     info.dSize = strlen(data_full);
258     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
259     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
260
261     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
262     ok(val == 3, "Wrong number of vertices %d (must be 3)\n", val);
263
264     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
265     ok(val == 1, "Wrong number of faces %d (must be 1)\n", val);
266
267     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, v, &val2, n, &val3, f);
268     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
269     ok(val1 == 3, "Wrong number of vertices %d (must be 3)\n", val1);
270     ok(val2 == 3, "Wrong number of normals %d (must be 3)\n", val2);
271     ok(val3 == 8, "Wrong number of face data bytes %d (must be 8)\n", val3);
272     ok(U1(v[0]).x == 0.1f, "Wrong component v[0].x = %f (expected 0.1)\n", U1(v[0]).x);
273     ok(U2(v[0]).y == 0.2f, "Wrong component v[0].y = %f (expected 0.2)\n", U2(v[0]).y);
274     ok(U3(v[0]).z == 0.3f, "Wrong component v[0].z = %f (expected 0.3)\n", U3(v[0]).z);
275     ok(U1(v[1]).x == 0.4f, "Wrong component v[1].x = %f (expected 0.4)\n", U1(v[1]).x);
276     ok(U2(v[1]).y == 0.5f, "Wrong component v[1].y = %f (expected 0.5)\n", U2(v[1]).y);
277     ok(U3(v[1]).z == 0.6f, "Wrong component v[1].z = %f (expected 0.6)\n", U3(v[1]).z);
278     ok(U1(v[2]).x == 0.7f, "Wrong component v[2].x = %f (expected 0.7)\n", U1(v[2]).x);
279     ok(U2(v[2]).y == 0.8f, "Wrong component v[2].y = %f (expected 0.8)\n", U2(v[2]).y);
280     ok(U3(v[2]).z == 0.9f, "Wrong component v[2].z = %f (expected 0.9)\n", U3(v[2]).z);
281     ok(U1(n[0]).x == 1.1f, "Wrong component n[0].x = %f (expected 1.1)\n", U1(n[0]).x);
282     ok(U2(n[0]).y == 1.2f, "Wrong component n[0].y = %f (expected 1.2)\n", U2(n[0]).y);
283     ok(U3(n[0]).z == 1.3f, "Wrong component n[0].z = %f (expected 1.3)\n", U3(n[0]).z);
284     ok(U1(n[1]).x == 1.4f, "Wrong component n[1].x = %f (expected 1.4)\n", U1(n[1]).x);
285     ok(U2(n[1]).y == 1.5f, "Wrong component n[1].y = %f (expected 1.5)\n", U2(n[1]).y);
286     ok(U3(n[1]).z == 1.6f, "Wrong component n[1].z = %f (expected 1.6)\n", U3(n[1]).z);
287     ok(U1(n[2]).x == 1.7f, "Wrong component n[2].x = %f (expected 1.7)\n", U1(n[2]).x);
288     ok(U2(n[2]).y == 1.8f, "Wrong component n[2].y = %f (expected 1.8)\n", U2(n[2]).y);
289     ok(U3(n[2]).z == 1.9f, "Wrong component n[2].z = %f (expected 1.9)\n", U3(n[2]).z);
290     ok(f[0] == 3 , "Wrong component f[0] = %d (expected 3)\n", f[0]);
291     ok(f[1] == 0 , "Wrong component f[1] = %d (expected 0)\n", f[1]);
292     ok(f[2] == 0 , "Wrong component f[2] = %d (expected 0)\n", f[2]);
293     ok(f[3] == 1 , "Wrong component f[3] = %d (expected 1)\n", f[3]);
294     ok(f[4] == 1 , "Wrong component f[4] = %d (expected 1)\n", f[4]);
295     ok(f[5] == 2 , "Wrong component f[5] = %d (expected 2)\n", f[5]);
296     ok(f[6] == 2 , "Wrong component f[6] = %d (expected 2)\n", f[6]);
297     ok(f[7] == 0 , "Wrong component f[7] = %d (expected 0)\n", f[7]);
298
299     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
300
301     IDirect3DRM_Release(pD3DRM);
302 }
303
304 static void test_MeshBuilder3(void)
305 {
306     HRESULT hr;
307     LPDIRECT3DRM pD3DRM;
308     LPDIRECT3DRM3 pD3DRM3;
309     LPDIRECT3DRMMESHBUILDER3 pMeshBuilder3;
310     D3DRMLOADMEMORY info;
311     int val;
312     DWORD val1;
313     D3DVALUE valu, valv;
314
315     hr = pDirect3DRMCreate(&pD3DRM);
316     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
317
318     hr = IDirect3DRM_QueryInterface(pD3DRM, &IID_IDirect3DRM3, (LPVOID*)&pD3DRM3);
319     if (FAILED(hr))
320     {
321         win_skip("Cannot get IDirect3DRM3 interface (hr = %x), skipping tests\n", hr);
322         IDirect3DRM_Release(pD3DRM);
323         return;
324     }
325
326     hr = IDirect3DRM3_CreateMeshBuilder(pD3DRM3, &pMeshBuilder3);
327     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder3 interface (hr = %x)\n", hr);
328
329     info.lpMemory = data_bad_version;
330     info.dSize = strlen(data_bad_version);
331     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
332     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
333
334     info.lpMemory = data_no_mesh;
335     info.dSize = strlen(data_no_mesh);
336     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
337     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
338
339     info.lpMemory = data_ok;
340     info.dSize = strlen(data_ok);
341     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
342     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
343
344     val = IDirect3DRMMeshBuilder3_GetVertexCount(pMeshBuilder3);
345     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
346
347     val = IDirect3DRMMeshBuilder3_GetFaceCount(pMeshBuilder3);
348     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
349
350     hr = IDirect3DRMMeshBuilder3_GetVertices(pMeshBuilder3, 0, &val1, NULL);
351     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
352     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
353
354     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
355     valu = 1.23f;
356     valv = 3.21f;
357     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
358     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
359     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
360     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
361     valu = 1.23f;
362     valv = 3.21f;
363     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 1, &valu, &valv);
364     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
365     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
366     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
367     valu = 1.23f;
368     valv = 3.21f;
369     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 2, &valu, &valv);
370     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
371     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
372     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
373     valu = 1.23f;
374     valv = 3.21f;
375     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 3, &valu, &valv);
376     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
377     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
378     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
379     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 4, &valu, &valv);
380     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
381
382     valu = 1.23f;
383     valv = 3.21f;
384     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 0, valu, valv);
385     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
386     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 4, valu, valv);
387     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
388
389     valu = 0.0f;
390     valv = 0.0f;
391     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
392     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
393     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
394     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
395
396     IDirect3DRMMeshBuilder3_Release(pMeshBuilder3);
397     IDirect3DRM3_Release(pD3DRM3);
398     IDirect3DRM_Release(pD3DRM);
399 }
400
401 static void test_Frame(void)
402 {
403     HRESULT hr;
404     LPDIRECT3DRM pD3DRM;
405     LPDIRECT3DRMFRAME pFrameC;
406     LPDIRECT3DRMFRAME pFrameP1;
407     LPDIRECT3DRMFRAME pFrameP2;
408     LPDIRECT3DRMFRAME pFrameTmp;
409     LPDIRECT3DRMFRAMEARRAY pArray;
410     DWORD count;
411
412     hr = pDirect3DRMCreate(&pD3DRM);
413     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
414
415     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameC);
416     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
417     CHECK_REFCOUNT(pFrameC, 1);
418
419     pFrameTmp = (void*)0xdeadbeef;
420     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
421     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
422     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
423     CHECK_REFCOUNT(pFrameC, 1);
424
425     pArray = NULL;
426     hr = IDirect3DRMFrame_GetChildren(pFrameC, &pArray);
427     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
428     todo_wine ok(pArray != NULL, "pArray = %p\n", pArray);
429     if (pArray)
430     {
431         count = IDirect3DRMFrameArray_GetSize(pArray);
432         ok(count == 0, "count = %u\n", count);
433         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
434         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
435         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
436         IDirect3DRMFrameArray_Release(pArray);
437     }
438
439     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP1);
440     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
441
442     /* GetParent with NULL pointer */
443     hr = IDirect3DRMFrame_GetParent(pFrameP1, NULL);
444     todo_wine ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
445     CHECK_REFCOUNT(pFrameP1, 1);
446
447     /* [Add/Delete]Child with NULL pointer */
448     hr = IDirect3DRMFrame_AddChild(pFrameP1, NULL);
449     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
450     CHECK_REFCOUNT(pFrameP1, 1);
451
452     hr = IDirect3DRMFrame_DeleteChild(pFrameP1, NULL);
453     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
454     CHECK_REFCOUNT(pFrameP1, 1);
455
456     /* Add child to first parent */
457     pFrameTmp = (void*)0xdeadbeef;
458     hr = IDirect3DRMFrame_GetParent(pFrameP1, &pFrameTmp);
459     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
460     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
461
462     hr = IDirect3DRMFrame_AddChild(pFrameP1, pFrameC);
463     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
464     CHECK_REFCOUNT(pFrameP1, 1);
465     CHECK_REFCOUNT(pFrameC, 2);
466
467     pArray = NULL;
468     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
469     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
470     if (pArray)
471     {
472         count = IDirect3DRMFrameArray_GetSize(pArray);
473         ok(count == 1, "count = %u\n", count);
474         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
475         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
476         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
477         IDirect3DRMFrame_Release(pFrameTmp);
478         IDirect3DRMFrameArray_Release(pArray);
479     }
480
481     pFrameTmp = (void*)0xdeadbeef;
482     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
483     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
484     todo_wine ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
485     todo_wine CHECK_REFCOUNT(pFrameP1, 2);
486
487     /* Add child to second parent */
488     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP2);
489     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
490
491     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
492     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
493     todo_wine CHECK_REFCOUNT(pFrameC, 2);
494
495     pArray = NULL;
496     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
497     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
498     if (pArray)
499     {
500         count = IDirect3DRMFrameArray_GetSize(pArray);
501         ok(count == 1, "count = %u\n", count);
502         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
503         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
504         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
505         IDirect3DRMFrame_Release(pFrameTmp);
506         IDirect3DRMFrameArray_Release(pArray);
507     }
508
509     pArray = NULL;
510     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
511     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
512     if (pArray)
513     {
514         count = IDirect3DRMFrameArray_GetSize(pArray);
515         ok(count == 0, "count = %u\n", count);
516         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
517         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
518         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
519         IDirect3DRMFrameArray_Release(pArray);
520     }
521
522     pFrameTmp = (void*)0xdeadbeef;
523     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
524     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
525     todo_wine ok(pFrameTmp == pFrameP2, "pFrameTmp = %p\n", pFrameTmp);
526     todo_wine CHECK_REFCOUNT(pFrameP2, 2);
527     todo_wine CHECK_REFCOUNT(pFrameC, 2);
528
529     /* Add child again */
530     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
531     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
532     todo_wine CHECK_REFCOUNT(pFrameC, 2);
533
534     pArray = NULL;
535     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
536     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
537     if (pArray)
538     {
539         count = IDirect3DRMFrameArray_GetSize(pArray);
540         ok(count == 1, "count = %u\n", count);
541         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
542         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
543         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
544         IDirect3DRMFrame_Release(pFrameTmp);
545         IDirect3DRMFrameArray_Release(pArray);
546     }
547
548     /* Delete child */
549     hr = IDirect3DRMFrame_DeleteChild(pFrameP2, pFrameC);
550     ok(hr == D3DRM_OK, "Cannot delete child frame (hr = %x)\n", hr);
551     todo_wine CHECK_REFCOUNT(pFrameC, 1);
552
553     pArray = NULL;
554     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
555     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
556     if (pArray)
557     {
558         count = IDirect3DRMFrameArray_GetSize(pArray);
559         ok(count == 0, "count = %u\n", count);
560         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
561         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
562         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
563         IDirect3DRMFrameArray_Release(pArray);
564     }
565
566     pFrameTmp = (void*)0xdeadbeef;
567     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
568     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
569     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
570
571     /* Add two children */
572     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
573     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
574     todo_wine CHECK_REFCOUNT(pFrameC, 2);
575
576     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameP1);
577     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
578     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
579
580     pArray = NULL;
581     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
582     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
583     if (pArray)
584     {
585         count = IDirect3DRMFrameArray_GetSize(pArray);
586         ok(count == 2, "count = %u\n", count);
587         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
588         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
589         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
590         IDirect3DRMFrame_Release(pFrameTmp);
591         hr = IDirect3DRMFrameArray_GetElement(pArray, 1, &pFrameTmp);
592         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
593         ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
594         IDirect3DRMFrame_Release(pFrameTmp);
595         IDirect3DRMFrameArray_Release(pArray);
596     }
597
598     IDirect3DRMMeshBuilder_Release(pFrameP2);
599     CHECK_REFCOUNT(pFrameC, 2);
600     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
601
602     IDirect3DRMMeshBuilder_Release(pFrameC);
603     IDirect3DRMMeshBuilder_Release(pFrameP1);
604
605     IDirect3DRM_Release(pD3DRM);
606 }
607
608 static int nb_objects = 0;
609 static const GUID* refiids[] =
610 {
611     &IID_IDirect3DRMMeshBuilder,
612     &IID_IDirect3DRMMeshBuilder,
613     &IID_IDirect3DRMFrame
614 };
615
616 static void __cdecl object_load_callback(LPDIRECT3DRMOBJECT object, REFIID objectguid, LPVOID arg)
617 {
618     ok(object != NULL, "Arg 1 should not be null\n");
619     ok(IsEqualGUID(objectguid, refiids[nb_objects]), "Arg 2 should is incorrect\n");
620     ok(arg == (LPVOID)0xdeadbeef, "Arg 3 should be 0xdeadbeef (got %p)\n", arg);
621     nb_objects++;
622 }
623
624 static void test_d3drm_load(void)
625 {
626     HRESULT hr;
627     LPDIRECT3DRM pD3DRM;
628     D3DRMLOADMEMORY info;
629     const GUID* req_refiids[] = { &IID_IDirect3DRMMeshBuilder, &IID_IDirect3DRMFrame };
630
631     hr = pDirect3DRMCreate(&pD3DRM);
632     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
633
634     info.lpMemory = data_d3drm_load;
635     info.dSize = strlen(data_d3drm_load);
636     hr = IDirect3DRM_Load(pD3DRM, &info, NULL, (GUID**)req_refiids, 2, D3DRMLOAD_FROMMEMORY, object_load_callback, (LPVOID)0xdeadbeef, NULL, NULL, NULL);
637     ok(hr == D3DRM_OK, "Cannot load data (hr = %x)\n", hr);
638     ok(nb_objects == 3, "Should have loaded 3 objects (got %d)\n", nb_objects);
639
640     IDirect3DRM_Release(pD3DRM);
641 }
642
643 START_TEST(d3drm)
644 {
645     if (!InitFunctionPtrs())
646         return;
647
648     test_MeshBuilder();
649     test_MeshBuilder3();
650     test_Frame();
651     test_d3drm_load();
652
653     FreeLibrary(d3drm_handle);
654 }