uxtheme: Remove unused variable.
[wine] / dlls / d3drm / tests / d3drm.c
1 /*
2  * Copyright 2010, 2012 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 #include <d3drmwin.h>
24
25 #include "wine/test.h"
26
27 static HMODULE d3drm_handle = 0;
28
29 static HRESULT (WINAPI * pDirect3DRMCreate)(LPDIRECT3DRM* ppDirect3DRM);
30
31 #define CHECK_REFCOUNT(obj,rc) \
32     { \
33         int rc_new = rc; \
34         int count = get_refcount( (IUnknown *)obj ); \
35         ok(count == rc_new, "Invalid refcount. Expected %d got %d\n", rc_new, count); \
36     }
37
38 #define D3DRM_GET_PROC(func) \
39     p ## func = (void*)GetProcAddress(d3drm_handle, #func); \
40     if(!p ## func) { \
41       trace("GetProcAddress(%s) failed\n", #func); \
42       FreeLibrary(d3drm_handle); \
43       return FALSE; \
44     }
45
46 static BOOL InitFunctionPtrs(void)
47 {
48     d3drm_handle = LoadLibraryA("d3drm.dll");
49
50     if(!d3drm_handle)
51     {
52         skip("Could not load d3drm.dll\n");
53         return FALSE;
54     }
55
56     D3DRM_GET_PROC(Direct3DRMCreate)
57
58     return TRUE;
59 }
60
61 static int get_refcount(IUnknown *object)
62 {
63     IUnknown_AddRef( object );
64     return IUnknown_Release( object );
65 }
66
67 static BOOL match_float(float a, float b)
68 {
69     return (a - b) < 0.000001f;
70 }
71
72 static D3DRMMATRIX4D identity = {
73     { 1.0f, 0.0f, 0.0f, 0.0f },
74     { 0.0f, 1.0f, 0.0f, 0.0f },
75     { 0.0f, 0.0f, 1.0f, 0.0f },
76     { 0.0f, 0.0f, 0.0f, 1.0f }
77 };
78
79 static char data_bad_version[] =
80 "xof 0302txt 0064\n"
81 "Header Object\n"
82 "{\n"
83 "1; 2; 3;\n"
84 "}\n";
85
86 static char data_no_mesh[] =
87 "xof 0302txt 0064\n"
88 "Header Object\n"
89 "{\n"
90 "1; 0; 1;\n"
91 "}\n";
92
93 static char data_ok[] =
94 "xof 0302txt 0064\n"
95 "Header Object\n"
96 "{\n"
97 "1; 0; 1;\n"
98 "}\n"
99 "Mesh Object\n"
100 "{\n"
101 "4;\n"
102 "1.0; 0.0; 0.0;,\n"
103 "0.0; 1.0; 0.0;,\n"
104 "0.0; 0.0; 1.0;,\n"
105 "1.0; 1.0; 1.0;;\n"
106 "3;\n"
107 "3; 0, 1, 2;,\n"
108 "3; 1, 2, 3;,\n"
109 "3; 3, 1, 2;;\n"
110 "}\n";
111
112 static char data_full[] =
113 "xof 0302txt 0064\n"
114 "Header { 1; 0; 1; }\n"
115 "Mesh {\n"
116 " 3;\n"
117 " 0.1; 0.2; 0.3;,\n"
118 " 0.4; 0.5; 0.6;,\n"
119 " 0.7; 0.8; 0.9;;\n"
120 " 1;\n"
121 " 3; 0, 1, 2;;\n"
122 " MeshMaterialList {\n"
123 "  1; 1; 0;\n"
124 "  Material {\n"
125 "   0.0; 1.0; 0.0; 1.0;;\n"
126 "   30.0;\n"
127 "   1.0; 0.0; 0.0;;\n"
128 "   0.5; 0.5; 0.5;;\n"
129 "   TextureFileName {\n"
130 "    \"Texture.bmp\";\n"
131 "   }\n"
132 "  }\n"
133 " }\n"
134 " MeshNormals {\n"
135 "  3;\n"
136 "  1.1; 1.2; 1.3;,\n"
137 "  1.4; 1.5; 1.6;,\n"
138 "  1.7; 1.8; 1.9;;\n"
139 "  1;"
140 "  3; 0, 1, 2;;\n"
141 " }\n"
142 " MeshTextureCoords {\n"
143 "  3;\n"
144 "  0.13; 0.17;,\n"
145 "  0.23; 0.27;,\n"
146 "  0.33; 0.37;;\n"
147 " }\n"
148 "}\n";
149
150 static char data_d3drm_load[] =
151 "xof 0302txt 0064\n"
152 "Header Object\n"
153 "{\n"
154 "1; 0; 1;\n"
155 "}\n"
156 "Mesh Object1\n"
157 "{\n"
158 " 1;\n"
159 " 0.1; 0.2; 0.3;,\n"
160 " 1;\n"
161 " 3; 0, 1, 2;;\n"
162 "}\n"
163 "Mesh Object2\n"
164 "{\n"
165 " 1;\n"
166 " 0.1; 0.2; 0.3;,\n"
167 " 1;\n"
168 " 3; 0, 1, 2;;\n"
169 "}\n"
170 "Frame Scene\n"
171 "{\n"
172 " {Object1}\n"
173 " {Object2}\n"
174 "}\n"
175 "Material\n"
176 "{\n"
177 " 0.1, 0.2, 0.3, 0.4;;\n"
178 " 0.5;\n"
179 " 0.6, 0.7, 0.8;;\n"
180 " 0.9, 1.0, 1.1;;\n"
181 "}\n";
182
183 static void test_MeshBuilder(void)
184 {
185     HRESULT hr;
186     LPDIRECT3DRM pD3DRM;
187     LPDIRECT3DRMMESHBUILDER pMeshBuilder;
188     LPDIRECT3DRMMESH mesh;
189     D3DRMLOADMEMORY info;
190     int val;
191     DWORD val1, val2, val3;
192     D3DVALUE valu, valv;
193     D3DVECTOR v[3];
194     D3DVECTOR n[4];
195     DWORD f[8];
196     char name[10];
197     DWORD size;
198     D3DCOLOR color;
199     CHAR cname[64] = {0};
200
201     hr = pDirect3DRMCreate(&pD3DRM);
202     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
203
204     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
205     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
206
207     hr = IDirect3DRMMeshBuilder_GetClassName(pMeshBuilder, NULL, cname);
208     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
209     hr = IDirect3DRMMeshBuilder_GetClassName(pMeshBuilder, NULL, NULL);
210     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
211     size = 1;
212     hr = IDirect3DRMMeshBuilder_GetClassName(pMeshBuilder, &size, cname);
213     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
214     size = sizeof(cname);
215     hr = IDirect3DRMMeshBuilder_GetClassName(pMeshBuilder, &size, cname);
216     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
217     ok(size == sizeof("Builder"), "wrong size: %u\n", size);
218     ok(!strcmp(cname, "Builder"), "Expected cname to be \"Builder\", but got \"%s\"\n", cname);
219
220     info.lpMemory = data_bad_version;
221     info.dSize = strlen(data_bad_version);
222     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
223     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
224
225     info.lpMemory = data_no_mesh;
226     info.dSize = strlen(data_no_mesh);
227     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
228     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
229
230     info.lpMemory = data_ok;
231     info.dSize = strlen(data_ok);
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     size = sizeof(name);
236     hr = IDirect3DRMMeshBuilder_GetName(pMeshBuilder, &size, name);
237     ok(hr == D3DRM_OK, "IDirect3DRMMeshBuilder_GetName returned hr = %x\n", hr);
238     ok(!strcmp(name, "Object"), "Retrieved name '%s' instead of 'Object'\n", name);
239     size = strlen("Object"); /* No space for null character */
240     hr = IDirect3DRMMeshBuilder_GetName(pMeshBuilder, &size, name);
241     ok(hr == E_INVALIDARG, "IDirect3DRMMeshBuilder_GetName returned hr = %x\n", hr);
242     hr = IDirect3DRMMeshBuilder_SetName(pMeshBuilder, NULL);
243     ok(hr == D3DRM_OK, "IDirect3DRMMeshBuilder_SetName returned hr = %x\n", hr);
244     size = sizeof(name);
245     hr = IDirect3DRMMeshBuilder_GetName(pMeshBuilder, &size, name);
246     ok(hr == D3DRM_OK, "IDirect3DRMMeshBuilder_GetName returned hr = %x\n", hr);
247     ok(size == 0, "Size should be 0 instead of %u\n", size);
248     hr = IDirect3DRMMeshBuilder_SetName(pMeshBuilder, "");
249     ok(hr == D3DRM_OK, "IDirect3DRMMeshBuilder_SetName returned hr = %x\n", hr);
250     size = sizeof(name);
251     hr = IDirect3DRMMeshBuilder_GetName(pMeshBuilder, &size, name);
252     ok(hr == D3DRM_OK, "IDirect3DRMMeshBuilder_GetName returned hr = %x\n", hr);
253     ok(!strcmp(name, ""), "Retrieved name '%s' instead of ''\n", name);
254
255     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
256     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
257
258     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
259     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
260
261     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, NULL, &val2, NULL, &val3, NULL);
262     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
263     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
264     ok(val2 == 4, "Wrong number of normals %d (must be 4)\n", val2);
265     ok(val3 == 22, "Wrong number of face data bytes %d (must be 22)\n", val3);
266
267     /* Check that Load method generated default normals */
268     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, NULL, NULL, &val2, n, NULL, NULL);
269     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
270     ok(match_float(U1(n[0]).x, 0.577350f),  "Wrong component n[0].x = %f (expected %f)\n", U1(n[0]).x, 0.577350f);
271     ok(match_float(U2(n[0]).y, 0.577350f),  "Wrong component n[0].y = %f (expected %f)\n", U2(n[0]).y, 0.577350f);
272     ok(match_float(U3(n[0]).z, 0.577350f),  "Wrong component n[0].z = %f (expected %f)\n", U3(n[0]).z, 0.577350f);
273     ok(match_float(U1(n[1]).x, -0.229416f), "Wrong component n[1].x = %f (expected %f)\n", U1(n[1]).x, -0.229416f);
274     ok(match_float(U2(n[1]).y, 0.688247f),  "Wrong component n[1].y = %f (expected %f)\n", U2(n[1]).y, 0.688247f);
275     ok(match_float(U3(n[1]).z, 0.688247f),  "Wrong component n[1].z = %f (expected %f)\n", U3(n[1]).z, 0.688247f);
276     ok(match_float(U1(n[2]).x, -0.229416f), "Wrong component n[2].x = %f (expected %f)\n", U1(n[2]).x, -0.229416f);
277     ok(match_float(U2(n[2]).y, 0.688247f),  "Wrong component n[2].y = %f (expected %f)\n", U2(n[2]).y, 0.688247f);
278     ok(match_float(U3(n[2]).z, 0.688247f),  "Wrong component n[2].z = %f (expected %f)\n", U3(n[2]).z, 0.688247f);
279     ok(match_float(U1(n[3]).x, -0.577350f), "Wrong component n[3].x = %f (expected %f)\n", U1(n[3]).x, -0.577350f);
280     ok(match_float(U2(n[3]).y, 0.577350f),  "Wrong component n[3].y = %f (expected %f)\n", U2(n[3]).y, 0.577350f);
281     ok(match_float(U3(n[3]).z, 0.577350f),  "Wrong component n[3].z = %f (expected %f)\n", U3(n[3]).z, 0.577350f);
282
283     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
284     valu = 1.23f;
285     valv = 3.21f;
286     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
287     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
288     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
289     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
290     valu = 1.23f;
291     valv = 3.21f;
292     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 1, &valu, &valv);
293     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
294     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
295     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
296     valu = 1.23f;
297     valv = 3.21f;
298     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 2, &valu, &valv);
299     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
300     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
301     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
302     valu = 1.23f;
303     valv = 3.21f;
304     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 3, &valu, &valv);
305     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
306     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
307     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
308     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 4, &valu, &valv);
309     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
310
311     valu = 1.23f;
312     valv = 3.21f;
313     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 0, valu, valv);
314     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
315     hr = IDirect3DRMMeshBuilder_SetTextureCoordinates(pMeshBuilder, 4, valu, valv);
316     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
317
318     valu = 0.0f;
319     valv = 0.0f;
320     hr = IDirect3DRMMeshBuilder_GetTextureCoordinates(pMeshBuilder, 0, &valu, &valv);
321     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
322     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
323     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
324
325     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
326
327     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
328     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
329
330     /* No group in mesh when mesh builder is not loaded */
331     hr = IDirect3DRMMeshBuilder_CreateMesh(pMeshBuilder, &mesh);
332     ok(hr == D3DRM_OK, "CreateMesh failed returning hr = %x\n", hr);
333     if (hr == D3DRM_OK)
334     {
335         DWORD nb_groups;
336
337         nb_groups = IDirect3DRMMesh_GetGroupCount(mesh);
338         ok(nb_groups == 0, "GetCroupCount returned %u\n", nb_groups);
339
340         IDirect3DRMMesh_Release(mesh);
341     }
342
343     info.lpMemory = data_full;
344     info.dSize = strlen(data_full);
345     hr = IDirect3DRMMeshBuilder_Load(pMeshBuilder, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
346     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
347
348     val = IDirect3DRMMeshBuilder_GetVertexCount(pMeshBuilder);
349     ok(val == 3, "Wrong number of vertices %d (must be 3)\n", val);
350
351     val = IDirect3DRMMeshBuilder_GetFaceCount(pMeshBuilder);
352     ok(val == 1, "Wrong number of faces %d (must be 1)\n", val);
353
354     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, v, &val2, n, &val3, f);
355     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
356     ok(val1 == 3, "Wrong number of vertices %d (must be 3)\n", val1);
357     ok(val2 == 3, "Wrong number of normals %d (must be 3)\n", val2);
358     ok(val3 == 8, "Wrong number of face data bytes %d (must be 8)\n", val3);
359     ok(U1(v[0]).x == 0.1f, "Wrong component v[0].x = %f (expected 0.1)\n", U1(v[0]).x);
360     ok(U2(v[0]).y == 0.2f, "Wrong component v[0].y = %f (expected 0.2)\n", U2(v[0]).y);
361     ok(U3(v[0]).z == 0.3f, "Wrong component v[0].z = %f (expected 0.3)\n", U3(v[0]).z);
362     ok(U1(v[1]).x == 0.4f, "Wrong component v[1].x = %f (expected 0.4)\n", U1(v[1]).x);
363     ok(U2(v[1]).y == 0.5f, "Wrong component v[1].y = %f (expected 0.5)\n", U2(v[1]).y);
364     ok(U3(v[1]).z == 0.6f, "Wrong component v[1].z = %f (expected 0.6)\n", U3(v[1]).z);
365     ok(U1(v[2]).x == 0.7f, "Wrong component v[2].x = %f (expected 0.7)\n", U1(v[2]).x);
366     ok(U2(v[2]).y == 0.8f, "Wrong component v[2].y = %f (expected 0.8)\n", U2(v[2]).y);
367     ok(U3(v[2]).z == 0.9f, "Wrong component v[2].z = %f (expected 0.9)\n", U3(v[2]).z);
368     ok(U1(n[0]).x == 1.1f, "Wrong component n[0].x = %f (expected 1.1)\n", U1(n[0]).x);
369     ok(U2(n[0]).y == 1.2f, "Wrong component n[0].y = %f (expected 1.2)\n", U2(n[0]).y);
370     ok(U3(n[0]).z == 1.3f, "Wrong component n[0].z = %f (expected 1.3)\n", U3(n[0]).z);
371     ok(U1(n[1]).x == 1.4f, "Wrong component n[1].x = %f (expected 1.4)\n", U1(n[1]).x);
372     ok(U2(n[1]).y == 1.5f, "Wrong component n[1].y = %f (expected 1.5)\n", U2(n[1]).y);
373     ok(U3(n[1]).z == 1.6f, "Wrong component n[1].z = %f (expected 1.6)\n", U3(n[1]).z);
374     ok(U1(n[2]).x == 1.7f, "Wrong component n[2].x = %f (expected 1.7)\n", U1(n[2]).x);
375     ok(U2(n[2]).y == 1.8f, "Wrong component n[2].y = %f (expected 1.8)\n", U2(n[2]).y);
376     ok(U3(n[2]).z == 1.9f, "Wrong component n[2].z = %f (expected 1.9)\n", U3(n[2]).z);
377     ok(f[0] == 3 , "Wrong component f[0] = %d (expected 3)\n", f[0]);
378     ok(f[1] == 0 , "Wrong component f[1] = %d (expected 0)\n", f[1]);
379     ok(f[2] == 0 , "Wrong component f[2] = %d (expected 0)\n", f[2]);
380     ok(f[3] == 1 , "Wrong component f[3] = %d (expected 1)\n", f[3]);
381     ok(f[4] == 1 , "Wrong component f[4] = %d (expected 1)\n", f[4]);
382     ok(f[5] == 2 , "Wrong component f[5] = %d (expected 2)\n", f[5]);
383     ok(f[6] == 2 , "Wrong component f[6] = %d (expected 2)\n", f[6]);
384     ok(f[7] == 0 , "Wrong component f[7] = %d (expected 0)\n", f[7]);
385
386     hr = IDirect3DRMMeshBuilder_CreateMesh(pMeshBuilder, &mesh);
387     ok(hr == D3DRM_OK, "CreateMesh failed returning hr = %x\n", hr);
388     if (hr == D3DRM_OK)
389     {
390         DWORD nb_groups;
391         unsigned nb_vertices, nb_faces, nb_face_vertices;
392         DWORD data_size;
393         LPDIRECT3DRMMATERIAL material = (LPDIRECT3DRMMATERIAL)0xdeadbeef;
394         LPDIRECT3DRMTEXTURE texture = (LPDIRECT3DRMTEXTURE)0xdeadbeef;
395         D3DVALUE values[3];
396
397         nb_groups = IDirect3DRMMesh_GetGroupCount(mesh);
398         ok(nb_groups == 1, "GetCroupCount returned %u\n", nb_groups);
399         hr = IDirect3DRMMesh_GetGroup(mesh, 1, &nb_vertices, &nb_faces, &nb_face_vertices, &data_size, NULL);
400         ok(hr == D3DRMERR_BADVALUE, "GetCroup returned hr = %x\n", hr);
401         hr = IDirect3DRMMesh_GetGroup(mesh, 0, &nb_vertices, &nb_faces, &nb_face_vertices, &data_size, NULL);
402         ok(hr == D3DRM_OK, "GetCroup failed returning hr = %x\n", hr);
403         ok(nb_vertices == 3, "Wrong number of vertices %u (must be 3)\n", nb_vertices);
404         ok(nb_faces == 1, "Wrong number of faces %u (must be 1)\n", nb_faces);
405         ok(nb_face_vertices == 3, "Wrong number of vertices per face %u (must be 3)\n", nb_face_vertices);
406         ok(data_size == 3, "Wrong number of face data bytes %u (must be 3)\n", data_size);
407         color = IDirect3DRMMesh_GetGroupColor(mesh, 0);
408         ok(color == 0xff00ff00, "Wrong color returned %#x instead of %#x\n", color, 0xff00ff00);
409         hr = IDirect3DRMMesh_GetGroupTexture(mesh, 0, &texture);
410         ok(hr == D3DRM_OK, "GetCroupTexture failed returning hr = %x\n", hr);
411         ok(texture == NULL, "No texture should be present\n");
412         hr = IDirect3DRMMesh_GetGroupMaterial(mesh, 0, &material);
413         ok(hr == D3DRM_OK, "GetCroupMaterial failed returning hr = %x\n", hr);
414         ok(material != NULL, "No material present\n");
415         if ((hr == D3DRM_OK) && material)
416         {
417             hr = IDirect3DRMMaterial_GetEmissive(material, &values[0], &values[1], &values[2]);
418             ok(hr == D3DRM_OK, "GetMaterialEmissive failed returning hr = %x\n", hr);
419             ok(values[0] == 0.5f, "Emissive red component should be %f instead of %f\n", 0.5f, values[0]);
420             ok(values[1] == 0.5f, "Emissive green component should be %f instead of %f\n", 0.5f, values[1]);
421             ok(values[2] == 0.5f, "Emissive blue component should be %f instead of %f\n", 0.5f, values[2]);
422             hr = IDirect3DRMMaterial_GetSpecular(material, &values[0], &values[1], &values[2]);
423             ok(hr == D3DRM_OK, "GetMaterialEmissive failed returning hr = %x\n", hr);
424             ok(values[0] == 1.0f, "Specular red component should be %f instead of %f\n", 1.0f, values[0]);
425             ok(values[1] == 0.0f, "Specular green component should be %f instead of %f\n", 0.0f, values[1]);
426             ok(values[2] == 0.0f, "Specular blue component should be %f instead of %f\n", 0.0f, values[2]);
427             values[0] = IDirect3DRMMaterial_GetPower(material);
428             ok(values[0] == 30.0f, "Power value should be %f instead of %f\n", 30.0f, values[0]);
429         }
430
431         IDirect3DRMMesh_Release(mesh);
432     }
433
434     hr = IDirect3DRMMeshBuilder_Scale(pMeshBuilder, 2, 3 ,4);
435     ok(hr == D3DRM_OK, "Scale failed returning hr = %x\n", hr);
436
437     hr = IDirect3DRMMeshBuilder_GetVertices(pMeshBuilder, &val1, v, &val2, n, &val3, f);
438     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
439     ok(val2 == 3, "Wrong number of normals %d (must be 3)\n", val2);
440     ok(val1 == 3, "Wrong number of vertices %d (must be 3)\n", val1);
441     ok(U1(v[0]).x == 0.1f*2, "Wrong component v[0].x = %f (expected %f)\n", U1(v[0]).x, 0.1f*2);
442     ok(U2(v[0]).y == 0.2f*3, "Wrong component v[0].y = %f (expected %f)\n", U2(v[0]).y, 0.2f*3);
443     ok(U3(v[0]).z == 0.3f*4, "Wrong component v[0].z = %f (expected %f)\n", U3(v[0]).z, 0.3f*4);
444     ok(U1(v[1]).x == 0.4f*2, "Wrong component v[1].x = %f (expected %f)\n", U1(v[1]).x, 0.4f*2);
445     ok(U2(v[1]).y == 0.5f*3, "Wrong component v[1].y = %f (expected %f)\n", U2(v[1]).y, 0.5f*3);
446     ok(U3(v[1]).z == 0.6f*4, "Wrong component v[1].z = %f (expected %f)\n", U3(v[1]).z, 0.6f*4);
447     ok(U1(v[2]).x == 0.7f*2, "Wrong component v[2].x = %f (expected %f)\n", U1(v[2]).x, 0.7f*2);
448     ok(U2(v[2]).y == 0.8f*3, "Wrong component v[2].y = %f (expected %f)\n", U2(v[2]).y, 0.8f*3);
449     ok(U3(v[2]).z == 0.9f*4, "Wrong component v[2].z = %f (expected %f)\n", U3(v[2]).z, 0.9f*4);
450     /* Normals are not affected by Scale */
451     ok(U1(n[0]).x == 1.1f, "Wrong component n[0].x = %f (expected %f)\n", U1(n[0]).x, 1.1f);
452     ok(U2(n[0]).y == 1.2f, "Wrong component n[0].y = %f (expected %f)\n", U2(n[0]).y, 1.2f);
453     ok(U3(n[0]).z == 1.3f, "Wrong component n[0].z = %f (expected %f)\n", U3(n[0]).z, 1.3f);
454     ok(U1(n[1]).x == 1.4f, "Wrong component n[1].x = %f (expected %f)\n", U1(n[1]).x, 1.4f);
455     ok(U2(n[1]).y == 1.5f, "Wrong component n[1].y = %f (expected %f)\n", U2(n[1]).y, 1.5f);
456     ok(U3(n[1]).z == 1.6f, "Wrong component n[1].z = %f (expected %f)\n", U3(n[1]).z, 1.6f);
457     ok(U1(n[2]).x == 1.7f, "Wrong component n[2].x = %f (expected %f)\n", U1(n[2]).x, 1.7f);
458     ok(U2(n[2]).y == 1.8f, "Wrong component n[2].y = %f (expected %f)\n", U2(n[2]).y, 1.8f);
459     ok(U3(n[2]).z == 1.9f, "Wrong component n[2].z = %f (expected %f)\n", U3(n[2]).z, 1.9f);
460
461     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
462
463     IDirect3DRM_Release(pD3DRM);
464 }
465
466 static void test_MeshBuilder3(void)
467 {
468     HRESULT hr;
469     LPDIRECT3DRM pD3DRM;
470     LPDIRECT3DRM3 pD3DRM3;
471     LPDIRECT3DRMMESHBUILDER3 pMeshBuilder3;
472     D3DRMLOADMEMORY info;
473     int val;
474     DWORD val1;
475     D3DVALUE valu, valv;
476     DWORD size;
477     CHAR cname[64] = {0};
478
479     hr = pDirect3DRMCreate(&pD3DRM);
480     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
481
482     hr = IDirect3DRM_QueryInterface(pD3DRM, &IID_IDirect3DRM3, (LPVOID*)&pD3DRM3);
483     if (FAILED(hr))
484     {
485         win_skip("Cannot get IDirect3DRM3 interface (hr = %x), skipping tests\n", hr);
486         IDirect3DRM_Release(pD3DRM);
487         return;
488     }
489
490     hr = IDirect3DRM3_CreateMeshBuilder(pD3DRM3, &pMeshBuilder3);
491     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder3 interface (hr = %x)\n", hr);
492
493     hr = IDirect3DRMMeshBuilder3_GetClassName(pMeshBuilder3, NULL, cname);
494     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
495     hr = IDirect3DRMMeshBuilder3_GetClassName(pMeshBuilder3, NULL, NULL);
496     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
497     size = 1;
498     hr = IDirect3DRMMeshBuilder3_GetClassName(pMeshBuilder3, &size, cname);
499     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
500     size = sizeof(cname);
501     hr = IDirect3DRMMeshBuilder3_GetClassName(pMeshBuilder3, &size, cname);
502     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
503     ok(size == sizeof("Builder"), "wrong size: %u\n", size);
504     ok(!strcmp(cname, "Builder"), "Expected cname to be \"Builder\", but got \"%s\"\n", cname);
505
506     info.lpMemory = data_bad_version;
507     info.dSize = strlen(data_bad_version);
508     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
509     ok(hr == D3DRMERR_BADFILE, "Should have returned D3DRMERR_BADFILE (hr = %x)\n", hr);
510
511     info.lpMemory = data_no_mesh;
512     info.dSize = strlen(data_no_mesh);
513     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
514     ok(hr == D3DRMERR_NOTFOUND, "Should have returned D3DRMERR_NOTFOUND (hr = %x)\n", hr);
515
516     info.lpMemory = data_ok;
517     info.dSize = strlen(data_ok);
518     hr = IDirect3DRMMeshBuilder3_Load(pMeshBuilder3, &info, NULL, D3DRMLOAD_FROMMEMORY, NULL, NULL);
519     ok(hr == D3DRM_OK, "Cannot load mesh data (hr = %x)\n", hr);
520
521     val = IDirect3DRMMeshBuilder3_GetVertexCount(pMeshBuilder3);
522     ok(val == 4, "Wrong number of vertices %d (must be 4)\n", val);
523
524     val = IDirect3DRMMeshBuilder3_GetFaceCount(pMeshBuilder3);
525     ok(val == 3, "Wrong number of faces %d (must be 3)\n", val);
526
527     hr = IDirect3DRMMeshBuilder3_GetVertices(pMeshBuilder3, 0, &val1, NULL);
528     ok(hr == D3DRM_OK, "Cannot get vertices information (hr = %x)\n", hr);
529     ok(val1 == 4, "Wrong number of vertices %d (must be 4)\n", val1);
530
531     /* Check that Load method generated default texture coordinates (0.0f, 0.0f) for each vertex */
532     valu = 1.23f;
533     valv = 3.21f;
534     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
535     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
536     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
537     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
538     valu = 1.23f;
539     valv = 3.21f;
540     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 1, &valu, &valv);
541     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
542     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
543     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
544     valu = 1.23f;
545     valv = 3.21f;
546     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 2, &valu, &valv);
547     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
548     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
549     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
550     valu = 1.23f;
551     valv = 3.21f;
552     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 3, &valu, &valv);
553     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
554     ok(valu == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valu);
555     ok(valv == 0.0f, "Wrong coordinate %f (must be 0.0)\n", valv);
556     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 4, &valu, &valv);
557     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
558
559     valu = 1.23f;
560     valv = 3.21f;
561     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 0, valu, valv);
562     ok(hr == D3DRM_OK, "Cannot set texture coordinates (hr = %x)\n", hr);
563     hr = IDirect3DRMMeshBuilder3_SetTextureCoordinates(pMeshBuilder3, 4, valu, valv);
564     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
565
566     valu = 0.0f;
567     valv = 0.0f;
568     hr = IDirect3DRMMeshBuilder3_GetTextureCoordinates(pMeshBuilder3, 0, &valu, &valv);
569     ok(hr == D3DRM_OK, "Cannot get texture coordinates (hr = %x)\n", hr);
570     ok(valu == 1.23f, "Wrong coordinate %f (must be 1.23)\n", valu);
571     ok(valv == 3.21f, "Wrong coordinate %f (must be 3.21)\n", valv);
572
573     IDirect3DRMMeshBuilder3_Release(pMeshBuilder3);
574     IDirect3DRM3_Release(pD3DRM3);
575     IDirect3DRM_Release(pD3DRM);
576 }
577
578 static void test_Mesh(void)
579 {
580     HRESULT hr;
581     LPDIRECT3DRM pD3DRM;
582     LPDIRECT3DRMMESH pMesh;
583     DWORD size;
584     CHAR cname[64] = {0};
585
586     hr = pDirect3DRMCreate(&pD3DRM);
587     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
588
589     hr = IDirect3DRM_CreateMesh(pD3DRM, &pMesh);
590     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMesh interface (hr = %x)\n", hr);
591
592     hr = IDirect3DRMMesh_GetClassName(pMesh, NULL, cname);
593     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
594     hr = IDirect3DRMMesh_GetClassName(pMesh, NULL, NULL);
595     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
596     size = 1;
597     hr = IDirect3DRMMesh_GetClassName(pMesh, &size, cname);
598     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
599     size = sizeof(cname);
600     hr = IDirect3DRMMesh_GetClassName(pMesh, &size, cname);
601     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
602     ok(size == sizeof("Mesh"), "wrong size: %u\n", size);
603     ok(!strcmp(cname, "Mesh"), "Expected cname to be \"Mesh\", but got \"%s\"\n", cname);
604
605     IDirect3DRMMesh_Release(pMesh);
606
607     IDirect3DRM_Release(pD3DRM);
608 }
609
610 static void test_Frame(void)
611 {
612     HRESULT hr;
613     LPDIRECT3DRM pD3DRM;
614     LPDIRECT3DRMFRAME pFrameC;
615     LPDIRECT3DRMFRAME pFrameP1;
616     LPDIRECT3DRMFRAME pFrameP2;
617     LPDIRECT3DRMFRAME pFrameTmp;
618     LPDIRECT3DRMFRAMEARRAY pArray;
619     LPDIRECT3DRMMESHBUILDER pMeshBuilder;
620     LPDIRECT3DRMVISUAL pVisual1;
621     LPDIRECT3DRMVISUAL pVisualTmp;
622     LPDIRECT3DRMVISUALARRAY pVisualArray;
623     LPDIRECT3DRMLIGHT pLight1;
624     LPDIRECT3DRMLIGHT pLightTmp;
625     LPDIRECT3DRMLIGHTARRAY pLightArray;
626     D3DCOLOR color;
627     DWORD count;
628     CHAR cname[64] = {0};
629
630     hr = pDirect3DRMCreate(&pD3DRM);
631     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
632
633     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameC);
634     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
635     CHECK_REFCOUNT(pFrameC, 1);
636
637     hr = IDirect3DRMFrame_GetClassName(pFrameC, NULL, cname);
638     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
639     hr = IDirect3DRMFrame_GetClassName(pFrameC, NULL, NULL);
640     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
641     count = 1;
642     hr = IDirect3DRMFrame_GetClassName(pFrameC, &count, cname);
643     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
644     count = sizeof(cname);
645     hr = IDirect3DRMFrame_GetClassName(pFrameC, &count, cname);
646     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
647     ok(count == sizeof("Frame"), "wrong size: %u\n", count);
648     ok(!strcmp(cname, "Frame"), "Expected cname to be \"Frame\", but got \"%s\"\n", cname);
649
650     hr = IDirect3DRMFrame_GetParent(pFrameC, NULL);
651     ok(hr == D3DRMERR_BADVALUE, "Should fail and return D3DRM_BADVALUE (hr = %x)\n", hr);
652     pFrameTmp = (void*)0xdeadbeef;
653     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
654     ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
655     ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
656     CHECK_REFCOUNT(pFrameC, 1);
657
658     pArray = NULL;
659     hr = IDirect3DRMFrame_GetChildren(pFrameC, &pArray);
660     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
661     ok(pArray != NULL, "pArray = %p\n", pArray);
662     if (pArray)
663     {
664         count = IDirect3DRMFrameArray_GetSize(pArray);
665         ok(count == 0, "count = %u\n", count);
666         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
667         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
668         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
669         IDirect3DRMFrameArray_Release(pArray);
670     }
671
672     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP1);
673     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
674
675     /* GetParent with NULL pointer */
676     hr = IDirect3DRMFrame_GetParent(pFrameP1, NULL);
677     ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
678     CHECK_REFCOUNT(pFrameP1, 1);
679
680     /* [Add/Delete]Child with NULL pointer */
681     hr = IDirect3DRMFrame_AddChild(pFrameP1, NULL);
682     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
683     CHECK_REFCOUNT(pFrameP1, 1);
684
685     hr = IDirect3DRMFrame_DeleteChild(pFrameP1, NULL);
686     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
687     CHECK_REFCOUNT(pFrameP1, 1);
688
689     /* Add child to first parent */
690     pFrameTmp = (void*)0xdeadbeef;
691     hr = IDirect3DRMFrame_GetParent(pFrameP1, &pFrameTmp);
692     ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
693     ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
694
695     hr = IDirect3DRMFrame_AddChild(pFrameP1, pFrameC);
696     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
697     CHECK_REFCOUNT(pFrameP1, 1);
698     CHECK_REFCOUNT(pFrameC, 2);
699
700     pArray = NULL;
701     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
702     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
703     /* In some older version of d3drm, creating IDirect3DRMFrameArray object with GetChildren does not increment refcount of children frames */
704     ok((get_refcount((IUnknown*)pFrameC) == 3) || broken(get_refcount((IUnknown*)pFrameC) == 2),
705             "Invalid refcount. Expected 3 (or 2) got %d\n", get_refcount((IUnknown*)pFrameC));
706     if (pArray)
707     {
708         count = IDirect3DRMFrameArray_GetSize(pArray);
709         ok(count == 1, "count = %u\n", count);
710         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
711         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
712         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
713         ok((get_refcount((IUnknown*)pFrameC) == 4) || broken(get_refcount((IUnknown*)pFrameC) == 3),
714                 "Invalid refcount. Expected 4 (or 3) got %d\n", get_refcount((IUnknown*)pFrameC));
715         IDirect3DRMFrame_Release(pFrameTmp);
716         ok((get_refcount((IUnknown*)pFrameC) == 3) || broken(get_refcount((IUnknown*)pFrameC) == 2),
717                 "Invalid refcount. Expected 3 (or 2) got %d\n", get_refcount((IUnknown*)pFrameC));
718         IDirect3DRMFrameArray_Release(pArray);
719         CHECK_REFCOUNT(pFrameC, 2);
720     }
721
722     pFrameTmp = (void*)0xdeadbeef;
723     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
724     ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
725     ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
726     CHECK_REFCOUNT(pFrameP1, 2);
727
728     /* Add child to second parent */
729     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrameP2);
730     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
731
732     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
733     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
734     CHECK_REFCOUNT(pFrameC, 2);
735
736     pArray = NULL;
737     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
738     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
739     if (pArray)
740     {
741         count = IDirect3DRMFrameArray_GetSize(pArray);
742         ok(count == 1, "count = %u\n", count);
743         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
744         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
745         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
746         IDirect3DRMFrame_Release(pFrameTmp);
747         IDirect3DRMFrameArray_Release(pArray);
748     }
749
750     pArray = NULL;
751     hr = IDirect3DRMFrame_GetChildren(pFrameP1, &pArray);
752     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
753     if (pArray)
754     {
755         count = IDirect3DRMFrameArray_GetSize(pArray);
756         ok(count == 0, "count = %u\n", count);
757         pFrameTmp = (void*)0xdeadbeef;
758         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
759         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
760         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
761         IDirect3DRMFrameArray_Release(pArray);
762     }
763
764     pFrameTmp = (void*)0xdeadbeef;
765     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
766     ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
767     ok(pFrameTmp == pFrameP2, "pFrameTmp = %p\n", pFrameTmp);
768     CHECK_REFCOUNT(pFrameP2, 2);
769     CHECK_REFCOUNT(pFrameC, 2);
770
771     /* Add child again */
772     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
773     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
774     CHECK_REFCOUNT(pFrameC, 2);
775
776     pArray = NULL;
777     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
778     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
779     if (pArray)
780     {
781         count = IDirect3DRMFrameArray_GetSize(pArray);
782         ok(count == 1, "count = %u\n", count);
783         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
784         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
785         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
786         IDirect3DRMFrame_Release(pFrameTmp);
787         IDirect3DRMFrameArray_Release(pArray);
788     }
789
790     /* Delete child */
791     hr = IDirect3DRMFrame_DeleteChild(pFrameP2, pFrameC);
792     ok(hr == D3DRM_OK, "Cannot delete child frame (hr = %x)\n", hr);
793     CHECK_REFCOUNT(pFrameC, 1);
794
795     pArray = NULL;
796     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
797     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
798     if (pArray)
799     {
800         count = IDirect3DRMFrameArray_GetSize(pArray);
801         ok(count == 0, "count = %u\n", count);
802         pFrameTmp = (void*)0xdeadbeef;
803         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
804         ok(hr == D3DRMERR_BADVALUE, "Should have returned D3DRMERR_BADVALUE (hr = %x)\n", hr);
805         ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
806         IDirect3DRMFrameArray_Release(pArray);
807     }
808
809     pFrameTmp = (void*)0xdeadbeef;
810     hr = IDirect3DRMFrame_GetParent(pFrameC, &pFrameTmp);
811     ok(hr == D3DRM_OK, "Cannot get parent frame (hr = %x)\n", hr);
812     ok(pFrameTmp == NULL, "pFrameTmp = %p\n", pFrameTmp);
813
814     /* Add two children */
815     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameC);
816     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
817     CHECK_REFCOUNT(pFrameC, 2);
818
819     hr = IDirect3DRMFrame_AddChild(pFrameP2, pFrameP1);
820     ok(hr == D3DRM_OK, "Cannot add child frame (hr = %x)\n", hr);
821     CHECK_REFCOUNT(pFrameP1, 3);
822
823     pArray = NULL;
824     hr = IDirect3DRMFrame_GetChildren(pFrameP2, &pArray);
825     ok(hr == D3DRM_OK, "Cannot get children (hr = %x)\n", hr);
826     if (pArray)
827     {
828         count = IDirect3DRMFrameArray_GetSize(pArray);
829         ok(count == 2, "count = %u\n", count);
830         hr = IDirect3DRMFrameArray_GetElement(pArray, 0, &pFrameTmp);
831         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
832         ok(pFrameTmp == pFrameC, "pFrameTmp = %p\n", pFrameTmp);
833         IDirect3DRMFrame_Release(pFrameTmp);
834         hr = IDirect3DRMFrameArray_GetElement(pArray, 1, &pFrameTmp);
835         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
836         ok(pFrameTmp == pFrameP1, "pFrameTmp = %p\n", pFrameTmp);
837         IDirect3DRMFrame_Release(pFrameTmp);
838         IDirect3DRMFrameArray_Release(pArray);
839     }
840
841     /* [Add/Delete]Visual with NULL pointer */
842     hr = IDirect3DRMFrame_AddVisual(pFrameP1, NULL);
843     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
844     CHECK_REFCOUNT(pFrameP1, 3);
845
846     hr = IDirect3DRMFrame_DeleteVisual(pFrameP1, NULL);
847     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
848     CHECK_REFCOUNT(pFrameP1, 3);
849
850     /* Create Visual */
851     hr = IDirect3DRM_CreateMeshBuilder(pD3DRM, &pMeshBuilder);
852     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMeshBuilder interface (hr = %x)\n", hr);
853     pVisual1 = (LPDIRECT3DRMVISUAL)pMeshBuilder;
854
855     /* Add Visual to first parent */
856     hr = IDirect3DRMFrame_AddVisual(pFrameP1, pVisual1);
857     ok(hr == D3DRM_OK, "Cannot add visual (hr = %x)\n", hr);
858     CHECK_REFCOUNT(pFrameP1, 3);
859     CHECK_REFCOUNT(pVisual1, 2);
860
861     pVisualArray = NULL;
862     hr = IDirect3DRMFrame_GetVisuals(pFrameP1, &pVisualArray);
863     ok(hr == D3DRM_OK, "Cannot get visuals (hr = %x)\n", hr);
864     if (pVisualArray)
865     {
866         count = IDirect3DRMVisualArray_GetSize(pVisualArray);
867         ok(count == 1, "count = %u\n", count);
868         hr = IDirect3DRMVisualArray_GetElement(pVisualArray, 0, &pVisualTmp);
869         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
870         ok(pVisualTmp == pVisual1, "pVisualTmp = %p\n", pVisualTmp);
871         IDirect3DRMVisual_Release(pVisualTmp);
872         IDirect3DRMVisualArray_Release(pVisualArray);
873     }
874
875     /* Delete Visual */
876     hr = IDirect3DRMFrame_DeleteVisual(pFrameP1, pVisual1);
877     ok(hr == D3DRM_OK, "Cannot delete visual (hr = %x)\n", hr);
878     CHECK_REFCOUNT(pFrameP1, 3);
879     IDirect3DRMMeshBuilder_Release(pMeshBuilder);
880
881     /* [Add/Delete]Light with NULL pointer */
882     hr = IDirect3DRMFrame_AddLight(pFrameP1, NULL);
883     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
884     CHECK_REFCOUNT(pFrameP1, 3);
885
886     hr = IDirect3DRMFrame_DeleteLight(pFrameP1, NULL);
887     ok(hr == D3DRMERR_BADOBJECT, "Should have returned D3DRMERR_BADOBJECT (hr = %x)\n", hr);
888     CHECK_REFCOUNT(pFrameP1, 3);
889
890     /* Create Light */
891     hr = IDirect3DRM_CreateLightRGB(pD3DRM, D3DRMLIGHT_SPOT, 0.1, 0.2, 0.3, &pLight1);
892     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMLight interface (hr = %x)\n", hr);
893
894     /* Add Light to first parent */
895     hr = IDirect3DRMFrame_AddLight(pFrameP1, pLight1);
896     ok(hr == D3DRM_OK, "Cannot add light (hr = %x)\n", hr);
897     CHECK_REFCOUNT(pFrameP1, 3);
898     CHECK_REFCOUNT(pLight1, 2);
899
900     pLightArray = NULL;
901     hr = IDirect3DRMFrame_GetLights(pFrameP1, &pLightArray);
902     ok(hr == D3DRM_OK, "Cannot get lights (hr = %x)\n", hr);
903     if (pLightArray)
904     {
905         count = IDirect3DRMLightArray_GetSize(pLightArray);
906         ok(count == 1, "count = %u\n", count);
907         hr = IDirect3DRMLightArray_GetElement(pLightArray, 0, &pLightTmp);
908         ok(hr == D3DRM_OK, "Cannot get element (hr = %x)\n", hr);
909         ok(pLightTmp == pLight1, "pLightTmp = %p\n", pLightTmp);
910         IDirect3DRMLight_Release(pLightTmp);
911         IDirect3DRMLightArray_Release(pLightArray);
912     }
913
914     /* Delete Light */
915     hr = IDirect3DRMFrame_DeleteLight(pFrameP1, pLight1);
916     ok(hr == D3DRM_OK, "Cannot delete light (hr = %x)\n", hr);
917     CHECK_REFCOUNT(pFrameP1, 3);
918     IDirect3DRMLight_Release(pLight1);
919
920     /* Test SceneBackground on first parent */
921     color = IDirect3DRMFrame_GetSceneBackground(pFrameP1);
922     ok(color == 0xff000000, "wrong color (%x)\n", color);
923
924     hr = IDirect3DRMFrame_SetSceneBackground(pFrameP1, 0xff180587);
925     ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
926     color = IDirect3DRMFrame_GetSceneBackground(pFrameP1);
927     ok(color == 0xff180587, "wrong color (%x)\n", color);
928
929     hr = IDirect3DRMFrame_SetSceneBackgroundRGB(pFrameP1, 0.5, 0.5, 0.5);
930     ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
931     color = IDirect3DRMFrame_GetSceneBackground(pFrameP1);
932     ok(color == 0xff7f7f7f, "wrong color (%x)\n", color);
933
934     /* Cleanup */
935     IDirect3DRMFrame_Release(pFrameP2);
936     CHECK_REFCOUNT(pFrameC, 2);
937     CHECK_REFCOUNT(pFrameP1, 3);
938
939     IDirect3DRMFrame_Release(pFrameC);
940     IDirect3DRMFrame_Release(pFrameP1);
941
942     IDirect3DRM_Release(pD3DRM);
943 }
944
945 static void test_Viewport(void)
946 {
947     IDirectDrawClipper *pClipper;
948     HRESULT hr;
949     LPDIRECT3DRM pD3DRM;
950     LPDIRECT3DRMDEVICE pDevice;
951     LPDIRECT3DRMFRAME pFrame;
952     LPDIRECT3DRMVIEWPORT pViewport;
953     GUID driver;
954     HWND window;
955     RECT rc;
956     DWORD size;
957     CHAR cname[64] = {0};
958
959     window = CreateWindowA("static", "d3drm_test", WS_OVERLAPPEDWINDOW, 0, 0, 300, 200, 0, 0, 0, 0);
960     GetClientRect(window, &rc);
961
962     hr = pDirect3DRMCreate(&pD3DRM);
963     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
964
965     hr = DirectDrawCreateClipper(0, &pClipper, NULL);
966     ok(hr == DD_OK, "Cannot get IDirectDrawClipper interface (hr = %x)\n", hr);
967
968     hr = IDirectDrawClipper_SetHWnd(pClipper, 0, window);
969     ok(hr == DD_OK, "Cannot set HWnd to Clipper (hr = %x)\n", hr);
970
971     memcpy(&driver, &IID_IDirect3DRGBDevice, sizeof(GUID));
972     hr = IDirect3DRM3_CreateDeviceFromClipper(pD3DRM, pClipper, &driver, rc.right, rc.bottom, &pDevice);
973     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMDevice interface (hr = %x)\n", hr);
974
975     hr = IDirect3DRM_CreateFrame(pD3DRM, NULL, &pFrame);
976     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
977
978     hr = IDirect3DRM_CreateViewport(pD3DRM, pDevice, pFrame, rc.left, rc.top, rc.right, rc.bottom, &pViewport);
979     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMViewport interface (hr = %x)\n", hr);
980
981     hr = IDirect3DRMViewport_GetClassName(pViewport, NULL, cname);
982     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
983     hr = IDirect3DRMViewport_GetClassName(pViewport, NULL, NULL);
984     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
985     size = 1;
986     hr = IDirect3DRMViewport_GetClassName(pViewport, &size, cname);
987     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
988     size = sizeof(cname);
989     hr = IDirect3DRMViewport_GetClassName(pViewport, &size, cname);
990     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
991     ok(size == sizeof("Viewport"), "wrong size: %u\n", size);
992     ok(!strcmp(cname, "Viewport"), "Expected cname to be \"Viewport\", but got \"%s\"\n", cname);
993
994     IDirect3DRMViewport_Release(pViewport);
995     IDirect3DRMFrame_Release(pFrame);
996     IDirect3DRMDevice_Release(pDevice);
997     IDirectDrawClipper_Release(pClipper);
998
999     IDirect3DRM_Release(pD3DRM);
1000     DestroyWindow(window);
1001 }
1002
1003 static void test_Light(void)
1004 {
1005     HRESULT hr;
1006     LPDIRECT3DRM pD3DRM;
1007     LPDIRECT3DRMLIGHT pLight;
1008     D3DRMLIGHTTYPE type;
1009     D3DCOLOR color;
1010     DWORD size;
1011     CHAR cname[64] = {0};
1012
1013     hr = pDirect3DRMCreate(&pD3DRM);
1014     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1015
1016     hr = IDirect3DRM_CreateLightRGB(pD3DRM, D3DRMLIGHT_SPOT, 0.5, 0.5, 0.5, &pLight);
1017     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMLight interface (hr = %x)\n", hr);
1018
1019     hr = IDirect3DRMLight_GetClassName(pLight, NULL, cname);
1020     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1021     hr = IDirect3DRMLight_GetClassName(pLight, NULL, NULL);
1022     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1023     size = 1;
1024     hr = IDirect3DRMLight_GetClassName(pLight, &size, cname);
1025     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1026     size = sizeof(cname);
1027     hr = IDirect3DRMLight_GetClassName(pLight, &size, cname);
1028     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
1029     ok(size == sizeof("Light"), "wrong size: %u\n", size);
1030     ok(!strcmp(cname, "Light"), "Expected cname to be \"Light\", but got \"%s\"\n", cname);
1031
1032     type = IDirect3DRMLight_GetType(pLight);
1033     ok(type == D3DRMLIGHT_SPOT, "wrong type (%u)\n", type);
1034
1035     color = IDirect3DRMLight_GetColor(pLight);
1036     ok(color == 0xff7f7f7f, "wrong color (%x)\n", color);
1037
1038     hr = IDirect3DRMLight_SetType(pLight, D3DRMLIGHT_POINT);
1039     ok(hr == D3DRM_OK, "Cannot set type (hr = %x)\n", hr);
1040     type = IDirect3DRMLight_GetType(pLight);
1041     ok(type == D3DRMLIGHT_POINT, "wrong type (%u)\n", type);
1042
1043     hr = IDirect3DRMLight_SetColor(pLight, 0xff180587);
1044     ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
1045     color = IDirect3DRMLight_GetColor(pLight);
1046     ok(color == 0xff180587, "wrong color (%x)\n", color);
1047
1048     hr = IDirect3DRMLight_SetColorRGB(pLight, 0.5, 0.5, 0.5);
1049     ok(hr == D3DRM_OK, "Cannot set color (hr = %x)\n", hr);
1050     color = IDirect3DRMLight_GetColor(pLight);
1051     ok(color == 0xff7f7f7f, "wrong color (%x)\n", color);
1052
1053     IDirect3DRMLight_Release(pLight);
1054
1055     IDirect3DRM_Release(pD3DRM);
1056 }
1057
1058 static void test_Material2(void)
1059 {
1060     HRESULT hr;
1061     LPDIRECT3DRM pD3DRM;
1062     LPDIRECT3DRM3 pD3DRM3;
1063     LPDIRECT3DRMMATERIAL2 pMaterial2;
1064     D3DVALUE r, g, b;
1065     DWORD size;
1066     CHAR cname[64] = {0};
1067
1068     hr = pDirect3DRMCreate(&pD3DRM);
1069     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1070
1071     hr = IDirect3DRM_QueryInterface(pD3DRM, &IID_IDirect3DRM3, (LPVOID*)&pD3DRM3);
1072     if (FAILED(hr))
1073     {
1074         win_skip("Cannot get IDirect3DRM3 interface (hr = %x), skipping tests\n", hr);
1075         IDirect3DRM_Release(pD3DRM);
1076         return;
1077     }
1078
1079     hr = IDirect3DRM3_CreateMaterial(pD3DRM3, 18.5f, &pMaterial2);
1080     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMMaterial2 interface (hr = %x)\n", hr);
1081
1082     hr = IDirect3DRMMaterial2_GetClassName(pMaterial2, NULL, cname);
1083     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1084     hr = IDirect3DRMMaterial2_GetClassName(pMaterial2, NULL, NULL);
1085     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1086     size = 1;
1087     hr = IDirect3DRMMaterial2_GetClassName(pMaterial2, &size, cname);
1088     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1089     size = sizeof(cname);
1090     hr = IDirect3DRMMaterial2_GetClassName(pMaterial2, &size, cname);
1091     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
1092     ok(size == sizeof("Material"), "wrong size: %u\n", size);
1093     ok(!strcmp(cname, "Material"), "Expected cname to be \"Material\", but got \"%s\"\n", cname);
1094
1095     r = IDirect3DRMMaterial2_GetPower(pMaterial2);
1096     ok(r == 18.5f, "wrong power (%f)\n", r);
1097
1098     hr = IDirect3DRMMaterial2_GetEmissive(pMaterial2, &r, &g, &b);
1099     ok(hr == D3DRM_OK, "Cannot get emissive (hr = %x)\n", hr);
1100     ok(r == 0.0f && g == 0.0f && b == 0.0f, "wrong emissive r=%f g=%f b=%f, expected r=0.0 g=0.0 b=0.0\n", r, g, b);
1101
1102     hr = IDirect3DRMMaterial2_GetSpecular(pMaterial2, &r, &g, &b);
1103     ok(hr == D3DRM_OK, "Cannot get emissive (hr = %x)\n", hr);
1104     ok(r == 1.0f && g == 1.0f && b == 1.0f, "wrong specular r=%f g=%f b=%f, expected r=1.0 g=1.0 b=1.0\n", r, g, b);
1105
1106     hr = IDirect3DRMMaterial2_GetAmbient(pMaterial2, &r, &g, &b);
1107     ok(hr == D3DRM_OK, "Cannot get emissive (hr = %x)\n", hr);
1108     ok(r == 0.0f && g == 0.0f && b == 0.0f, "wrong ambient r=%f g=%f b=%f, expected r=0.0 g=0.0 b=0.0\n", r, g, b);
1109
1110     hr = IDirect3DRMMaterial2_SetPower(pMaterial2, 5.87f);
1111     ok(hr == D3DRM_OK, "Cannot set power (hr = %x)\n", hr);
1112     r = IDirect3DRMMaterial2_GetPower(pMaterial2);
1113     ok(r == 5.87f, "wrong power (%f)\n", r);
1114
1115     hr = IDirect3DRMMaterial2_SetEmissive(pMaterial2, 0.5f, 0.5f, 0.5f);
1116     ok(hr == D3DRM_OK, "Cannot set emissive (hr = %x)\n", hr);
1117     hr = IDirect3DRMMaterial2_GetEmissive(pMaterial2, &r, &g, &b);
1118     ok(hr == D3DRM_OK, "Cannot get emissive (hr = %x)\n", hr);
1119     ok(r == 0.5f && g == 0.5f && b == 0.5f, "wrong emissive r=%f g=%f b=%f, expected r=0.5 g=0.5 b=0.5\n", r, g, b);
1120
1121     hr = IDirect3DRMMaterial2_SetSpecular(pMaterial2, 0.6f, 0.6f, 0.6f);
1122     ok(hr == D3DRM_OK, "Cannot set specular (hr = %x)\n", hr);
1123     hr = IDirect3DRMMaterial2_GetSpecular(pMaterial2, &r, &g, &b);
1124     ok(hr == D3DRM_OK, "Cannot get specular (hr = %x)\n", hr);
1125     ok(r == 0.6f && g == 0.6f && b == 0.6f, "wrong specular r=%f g=%f b=%f, expected r=0.6 g=0.6 b=0.6\n", r, g, b);
1126
1127     hr = IDirect3DRMMaterial2_SetAmbient(pMaterial2, 0.7f, 0.7f, 0.7f);
1128     ok(hr == D3DRM_OK, "Cannot set ambient (hr = %x)\n", hr);
1129     hr = IDirect3DRMMaterial2_GetAmbient(pMaterial2, &r, &g, &b);
1130     ok(hr == D3DRM_OK, "Cannot get ambient (hr = %x)\n", hr);
1131     ok(r == 0.7f && g == 0.7f && b == 0.7f, "wrong ambient r=%f g=%f b=%f, expected r=0.7 g=0.7 b=0.7\n", r, g, b);
1132
1133     IDirect3DRMMaterial2_Release(pMaterial2);
1134
1135     IDirect3DRM3_Release(pD3DRM3);
1136     IDirect3DRM_Release(pD3DRM);
1137 }
1138
1139 static void test_Texture(void)
1140 {
1141     HRESULT hr;
1142     LPDIRECT3DRM pD3DRM;
1143     LPDIRECT3DRMTEXTURE pTexture;
1144     D3DRMIMAGE initimg = {
1145         2, 2, 1, 1, 32,
1146         TRUE, 2 * sizeof(DWORD), NULL, NULL,
1147         0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000, 0, NULL
1148     };
1149     DWORD pixel[4] = { 20000, 30000, 10000, 0 };
1150     DWORD size;
1151     CHAR cname[64] = {0};
1152
1153     hr = pDirect3DRMCreate(&pD3DRM);
1154     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1155
1156     initimg.buffer1 = &pixel;
1157     hr = IDirect3DRM_CreateTexture(pD3DRM, &initimg, &pTexture);
1158     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMTexture interface (hr = %x)\n", hr);
1159
1160     hr = IDirect3DRMTexture_GetClassName(pTexture, NULL, cname);
1161     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1162     hr = IDirect3DRMTexture_GetClassName(pTexture, NULL, NULL);
1163     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1164     size = 1;
1165     hr = IDirect3DRMTexture_GetClassName(pTexture, &size, cname);
1166     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1167     size = sizeof(cname);
1168     hr = IDirect3DRMTexture_GetClassName(pTexture, &size, cname);
1169     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
1170     ok(size == sizeof("Texture"), "wrong size: %u\n", size);
1171     ok(!strcmp(cname, "Texture"), "Expected cname to be \"Texture\", but got \"%s\"\n", cname);
1172
1173     IDirect3DRMTexture_Release(pTexture);
1174
1175     IDirect3DRM_Release(pD3DRM);
1176 }
1177
1178 static void test_Device(void)
1179 {
1180     IDirectDrawClipper *pClipper;
1181     HRESULT hr;
1182     LPDIRECT3DRM pD3DRM;
1183     LPDIRECT3DRMDEVICE pDevice;
1184     LPDIRECT3DRMWINDEVICE pWinDevice;
1185     GUID driver;
1186     HWND window;
1187     RECT rc;
1188     DWORD size;
1189     CHAR cname[64] = {0};
1190
1191     window = CreateWindowA("static", "d3drm_test", WS_OVERLAPPEDWINDOW, 0, 0, 300, 200, 0, 0, 0, 0);
1192     GetClientRect(window, &rc);
1193
1194     hr = pDirect3DRMCreate(&pD3DRM);
1195     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1196
1197     hr = DirectDrawCreateClipper(0, &pClipper, NULL);
1198     ok(hr == DD_OK, "Cannot get IDirectDrawClipper interface (hr = %x)\n", hr);
1199
1200     hr = IDirectDrawClipper_SetHWnd(pClipper, 0, window);
1201     ok(hr == DD_OK, "Cannot set HWnd to Clipper (hr = %x)\n", hr);
1202
1203     memcpy(&driver, &IID_IDirect3DRGBDevice, sizeof(GUID));
1204     hr = IDirect3DRM3_CreateDeviceFromClipper(pD3DRM, pClipper, &driver, rc.right, rc.bottom, &pDevice);
1205     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMDevice interface (hr = %x)\n", hr);
1206
1207     hr = IDirect3DRMDevice_GetClassName(pDevice, NULL, cname);
1208     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1209     hr = IDirect3DRMDevice_GetClassName(pDevice, NULL, NULL);
1210     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1211     size = 1;
1212     hr = IDirect3DRMDevice_GetClassName(pDevice, &size, cname);
1213     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1214     size = sizeof(cname);
1215     hr = IDirect3DRMDevice_GetClassName(pDevice, &size, cname);
1216     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
1217     ok(size == sizeof("Device"), "wrong size: %u\n", size);
1218     ok(!strcmp(cname, "Device"), "Expected cname to be \"Device\", but got \"%s\"\n", cname);
1219
1220     /* WinDevice */
1221     hr = IDirect3DRMDevice_QueryInterface(pDevice, &IID_IDirect3DRMWinDevice, (LPVOID*)&pWinDevice);
1222     if (FAILED(hr))
1223     {
1224         win_skip("Cannot get IDirect3DRMWinDevice interface (hr = %x), skipping tests\n", hr);
1225         goto cleanup;
1226     }
1227
1228     hr = IDirect3DRMWinDevice_GetClassName(pWinDevice, NULL, cname);
1229     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1230     hr = IDirect3DRMWinDevice_GetClassName(pWinDevice, NULL, NULL);
1231     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1232     size = 1;
1233     hr = IDirect3DRMWinDevice_GetClassName(pWinDevice, &size, cname);
1234     ok(hr == E_INVALIDARG, "GetClassName failed with %x\n", hr);
1235     size = sizeof(cname);
1236     hr = IDirect3DRMWinDevice_GetClassName(pWinDevice, &size, cname);
1237     ok(hr == D3DRM_OK, "Cannot get classname (hr = %x)\n", hr);
1238     ok(size == sizeof("Device"), "wrong size: %u\n", size);
1239     ok(!strcmp(cname, "Device"), "Expected cname to be \"Device\", but got \"%s\"\n", cname);
1240
1241     IDirect3DRMWinDevice_Release(pWinDevice);
1242
1243 cleanup:
1244     IDirect3DRMDevice_Release(pDevice);
1245     IDirectDrawClipper_Release(pClipper);
1246
1247     IDirect3DRM_Release(pD3DRM);
1248     DestroyWindow(window);
1249 }
1250
1251 static void test_frame_transform(void)
1252 {
1253     HRESULT hr;
1254     LPDIRECT3DRM d3drm;
1255     LPDIRECT3DRMFRAME frame;
1256     D3DRMMATRIX4D matrix;
1257
1258     hr = pDirect3DRMCreate(&d3drm);
1259     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1260
1261     hr = IDirect3DRM_CreateFrame(d3drm, NULL, &frame);
1262     ok(hr == D3DRM_OK, "Cannot get IDirect3DRMFrame interface (hr = %x)\n", hr);
1263
1264     hr = IDirect3DRMFrame_GetTransform(frame, matrix);
1265     ok(hr == D3DRM_OK, "IDirect3DRMFrame_GetTransform returned hr = %x\n", hr);
1266     ok(!memcmp(matrix, identity, sizeof(D3DRMMATRIX4D)), "Returned matrix is not identity\n");
1267
1268     IDirect3DRM_Release(d3drm);
1269 }
1270
1271 static int nb_objects = 0;
1272 static const GUID* refiids[] =
1273 {
1274     &IID_IDirect3DRMMeshBuilder,
1275     &IID_IDirect3DRMMeshBuilder,
1276     &IID_IDirect3DRMFrame,
1277     &IID_IDirect3DRMMaterial /* Not taken into account and not notified */
1278 };
1279
1280 static void __cdecl object_load_callback(LPDIRECT3DRMOBJECT object, REFIID objectguid, LPVOID arg)
1281 {
1282     ok(object != NULL, "Arg 1 should not be null\n");
1283     ok(IsEqualGUID(objectguid, refiids[nb_objects]), "Arg 2 is incorrect\n");
1284     ok(arg == (LPVOID)0xdeadbeef, "Arg 3 should be 0xdeadbeef (got %p)\n", arg);
1285     nb_objects++;
1286 }
1287
1288 static void test_d3drm_load(void)
1289 {
1290     HRESULT hr;
1291     LPDIRECT3DRM pD3DRM;
1292     D3DRMLOADMEMORY info;
1293     const GUID* req_refiids[] = { &IID_IDirect3DRMMeshBuilder, &IID_IDirect3DRMFrame, &IID_IDirect3DRMMaterial };
1294
1295     hr = pDirect3DRMCreate(&pD3DRM);
1296     ok(hr == D3DRM_OK, "Cannot get IDirect3DRM interface (hr = %x)\n", hr);
1297
1298     info.lpMemory = data_d3drm_load;
1299     info.dSize = strlen(data_d3drm_load);
1300     hr = IDirect3DRM_Load(pD3DRM, &info, NULL, (GUID**)req_refiids, 3, D3DRMLOAD_FROMMEMORY, object_load_callback, (LPVOID)0xdeadbeef, NULL, NULL, NULL);
1301     ok(hr == D3DRM_OK, "Cannot load data (hr = %x)\n", hr);
1302     ok(nb_objects == 3, "Should have loaded 3 objects (got %d)\n", nb_objects);
1303
1304     IDirect3DRM_Release(pD3DRM);
1305 }
1306
1307 START_TEST(d3drm)
1308 {
1309     if (!InitFunctionPtrs())
1310         return;
1311
1312     test_MeshBuilder();
1313     test_MeshBuilder3();
1314     test_Mesh();
1315     test_Frame();
1316     test_Device();
1317     test_Viewport();
1318     test_Light();
1319     test_Material2();
1320     test_Texture();
1321     test_frame_transform();
1322     test_d3drm_load();
1323
1324     FreeLibrary(d3drm_handle);
1325 }