d3drm: Generate normals automatically when there are not present in the x file.
[wine] / dlls / d3drm / meshbuilder.c
1 /*
2  * Implementation of IDirect3DRMMeshBuilderX and IDirect3DRMMesh interfaces
3  *
4  * Copyright 2010, 2012 Christian Costa
5  * Copyright 2011 AndrĂ© Hentschel
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21
22 #define NONAMELESSUNION
23
24 #define COBJMACROS
25
26 #include "wine/debug.h"
27
28 #include "winbase.h"
29 #include "wingdi.h"
30 #include "dxfile.h"
31 #include "rmxfguid.h"
32
33 #include "d3drm_private.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(d3drm);
36
37 static HRESULT Direct3DRMMesh_create(IDirect3DRMMesh** obj);
38
39 typedef struct {
40     unsigned nb_vertices;
41     D3DRMVERTEX* vertices;
42     unsigned nb_faces;
43     unsigned vertex_per_face;
44     DWORD face_data_size;
45     unsigned* face_data;
46     IDirect3DRMMaterial2* material;
47     IDirect3DRMTexture3* texture;
48 } mesh_group;
49
50 typedef struct {
51     IDirect3DRMMesh IDirect3DRMMesh_iface;
52     LONG ref;
53     DWORD groups_capacity;
54     DWORD nb_groups;
55     mesh_group* groups;
56 } IDirect3DRMMeshImpl;
57
58 typedef struct {
59     D3DVALUE u;
60     D3DVALUE v;
61 } Coords2d;
62
63 typedef struct {
64     IDirect3DRMMeshBuilder2 IDirect3DRMMeshBuilder2_iface;
65     IDirect3DRMMeshBuilder3 IDirect3DRMMeshBuilder3_iface;
66     LONG ref;
67     char* name;
68     DWORD nb_vertices;
69     D3DVECTOR* pVertices;
70     DWORD nb_normals;
71     D3DVECTOR* pNormals;
72     DWORD nb_faces;
73     DWORD face_data_size;
74     LPVOID pFaceData;
75     DWORD nb_coords2d;
76     Coords2d* pCoords2d;
77 } IDirect3DRMMeshBuilderImpl;
78
79 char templates[] = {
80 "xof 0302txt 0064"
81 "template Header"
82 "{"
83 "<3D82AB43-62DA-11CF-AB39-0020AF71E433>"
84 "WORD major;"
85 "WORD minor;"
86 "DWORD flags;"
87 "}"
88 "template Vector"
89 "{"
90 "<3D82AB5E-62DA-11CF-AB39-0020AF71E433>"
91 "FLOAT x;"
92 "FLOAT y;"
93 "FLOAT z;"
94 "}"
95 "template Coords2d"
96 "{"
97 "<F6F23F44-7686-11CF-8F52-0040333594A3>"
98 "FLOAT u;"
99 "FLOAT v;"
100 "}"
101 "template Matrix4x4"
102 "{"
103 "<F6F23F45-7686-11CF-8F52-0040333594A3>"
104 "array FLOAT matrix[16];"
105 "}"
106 "template ColorRGBA"
107 "{"
108 "<35FF44E0-6C7C-11CF-8F52-0040333594A3>"
109 "FLOAT red;"
110 "FLOAT green;"
111 "FLOAT blue;"
112 "FLOAT alpha;"
113 "}"
114 "template ColorRGB"
115 "{"
116 "<D3E16E81-7835-11CF-8F52-0040333594A3>"
117 "FLOAT red;"
118 "FLOAT green;"
119 "FLOAT blue;"
120 "}"
121 "template IndexedColor"
122 "{"
123 "<1630B820-7842-11CF-8F52-0040333594A3>"
124 "DWORD index;"
125 "ColorRGBA indexColor;"
126 "}"
127 "template Boolean"
128 "{"
129 "<537DA6A0-CA37-11D0-941C-0080C80CFA7B>"
130 "DWORD truefalse;"
131 "}"
132 "template Boolean2d"
133 "{"
134 "<4885AE63-78E8-11CF-8F52-0040333594A3>"
135 "Boolean u;"
136 "Boolean v;"
137 "}"
138 "template MaterialWrap"
139 "{"
140 "<4885AE60-78E8-11CF-8F52-0040333594A3>"
141 "Boolean u;"
142 "Boolean v;"
143 "}"
144 "template TextureFilename"
145 "{"
146 "<A42790E1-7810-11CF-8F52-0040333594A3>"
147 "STRING filename;"
148 "}"
149 "template Material"
150 "{"
151 "<3D82AB4D-62DA-11CF-AB39-0020AF71E433>"
152 "ColorRGBA faceColor;"
153 "FLOAT power;"
154 "ColorRGB specularColor;"
155 "ColorRGB emissiveColor;"
156 "[...]"
157 "}"
158 "template MeshFace"
159 "{"
160 "<3D82AB5F-62DA-11CF-AB39-0020AF71E433>"
161 "DWORD nFaceVertexIndices;"
162 "array DWORD faceVertexIndices[nFaceVertexIndices];"
163 "}"
164 "template MeshFaceWraps"
165 "{"
166 "<ED1EC5C0-C0A8-11D0-941C-0080C80CFA7B>"
167 "DWORD nFaceWrapValues;"
168 "array Boolean2d faceWrapValues[nFaceWrapValues];"
169 "}"
170 "template MeshTextureCoords"
171 "{"
172 "<F6F23F40-7686-11CF-8F52-0040333594A3>"
173 "DWORD nTextureCoords;"
174 "array Coords2d textureCoords[nTextureCoords];"
175 "}"
176 "template MeshMaterialList"
177 "{"
178 "<F6F23F42-7686-11CF-8F52-0040333594A3>"
179 "DWORD nMaterials;"
180 "DWORD nFaceIndexes;"
181 "array DWORD faceIndexes[nFaceIndexes];"
182 "[Material]"
183 "}"
184 "template MeshNormals"
185 "{"
186 "<F6F23F43-7686-11CF-8F52-0040333594A3>"
187 "DWORD nNormals;"
188 "array Vector normals[nNormals];"
189 "DWORD nFaceNormals;"
190 "array MeshFace faceNormals[nFaceNormals];"
191 "}"
192 "template MeshVertexColors"
193 "{"
194 "<1630B821-7842-11CF-8F52-0040333594A3>"
195 "DWORD nVertexColors;"
196 "array IndexedColor vertexColors[nVertexColors];"
197 "}"
198 "template Mesh"
199 "{"
200 "<3D82AB44-62DA-11CF-AB39-0020AF71E433>"
201 "DWORD nVertices;"
202 "array Vector vertices[nVertices];"
203 "DWORD nFaces;"
204 "array MeshFace faces[nFaces];"
205 "[...]"
206 "}"
207 "template FrameTransformMatrix"
208 "{"
209 "<F6F23F41-7686-11CF-8F52-0040333594A3>"
210 "Matrix4x4 frameMatrix;"
211 "}"
212 "template Frame"
213 "{"
214 "<3D82AB46-62DA-11CF-AB39-0020AF71E433>"
215 "[...]"
216 "}"
217 "template FloatKeys"
218 "{"
219 "<10DD46A9-775B-11CF-8F52-0040333594A3>"
220 "DWORD nValues;"
221 "array FLOAT values[nValues];"
222 "}"
223 "template TimedFloatKeys"
224 "{"
225 "<F406B180-7B3B-11CF-8F52-0040333594A3>"
226 "DWORD time;"
227 "FloatKeys tfkeys;"
228 "}"
229 "template AnimationKey"
230 "{"
231 "<10DD46A8-775B-11CF-8F52-0040333594A3>"
232 "DWORD keyType;"
233 "DWORD nKeys;"
234 "array TimedFloatKeys keys[nKeys];"
235 "}"
236 "template AnimationOptions"
237 "{"
238 "<E2BF56C0-840F-11CF-8F52-0040333594A3>"
239 "DWORD openclosed;"
240 "DWORD positionquality;"
241 "}"
242 "template Animation"
243 "{"
244 "<3D82AB4F-62DA-11CF-AB39-0020AF71E433>"
245 "[...]"
246 "}"
247 "template AnimationSet"
248 "{"
249 "<3D82AB50-62DA-11CF-AB39-0020AF71E433>"
250 "[Animation]"
251 "}"
252 "template InlineData"
253 "{"
254 "<3A23EEA0-94B1-11D0-AB39-0020AF71E433>"
255 "[BINARY]"
256 "}"
257 "template Url"
258 "{"
259 "<3A23EEA1-94B1-11D0-AB39-0020AF71E433>"
260 "DWORD nUrls;"
261 "array STRING urls[nUrls];"
262 "}"
263 "template ProgressiveMesh"
264 "{"
265 "<8A63C360-997D-11D0-941C-0080C80CFA7B>"
266 "[Url,InlineData]"
267 "}"
268 "template Guid"
269 "{"
270 "<A42790E0-7810-11CF-8F52-0040333594A3>"
271 "DWORD data1;"
272 "WORD data2;"
273 "WORD data3;"
274 "array UCHAR data4[8];"
275 "}"
276 "template StringProperty"
277 "{"
278 "<7F0F21E0-BFE1-11D1-82C0-00A0C9697271>"
279 "STRING key;"
280 "STRING value;"
281 "}"
282 "template PropertyBag"
283 "{"
284 "<7F0F21E1-BFE1-11D1-82C0-00A0C9697271>"
285 "[StringProperty]"
286 "}"
287 "template ExternalVisual"
288 "{"
289 "<98116AA0-BDBA-11D1-82C0-00A0C9697271>"
290 "Guid guidExternalVisual;"
291 "[...]"
292 "}"
293 "template RightHanded"
294 "{"
295 "<7F5D5EA0-D53A-11D1-82C0-00A0C9697271>"
296 "DWORD bRightHanded;"
297 "}"
298 };
299
300 static inline IDirect3DRMMeshImpl *impl_from_IDirect3DRMMesh(IDirect3DRMMesh *iface)
301 {
302     return CONTAINING_RECORD(iface, IDirect3DRMMeshImpl, IDirect3DRMMesh_iface);
303 }
304
305 static inline IDirect3DRMMeshBuilderImpl *impl_from_IDirect3DRMMeshBuilder2(IDirect3DRMMeshBuilder2 *iface)
306 {
307     return CONTAINING_RECORD(iface, IDirect3DRMMeshBuilderImpl, IDirect3DRMMeshBuilder2_iface);
308 }
309
310 static inline IDirect3DRMMeshBuilderImpl *impl_from_IDirect3DRMMeshBuilder3(IDirect3DRMMeshBuilder3 *iface)
311 {
312     return CONTAINING_RECORD(iface, IDirect3DRMMeshBuilderImpl, IDirect3DRMMeshBuilder3_iface);
313 }
314
315 /*** IUnknown methods ***/
316 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_QueryInterface(IDirect3DRMMeshBuilder2* iface,
317                                                                  REFIID riid, void** ppvObject)
318 {
319     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
320
321     TRACE("(%p)->(%s,%p)\n", This, debugstr_guid(riid), ppvObject);
322
323     *ppvObject = NULL;
324
325     if(IsEqualGUID(riid, &IID_IUnknown) ||
326        IsEqualGUID(riid, &IID_IDirect3DRMMeshBuilder) ||
327        IsEqualGUID(riid, &IID_IDirect3DRMMeshBuilder2))
328     {
329         *ppvObject = &This->IDirect3DRMMeshBuilder2_iface;
330     }
331     else if(IsEqualGUID(riid, &IID_IDirect3DRMMeshBuilder3))
332     {
333         *ppvObject = &This->IDirect3DRMMeshBuilder3_iface;
334     }
335     else
336     {
337         FIXME("interface %s not implemented\n", debugstr_guid(riid));
338         return E_NOINTERFACE;
339     }
340
341     IDirect3DRMMeshBuilder_AddRef(iface);
342     return S_OK;
343 }
344
345 static ULONG WINAPI IDirect3DRMMeshBuilder2Impl_AddRef(IDirect3DRMMeshBuilder2* iface)
346 {
347     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
348     ULONG ref = InterlockedIncrement(&This->ref);
349
350     TRACE("(%p)->(): new ref = %d\n", This, ref);
351
352     return ref;
353 }
354
355 static ULONG WINAPI IDirect3DRMMeshBuilder2Impl_Release(IDirect3DRMMeshBuilder2* iface)
356 {
357     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
358     ULONG ref = InterlockedDecrement(&This->ref);
359
360     TRACE("(%p)->(): new ref = %d\n", This, ref);
361
362     if (!ref)
363     {
364         HeapFree(GetProcessHeap(), 0, This->name);
365         HeapFree(GetProcessHeap(), 0, This->pVertices);
366         HeapFree(GetProcessHeap(), 0, This->pNormals);
367         HeapFree(GetProcessHeap(), 0, This->pFaceData);
368         HeapFree(GetProcessHeap(), 0, This->pCoords2d);
369         HeapFree(GetProcessHeap(), 0, This);
370     }
371
372     return ref;
373 }
374
375 /*** IDirect3DRMObject methods ***/
376 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_Clone(IDirect3DRMMeshBuilder2* iface,
377                                                         LPUNKNOWN pUnkOuter, REFIID riid,
378                                                         LPVOID *ppvObj)
379 {
380     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
381
382     FIXME("(%p)->(%p,%s,%p): stub\n", This, pUnkOuter, debugstr_guid(riid), ppvObj);
383
384     return E_NOTIMPL;
385 }
386
387 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddDestroyCallback(IDirect3DRMMeshBuilder2* iface,
388                                                                      D3DRMOBJECTCALLBACK cb,
389                                                                      LPVOID argument)
390 {
391     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
392
393     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
394
395     return E_NOTIMPL;
396 }
397
398 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_DeleteDestroyCallback(IDirect3DRMMeshBuilder2* iface,
399                                                                         D3DRMOBJECTCALLBACK cb,
400                                                                         LPVOID argument)
401 {
402     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
403
404     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
405
406     return E_NOTIMPL;
407 }
408
409 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetAppData(IDirect3DRMMeshBuilder2* iface,
410                                                              DWORD data)
411 {
412     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
413
414     FIXME("(%p)->(%u): stub\n", This, data);
415
416     return E_NOTIMPL;
417 }
418
419 static DWORD WINAPI IDirect3DRMMeshBuilder2Impl_GetAppData(IDirect3DRMMeshBuilder2* iface)
420 {
421     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
422
423     FIXME("(%p)->(): stub\n", This);
424
425     return 0;
426 }
427
428 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetName(IDirect3DRMMeshBuilder2* iface,
429                                                           LPCSTR name)
430 {
431     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
432
433     TRACE("(%p)->(%s)\n", This, debugstr_a(name));
434
435     return IDirect3DRMMeshBuilder3_SetName(&This->IDirect3DRMMeshBuilder3_iface, name);
436 }
437
438 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetName(IDirect3DRMMeshBuilder2* iface,
439                                                           LPDWORD size, LPSTR name)
440 {
441     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
442
443     TRACE("(%p)->(%p,%p)\n", This, size, name);
444
445     return IDirect3DRMMeshBuilder3_GetName(&This->IDirect3DRMMeshBuilder3_iface, size, name);
446 }
447
448 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetClassName(IDirect3DRMMeshBuilder2* iface,
449                                                                LPDWORD lpdwSize, LPSTR lpName)
450 {
451     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
452
453     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
454
455     return E_NOTIMPL;
456 }
457
458 /*** IDirect3DRMMeshBuilder2 methods ***/
459 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_Load(IDirect3DRMMeshBuilder2* iface,
460                                                        LPVOID filename, LPVOID name,
461                                                        D3DRMLOADOPTIONS loadflags,
462                                                        D3DRMLOADTEXTURECALLBACK cb, LPVOID arg)
463 {
464     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
465
466     TRACE("(%p)->(%p,%p,%x,%p,%p)\n", This, filename, name, loadflags, cb, arg);
467
468     if (cb)
469         FIXME("Texture callback is not yet supported\n");
470
471     return IDirect3DRMMeshBuilder3_Load(&This->IDirect3DRMMeshBuilder3_iface, filename, name,
472                                         loadflags, NULL, arg);
473 }
474
475 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_Save(IDirect3DRMMeshBuilder2* iface,
476                                                        const char *filename, D3DRMXOFFORMAT format,
477                                                        D3DRMSAVEOPTIONS save)
478 {
479     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
480
481     FIXME("(%p)->(%s,%d,%d): stub\n", This, filename, format, save);
482
483     return E_NOTIMPL;
484 }
485
486 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_Scale(IDirect3DRMMeshBuilder2* iface,
487                                                         D3DVALUE sx, D3DVALUE sy, D3DVALUE sz)
488 {
489     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
490
491     TRACE("(%p)->(%f,%f,%f)\n", This, sx, sy, sz);
492
493     return IDirect3DRMMeshBuilder3_Scale(&This->IDirect3DRMMeshBuilder3_iface, sx, sy, sz);
494 }
495
496 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_Translate(IDirect3DRMMeshBuilder2* iface,
497                                                             D3DVALUE tx, D3DVALUE ty, D3DVALUE tz)
498 {
499     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
500
501     FIXME("(%p)->(%f,%f,%f): stub\n", This, tx, ty, tz);
502
503     return E_NOTIMPL;
504 }
505
506 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetColorSource(IDirect3DRMMeshBuilder2* iface,
507                                                                  D3DRMCOLORSOURCE color)
508 {
509     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
510
511     FIXME("(%p)->(%x): stub\n", This, color);
512
513     return E_NOTIMPL;
514 }
515
516 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetBox(IDirect3DRMMeshBuilder2* iface,
517                                                          D3DRMBOX *pBox)
518 {
519     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
520
521     FIXME("(%p)->(%p): stub\n", This, pBox);
522
523     return E_NOTIMPL;
524 }
525
526 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GenerateNormals(IDirect3DRMMeshBuilder2* iface)
527 {
528     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
529
530     FIXME("(%p)->(): stub\n", This);
531
532     return E_NOTIMPL;
533 }
534
535 static D3DRMCOLORSOURCE WINAPI IDirect3DRMMeshBuilder2Impl_GetColorSource(IDirect3DRMMeshBuilder2* iface)
536 {
537     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
538
539     FIXME("(%p)->(): stub\n", This);
540
541     return E_NOTIMPL;
542 }
543
544 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddMesh(IDirect3DRMMeshBuilder2* iface,
545                                                           LPDIRECT3DRMMESH pMesh)
546 {
547     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
548
549     FIXME("(%p)->(%p): stub\n", This, pMesh);
550
551     return E_NOTIMPL;
552 }
553
554 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddMeshBuilder(IDirect3DRMMeshBuilder2* iface,
555                                                                  LPDIRECT3DRMMESHBUILDER pMeshBuilder)
556 {
557     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
558
559     FIXME("(%p)->(%p): stub\n", This, pMeshBuilder);
560
561     return E_NOTIMPL;
562 }
563
564 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddFrame(IDirect3DRMMeshBuilder2* iface,
565                                                            LPDIRECT3DRMFRAME pFrame)
566 {
567     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
568
569     FIXME("(%p)->(%p): stub\n", This, pFrame);
570
571     return E_NOTIMPL;
572 }
573
574 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddFace(IDirect3DRMMeshBuilder2* iface,
575                                                           LPDIRECT3DRMFACE pFace)
576 {
577     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
578
579     FIXME("(%p)->(%p): stub\n", This, pFace);
580
581     return E_NOTIMPL;
582 }
583
584 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_AddFaces(IDirect3DRMMeshBuilder2* iface,
585                                                            DWORD vcount, D3DVECTOR *vertices,
586                                                            DWORD ncount, D3DVECTOR *normals,
587                                                            DWORD *data,
588                                                            LPDIRECT3DRMFACEARRAY* pFaceArray)
589 {
590     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
591
592     FIXME("(%p)->(%d,%p,%d,%p,%p,%p): stub\n", This, vcount, vertices, ncount, normals, data, pFaceArray);
593
594     return E_NOTIMPL;
595 }
596
597 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_ReserveSpace(IDirect3DRMMeshBuilder2* iface,
598                                                                DWORD vertex_Count,
599                                                                DWORD normal_count,
600                                                                DWORD face_count)
601 {
602     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
603
604     FIXME("(%p)->(%d,%d,%d): stub\n", This, vertex_Count, normal_count, face_count);
605
606     return E_NOTIMPL;
607 }
608
609 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetColorRGB(IDirect3DRMMeshBuilder2* iface,
610                                                               D3DVALUE red, D3DVALUE green,
611                                                               D3DVALUE blue)
612 {
613     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
614
615     FIXME("(%p)->(%f,%f,%f): stub\n", This, red, green, blue);
616
617     return E_NOTIMPL;
618 }
619
620 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetColor(IDirect3DRMMeshBuilder2* iface,
621                                                            D3DCOLOR color)
622 {
623     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
624
625     FIXME("(%p)->(%x): stub\n", This, color);
626
627     return E_NOTIMPL;
628 }
629
630 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetTexture(IDirect3DRMMeshBuilder2* iface,
631                                                              LPDIRECT3DRMTEXTURE pTexture)
632 {
633     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
634
635     FIXME("(%p)->(%p): stub\n", This, pTexture);
636
637     return E_NOTIMPL;
638 }
639
640 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetMaterial(IDirect3DRMMeshBuilder2* iface,
641                                                               LPDIRECT3DRMMATERIAL pMaterial)
642 {
643     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
644
645     FIXME("(%p)->(%p): stub\n", This, pMaterial);
646
647     return E_NOTIMPL;
648 }
649
650 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetTextureTopology(IDirect3DRMMeshBuilder2* iface,
651                                                                      BOOL wrap_u, BOOL wrap_v)
652 {
653     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
654
655     FIXME("(%p)->(%d,%d): stub\n", This, wrap_u, wrap_v);
656
657     return E_NOTIMPL;
658 }
659
660 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetQuality(IDirect3DRMMeshBuilder2* iface,
661                                                              D3DRMRENDERQUALITY quality)
662 {
663     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
664
665     FIXME("(%p)->(%d): stub\n", This, quality);
666
667     return E_NOTIMPL;
668 }
669
670 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetPerspective(IDirect3DRMMeshBuilder2* iface,
671                                                                  BOOL enable)
672 {
673     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
674
675     FIXME("(%p)->(%d): stub\n", This, enable);
676
677     return E_NOTIMPL;
678 }
679
680 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetVertex(IDirect3DRMMeshBuilder2* iface,
681                                                             DWORD index,
682                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
683 {
684     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
685
686     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
687
688     return E_NOTIMPL;
689 }
690
691 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetNormal(IDirect3DRMMeshBuilder2* iface,
692                                                             DWORD index,
693                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
694 {
695     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
696
697     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
698
699     return E_NOTIMPL;
700 }
701
702 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetTextureCoordinates(IDirect3DRMMeshBuilder2* iface,
703                                                                         DWORD index,
704                                                                         D3DVALUE u, D3DVALUE v)
705 {
706     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
707
708     TRACE("(%p)->(%u,%f,%f)\n", This, index, u, v);
709
710     return IDirect3DRMMeshBuilder3_SetTextureCoordinates(&This->IDirect3DRMMeshBuilder3_iface,
711                                                          index, u, v);
712 }
713
714 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetVertexColor(IDirect3DRMMeshBuilder2* iface,
715                                                                  DWORD index, D3DCOLOR color)
716 {
717     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
718
719     FIXME("(%p)->(%d,%x): stub\n", This, index, color);
720
721     return E_NOTIMPL;
722 }
723
724 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_SetVertexColorRGB(IDirect3DRMMeshBuilder2* iface,
725                                                                     DWORD index, D3DVALUE red,
726                                                                     D3DVALUE green, D3DVALUE blue)
727 {
728     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
729
730     FIXME("(%p)->(%d,%f,%f,%f): stub\n", This, index, red, green, blue);
731
732     return E_NOTIMPL;
733 }
734
735 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetFaces(IDirect3DRMMeshBuilder2* iface,
736                                                            LPDIRECT3DRMFACEARRAY* pFaceArray)
737 {
738     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
739
740     FIXME("(%p)->(%p): stub\n", This, pFaceArray);
741
742     return E_NOTIMPL;
743 }
744
745 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetVertices(IDirect3DRMMeshBuilder2* iface,
746                                                               DWORD *vcount, D3DVECTOR *vertices,
747                                                               DWORD *ncount, D3DVECTOR *normals,
748                                                               DWORD *face_data_size,
749                                                               DWORD *face_data)
750 {
751     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
752
753     TRACE("(%p)->(%p,%p,%p,%p,%p,%p)\n", This, vcount, vertices, ncount, normals, face_data_size, face_data);
754
755     if (vcount)
756         *vcount = This->nb_vertices;
757     if (vertices && This->nb_vertices)
758         memcpy(vertices, This->pVertices, This->nb_vertices * sizeof(D3DVECTOR));
759     if (ncount)
760         *ncount = This->nb_normals;
761     if (normals && This->nb_normals)
762         memcpy(normals, This->pNormals, This->nb_normals * sizeof(D3DVECTOR));
763     if (face_data_size)
764         *face_data_size = This->face_data_size;
765     if (face_data && This->face_data_size)
766         memcpy(face_data, This->pFaceData, This->face_data_size * sizeof(DWORD));
767
768     return D3DRM_OK;
769 }
770
771 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetTextureCoordinates(IDirect3DRMMeshBuilder2* iface,
772                                                                         DWORD index,
773                                                                         D3DVALUE *u, D3DVALUE *v)
774 {
775     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
776
777     TRACE("(%p)->(%u,%p,%p)\n", This, index, u, v);
778
779     return IDirect3DRMMeshBuilder3_GetTextureCoordinates(&This->IDirect3DRMMeshBuilder3_iface,
780                                                          index, u, v);
781 }
782
783 static int WINAPI IDirect3DRMMeshBuilder2Impl_AddVertex(IDirect3DRMMeshBuilder2* iface,
784                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
785 {
786     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
787
788     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
789
790     return 0;
791 }
792
793 static int WINAPI IDirect3DRMMeshBuilder2Impl_AddNormal(IDirect3DRMMeshBuilder2* iface,
794                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
795 {
796     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
797
798     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
799
800     return 0;
801 }
802
803 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_CreateFace(IDirect3DRMMeshBuilder2* iface,
804                                                              LPDIRECT3DRMFACE* ppFace)
805 {
806     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
807
808     FIXME("(%p)->(%p): stub\n", This, ppFace);
809
810     return E_NOTIMPL;
811 }
812
813 static D3DRMRENDERQUALITY WINAPI IDirect3DRMMeshBuilder2Impl_GetQuality(IDirect3DRMMeshBuilder2* iface)
814 {
815     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
816
817     FIXME("(%p)->(): stub\n", This);
818
819     return 0;
820 }
821
822 static BOOL WINAPI IDirect3DRMMeshBuilder2Impl_GetPerspective(IDirect3DRMMeshBuilder2* iface)
823 {
824     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
825
826     FIXME("(%p)->(): stub\n", This);
827
828     return FALSE;
829 }
830
831 static int WINAPI IDirect3DRMMeshBuilder2Impl_GetFaceCount(IDirect3DRMMeshBuilder2* iface)
832 {
833     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
834
835     TRACE("(%p)->()\n", This);
836
837     return This->nb_faces;
838 }
839
840 static int WINAPI IDirect3DRMMeshBuilder2Impl_GetVertexCount(IDirect3DRMMeshBuilder2* iface)
841 {
842     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
843
844     TRACE("(%p)->()\n", This);
845
846     return This->nb_vertices;
847 }
848
849 static D3DCOLOR WINAPI IDirect3DRMMeshBuilder2Impl_GetVertexColor(IDirect3DRMMeshBuilder2* iface,
850                                                                   DWORD index)
851 {
852     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
853
854     FIXME("(%p)->(%d): stub\n", This, index);
855
856     return 0;
857 }
858
859 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_CreateMesh(IDirect3DRMMeshBuilder2* iface,
860                                                              LPDIRECT3DRMMESH* mesh)
861 {
862     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
863
864     TRACE("(%p)->(%p)\n", This, mesh);
865
866     return IDirect3DRMMeshBuilder3_CreateMesh(&This->IDirect3DRMMeshBuilder3_iface, mesh);
867 }
868
869 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GenerateNormals2(IDirect3DRMMeshBuilder2* iface,
870                                                                    D3DVALUE crease, DWORD dwFlags)
871 {
872     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
873
874     FIXME("(%p)->(%f,%u): stub\n", This, crease, dwFlags);
875
876     return E_NOTIMPL;
877 }
878
879 static HRESULT WINAPI IDirect3DRMMeshBuilder2Impl_GetFace(IDirect3DRMMeshBuilder2* iface,
880                                                           DWORD index, LPDIRECT3DRMFACE* ppFace)
881 {
882     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder2(iface);
883
884     FIXME("(%p)->(%u,%p): stub\n", This, index, ppFace);
885
886     return E_NOTIMPL;
887 }
888
889 static const struct IDirect3DRMMeshBuilder2Vtbl Direct3DRMMeshBuilder2_Vtbl =
890 {
891     /*** IUnknown methods ***/
892     IDirect3DRMMeshBuilder2Impl_QueryInterface,
893     IDirect3DRMMeshBuilder2Impl_AddRef,
894     IDirect3DRMMeshBuilder2Impl_Release,
895     /*** IDirect3DRMObject methods ***/
896     IDirect3DRMMeshBuilder2Impl_Clone,
897     IDirect3DRMMeshBuilder2Impl_AddDestroyCallback,
898     IDirect3DRMMeshBuilder2Impl_DeleteDestroyCallback,
899     IDirect3DRMMeshBuilder2Impl_SetAppData,
900     IDirect3DRMMeshBuilder2Impl_GetAppData,
901     IDirect3DRMMeshBuilder2Impl_SetName,
902     IDirect3DRMMeshBuilder2Impl_GetName,
903     IDirect3DRMMeshBuilder2Impl_GetClassName,
904     /*** IDirect3DRMMeshBuilder methods ***/
905     IDirect3DRMMeshBuilder2Impl_Load,
906     IDirect3DRMMeshBuilder2Impl_Save,
907     IDirect3DRMMeshBuilder2Impl_Scale,
908     IDirect3DRMMeshBuilder2Impl_Translate,
909     IDirect3DRMMeshBuilder2Impl_SetColorSource,
910     IDirect3DRMMeshBuilder2Impl_GetBox,
911     IDirect3DRMMeshBuilder2Impl_GenerateNormals,
912     IDirect3DRMMeshBuilder2Impl_GetColorSource,
913     IDirect3DRMMeshBuilder2Impl_AddMesh,
914     IDirect3DRMMeshBuilder2Impl_AddMeshBuilder,
915     IDirect3DRMMeshBuilder2Impl_AddFrame,
916     IDirect3DRMMeshBuilder2Impl_AddFace,
917     IDirect3DRMMeshBuilder2Impl_AddFaces,
918     IDirect3DRMMeshBuilder2Impl_ReserveSpace,
919     IDirect3DRMMeshBuilder2Impl_SetColorRGB,
920     IDirect3DRMMeshBuilder2Impl_SetColor,
921     IDirect3DRMMeshBuilder2Impl_SetTexture,
922     IDirect3DRMMeshBuilder2Impl_SetMaterial,
923     IDirect3DRMMeshBuilder2Impl_SetTextureTopology,
924     IDirect3DRMMeshBuilder2Impl_SetQuality,
925     IDirect3DRMMeshBuilder2Impl_SetPerspective,
926     IDirect3DRMMeshBuilder2Impl_SetVertex,
927     IDirect3DRMMeshBuilder2Impl_SetNormal,
928     IDirect3DRMMeshBuilder2Impl_SetTextureCoordinates,
929     IDirect3DRMMeshBuilder2Impl_SetVertexColor,
930     IDirect3DRMMeshBuilder2Impl_SetVertexColorRGB,
931     IDirect3DRMMeshBuilder2Impl_GetFaces,
932     IDirect3DRMMeshBuilder2Impl_GetVertices,
933     IDirect3DRMMeshBuilder2Impl_GetTextureCoordinates,
934     IDirect3DRMMeshBuilder2Impl_AddVertex,
935     IDirect3DRMMeshBuilder2Impl_AddNormal,
936     IDirect3DRMMeshBuilder2Impl_CreateFace,
937     IDirect3DRMMeshBuilder2Impl_GetQuality,
938     IDirect3DRMMeshBuilder2Impl_GetPerspective,
939     IDirect3DRMMeshBuilder2Impl_GetFaceCount,
940     IDirect3DRMMeshBuilder2Impl_GetVertexCount,
941     IDirect3DRMMeshBuilder2Impl_GetVertexColor,
942     IDirect3DRMMeshBuilder2Impl_CreateMesh,
943     /*** IDirect3DRMMeshBuilder2 methods ***/
944     IDirect3DRMMeshBuilder2Impl_GenerateNormals2,
945     IDirect3DRMMeshBuilder2Impl_GetFace
946 };
947
948
949 /*** IUnknown methods ***/
950 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_QueryInterface(IDirect3DRMMeshBuilder3* iface,
951                                                                  REFIID riid, void** ppvObject)
952 {
953     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
954     return IDirect3DRMMeshBuilder2_QueryInterface(&This->IDirect3DRMMeshBuilder2_iface, riid, ppvObject);
955 }
956
957 static ULONG WINAPI IDirect3DRMMeshBuilder3Impl_AddRef(IDirect3DRMMeshBuilder3* iface)
958 {
959     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
960     return IDirect3DRMMeshBuilder2_AddRef(&This->IDirect3DRMMeshBuilder2_iface);
961 }
962
963 static ULONG WINAPI IDirect3DRMMeshBuilder3Impl_Release(IDirect3DRMMeshBuilder3* iface)
964 {
965     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
966     return IDirect3DRMMeshBuilder2_Release(&This->IDirect3DRMMeshBuilder2_iface);
967 }
968
969 /*** IDirect3DRMObject methods ***/
970 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Clone(IDirect3DRMMeshBuilder3* iface,
971                                                         LPUNKNOWN pUnkOuter, REFIID riid,
972                                                         LPVOID *ppvObj)
973 {
974     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
975
976     FIXME("(%p)->(%p,%s,%p): stub\n", This, pUnkOuter, debugstr_guid(riid), ppvObj);
977
978     return E_NOTIMPL;
979 }
980
981 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddDestroyCallback(IDirect3DRMMeshBuilder3* iface,
982                                                                      D3DRMOBJECTCALLBACK cb,
983                                                                      LPVOID argument)
984 {
985     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
986
987     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
988
989     return E_NOTIMPL;
990 }
991
992 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteDestroyCallback(IDirect3DRMMeshBuilder3* iface,
993                                                                         D3DRMOBJECTCALLBACK cb,
994                                                                         LPVOID argument)
995 {
996     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
997
998     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
999
1000     return E_NOTIMPL;
1001 }
1002
1003 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetAppData(IDirect3DRMMeshBuilder3* iface,
1004                                                              DWORD data)
1005 {
1006     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1007
1008     FIXME("(%p)->(%u): stub\n", This, data);
1009
1010     return E_NOTIMPL;
1011 }
1012
1013 static DWORD WINAPI IDirect3DRMMeshBuilder3Impl_GetAppData(IDirect3DRMMeshBuilder3* iface)
1014 {
1015     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1016
1017     FIXME("(%p)->(): stub\n", This);
1018
1019     return 0;
1020 }
1021
1022 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetName(IDirect3DRMMeshBuilder3* iface,
1023                                                           LPCSTR name)
1024 {
1025     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1026     char *string = NULL;
1027
1028     TRACE("(%p)->(%s)\n", This, debugstr_a(name));
1029
1030     if (name)
1031     {
1032         string = HeapAlloc(GetProcessHeap(), 0, strlen(name) + 1);
1033         if (!string) return E_OUTOFMEMORY;
1034         strcpy(string, name);
1035     }
1036     HeapFree(GetProcessHeap(), 0, This->name);
1037     This->name = string;
1038
1039     return D3DRM_OK;
1040 }
1041
1042 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetName(IDirect3DRMMeshBuilder3* iface,
1043                                                           LPDWORD size, LPSTR name)
1044 {
1045     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1046
1047     TRACE("(%p)->(%p,%p)\n", This, size, name);
1048
1049     if (!size)
1050         return E_POINTER;
1051
1052     if (!This->name)
1053     {
1054         *size = 0;
1055         return D3DRM_OK;
1056     }
1057
1058     if (*size < (strlen(This->name) + 1))
1059         return E_INVALIDARG;
1060
1061     strcpy(name, This->name);
1062     *size = strlen(This->name) + 1;
1063
1064     return D3DRM_OK;
1065 }
1066
1067 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetClassName(IDirect3DRMMeshBuilder3* iface,
1068                                                                LPDWORD lpdwSize, LPSTR lpName)
1069 {
1070     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1071
1072     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
1073
1074     return E_NOTIMPL;
1075 }
1076
1077 HRESULT load_mesh_data(IDirect3DRMMeshBuilder3* iface, LPDIRECTXFILEDATA pData)
1078 {
1079     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1080     LPDIRECTXFILEOBJECT pObject = NULL;
1081     LPDIRECTXFILEDATA pData2 = NULL;
1082     const GUID* pGuid;
1083     DWORD size;
1084     LPBYTE ptr;
1085     HRESULT hr;
1086     HRESULT ret = D3DRMERR_BADOBJECT;
1087     DWORD* faces_vertex_idx_data = NULL;
1088     DWORD* faces_vertex_idx_ptr;
1089     DWORD faces_vertex_idx_size;
1090     DWORD* faces_normal_idx_data = NULL;
1091     DWORD* faces_normal_idx_ptr = NULL;
1092     DWORD* faces_data_ptr;
1093     DWORD faces_data_size = 0;
1094     DWORD i;
1095
1096     TRACE("(%p)->(%p)\n", This, pData);
1097
1098     /* Remove previous name */
1099     HeapFree(GetProcessHeap(), 0, This->name);
1100     This->name = NULL;
1101     hr = IDirectXFileData_GetName(pData, NULL, &size);
1102     if (hr != DXFILE_OK)
1103         return hr;
1104     if (size)
1105     {
1106         This->name = HeapAlloc(GetProcessHeap(), 0, size);
1107         if (!This->name)
1108             return E_OUTOFMEMORY;
1109
1110         hr = IDirectXFileData_GetName(pData, This->name, &size);
1111         if (hr != DXFILE_OK)
1112         {
1113             HeapFree(GetProcessHeap(), 0, This->name);
1114             This->name = NULL;
1115             return hr;
1116         }
1117     }
1118
1119     TRACE("Mesh name is '%s'\n", This->name ? This->name : "");
1120
1121     This->nb_normals = 0;
1122
1123     hr = IDirectXFileData_GetData(pData, NULL, &size, (void**)&ptr);
1124     if (hr != DXFILE_OK)
1125         goto end;
1126
1127     This->nb_vertices = *(DWORD*)ptr;
1128     This->nb_faces = *(DWORD*)(ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR));
1129     faces_vertex_idx_size = size - sizeof(DWORD) - This->nb_vertices * sizeof(D3DVECTOR) - sizeof(DWORD);
1130     faces_vertex_idx_ptr = (DWORD*)(ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR) + sizeof(DWORD));
1131
1132     TRACE("Mesh: nb_vertices = %d, nb_faces = %d, faces_vertex_idx_size = %d\n", This->nb_vertices, This->nb_faces, faces_vertex_idx_size);
1133
1134     This->pVertices = HeapAlloc(GetProcessHeap(), 0, This->nb_vertices * sizeof(D3DVECTOR));
1135     memcpy(This->pVertices, ptr + sizeof(DWORD), This->nb_vertices * sizeof(D3DVECTOR));
1136
1137     faces_vertex_idx_ptr = faces_vertex_idx_data = HeapAlloc(GetProcessHeap(), 0, faces_vertex_idx_size);
1138     memcpy(faces_vertex_idx_data, ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR) + sizeof(DWORD), faces_vertex_idx_size);
1139
1140     /* Each vertex index will have its normal index counterpart so just allocate twice the size */
1141     This->pFaceData = HeapAlloc(GetProcessHeap(), 0, faces_vertex_idx_size*2);
1142     faces_data_ptr = (DWORD*)This->pFaceData;
1143
1144     while (1)
1145     {
1146         hr =  IDirectXFileData_GetNextObject(pData, &pObject);
1147         if (hr == DXFILEERR_NOMOREOBJECTS)
1148         {
1149             TRACE("No more object\n");
1150             break;
1151         }
1152         if (hr != DXFILE_OK)
1153            goto end;
1154
1155         hr = IDirectXFileObject_QueryInterface(pObject, &IID_IDirectXFileData, (void**)&pData2);
1156         IDirectXFileObject_Release(pObject);
1157         if (hr != DXFILE_OK)
1158             goto end;
1159
1160         hr = IDirectXFileData_GetType(pData2, &pGuid);
1161         if (hr != DXFILE_OK)
1162             goto end;
1163
1164         TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1165
1166         if (IsEqualGUID(pGuid, &TID_D3DRMMeshNormals))
1167         {
1168             DWORD nb_faces_normals;
1169             DWORD faces_normal_idx_size;
1170
1171             hr = IDirectXFileData_GetData(pData2, NULL, &size, (void**)&ptr);
1172             if (hr != DXFILE_OK)
1173                 goto end;
1174
1175             This->nb_normals = *(DWORD*)ptr;
1176             nb_faces_normals = *(DWORD*)(ptr + sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR));
1177
1178             TRACE("MeshNormals: nb_normals = %d, nb_faces_normals = %d\n", This->nb_normals, nb_faces_normals);
1179             if (nb_faces_normals != This->nb_faces)
1180                 WARN("nb_face_normals (%d) != nb_faces (%d)\n", nb_faces_normals, This->nb_normals);
1181
1182             This->pNormals = HeapAlloc(GetProcessHeap(), 0, This->nb_normals * sizeof(D3DVECTOR));
1183             memcpy(This->pNormals, ptr + sizeof(DWORD), This->nb_normals * sizeof(D3DVECTOR));
1184
1185             faces_normal_idx_size = size - (2*sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR));
1186             faces_normal_idx_ptr = faces_normal_idx_data = HeapAlloc(GetProcessHeap(), 0, faces_normal_idx_size);
1187             memcpy(faces_normal_idx_data, ptr + sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR) + sizeof(DWORD), faces_normal_idx_size);
1188         }
1189         else if (IsEqualGUID(pGuid, &TID_D3DRMMeshTextureCoords))
1190         {
1191             hr = IDirectXFileData_GetData(pData2, NULL, &size, (void**)&ptr);
1192             if (hr != DXFILE_OK)
1193                 goto end;
1194
1195             This->nb_coords2d = *(DWORD*)ptr;
1196
1197             TRACE("MeshTextureCoords: nb_coords2d = %d\n", This->nb_coords2d);
1198
1199             This->pCoords2d = HeapAlloc(GetProcessHeap(), 0, This->nb_coords2d * sizeof(Coords2d));
1200             memcpy(This->pCoords2d, ptr + sizeof(DWORD), This->nb_coords2d * sizeof(Coords2d));
1201
1202         }
1203         else if (IsEqualGUID(pGuid, &TID_D3DRMMeshMaterialList))
1204         {
1205             FIXME("MeshMaterialList not supported yet, ignoring...\n");
1206         }
1207         else
1208         {
1209             FIXME("Unknown GUID %s, ignoring...\n", debugstr_guid(pGuid));
1210         }
1211
1212         IDirectXFileData_Release(pData2);
1213         pData2 = NULL;
1214     }
1215
1216     if (!This->nb_normals)
1217     {
1218         /* Allocate normals, one per vertex */
1219         This->pNormals = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->nb_vertices * sizeof(D3DVECTOR));
1220         if (!This->pNormals)
1221             goto end;
1222     }
1223
1224     for (i = 0; i < This->nb_faces; i++)
1225     {
1226         DWORD j;
1227         DWORD nb_face_indexes;
1228         D3DVECTOR face_normal;
1229
1230         if (faces_vertex_idx_size < sizeof(DWORD))
1231             WARN("Not enough data to read number of indices of face %d\n", i);
1232
1233         nb_face_indexes  = *(faces_data_ptr + faces_data_size++) = *(faces_vertex_idx_ptr++);
1234         faces_vertex_idx_size--;
1235         if (faces_normal_idx_data && (*(faces_normal_idx_ptr++) != nb_face_indexes))
1236             WARN("Faces indices number mismatch\n");
1237
1238         if (faces_vertex_idx_size < (nb_face_indexes * sizeof(DWORD)))
1239             WARN("Not enough data to read all indices of face %d\n", i);
1240
1241         if (!This->nb_normals)
1242         {
1243             /* Compute face normal */
1244             if (nb_face_indexes > 2)
1245             {
1246                 D3DVECTOR a, b;
1247
1248                 D3DRMVectorSubtract(&a, &This->pVertices[faces_vertex_idx_ptr[2]], &This->pVertices[faces_vertex_idx_ptr[1]]);
1249                 D3DRMVectorSubtract(&a, &This->pVertices[faces_vertex_idx_ptr[1]], &This->pVertices[faces_vertex_idx_ptr[0]]);
1250                 D3DRMVectorCrossProduct(&face_normal, &a, &b);
1251                 D3DRMVectorNormalize(&face_normal);
1252             }
1253             else
1254             {
1255                 face_normal.u1.x = 0.0f;
1256                 face_normal.u2.y = 0.0f;
1257                 face_normal.u3.z = 0.0f;
1258             }
1259         }
1260
1261         for (j = 0; j < nb_face_indexes; j++)
1262         {
1263             /* Copy vertex index */
1264             *(faces_data_ptr + faces_data_size++) = *faces_vertex_idx_ptr;
1265             /* Copy normal index */
1266             if (This->nb_normals)
1267             {
1268                 /* Read from x file */
1269                 *(faces_data_ptr + faces_data_size++) = *(faces_normal_idx_ptr++);
1270             }
1271             else
1272             {
1273                 DWORD vertex_idx = *faces_vertex_idx_ptr;
1274                 if (vertex_idx > This->nb_vertices)
1275                     vertex_idx = 0;
1276                 *(faces_data_ptr + faces_data_size++) = vertex_idx;
1277                 /* Add face normal to vertex normal */
1278                 D3DRMVectorAdd(&This->pNormals[vertex_idx], &This->pNormals[vertex_idx], &face_normal);
1279             }
1280             faces_vertex_idx_ptr++;
1281         }
1282         faces_vertex_idx_size -= nb_face_indexes;
1283     }
1284
1285     /* Last DWORD must be 0 */
1286     *(faces_data_ptr + faces_data_size++) = 0;
1287
1288     /* Set size (in number of DWORD) of all faces data */
1289     This->face_data_size = faces_data_size;
1290
1291     if (!This->nb_normals)
1292     {
1293         /* Normalize all normals */
1294         for (i = 0; i < This->nb_vertices; i++)
1295         {
1296             D3DRMVectorNormalize(&This->pNormals[i]);
1297         }
1298         This->nb_normals = This->nb_vertices;
1299     }
1300
1301     /* If there is no texture coordinates, generate default texture coordinates (0.0f, 0.0f) for each vertex */
1302     if (!This->pCoords2d)
1303     {
1304         This->nb_coords2d = This->nb_vertices;
1305         This->pCoords2d = HeapAlloc(GetProcessHeap(), 0, This->nb_coords2d * sizeof(Coords2d));
1306         for (i = 0; i < This->nb_coords2d; i++)
1307         {
1308             This->pCoords2d[i].u = 0.0f;
1309             This->pCoords2d[i].v = 0.0f;
1310         }
1311     }
1312
1313     TRACE("Mesh data loaded successfully\n");
1314
1315     ret = D3DRM_OK;
1316
1317 end:
1318
1319     HeapFree(GetProcessHeap(), 0, faces_normal_idx_data);
1320     HeapFree(GetProcessHeap(), 0, faces_vertex_idx_data);
1321
1322     return ret;
1323 }
1324
1325 /*** IDirect3DRMMeshBuilder3 methods ***/
1326 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Load(IDirect3DRMMeshBuilder3* iface,
1327                                                        LPVOID filename, LPVOID name,
1328                                                        D3DRMLOADOPTIONS loadflags,
1329                                                        D3DRMLOADTEXTURE3CALLBACK cb, LPVOID arg)
1330 {
1331     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1332     DXFILELOADOPTIONS load_options;
1333     LPDIRECTXFILE pDXFile = NULL;
1334     LPDIRECTXFILEENUMOBJECT pEnumObject = NULL;
1335     LPDIRECTXFILEDATA pData = NULL;
1336     LPDIRECTXFILEDATA pData2 = NULL;
1337     const GUID* pGuid;
1338     DWORD size;
1339     Header* pHeader;
1340     HRESULT hr;
1341     HRESULT ret = D3DRMERR_BADOBJECT;
1342
1343     TRACE("(%p)->(%p,%p,%x,%p,%p)\n", This, filename, name, loadflags, cb, arg);
1344
1345     /* First free allocated buffers of previous mesh data */
1346     HeapFree(GetProcessHeap(), 0, This->pVertices);
1347     This->pVertices = NULL;
1348     HeapFree(GetProcessHeap(), 0, This->pNormals);
1349     This->pNormals = NULL;
1350     HeapFree(GetProcessHeap(), 0, This->pFaceData);
1351     This->pFaceData = NULL;
1352     HeapFree(GetProcessHeap(), 0, This->pCoords2d);
1353     This->pCoords2d = NULL;
1354
1355     if (loadflags == D3DRMLOAD_FROMMEMORY)
1356     {
1357         load_options = DXFILELOAD_FROMMEMORY;
1358     }
1359     else if (loadflags == D3DRMLOAD_FROMFILE)
1360     {
1361         load_options = DXFILELOAD_FROMFILE;
1362         TRACE("Loading from file %s\n", debugstr_a(filename));
1363     }
1364     else
1365     {
1366         FIXME("Load options %d not supported yet\n", loadflags);
1367         return E_NOTIMPL;
1368     }
1369
1370     hr = DirectXFileCreate(&pDXFile);
1371     if (hr != DXFILE_OK)
1372         goto end;
1373
1374     hr = IDirectXFile_RegisterTemplates(pDXFile, templates, strlen(templates));
1375     if (hr != DXFILE_OK)
1376         goto end;
1377
1378     hr = IDirectXFile_CreateEnumObject(pDXFile, filename, load_options, &pEnumObject);
1379     if (hr != DXFILE_OK)
1380         goto end;
1381
1382     hr = IDirectXFileEnumObject_GetNextDataObject(pEnumObject, &pData);
1383     if (hr != DXFILE_OK)
1384         goto end;
1385
1386     hr = IDirectXFileData_GetType(pData, &pGuid);
1387     if (hr != DXFILE_OK)
1388         goto end;
1389
1390     TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1391
1392     if (!IsEqualGUID(pGuid, &TID_DXFILEHeader))
1393     {
1394         ret = D3DRMERR_BADFILE;
1395         goto end;
1396     }
1397
1398     hr = IDirectXFileData_GetData(pData, NULL, &size, (void**)&pHeader);
1399     if ((hr != DXFILE_OK) || (size != sizeof(Header)))
1400         goto end;
1401
1402     TRACE("Version is %d %d %d\n", pHeader->major, pHeader->minor, pHeader->flags);
1403
1404     /* Version must be 1.0.x */
1405     if ((pHeader->major != 1) || (pHeader->minor != 0))
1406     {
1407         ret = D3DRMERR_BADFILE;
1408         goto end;
1409     }
1410
1411     IDirectXFileData_Release(pData);
1412     pData = NULL;
1413
1414     hr = IDirectXFileEnumObject_GetNextDataObject(pEnumObject, &pData);
1415     if (hr != DXFILE_OK)
1416     {
1417         ret = D3DRMERR_NOTFOUND;
1418         goto end;
1419     }
1420
1421     hr = IDirectXFileData_GetType(pData, &pGuid);
1422     if (hr != DXFILE_OK)
1423         goto end;
1424
1425     TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1426
1427     if (!IsEqualGUID(pGuid, &TID_D3DRMMesh))
1428     {
1429         ret = D3DRMERR_NOTFOUND;
1430         goto end;
1431     }
1432
1433     hr = load_mesh_data(iface, pData);
1434     if (hr == S_OK)
1435         ret = D3DRM_OK;
1436
1437 end:
1438
1439     if (pData2)
1440         IDirectXFileData_Release(pData2);
1441     if (pData)
1442         IDirectXFileData_Release(pData);
1443     if (pEnumObject)
1444         IDirectXFileEnumObject_Release(pEnumObject);
1445     if (pDXFile)
1446         IDirectXFile_Release(pDXFile);
1447
1448     if (ret != D3DRM_OK)
1449     {
1450         /* Clean mesh data */
1451         This->nb_vertices = 0;
1452         This->nb_normals = 0;
1453         This->nb_faces = 0;
1454         This->face_data_size = 0;
1455         This->nb_coords2d = 0;
1456         HeapFree(GetProcessHeap(), 0, This->pVertices);
1457         This->pVertices = NULL;
1458         HeapFree(GetProcessHeap(), 0, This->pNormals);
1459         This->pNormals = NULL;
1460         HeapFree(GetProcessHeap(), 0, This->pFaceData);
1461         This->pFaceData = NULL;
1462         HeapFree(GetProcessHeap(), 0, This->pCoords2d);
1463         This->pCoords2d = NULL;
1464     }
1465
1466     return ret;
1467 }
1468
1469 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Save(IDirect3DRMMeshBuilder3* iface,
1470                                                        const char* filename, D3DRMXOFFORMAT format,
1471                                                        D3DRMSAVEOPTIONS save)
1472 {
1473     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1474
1475     FIXME("(%p)->(%s,%d,%d): stub\n", This, filename, format, save);
1476
1477     return E_NOTIMPL;
1478 }
1479
1480 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Scale(IDirect3DRMMeshBuilder3* iface,
1481                                                         D3DVALUE sx, D3DVALUE sy, D3DVALUE sz)
1482 {
1483     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1484     int i;
1485
1486     TRACE("(%p)->(%f,%f,%f)\n", This, sx, sy, sz);
1487
1488     for (i = 0; i < This->nb_vertices; i++)
1489     {
1490         This->pVertices[i].u1.x *= sx;
1491         This->pVertices[i].u2.y *= sy;
1492         This->pVertices[i].u3.z *= sz;
1493     }
1494
1495     /* Normals are not affected by Scale */
1496
1497     return D3DRM_OK;
1498 }
1499
1500 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Translate(IDirect3DRMMeshBuilder3* iface,
1501                                                             D3DVALUE tx, D3DVALUE ty, D3DVALUE tz)
1502 {
1503     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1504
1505     FIXME("(%p)->(%f,%f,%f): stub\n", This, tx, ty, tz);
1506
1507     return E_NOTIMPL;
1508 }
1509
1510 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColorSource(IDirect3DRMMeshBuilder3* iface,
1511                                                                  D3DRMCOLORSOURCE color)
1512 {
1513     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1514
1515     FIXME("(%p)->(%x): stub\n", This, color);
1516
1517     return E_NOTIMPL;
1518 }
1519
1520 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetBox(IDirect3DRMMeshBuilder3* iface,
1521                                                          D3DRMBOX* box)
1522 {
1523     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1524
1525     FIXME("(%p)->(%p): stub\n", This, box);
1526
1527     return E_NOTIMPL;
1528 }
1529
1530 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GenerateNormals(IDirect3DRMMeshBuilder3* iface,
1531                                                                   D3DVALUE crease, DWORD flags)
1532 {
1533     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1534
1535     FIXME("(%p)->(%f,%u): stub\n", This, crease, flags);
1536
1537     return E_NOTIMPL;
1538 }
1539
1540 static D3DRMCOLORSOURCE WINAPI IDirect3DRMMeshBuilder3Impl_GetColorSource(IDirect3DRMMeshBuilder3* iface)
1541 {
1542     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1543
1544     FIXME("(%p)->(): stub\n", This);
1545
1546     return E_NOTIMPL;
1547 }
1548
1549 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddMesh(IDirect3DRMMeshBuilder3* iface,
1550                                                           LPDIRECT3DRMMESH Mesh)
1551 {
1552     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1553
1554     FIXME("(%p)->(%p): stub\n", This, Mesh);
1555
1556     return E_NOTIMPL;
1557 }
1558
1559 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddMeshBuilder(IDirect3DRMMeshBuilder3* iface,
1560                                                                  LPDIRECT3DRMMESHBUILDER3 MeshBuilder,
1561                                                                  DWORD flags)
1562 {
1563     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1564
1565     FIXME("(%p)->(%p,%u): stub\n", This, MeshBuilder, flags);
1566
1567     return E_NOTIMPL;
1568 }
1569
1570 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFrame(IDirect3DRMMeshBuilder3* iface,
1571                                                            LPDIRECT3DRMFRAME3 Frame)
1572 {
1573     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1574
1575     FIXME("(%p)->(%p): stub\n", This, Frame);
1576
1577     return E_NOTIMPL;
1578 }
1579
1580 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFace(IDirect3DRMMeshBuilder3* iface,
1581                                                           LPDIRECT3DRMFACE2 Face)
1582 {
1583     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1584
1585     FIXME("(%p)->(%p): stub\n", This, Face);
1586
1587     return E_NOTIMPL;
1588 }
1589
1590 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFaces(IDirect3DRMMeshBuilder3* iface,
1591                                                            DWORD vcount, D3DVECTOR* vertices,
1592                                                            DWORD ncount, D3DVECTOR* normals,
1593                                                            DWORD* data,
1594                                                            LPDIRECT3DRMFACEARRAY* FaceArray)
1595 {
1596     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1597
1598     FIXME("(%p)->(%d,%p,%d,%p,%p,%p): stub\n", This, vcount, vertices, ncount,
1599           normals, data, FaceArray);
1600
1601     return E_NOTIMPL;
1602 }
1603
1604 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_ReserveSpace(IDirect3DRMMeshBuilder3* iface,
1605                                                                DWORD vertex_Count,
1606                                                                DWORD normal_count,
1607                                                                DWORD face_count)
1608 {
1609     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1610
1611     FIXME("(%p)->(%d,%d,%d): stub\n", This, vertex_Count, normal_count, face_count);
1612
1613     return E_NOTIMPL;
1614 }
1615
1616 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColorRGB(IDirect3DRMMeshBuilder3* iface,
1617                                                               D3DVALUE red, D3DVALUE green,
1618                                                               D3DVALUE blue)
1619 {
1620     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1621
1622     FIXME("(%p)->(%f,%f,%f): stub\n", This, red, green, blue);
1623
1624     return E_NOTIMPL;
1625 }
1626
1627 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColor(IDirect3DRMMeshBuilder3* iface,
1628                                                            D3DCOLOR color)
1629 {
1630     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1631
1632     FIXME("(%p)->(%x): stub\n", This, color);
1633
1634     return E_NOTIMPL;
1635 }
1636
1637 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTexture(IDirect3DRMMeshBuilder3* iface,
1638                                                              LPDIRECT3DRMTEXTURE3 texture)
1639 {
1640     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1641
1642     FIXME("(%p)->(%p): stub\n", This, texture);
1643
1644     return E_NOTIMPL;
1645 }
1646
1647 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetMaterial(IDirect3DRMMeshBuilder3* iface,
1648                                                               LPDIRECT3DRMMATERIAL2 material)
1649 {
1650     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1651
1652     FIXME("(%p)->(%p): stub\n", This, material);
1653
1654     return E_NOTIMPL;
1655 }
1656
1657 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTextureTopology(IDirect3DRMMeshBuilder3* iface,
1658                                                                      BOOL wrap_u, BOOL wrap_v)
1659 {
1660     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1661
1662     FIXME("(%p)->(%d,%d): stub\n", This, wrap_u, wrap_v);
1663
1664     return E_NOTIMPL;
1665 }
1666
1667 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetQuality(IDirect3DRMMeshBuilder3* iface,
1668                                                              D3DRMRENDERQUALITY quality)
1669 {
1670     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1671
1672     FIXME("(%p)->(%d): stub\n", This, quality);
1673
1674     return E_NOTIMPL;
1675 }
1676
1677 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetPerspective(IDirect3DRMMeshBuilder3* iface,
1678                                                                  BOOL enable)
1679 {
1680     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1681
1682     FIXME("(%p)->(%d): stub\n", This, enable);
1683
1684     return E_NOTIMPL;
1685 }
1686
1687 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertex(IDirect3DRMMeshBuilder3* iface,
1688                                                             DWORD index,
1689                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
1690 {
1691     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1692
1693     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1694
1695     return E_NOTIMPL;
1696 }
1697
1698 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetNormal(IDirect3DRMMeshBuilder3* iface,
1699                                                             DWORD index,
1700                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
1701 {
1702     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1703
1704     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1705
1706     return E_NOTIMPL;
1707 }
1708
1709 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTextureCoordinates(IDirect3DRMMeshBuilder3* iface,
1710                                                                         DWORD index, D3DVALUE u,
1711                                                                         D3DVALUE v)
1712 {
1713     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1714
1715     TRACE("(%p)->(%u,%f,%f)\n", This, index, u, v);
1716
1717     if (index >= This->nb_coords2d)
1718         return D3DRMERR_BADVALUE;
1719
1720     This->pCoords2d[index].u = u;
1721     This->pCoords2d[index].v = v;
1722
1723     return D3DRM_OK;
1724 }
1725
1726 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertexColor(IDirect3DRMMeshBuilder3* iface,
1727                                                                  DWORD index, D3DCOLOR color)
1728 {
1729     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1730
1731     FIXME("(%p)->(%d,%x): stub\n", This, index, color);
1732
1733     return E_NOTIMPL;
1734 }
1735
1736 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertexColorRGB(IDirect3DRMMeshBuilder3* iface,
1737                                                                     DWORD index,
1738                                                                     D3DVALUE red, D3DVALUE green,
1739                                                                     D3DVALUE blue)
1740 {
1741     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1742
1743     FIXME("(%p)->(%d,%f,%f,%f): stub\n", This, index, red, green, blue);
1744
1745     return E_NOTIMPL;
1746 }
1747
1748 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetFaces(IDirect3DRMMeshBuilder3* iface,
1749                                                            LPDIRECT3DRMFACEARRAY* FaceArray)
1750 {
1751     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1752
1753     FIXME("(%p)->(%p): stub\n", This, FaceArray);
1754
1755     return E_NOTIMPL;
1756 }
1757
1758 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetGeometry(IDirect3DRMMeshBuilder3* iface,
1759                                                               DWORD* vcount, D3DVECTOR* vertices,
1760                                                               DWORD* ncount, D3DVECTOR* normals,
1761                                                               DWORD* face_data_size,
1762                                                               DWORD* face_data)
1763 {
1764     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1765
1766     FIXME("(%p)->(%p,%p,%p,%p,%p,%p): stub\n", This, vcount, vertices, ncount, normals,
1767           face_data_size, face_data);
1768
1769     return E_NOTIMPL;
1770 }
1771
1772 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetTextureCoordinates(IDirect3DRMMeshBuilder3* iface,
1773                                                                         DWORD index, D3DVALUE* u,
1774                                                                         D3DVALUE* v)
1775 {
1776     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1777
1778     TRACE("(%p)->(%d,%p,%p)\n", This, index, u, v);
1779
1780     if (index >= This->nb_coords2d)
1781         return D3DRMERR_BADVALUE;
1782
1783     *u = This->pCoords2d[index].u;
1784     *v = This->pCoords2d[index].v;
1785
1786     return D3DRM_OK;
1787 }
1788
1789
1790 static int WINAPI IDirect3DRMMeshBuilder3Impl_AddVertex(IDirect3DRMMeshBuilder3* iface,
1791                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
1792 {
1793     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1794
1795     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1796
1797     return 0;
1798 }
1799
1800 static int WINAPI IDirect3DRMMeshBuilder3Impl_AddNormal(IDirect3DRMMeshBuilder3* iface,
1801                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
1802 {
1803     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1804
1805     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1806
1807     return 0;
1808 }
1809
1810 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateFace(IDirect3DRMMeshBuilder3* iface,
1811                                                              LPDIRECT3DRMFACE2* Face)
1812 {
1813     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1814
1815     FIXME("(%p)->(%p): stub\n", This, Face);
1816
1817     return E_NOTIMPL;
1818 }
1819
1820 static D3DRMRENDERQUALITY WINAPI IDirect3DRMMeshBuilder3Impl_GetQuality(IDirect3DRMMeshBuilder3* iface)
1821 {
1822     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1823
1824     FIXME("(%p)->(): stub\n", This);
1825
1826     return 0;
1827 }
1828
1829 static BOOL WINAPI IDirect3DRMMeshBuilder3Impl_GetPerspective(IDirect3DRMMeshBuilder3* iface)
1830 {
1831     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1832
1833     FIXME("(%p)->(): stub\n", This);
1834
1835     return FALSE;
1836 }
1837
1838 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetFaceCount(IDirect3DRMMeshBuilder3* iface)
1839 {
1840     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1841
1842     TRACE("(%p)->()\n", This);
1843
1844     return This->nb_faces;
1845 }
1846
1847 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetVertexCount(IDirect3DRMMeshBuilder3* iface)
1848 {
1849     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1850
1851     TRACE("(%p)->()\n", This);
1852
1853     return This->nb_vertices;
1854 }
1855
1856 static D3DCOLOR WINAPI IDirect3DRMMeshBuilder3Impl_GetVertexColor(IDirect3DRMMeshBuilder3* iface,
1857                                                                   DWORD index)
1858 {
1859     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1860
1861     FIXME("(%p)->(%d): stub\n", This, index);
1862
1863     return 0;
1864 }
1865
1866 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateMesh(IDirect3DRMMeshBuilder3* iface,
1867                                                              LPDIRECT3DRMMESH* mesh)
1868 {
1869     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1870     HRESULT hr;
1871     D3DRMGROUPINDEX group;
1872
1873     TRACE("(%p)->(%p)\n", This, mesh);
1874
1875     if (!mesh)
1876         return E_POINTER;
1877
1878     hr = Direct3DRMMesh_create(mesh);
1879     if (FAILED(hr))
1880         return hr;
1881
1882     /* If there is mesh data, create a group and put data inside */
1883     if (This->nb_vertices)
1884     {
1885         unsigned* face_data;
1886         unsigned* out_ptr;
1887         DWORD* in_ptr = This->pFaceData;
1888         int i, j;
1889
1890         face_data = HeapAlloc(GetProcessHeap(), 0, This->face_data_size * sizeof(DWORD));
1891         if (!face_data)
1892         {
1893             IDirect3DRMMesh_Release(*mesh);
1894             return E_OUTOFMEMORY;
1895         }
1896         out_ptr = face_data;
1897
1898         /* Put only vertex indices */
1899         for (i = 0; i < This->nb_faces; i++)
1900         {
1901             DWORD nb_indices = *out_ptr++ = *in_ptr++;
1902             for (j = 0; j < nb_indices; j++)
1903             {
1904                 *out_ptr++ = *in_ptr++;
1905                 /* Skip normal index */
1906                 in_ptr++;
1907             }
1908         }
1909
1910         hr = IDirect3DRMMesh_AddGroup(*mesh, This->nb_vertices, This->nb_faces, 0, face_data, &group);
1911         HeapFree(GetProcessHeap(), 0, face_data);
1912         if (SUCCEEDED(hr))
1913         {
1914             D3DRMVERTEX* vertices;
1915
1916             vertices = HeapAlloc(GetProcessHeap(), 0, This->nb_vertices * sizeof(D3DRMVERTEX));
1917             if (vertices)
1918             {
1919                 for (i = 0; i < This->nb_vertices; i++)
1920                 {
1921                     vertices[i].position = This->pVertices[i];
1922                 }
1923                 hr = IDirect3DRMMesh_SetVertices(*mesh, 0, 0, This->nb_vertices, vertices);
1924                 HeapFree(GetProcessHeap(), 0, vertices);
1925             }
1926             else
1927             {
1928                 hr = E_OUTOFMEMORY;
1929             }
1930         }
1931         if (FAILED(hr))
1932             IDirect3DRMMesh_Release(*mesh);
1933     }
1934
1935     return hr;
1936 }
1937
1938 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetFace(IDirect3DRMMeshBuilder3* iface,
1939                                                           DWORD index, LPDIRECT3DRMFACE2* Face)
1940 {
1941     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1942
1943     FIXME("(%p)->(%u,%p): stub\n", This, index, Face);
1944
1945     return E_NOTIMPL;
1946 }
1947
1948 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetVertex(IDirect3DRMMeshBuilder3* iface,
1949                                                             DWORD index, LPD3DVECTOR vector)
1950 {
1951     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1952
1953     FIXME("(%p)->(%u,%p): stub\n", This, index, vector);
1954
1955     return E_NOTIMPL;
1956 }
1957
1958 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetNormal(IDirect3DRMMeshBuilder3* iface,
1959                                                             DWORD index, LPD3DVECTOR vector)
1960 {
1961     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1962
1963     FIXME("(%p)->(%u,%p): stub\n", This, index, vector);
1964
1965     return E_NOTIMPL;
1966 }
1967
1968 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteVertices(IDirect3DRMMeshBuilder3* iface,
1969                                                                  DWORD IndexFirst, DWORD count)
1970 {
1971     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1972
1973     FIXME("(%p)->(%u,%u): stub\n", This, IndexFirst, count);
1974
1975     return E_NOTIMPL;
1976 }
1977
1978 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteNormals(IDirect3DRMMeshBuilder3* iface,
1979                                                                 DWORD IndexFirst, DWORD count)
1980 {
1981     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1982
1983     FIXME("(%p)->(%u,%u): stub\n", This, IndexFirst, count);
1984
1985     return E_NOTIMPL;
1986 }
1987
1988 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteFace(IDirect3DRMMeshBuilder3* iface,
1989                                                              LPDIRECT3DRMFACE2 Face)
1990 {
1991     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1992
1993     FIXME("(%p)->(%p): stub\n", This, Face);
1994
1995     return E_NOTIMPL;
1996 }
1997
1998 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Empty(IDirect3DRMMeshBuilder3* iface, DWORD flags)
1999 {
2000     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2001
2002     FIXME("(%p)->(%u): stub\n", This, flags);
2003
2004     return E_NOTIMPL;
2005 }
2006
2007 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Optimize(IDirect3DRMMeshBuilder3* iface,
2008                                                            DWORD flags)
2009 {
2010     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2011
2012     FIXME("(%p)->(%u): stub\n", This, flags);
2013
2014     return E_NOTIMPL;
2015 }
2016
2017 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFacesIndexed(IDirect3DRMMeshBuilder3* iface,
2018                                                                   DWORD flags, DWORD* indices,
2019                                                                   DWORD* IndexFirst, DWORD* count)
2020 {
2021     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2022
2023     FIXME("(%p)->(%u,%p,%p,%p): stub\n", This, flags, indices, IndexFirst, count);
2024
2025     return E_NOTIMPL;
2026 }
2027
2028 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateSubMesh(IDirect3DRMMeshBuilder3* iface,
2029                                                                 LPUNKNOWN *unkwn)
2030 {
2031     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2032
2033     FIXME("(%p)->(%p): stub\n", This, unkwn);
2034
2035     return E_NOTIMPL;
2036 }
2037
2038 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetParentMesh(IDirect3DRMMeshBuilder3* iface,
2039                                                                 DWORD flags, LPUNKNOWN *unkwn)
2040 {
2041     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2042
2043     FIXME("(%p)->(%u,%p): stub\n", This, flags, unkwn);
2044
2045     return E_NOTIMPL;
2046 }
2047
2048 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetSubMeshes(IDirect3DRMMeshBuilder3* iface,
2049                                                                LPDWORD count, LPUNKNOWN *unkwn)
2050 {
2051     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2052
2053     FIXME("(%p)->(%p,%p): stub\n", This, count, unkwn);
2054
2055     return E_NOTIMPL;
2056 }
2057
2058 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteSubMesh(IDirect3DRMMeshBuilder3* iface,
2059                                                                 LPUNKNOWN unkwn)
2060 {
2061     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2062
2063     FIXME("(%p)->(%p): stub\n", This, unkwn);
2064
2065     return E_NOTIMPL;
2066 }
2067
2068 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Enable(IDirect3DRMMeshBuilder3* iface,
2069                                                          DWORD index)
2070 {
2071     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2072
2073     FIXME("(%p)->(%u): stub\n", This, index);
2074
2075     return E_NOTIMPL;
2076 }
2077
2078 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetEnable(IDirect3DRMMeshBuilder3* iface,
2079                                                             DWORD* indices)
2080 {
2081     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2082
2083     FIXME("(%p)->(%p): stub\n", This, indices);
2084
2085     return E_NOTIMPL;
2086 }
2087
2088 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddTriangles(IDirect3DRMMeshBuilder3* iface,
2089                                                                DWORD flags, DWORD format,
2090                                                                DWORD VertexCount, LPVOID data)
2091 {
2092     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2093
2094     FIXME("(%p)->(%u,%u,%u,%p): stub\n", This, flags, format, VertexCount, data);
2095
2096     return E_NOTIMPL;
2097 }
2098
2099 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertices(IDirect3DRMMeshBuilder3* iface,
2100                                                               DWORD IndexFirst, DWORD count,
2101                                                               LPD3DVECTOR vector)
2102 {
2103     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2104
2105     FIXME("(%p)->(%u,%u,%p): stub\n", This, IndexFirst, count, vector);
2106
2107     return E_NOTIMPL;
2108 }
2109
2110 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetVertices(IDirect3DRMMeshBuilder3* iface,
2111                                                               DWORD IndexFirst, LPDWORD vcount,
2112                                                               LPD3DVECTOR vertices)
2113 {
2114     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2115     DWORD count = This->nb_vertices - IndexFirst;
2116
2117     TRACE("(%p)->(%u,%p,%p)\n", This, IndexFirst, vcount, vertices);
2118
2119     if (vcount)
2120         *vcount = count;
2121     if (vertices && This->nb_vertices)
2122         memcpy(vertices, This->pVertices + IndexFirst, count * sizeof(D3DVECTOR));
2123
2124     return D3DRM_OK;
2125 }
2126
2127 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetNormals(IDirect3DRMMeshBuilder3* iface,
2128                                                              DWORD IndexFirst, DWORD count,
2129                                                              LPD3DVECTOR vector)
2130 {
2131     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2132
2133     FIXME("(%p)->(%u,%u,%p): stub\n", This, IndexFirst, count, vector);
2134
2135     return E_NOTIMPL;
2136 }
2137
2138 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetNormals(IDirect3DRMMeshBuilder3* iface,
2139                                                              DWORD IndexFirst, LPDWORD count,
2140                                                              LPD3DVECTOR vector)
2141 {
2142     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2143
2144     FIXME("(%p)->(%u,%p,%p): stub\n", This, IndexFirst, count, vector);
2145
2146     return E_NOTIMPL;
2147 }
2148
2149 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetNormalCount(IDirect3DRMMeshBuilder3* iface)
2150 {
2151     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2152
2153     FIXME("(%p)->(): stub\n", This);
2154
2155     return E_NOTIMPL;
2156 }
2157
2158 static const struct IDirect3DRMMeshBuilder3Vtbl Direct3DRMMeshBuilder3_Vtbl =
2159 {
2160     /*** IUnknown methods ***/
2161     IDirect3DRMMeshBuilder3Impl_QueryInterface,
2162     IDirect3DRMMeshBuilder3Impl_AddRef,
2163     IDirect3DRMMeshBuilder3Impl_Release,
2164     /*** IDirect3DRMObject methods ***/
2165     IDirect3DRMMeshBuilder3Impl_Clone,
2166     IDirect3DRMMeshBuilder3Impl_AddDestroyCallback,
2167     IDirect3DRMMeshBuilder3Impl_DeleteDestroyCallback,
2168     IDirect3DRMMeshBuilder3Impl_SetAppData,
2169     IDirect3DRMMeshBuilder3Impl_GetAppData,
2170     IDirect3DRMMeshBuilder3Impl_SetName,
2171     IDirect3DRMMeshBuilder3Impl_GetName,
2172     IDirect3DRMMeshBuilder3Impl_GetClassName,
2173     /*** IDirect3DRMMeshBuilder3 methods ***/
2174     IDirect3DRMMeshBuilder3Impl_Load,
2175     IDirect3DRMMeshBuilder3Impl_Save,
2176     IDirect3DRMMeshBuilder3Impl_Scale,
2177     IDirect3DRMMeshBuilder3Impl_Translate,
2178     IDirect3DRMMeshBuilder3Impl_SetColorSource,
2179     IDirect3DRMMeshBuilder3Impl_GetBox,
2180     IDirect3DRMMeshBuilder3Impl_GenerateNormals,
2181     IDirect3DRMMeshBuilder3Impl_GetColorSource,
2182     IDirect3DRMMeshBuilder3Impl_AddMesh,
2183     IDirect3DRMMeshBuilder3Impl_AddMeshBuilder,
2184     IDirect3DRMMeshBuilder3Impl_AddFrame,
2185     IDirect3DRMMeshBuilder3Impl_AddFace,
2186     IDirect3DRMMeshBuilder3Impl_AddFaces,
2187     IDirect3DRMMeshBuilder3Impl_ReserveSpace,
2188     IDirect3DRMMeshBuilder3Impl_SetColorRGB,
2189     IDirect3DRMMeshBuilder3Impl_SetColor,
2190     IDirect3DRMMeshBuilder3Impl_SetTexture,
2191     IDirect3DRMMeshBuilder3Impl_SetMaterial,
2192     IDirect3DRMMeshBuilder3Impl_SetTextureTopology,
2193     IDirect3DRMMeshBuilder3Impl_SetQuality,
2194     IDirect3DRMMeshBuilder3Impl_SetPerspective,
2195     IDirect3DRMMeshBuilder3Impl_SetVertex,
2196     IDirect3DRMMeshBuilder3Impl_SetNormal,
2197     IDirect3DRMMeshBuilder3Impl_SetTextureCoordinates,
2198     IDirect3DRMMeshBuilder3Impl_SetVertexColor,
2199     IDirect3DRMMeshBuilder3Impl_SetVertexColorRGB,
2200     IDirect3DRMMeshBuilder3Impl_GetFaces,
2201     IDirect3DRMMeshBuilder3Impl_GetGeometry,
2202     IDirect3DRMMeshBuilder3Impl_GetTextureCoordinates,
2203     IDirect3DRMMeshBuilder3Impl_AddVertex,
2204     IDirect3DRMMeshBuilder3Impl_AddNormal,
2205     IDirect3DRMMeshBuilder3Impl_CreateFace,
2206     IDirect3DRMMeshBuilder3Impl_GetQuality,
2207     IDirect3DRMMeshBuilder3Impl_GetPerspective,
2208     IDirect3DRMMeshBuilder3Impl_GetFaceCount,
2209     IDirect3DRMMeshBuilder3Impl_GetVertexCount,
2210     IDirect3DRMMeshBuilder3Impl_GetVertexColor,
2211     IDirect3DRMMeshBuilder3Impl_CreateMesh,
2212     IDirect3DRMMeshBuilder3Impl_GetFace,
2213     IDirect3DRMMeshBuilder3Impl_GetVertex,
2214     IDirect3DRMMeshBuilder3Impl_GetNormal,
2215     IDirect3DRMMeshBuilder3Impl_DeleteVertices,
2216     IDirect3DRMMeshBuilder3Impl_DeleteNormals,
2217     IDirect3DRMMeshBuilder3Impl_DeleteFace,
2218     IDirect3DRMMeshBuilder3Impl_Empty,
2219     IDirect3DRMMeshBuilder3Impl_Optimize,
2220     IDirect3DRMMeshBuilder3Impl_AddFacesIndexed,
2221     IDirect3DRMMeshBuilder3Impl_CreateSubMesh,
2222     IDirect3DRMMeshBuilder3Impl_GetParentMesh,
2223     IDirect3DRMMeshBuilder3Impl_GetSubMeshes,
2224     IDirect3DRMMeshBuilder3Impl_DeleteSubMesh,
2225     IDirect3DRMMeshBuilder3Impl_Enable,
2226     IDirect3DRMMeshBuilder3Impl_GetEnable,
2227     IDirect3DRMMeshBuilder3Impl_AddTriangles,
2228     IDirect3DRMMeshBuilder3Impl_SetVertices,
2229     IDirect3DRMMeshBuilder3Impl_GetVertices,
2230     IDirect3DRMMeshBuilder3Impl_SetNormals,
2231     IDirect3DRMMeshBuilder3Impl_GetNormals,
2232     IDirect3DRMMeshBuilder3Impl_GetNormalCount
2233 };
2234
2235 HRESULT Direct3DRMMeshBuilder_create(REFIID riid, IUnknown** ppObj)
2236 {
2237     IDirect3DRMMeshBuilderImpl* object;
2238
2239     TRACE("(%p)\n", ppObj);
2240
2241     object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirect3DRMMeshBuilderImpl));
2242     if (!object)
2243     {
2244         ERR("Out of memory\n");
2245         return E_OUTOFMEMORY;
2246     }
2247
2248     object->IDirect3DRMMeshBuilder2_iface.lpVtbl = &Direct3DRMMeshBuilder2_Vtbl;
2249     object->IDirect3DRMMeshBuilder3_iface.lpVtbl = &Direct3DRMMeshBuilder3_Vtbl;
2250     object->ref = 1;
2251
2252     if (IsEqualGUID(riid, &IID_IDirect3DRMMeshBuilder3))
2253         *ppObj = (IUnknown*)&object->IDirect3DRMMeshBuilder3_iface;
2254     else
2255         *ppObj = (IUnknown*)&object->IDirect3DRMMeshBuilder2_iface;
2256
2257     return S_OK;
2258 }
2259
2260 /*** IUnknown methods ***/
2261 static HRESULT WINAPI IDirect3DRMMeshImpl_QueryInterface(IDirect3DRMMesh* iface,
2262                                                          REFIID riid, void** ppvObject)
2263 {
2264     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2265
2266     TRACE("(%p)->(%s,%p)\n", This, debugstr_guid(riid), ppvObject);
2267
2268     *ppvObject = NULL;
2269
2270     if (IsEqualGUID(riid, &IID_IUnknown) ||
2271         IsEqualGUID(riid, &IID_IDirect3DRMMesh))
2272     {
2273         *ppvObject = &This->IDirect3DRMMesh_iface;
2274     }
2275     else
2276     {
2277         FIXME("interface %s not implemented\n", debugstr_guid(riid));
2278         return E_NOINTERFACE;
2279     }
2280
2281     IDirect3DRMMesh_AddRef(iface);
2282     return S_OK;
2283 }
2284
2285 static ULONG WINAPI IDirect3DRMMeshImpl_AddRef(IDirect3DRMMesh* iface)
2286 {
2287     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2288     ULONG ref = InterlockedIncrement(&This->ref);
2289
2290     TRACE("(%p)->(): new ref = %d\n", This, ref);
2291
2292     return ref;
2293 }
2294
2295 static ULONG WINAPI IDirect3DRMMeshImpl_Release(IDirect3DRMMesh* iface)
2296 {
2297     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2298     ULONG ref = InterlockedDecrement(&This->ref);
2299
2300     TRACE("(%p)->(): new ref = %d\n", This, ref);
2301
2302     if (!ref)
2303     {
2304         int i;
2305
2306         for (i = 0; i < This->nb_groups; i++)
2307             HeapFree(GetProcessHeap(), 0, This->groups[i].vertices);
2308         HeapFree(GetProcessHeap(), 0, This->groups);
2309         HeapFree(GetProcessHeap(), 0, This);
2310     }
2311
2312     return ref;
2313 }
2314
2315 /*** IDirect3DRMObject methods ***/
2316 static HRESULT WINAPI IDirect3DRMMeshImpl_Clone(IDirect3DRMMesh* iface,
2317                                                 LPUNKNOWN pUnkOuter, REFIID riid,
2318                                                 LPVOID *ppvObj)
2319 {
2320     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2321
2322     FIXME("(%p)->(%p,%s,%p): stub\n", This, pUnkOuter, debugstr_guid(riid), ppvObj);
2323
2324     return E_NOTIMPL;
2325 }
2326
2327 static HRESULT WINAPI IDirect3DRMMeshImpl_AddDestroyCallback(IDirect3DRMMesh* iface,
2328                                                              D3DRMOBJECTCALLBACK cb,
2329                                                              LPVOID argument)
2330 {
2331     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2332
2333     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
2334
2335     return E_NOTIMPL;
2336 }
2337
2338 static HRESULT WINAPI IDirect3DRMMeshImpl_DeleteDestroyCallback(IDirect3DRMMesh* iface,
2339                                                                  D3DRMOBJECTCALLBACK cb,
2340                                                                  LPVOID argument)
2341 {
2342     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2343
2344     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
2345
2346     return E_NOTIMPL;
2347 }
2348
2349 static HRESULT WINAPI IDirect3DRMMeshImpl_SetAppData(IDirect3DRMMesh* iface,
2350                                                      DWORD data)
2351 {
2352     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2353
2354     FIXME("(%p)->(%u): stub\n", This, data);
2355
2356     return E_NOTIMPL;
2357 }
2358
2359 static DWORD WINAPI IDirect3DRMMeshImpl_GetAppData(IDirect3DRMMesh* iface)
2360 {
2361     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2362
2363     FIXME("(%p)->(): stub\n", This);
2364
2365     return 0;
2366 }
2367
2368 static HRESULT WINAPI IDirect3DRMMeshImpl_SetName(IDirect3DRMMesh* iface,
2369                                                   LPCSTR pName)
2370 {
2371     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2372
2373     FIXME("(%p)->(%s): stub\n", This, pName);
2374
2375     return E_NOTIMPL;
2376 }
2377
2378 static HRESULT WINAPI IDirect3DRMMeshImpl_GetName(IDirect3DRMMesh* iface,
2379                                                   LPDWORD lpdwSize, LPSTR lpName)
2380 {
2381     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2382
2383     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
2384
2385     return E_NOTIMPL;
2386 }
2387
2388 static HRESULT WINAPI IDirect3DRMMeshImpl_GetClassName(IDirect3DRMMesh* iface,
2389                                                        LPDWORD lpdwSize, LPSTR lpName)
2390 {
2391     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2392
2393     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
2394
2395     return E_NOTIMPL;
2396 }
2397
2398 /*** IDirect3DRMMesh methods ***/
2399 static HRESULT WINAPI IDirect3DRMMeshImpl_Scale(IDirect3DRMMesh* iface,
2400                                                 D3DVALUE sx, D3DVALUE sy, D3DVALUE sz)
2401 {
2402     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2403
2404     FIXME("(%p)->(%f,%f,%f): stub\n", This, sx, sy,sz);
2405
2406     return E_NOTIMPL;
2407 }
2408
2409 static HRESULT WINAPI IDirect3DRMMeshImpl_Translate(IDirect3DRMMesh* iface,
2410                                                     D3DVALUE tx, D3DVALUE ty, D3DVALUE tz)
2411 {
2412     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2413
2414     FIXME("(%p)->(%f,%f,%f): stub\n", This, tx, ty,tz);
2415
2416     return E_NOTIMPL;
2417 }
2418
2419 static HRESULT WINAPI IDirect3DRMMeshImpl_GetBox(IDirect3DRMMesh* iface,
2420                                                  D3DRMBOX * box)
2421 {
2422     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2423
2424     FIXME("(%p)->(%p): stub\n", This, box);
2425
2426     return E_NOTIMPL;
2427 }
2428
2429 static HRESULT WINAPI IDirect3DRMMeshImpl_AddGroup(IDirect3DRMMesh* iface,
2430                                                    unsigned vertex_count, unsigned face_count, unsigned vertex_per_face,
2431                                                    unsigned *face_data, D3DRMGROUPINDEX *return_id)
2432 {
2433     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2434     mesh_group* group;
2435
2436     TRACE("(%p)->(%u,%u,%u,%p,%p)\n", This, vertex_count, face_count, vertex_per_face, face_data, return_id);
2437
2438     if (!face_data || !return_id)
2439         return E_POINTER;
2440
2441     if ((This->nb_groups + 1) > This->groups_capacity)
2442     {
2443         ULONG new_capacity;
2444         mesh_group* groups;
2445
2446         if (!This->groups_capacity)
2447         {
2448             new_capacity = 16;
2449             groups = HeapAlloc(GetProcessHeap(), 0, new_capacity * sizeof(mesh_group));
2450         }
2451         else
2452         {
2453             new_capacity = This->groups_capacity * 2;
2454             groups = HeapReAlloc(GetProcessHeap(), 0, This->groups, new_capacity * sizeof(mesh_group));
2455         }
2456
2457         if (!groups)
2458             return E_OUTOFMEMORY;
2459
2460         This->groups_capacity = new_capacity;
2461         This->groups = groups;
2462     }
2463
2464     group = This->groups + This->nb_groups;
2465
2466     group->vertices = HeapAlloc(GetProcessHeap(), 0, vertex_count * sizeof(D3DRMVERTEX));
2467     if (!group->vertices)
2468         return E_OUTOFMEMORY;
2469     group->nb_vertices = vertex_count;
2470     group->nb_faces = face_count;
2471     group->vertex_per_face = vertex_per_face;
2472
2473     if (vertex_per_face)
2474     {
2475         group->face_data_size = face_count * vertex_per_face;
2476     }
2477     else
2478     {
2479         int i;
2480         unsigned nb_indices;
2481         unsigned* face_data_ptr = face_data;
2482         group->face_data_size = 0;
2483
2484         for (i = 0; i < face_count; i++)
2485         {
2486             nb_indices = *face_data_ptr;
2487             group->face_data_size += nb_indices + 1;
2488             face_data_ptr += nb_indices;
2489         }
2490     }
2491
2492     group->face_data = HeapAlloc(GetProcessHeap(), 0, group->face_data_size * sizeof(unsigned));
2493     if (!group->face_data)
2494     {
2495         HeapFree(GetProcessHeap(), 0 , group->vertices);
2496         return E_OUTOFMEMORY;
2497     }
2498
2499     memcpy(group->face_data, face_data, group->face_data_size * sizeof(unsigned));
2500
2501     group->material = NULL;
2502     group->texture = NULL;
2503
2504     *return_id = This->nb_groups++;
2505
2506     return D3DRM_OK;
2507 }
2508
2509 static HRESULT WINAPI IDirect3DRMMeshImpl_SetVertices(IDirect3DRMMesh* iface,
2510                                                       D3DRMGROUPINDEX id, unsigned index, unsigned count,
2511                                                       D3DRMVERTEX *values)
2512 {
2513     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2514
2515     TRACE("(%p)->(%u,%u,%u,%p)\n", This, id, index, count, values);
2516
2517     if (id >= This->nb_groups)
2518         return D3DRMERR_BADVALUE;
2519
2520     if ((index + count - 1) >= This->groups[id].nb_vertices)
2521         return D3DRMERR_BADVALUE;
2522
2523     if (!values)
2524         return E_POINTER;
2525
2526     memcpy(This->groups[id].vertices + index, values, count * sizeof(D3DRMVERTEX));
2527
2528     return D3DRM_OK;
2529 }
2530
2531 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupColor(IDirect3DRMMesh* iface,
2532                                                         D3DRMGROUPINDEX id, D3DCOLOR value)
2533 {
2534     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2535
2536     FIXME("(%p)->(%u,%x): stub\n", This, id, value);
2537
2538     return E_NOTIMPL;
2539 }
2540
2541 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupColorRGB(IDirect3DRMMesh* iface,
2542                                                            D3DRMGROUPINDEX id, D3DVALUE red, D3DVALUE green, D3DVALUE blue)
2543 {
2544     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2545
2546     FIXME("(%p)->(%u,%f,%f,%f): stub\n", This, id, red, green, blue);
2547
2548     return E_NOTIMPL;
2549 }
2550
2551 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupMapping(IDirect3DRMMesh* iface,
2552                                                           D3DRMGROUPINDEX id, D3DRMMAPPING value)
2553 {
2554     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2555
2556     FIXME("(%p)->(%u,%u): stub\n", This, id, value);
2557
2558     return E_NOTIMPL;
2559 }
2560
2561 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupQuality(IDirect3DRMMesh* iface,
2562                                                           D3DRMGROUPINDEX id, D3DRMRENDERQUALITY value)
2563 {
2564     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2565
2566     FIXME("(%p)->(%u,%u): stub\n", This, id, value);
2567
2568     return E_NOTIMPL;
2569 }
2570
2571 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupMaterial(IDirect3DRMMesh* iface,
2572                                                            D3DRMGROUPINDEX id, LPDIRECT3DRMMATERIAL value)
2573 {
2574     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2575
2576     FIXME("(%p)->(%u,%p): stub\n", This, id, value);
2577
2578     return E_NOTIMPL;
2579 }
2580
2581 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupTexture(IDirect3DRMMesh* iface,
2582                                                           D3DRMGROUPINDEX id, LPDIRECT3DRMTEXTURE value)
2583 {
2584     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2585
2586     FIXME("(%p)->(%u,%p): stub\n", This, id, value);
2587
2588     return E_NOTIMPL;
2589 }
2590
2591 static DWORD WINAPI IDirect3DRMMeshImpl_GetGroupCount(IDirect3DRMMesh* iface)
2592 {
2593     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2594
2595     TRACE("(%p)->()\n", This);
2596
2597     return This->nb_groups;
2598 }
2599
2600 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroup(IDirect3DRMMesh* iface,
2601                                                    D3DRMGROUPINDEX id, unsigned *vertex_count, unsigned *face_count, unsigned *vertex_per_face,
2602                                                    DWORD *face_data_size, unsigned *face_data)
2603 {
2604     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2605
2606     TRACE("(%p)->(%u,%p,%p,%p,%p,%p)\n", This, id, vertex_count, face_count, vertex_per_face, face_data_size, face_data);
2607
2608     if (id >= This->nb_groups)
2609         return D3DRMERR_BADVALUE;
2610
2611     if (vertex_count)
2612         *vertex_count = This->groups[id].nb_vertices;
2613     if (face_count)
2614         *face_count = This->groups[id].nb_faces;
2615     if (vertex_per_face)
2616         *vertex_per_face = This->groups[id].vertex_per_face;
2617     if (face_data_size)
2618         *face_data_size = This->groups[id].face_data_size;
2619     if (face_data)
2620         memcpy(face_data, This->groups[id].face_data, This->groups[id].face_data_size);
2621
2622     return D3DRM_OK;
2623 }
2624
2625 static HRESULT WINAPI IDirect3DRMMeshImpl_GetVertices(IDirect3DRMMesh* iface,
2626                                                       D3DRMGROUPINDEX id, DWORD index, DWORD count, D3DRMVERTEX *return_ptr)
2627 {
2628     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2629
2630     TRACE("(%p)->(%u,%u,%u,%p)\n", This, id, index, count, return_ptr);
2631
2632     if (id >= This->nb_groups)
2633         return D3DRMERR_BADVALUE;
2634
2635     if ((index + count - 1) >= This->groups[id].nb_vertices)
2636         return D3DRMERR_BADVALUE;
2637
2638     if (!return_ptr)
2639         return E_POINTER;
2640
2641     memcpy(return_ptr, This->groups[id].vertices + index, count * sizeof(D3DRMVERTEX));
2642
2643     return D3DRM_OK;
2644 }
2645
2646 static D3DCOLOR WINAPI IDirect3DRMMeshImpl_GetGroupColor(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2647 {
2648     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2649
2650     FIXME("(%p)->(%u): stub\n", This, id);
2651
2652     return 0;
2653 }
2654
2655 static D3DRMMAPPING WINAPI IDirect3DRMMeshImpl_GetGroupMapping(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2656 {
2657     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2658
2659     FIXME("(%p)->(%u): stub\n", This, id);
2660
2661     return 0;
2662 }
2663 static D3DRMRENDERQUALITY WINAPI IDirect3DRMMeshImpl_GetGroupQuality(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2664 {
2665     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2666
2667     FIXME("(%p)->(%u): stub\n", This, id);
2668
2669     return 0;
2670 }
2671
2672 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroupMaterial(IDirect3DRMMesh* iface,
2673                                                            D3DRMGROUPINDEX id, LPDIRECT3DRMMATERIAL *material)
2674 {
2675     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2676
2677     TRACE("(%p)->(%u,%p)\n", This, id, material);
2678
2679     if (id >= This->nb_groups)
2680         return D3DRMERR_BADVALUE;
2681
2682     if (!material)
2683         return E_POINTER;
2684
2685     if (This->groups[id].material)
2686         IDirect3DRMTexture_QueryInterface(This->groups[id].material, &IID_IDirect3DRMMaterial, (void**)material);
2687     else
2688         *material = NULL;
2689
2690     return D3DRM_OK;
2691 }
2692
2693 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroupTexture(IDirect3DRMMesh* iface,
2694                                                           D3DRMGROUPINDEX id, LPDIRECT3DRMTEXTURE *texture)
2695 {
2696     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2697
2698     TRACE("(%p)->(%u,%p)\n", This, id, texture);
2699
2700     if (id >= This->nb_groups)
2701         return D3DRMERR_BADVALUE;
2702
2703     if (!texture)
2704         return E_POINTER;
2705
2706     if (This->groups[id].texture)
2707         IDirect3DRMTexture_QueryInterface(This->groups[id].texture, &IID_IDirect3DRMTexture, (void**)texture);
2708     else
2709         *texture = NULL;
2710
2711     return D3DRM_OK;
2712 }
2713
2714 static const struct IDirect3DRMMeshVtbl Direct3DRMMesh_Vtbl =
2715 {
2716     /*** IUnknown methods ***/
2717     IDirect3DRMMeshImpl_QueryInterface,
2718     IDirect3DRMMeshImpl_AddRef,
2719     IDirect3DRMMeshImpl_Release,
2720     /*** IDirect3DRMObject methods ***/
2721     IDirect3DRMMeshImpl_Clone,
2722     IDirect3DRMMeshImpl_AddDestroyCallback,
2723     IDirect3DRMMeshImpl_DeleteDestroyCallback,
2724     IDirect3DRMMeshImpl_SetAppData,
2725     IDirect3DRMMeshImpl_GetAppData,
2726     IDirect3DRMMeshImpl_SetName,
2727     IDirect3DRMMeshImpl_GetName,
2728     IDirect3DRMMeshImpl_GetClassName,
2729     /*** IDirect3DRMMesh methods ***/
2730     IDirect3DRMMeshImpl_Scale,
2731     IDirect3DRMMeshImpl_Translate,
2732     IDirect3DRMMeshImpl_GetBox,
2733     IDirect3DRMMeshImpl_AddGroup,
2734     IDirect3DRMMeshImpl_SetVertices,
2735     IDirect3DRMMeshImpl_SetGroupColor,
2736     IDirect3DRMMeshImpl_SetGroupColorRGB,
2737     IDirect3DRMMeshImpl_SetGroupMapping,
2738     IDirect3DRMMeshImpl_SetGroupQuality,
2739     IDirect3DRMMeshImpl_SetGroupMaterial,
2740     IDirect3DRMMeshImpl_SetGroupTexture,
2741     IDirect3DRMMeshImpl_GetGroupCount,
2742     IDirect3DRMMeshImpl_GetGroup,
2743     IDirect3DRMMeshImpl_GetVertices,
2744     IDirect3DRMMeshImpl_GetGroupColor,
2745     IDirect3DRMMeshImpl_GetGroupMapping,
2746     IDirect3DRMMeshImpl_GetGroupQuality,
2747     IDirect3DRMMeshImpl_GetGroupMaterial,
2748     IDirect3DRMMeshImpl_GetGroupTexture
2749 };
2750
2751 static HRESULT Direct3DRMMesh_create(IDirect3DRMMesh** obj)
2752 {
2753     IDirect3DRMMeshImpl* object;
2754
2755     TRACE("(%p)\n", obj);
2756
2757     object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirect3DRMMeshImpl));
2758     if (!object)
2759     {
2760         ERR("Out of memory\n");
2761         return E_OUTOFMEMORY;
2762     }
2763
2764     object->IDirect3DRMMesh_iface.lpVtbl = &Direct3DRMMesh_Vtbl;
2765     object->ref = 1;
2766
2767     *obj = &object->IDirect3DRMMesh_iface;
2768
2769     return S_OK;
2770 }