d3drm/tests: Add tests for handling lights.
[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     LPDIRECT3DRMMESHBUILDER pMeshBuilder;
411     LPDIRECT3DRMVISUAL pVisual1;
412     LPDIRECT3DRMVISUAL pVisualTmp;
413     LPDIRECT3DRMVISUALARRAY pVisualArray;
414     LPDIRECT3DRMLIGHT pLight1;
415     LPDIRECT3DRMLIGHT pLightTmp;
416     LPDIRECT3DRMLIGHTARRAY pLightArray;
417     DWORD count;
418
419     hr = pDirect3DRMCreate(&pD3DRM);
420     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
421
422     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameC);
423     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
424     CHECK_REFCOUNT(pFrameC, 1);
425
426     pFrameTmp = (void*)0xdeadbeef;
427     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
428     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
429     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
430     CHECK_REFCOUNT(pFrameC, 1);
431
432     pArray = NULL;
433     hr = IDirect3DRMFrame_GetChildren(pFrameC, &pArray);
434     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
435     todo_wine ok(pArray != NULL, "pArray = %p\n", pArray);
436     if (pArray)
437     {
438         count = IDirect3DRMFrameArray_GetSize(pArray);
439         ok(count == 0, "count = %u\n", count);
440         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
441         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
442         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
443         IDirect3DRMFrameArray_Release(pArray);
444     }
445
446     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP1);
447     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
448
449     /* GetParent with NULL pointer */
450     hr = IDirect3DRMFrame_GetParent(pFrameP1, NULL);
451     todo_wine ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
452     CHECK_REFCOUNT(pFrameP1, 1);
453
454     /* [Add/Delete]Child with NULL pointer */
455     hr = IDirect3DRMFrame_AddChild(pFrameP1, NULL);
456     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
457     CHECK_REFCOUNT(pFrameP1, 1);
458
459     hr = IDirect3DRMFrame_DeleteChild(pFrameP1, NULL);
460     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
461     CHECK_REFCOUNT(pFrameP1, 1);
462
463     /* Add child to first parent */
464     pFrameTmp = (void*)0xdeadbeef;
465     hr = IDirect3DRMFrame_GetParent(pFrameP1, &pFrameTmp);
466     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
467     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
468
469     hr = IDirect3DRMFrame_AddChild(pFrameP1, pFrameC);
470     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
471     CHECK_REFCOUNT(pFrameP1, 1);
472     CHECK_REFCOUNT(pFrameC, 2);
473
474     pArray = NULL;
475     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
476     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
477     if (pArray)
478     {
479         count = IDirect3DRMFrameArray_GetSize(pArray);
480         ok(count == 1, "count = %u\n", count);
481         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
482         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
483         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
484         IDirect3DRMFrame_Release(pFrameTmp);
485         IDirect3DRMFrameArray_Release(pArray);
486     }
487
488     pFrameTmp = (void*)0xdeadbeef;
489     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
490     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
491     todo_wine ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
492     todo_wine CHECK_REFCOUNT(pFrameP1, 2);
493
494     /* Add child to second parent */
495     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP2);
496     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
497
498     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
499     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
500     todo_wine CHECK_REFCOUNT(pFrameC, 2);
501
502     pArray = NULL;
503     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
504     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
505     if (pArray)
506     {
507         count = IDirect3DRMFrameArray_GetSize(pArray);
508         ok(count == 1, "count = %u\n", count);
509         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
510         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
511         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
512         IDirect3DRMFrame_Release(pFrameTmp);
513         IDirect3DRMFrameArray_Release(pArray);
514     }
515
516     pArray = NULL;
517     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
518     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
519     if (pArray)
520     {
521         count = IDirect3DRMFrameArray_GetSize(pArray);
522         ok(count == 0, "count = %u\n", count);
523         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
524         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
525         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
526         IDirect3DRMFrameArray_Release(pArray);
527     }
528
529     pFrameTmp = (void*)0xdeadbeef;
530     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
531     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
532     todo_wine ok(pFrameTmp == pFrameP2, "pFrameTmp = %p\n", pFrameTmp);
533     todo_wine CHECK_REFCOUNT(pFrameP2, 2);
534     todo_wine CHECK_REFCOUNT(pFrameC, 2);
535
536     /* Add child again */
537     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
538     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
539     todo_wine CHECK_REFCOUNT(pFrameC, 2);
540
541     pArray = NULL;
542     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
543     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
544     if (pArray)
545     {
546         count = IDirect3DRMFrameArray_GetSize(pArray);
547         ok(count == 1, "count = %u\n", count);
548         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
549         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
550         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
551         IDirect3DRMFrame_Release(pFrameTmp);
552         IDirect3DRMFrameArray_Release(pArray);
553     }
554
555     /* Delete child */
556     hr = IDirect3DRMFrame_DeleteChild(pFrameP2, pFrameC);
557     ok(hr == D3DRM_OK, "Cannot delete child frame (hr = %x)\n", hr);
558     todo_wine CHECK_REFCOUNT(pFrameC, 1);
559
560     pArray = NULL;
561     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
562     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
563     if (pArray)
564     {
565         count = IDirect3DRMFrameArray_GetSize(pArray);
566         ok(count == 0, "count = %u\n", count);
567         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
568         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
569         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
570         IDirect3DRMFrameArray_Release(pArray);
571     }
572
573     pFrameTmp = (void*)0xdeadbeef;
574     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
575     todo_wine ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
576     todo_wine ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
577
578     /* Add two children */
579     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
580     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
581     todo_wine CHECK_REFCOUNT(pFrameC, 2);
582
583     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameP1);
584     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
585     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
586
587     pArray = NULL;
588     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
589     todo_wine ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
590     if (pArray)
591     {
592         count = IDirect3DRMFrameArray_GetSize(pArray);
593         ok(count == 2, "count = %u\n", count);
594         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
595         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
596         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
597         IDirect3DRMFrame_Release(pFrameTmp);
598         hr = IDirect3DRMFrameArray_GetElement(pArray, 1, &pFrameTmp);
599         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
600         ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
601         IDirect3DRMFrame_Release(pFrameTmp);
602         IDirect3DRMFrameArray_Release(pArray);
603     }
604
605     /* [Add/Delete]Visual with NULL pointer */
606     hr = IDirect3DRMFrame_AddVisual(pFrameP1, NULL);
607     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
608     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
609
610     hr = IDirect3DRMFrame_DeleteVisual(pFrameP1, NULL);
611     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
612     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
613
614     /* Create Visual */
615     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
616     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
617     pVisual1 = (LPDIRECT3DRMVISUAL)pMeshBuilder;
618
619     /* Add Visual to first parent */
620     hr = IDirect3DRMFrame_AddVisual(pFrameP1, pVisual1);
621     ok(hr == D3DRM_OK, "Cannot add visual (hr = %x)\n", hr);
622     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
623     CHECK_REFCOUNT(pVisual1, 2);
624
625     pVisualArray = NULL;
626     hr = IDirect3DRMFrame_GetVisuals(pFrameP1, &pVisualArray);
627     todo_wine ok(hr == D3DRM_OK, "Cannot get visuals (hr = %x)\n", hr);
628     if (pVisualArray)
629     {
630         count = IDirect3DRMVisualArray_GetSize(pVisualArray);
631         ok(count == 1, "count = %u\n", count);
632         hr = IDirect3DRMVisualArray_GetElement(pVisualArray, 0, &pVisualTmp);
633         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
634         ok(pVisualTmp == pVisual1, "pVisualTmp = %p\n", pVisualTmp);
635         IDirect3DRMVisual_Release(pVisualTmp);
636         IDirect3DRMVisualArray_Release(pVisualArray);
637     }
638
639     /* Delete Visual */
640     hr = IDirect3DRMFrame_DeleteVisual(pFrameP1, pVisual1);
641     ok(hr == D3DRM_OK, "Cannot delete visual (hr = %x)\n", hr);
642     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
643     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
644
645     /* [Add/Delete]Light with NULL pointer */
646     hr = IDirect3DRMFrame_AddLight(pFrameP1, NULL);
647     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
648     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
649
650     hr = IDirect3DRMFrame_DeleteLight(pFrameP1, NULL);
651     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
652     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
653
654     /* Create Light */
655     hr = IDirect3DRM_CreateLightRGB(pD3DRM, D3DRMLIGHT_SPOT, 0.1, 0.2, 0.3, &pLight1);
656     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMLight interface (hr = %x)\n", hr);
657
658     /* Add Light to first parent */
659     hr = IDirect3DRMFrame_AddLight(pFrameP1, pLight1);
660     ok(hr == D3DRM_OK, "Cannot add light (hr = %x)\n", hr);
661     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
662     CHECK_REFCOUNT(pLight1, 2);
663
664     pLightArray = NULL;
665     hr = IDirect3DRMFrame_GetLights(pFrameP1, &pLightArray);
666     todo_wine ok(hr == D3DRM_OK, "Cannot get lights (hr = %x)\n", hr);
667     if (pLightArray)
668     {
669         count = IDirect3DRMLightArray_GetSize(pLightArray);
670         ok(count == 1, "count = %u\n", count);
671         hr = IDirect3DRMLightArray_GetElement(pLightArray, 0, &pLightTmp);
672         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
673         ok(pLightTmp == pLight1, "pLightTmp = %p\n", pLightTmp);
674         IDirect3DRMLight_Release(pLightTmp);
675         IDirect3DRMLightArray_Release(pLightArray);
676     }
677
678     /* Delete Light */
679     hr = IDirect3DRMFrame_DeleteLight(pFrameP1, pLight1);
680     ok(hr == D3DRM_OK, "Cannot delete light (hr = %x)\n", hr);
681     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
682     IDirect3DRMLight_Release(pLight1);
683
684     /* Cleanup */
685     IDirect3DRMFrame_Release(pFrameP2);
686     CHECK_REFCOUNT(pFrameC, 2);
687     todo_wine CHECK_REFCOUNT(pFrameP1, 3);
688
689     IDirect3DRMFrame_Release(pFrameC);
690     IDirect3DRMFrame_Release(pFrameP1);
691
692     IDirect3DRM_Release(pD3DRM);
693 }
694
695 static void test_Light(void)
696 {
697     HRESULT hr;
698     LPDIRECT3DRM pD3DRM;
699     LPDIRECT3DRMLIGHT pLight;
700     D3DRMLIGHTTYPE type;
701     D3DCOLOR color;
702
703     hr = pDirect3DRMCreate(&pD3DRM);
704     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
705
706     hr = IDirect3DRM_CreateLightRGB(pD3DRM, D3DRMLIGHT_SPOT, 0.5, 0.5, 0.5, &pLight);
707     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMLight interface (hr = %x)\n", hr);
708
709     type = IDirect3DRMLight_GetType(pLight);
710     todo_wine ok(type == D3DRMLIGHT_SPOT, "wrong type (%u)\n", type);
711
712     color = IDirect3DRMLight_GetColor(pLight);
713     todo_wine ok(color == 0xff7f7f7f, "wrong color (%x)\n", color);
714
715     hr = IDirect3DRMLight_SetType(pLight, D3DRMLIGHT_POINT);
716     todo_wine ok(hr == D3DRM_OK, "Cannot set type (hr = %x)\n", hr);
717     type = IDirect3DRMLight_GetType(pLight);
718     todo_wine ok(type == D3DRMLIGHT_POINT, "wrong type (%u)\n", type);
719
720     hr = IDirect3DRMLight_SetColor(pLight, 0xff180587);
721     todo_wine ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
722     color = IDirect3DRMLight_GetColor(pLight);
723     todo_wine ok(color == 0xff180587, "wrong color (%x)\n", color);
724
725     hr = IDirect3DRMLight_SetColorRGB(pLight, 0.5, 0.5, 0.5);
726     todo_wine ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
727     color = IDirect3DRMLight_GetColor(pLight);
728     todo_wine ok(color == 0xff7f7f7f, "wrong color (%x)\n", color);
729
730     IDirect3DRMLight_Release(pLight);
731
732     IDirect3DRM_Release(pD3DRM);
733 }
734
735 static int nb_objects = 0;
736 static const GUID* refiids[] =
737 {
738     &IID_IDirect3DRMMeshBuilder,
739     &IID_IDirect3DRMMeshBuilder,
740     &IID_IDirect3DRMFrame
741 };
742
743 static void __cdecl object_load_callback(LPDIRECT3DRMOBJECT object, REFIID objectguid, LPVOID arg)
744 {
745     ok(object != NULL, "Arg 1 should not be null\n");
746     ok(IsEqualGUID(objectguid, refiids[nb_objects]), "Arg 2 should is incorrect\n");
747     ok(arg == (LPVOID)0xdeadbeef, "Arg 3 should be 0xdeadbeef (got %p)\n", arg);
748     nb_objects++;
749 }
750
751 static void test_d3drm_load(void)
752 {
753     HRESULT hr;
754     LPDIRECT3DRM pD3DRM;
755     D3DRMLOADMEMORY info;
756     const GUID* req_refiids[] = { &IID_IDirect3DRMMeshBuilder, &IID_IDirect3DRMFrame };
757
758     hr = pDirect3DRMCreate(&pD3DRM);
759     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
760
761     info.lpMemory = data_d3drm_load;
762     info.dSize = strlen(data_d3drm_load);
763     hr = IDirect3DRM_Load(pD3DRM, &info, NULL, (GUID**)req_refiids, 2, D3DRMLOAD_FROMMEMORY, object_load_callback, (LPVOID)0xdeadbeef, NULL, NULL, NULL);
764     ok(hr == D3DRM_OK, "Cannot load data (hr = %x)\n", hr);
765     ok(nb_objects == 3, "Should have loaded 3 objects (got %d)\n", nb_objects);
766
767     IDirect3DRM_Release(pD3DRM);
768 }
769
770 START_TEST(d3drm)
771 {
772     if (!InitFunctionPtrs())
773         return;
774
775     test_MeshBuilder();
776     test_MeshBuilder3();
777     test_Frame();
778     test_Light();
779     test_d3drm_load();
780
781     FreeLibrary(d3drm_handle);
782 }