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