d3drm: Implement IDirect3DRMMeshImpl_GetGroupMaterial.
[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     hr = IDirectXFileData_GetData(pData, NULL, &size, (void**)&ptr);
1122     if (hr != DXFILE_OK)
1123         goto end;
1124
1125     This->nb_vertices = *(DWORD*)ptr;
1126     This->nb_faces = *(DWORD*)(ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR));
1127     faces_vertex_idx_size = size - sizeof(DWORD) - This->nb_vertices * sizeof(D3DVECTOR) - sizeof(DWORD);
1128     faces_vertex_idx_ptr = (DWORD*)(ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR) + sizeof(DWORD));
1129
1130     TRACE("Mesh: nb_vertices = %d, nb_faces = %d, faces_vertex_idx_size = %d\n", This->nb_vertices, This->nb_faces, faces_vertex_idx_size);
1131
1132     This->pVertices = HeapAlloc(GetProcessHeap(), 0, This->nb_vertices * sizeof(D3DVECTOR));
1133     memcpy(This->pVertices, ptr + sizeof(DWORD), This->nb_vertices * sizeof(D3DVECTOR));
1134
1135     faces_vertex_idx_ptr = faces_vertex_idx_data = HeapAlloc(GetProcessHeap(), 0, faces_vertex_idx_size);
1136     memcpy(faces_vertex_idx_data, ptr + sizeof(DWORD) + This->nb_vertices * sizeof(D3DVECTOR) + sizeof(DWORD), faces_vertex_idx_size);
1137
1138     /* Each vertex index will have its normal index counterpart so just allocate twice the size */
1139     This->pFaceData = HeapAlloc(GetProcessHeap(), 0, faces_vertex_idx_size*2);
1140     faces_data_ptr = (DWORD*)This->pFaceData;
1141
1142     while (1)
1143     {
1144         hr =  IDirectXFileData_GetNextObject(pData, &pObject);
1145         if (hr == DXFILEERR_NOMOREOBJECTS)
1146         {
1147             TRACE("No more object\n");
1148             break;
1149         }
1150         if (hr != DXFILE_OK)
1151            goto end;
1152
1153         hr = IDirectXFileObject_QueryInterface(pObject, &IID_IDirectXFileData, (void**)&pData2);
1154         IDirectXFileObject_Release(pObject);
1155         if (hr != DXFILE_OK)
1156             goto end;
1157
1158         hr = IDirectXFileData_GetType(pData2, &pGuid);
1159         if (hr != DXFILE_OK)
1160             goto end;
1161
1162         TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1163
1164         if (IsEqualGUID(pGuid, &TID_D3DRMMeshNormals))
1165         {
1166             DWORD nb_faces_normals;
1167             DWORD faces_normal_idx_size;
1168
1169             hr = IDirectXFileData_GetData(pData2, NULL, &size, (void**)&ptr);
1170             if (hr != DXFILE_OK)
1171                 goto end;
1172
1173             This->nb_normals = *(DWORD*)ptr;
1174             nb_faces_normals = *(DWORD*)(ptr + sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR));
1175
1176             TRACE("MeshNormals: nb_normals = %d, nb_faces_normals = %d\n", This->nb_normals, nb_faces_normals);
1177             if (nb_faces_normals != This->nb_faces)
1178                 WARN("nb_face_normals (%d) != nb_faces (%d)\n", nb_faces_normals, This->nb_normals);
1179
1180             This->pNormals = HeapAlloc(GetProcessHeap(), 0, This->nb_normals * sizeof(D3DVECTOR));
1181             memcpy(This->pNormals, ptr + sizeof(DWORD), This->nb_normals * sizeof(D3DVECTOR));
1182
1183             faces_normal_idx_size = size - (2*sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR));
1184             faces_normal_idx_ptr = faces_normal_idx_data = HeapAlloc(GetProcessHeap(), 0, faces_normal_idx_size);
1185             memcpy(faces_normal_idx_data, ptr + sizeof(DWORD) + This->nb_normals * sizeof(D3DVECTOR) + sizeof(DWORD), faces_normal_idx_size);
1186         }
1187         else if (IsEqualGUID(pGuid, &TID_D3DRMMeshTextureCoords))
1188         {
1189             hr = IDirectXFileData_GetData(pData2, NULL, &size, (void**)&ptr);
1190             if (hr != DXFILE_OK)
1191                 goto end;
1192
1193             This->nb_coords2d = *(DWORD*)ptr;
1194
1195             TRACE("MeshTextureCoords: nb_coords2d = %d\n", This->nb_coords2d);
1196
1197             This->pCoords2d = HeapAlloc(GetProcessHeap(), 0, This->nb_coords2d * sizeof(Coords2d));
1198             memcpy(This->pCoords2d, ptr + sizeof(DWORD), This->nb_coords2d * sizeof(Coords2d));
1199
1200         }
1201         else if (IsEqualGUID(pGuid, &TID_D3DRMMeshMaterialList))
1202         {
1203             FIXME("MeshMaterialList not supported yet, ignoring...\n");
1204         }
1205         else
1206         {
1207             FIXME("Unknown GUID %s, ignoring...\n", debugstr_guid(pGuid));
1208         }
1209
1210         IDirectXFileData_Release(pData2);
1211         pData2 = NULL;
1212     }
1213
1214     for (i = 0; i < This->nb_faces; i++)
1215     {
1216         DWORD j;
1217         DWORD nb_face_indexes;
1218
1219         if (faces_vertex_idx_size < sizeof(DWORD))
1220             WARN("Not enough data to read number of indices of face %d\n", i);
1221
1222         nb_face_indexes  = *(faces_data_ptr + faces_data_size++) = *(faces_vertex_idx_ptr++);
1223         faces_vertex_idx_size--;
1224         if (faces_normal_idx_data && (*(faces_normal_idx_ptr++) != nb_face_indexes))
1225             WARN("Faces indices number mismatch\n");
1226
1227         if (faces_vertex_idx_size < (nb_face_indexes * sizeof(DWORD)))
1228             WARN("Not enough data to read all indices of face %d\n", i);
1229
1230         for (j = 0; j < nb_face_indexes; j++)
1231         {
1232             /* Copy vertex index */
1233             *(faces_data_ptr + faces_data_size++) = *(faces_vertex_idx_ptr++);
1234             /* Copy normal index */
1235             if (faces_normal_idx_data)
1236             {
1237                 /* Read from x file */
1238                 *(faces_data_ptr + faces_data_size++) = *(faces_normal_idx_ptr++);
1239             }
1240             else
1241             {
1242                 FIXME("No normal available, generate a fake normal index\n");
1243                 /* Must be generated, put 0 for now */
1244                 *(faces_data_ptr + faces_data_size++) = 0;
1245             }
1246         }
1247         faces_vertex_idx_size -= nb_face_indexes;
1248     }
1249
1250     /* Last DWORD must be 0 */
1251     *(faces_data_ptr + faces_data_size++) = 0;
1252
1253     /* Set size (in number of DWORD) of all faces data */
1254     This->face_data_size = faces_data_size;
1255
1256     /* If there is no texture coordinates, generate default texture coordinates (0.0f, 0.0f) for each vertex */
1257     if (!This->pCoords2d)
1258     {
1259         This->nb_coords2d = This->nb_vertices;
1260         This->pCoords2d = HeapAlloc(GetProcessHeap(), 0, This->nb_coords2d * sizeof(Coords2d));
1261         for (i = 0; i < This->nb_coords2d; i++)
1262         {
1263             This->pCoords2d[i].u = 0.0f;
1264             This->pCoords2d[i].v = 0.0f;
1265         }
1266     }
1267
1268     TRACE("Mesh data loaded successfully\n");
1269
1270     ret = D3DRM_OK;
1271
1272 end:
1273
1274     HeapFree(GetProcessHeap(), 0, faces_normal_idx_data);
1275     HeapFree(GetProcessHeap(), 0, faces_vertex_idx_data);
1276
1277     return ret;
1278 }
1279
1280 /*** IDirect3DRMMeshBuilder3 methods ***/
1281 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Load(IDirect3DRMMeshBuilder3* iface,
1282                                                        LPVOID filename, LPVOID name,
1283                                                        D3DRMLOADOPTIONS loadflags,
1284                                                        D3DRMLOADTEXTURE3CALLBACK cb, LPVOID arg)
1285 {
1286     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1287     DXFILELOADOPTIONS load_options;
1288     LPDIRECTXFILE pDXFile = NULL;
1289     LPDIRECTXFILEENUMOBJECT pEnumObject = NULL;
1290     LPDIRECTXFILEDATA pData = NULL;
1291     LPDIRECTXFILEDATA pData2 = NULL;
1292     const GUID* pGuid;
1293     DWORD size;
1294     Header* pHeader;
1295     HRESULT hr;
1296     HRESULT ret = D3DRMERR_BADOBJECT;
1297
1298     TRACE("(%p)->(%p,%p,%x,%p,%p)\n", This, filename, name, loadflags, cb, arg);
1299
1300     /* First free allocated buffers of previous mesh data */
1301     HeapFree(GetProcessHeap(), 0, This->pVertices);
1302     This->pVertices = NULL;
1303     HeapFree(GetProcessHeap(), 0, This->pNormals);
1304     This->pNormals = NULL;
1305     HeapFree(GetProcessHeap(), 0, This->pFaceData);
1306     This->pFaceData = NULL;
1307     HeapFree(GetProcessHeap(), 0, This->pCoords2d);
1308     This->pCoords2d = NULL;
1309
1310     if (loadflags == D3DRMLOAD_FROMMEMORY)
1311     {
1312         load_options = DXFILELOAD_FROMMEMORY;
1313     }
1314     else if (loadflags == D3DRMLOAD_FROMFILE)
1315     {
1316         load_options = DXFILELOAD_FROMFILE;
1317         TRACE("Loading from file %s\n", debugstr_a(filename));
1318     }
1319     else
1320     {
1321         FIXME("Load options %d not supported yet\n", loadflags);
1322         return E_NOTIMPL;
1323     }
1324
1325     hr = DirectXFileCreate(&pDXFile);
1326     if (hr != DXFILE_OK)
1327         goto end;
1328
1329     hr = IDirectXFile_RegisterTemplates(pDXFile, templates, strlen(templates));
1330     if (hr != DXFILE_OK)
1331         goto end;
1332
1333     hr = IDirectXFile_CreateEnumObject(pDXFile, filename, load_options, &pEnumObject);
1334     if (hr != DXFILE_OK)
1335         goto end;
1336
1337     hr = IDirectXFileEnumObject_GetNextDataObject(pEnumObject, &pData);
1338     if (hr != DXFILE_OK)
1339         goto end;
1340
1341     hr = IDirectXFileData_GetType(pData, &pGuid);
1342     if (hr != DXFILE_OK)
1343         goto end;
1344
1345     TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1346
1347     if (!IsEqualGUID(pGuid, &TID_DXFILEHeader))
1348     {
1349         ret = D3DRMERR_BADFILE;
1350         goto end;
1351     }
1352
1353     hr = IDirectXFileData_GetData(pData, NULL, &size, (void**)&pHeader);
1354     if ((hr != DXFILE_OK) || (size != sizeof(Header)))
1355         goto end;
1356
1357     TRACE("Version is %d %d %d\n", pHeader->major, pHeader->minor, pHeader->flags);
1358
1359     /* Version must be 1.0.x */
1360     if ((pHeader->major != 1) || (pHeader->minor != 0))
1361     {
1362         ret = D3DRMERR_BADFILE;
1363         goto end;
1364     }
1365
1366     IDirectXFileData_Release(pData);
1367     pData = NULL;
1368
1369     hr = IDirectXFileEnumObject_GetNextDataObject(pEnumObject, &pData);
1370     if (hr != DXFILE_OK)
1371     {
1372         ret = D3DRMERR_NOTFOUND;
1373         goto end;
1374     }
1375
1376     hr = IDirectXFileData_GetType(pData, &pGuid);
1377     if (hr != DXFILE_OK)
1378         goto end;
1379
1380     TRACE("Found object type whose GUID = %s\n", debugstr_guid(pGuid));
1381
1382     if (!IsEqualGUID(pGuid, &TID_D3DRMMesh))
1383     {
1384         ret = D3DRMERR_NOTFOUND;
1385         goto end;
1386     }
1387
1388     hr = load_mesh_data(iface, pData);
1389     if (hr == S_OK)
1390         ret = D3DRM_OK;
1391
1392 end:
1393
1394     if (pData2)
1395         IDirectXFileData_Release(pData2);
1396     if (pData)
1397         IDirectXFileData_Release(pData);
1398     if (pEnumObject)
1399         IDirectXFileEnumObject_Release(pEnumObject);
1400     if (pDXFile)
1401         IDirectXFile_Release(pDXFile);
1402
1403     if (ret != D3DRM_OK)
1404     {
1405         /* Clean mesh data */
1406         This->nb_vertices = 0;
1407         This->nb_normals = 0;
1408         This->nb_faces = 0;
1409         This->face_data_size = 0;
1410         This->nb_coords2d = 0;
1411         HeapFree(GetProcessHeap(), 0, This->pVertices);
1412         This->pVertices = NULL;
1413         HeapFree(GetProcessHeap(), 0, This->pNormals);
1414         This->pNormals = NULL;
1415         HeapFree(GetProcessHeap(), 0, This->pFaceData);
1416         This->pFaceData = NULL;
1417         HeapFree(GetProcessHeap(), 0, This->pCoords2d);
1418         This->pCoords2d = NULL;
1419     }
1420
1421     return ret;
1422 }
1423
1424 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Save(IDirect3DRMMeshBuilder3* iface,
1425                                                        const char* filename, D3DRMXOFFORMAT format,
1426                                                        D3DRMSAVEOPTIONS save)
1427 {
1428     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1429
1430     FIXME("(%p)->(%s,%d,%d): stub\n", This, filename, format, save);
1431
1432     return E_NOTIMPL;
1433 }
1434
1435 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Scale(IDirect3DRMMeshBuilder3* iface,
1436                                                         D3DVALUE sx, D3DVALUE sy, D3DVALUE sz)
1437 {
1438     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1439     int i;
1440
1441     TRACE("(%p)->(%f,%f,%f)\n", This, sx, sy, sz);
1442
1443     for (i = 0; i < This->nb_vertices; i++)
1444     {
1445         This->pVertices[i].u1.x *= sx;
1446         This->pVertices[i].u2.y *= sy;
1447         This->pVertices[i].u3.z *= sz;
1448     }
1449
1450     /* Normals are not affected by Scale */
1451
1452     return D3DRM_OK;
1453 }
1454
1455 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Translate(IDirect3DRMMeshBuilder3* iface,
1456                                                             D3DVALUE tx, D3DVALUE ty, D3DVALUE tz)
1457 {
1458     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1459
1460     FIXME("(%p)->(%f,%f,%f): stub\n", This, tx, ty, tz);
1461
1462     return E_NOTIMPL;
1463 }
1464
1465 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColorSource(IDirect3DRMMeshBuilder3* iface,
1466                                                                  D3DRMCOLORSOURCE color)
1467 {
1468     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1469
1470     FIXME("(%p)->(%x): stub\n", This, color);
1471
1472     return E_NOTIMPL;
1473 }
1474
1475 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetBox(IDirect3DRMMeshBuilder3* iface,
1476                                                          D3DRMBOX* box)
1477 {
1478     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1479
1480     FIXME("(%p)->(%p): stub\n", This, box);
1481
1482     return E_NOTIMPL;
1483 }
1484
1485 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GenerateNormals(IDirect3DRMMeshBuilder3* iface,
1486                                                                   D3DVALUE crease, DWORD flags)
1487 {
1488     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1489
1490     FIXME("(%p)->(%f,%u): stub\n", This, crease, flags);
1491
1492     return E_NOTIMPL;
1493 }
1494
1495 static D3DRMCOLORSOURCE WINAPI IDirect3DRMMeshBuilder3Impl_GetColorSource(IDirect3DRMMeshBuilder3* iface)
1496 {
1497     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1498
1499     FIXME("(%p)->(): stub\n", This);
1500
1501     return E_NOTIMPL;
1502 }
1503
1504 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddMesh(IDirect3DRMMeshBuilder3* iface,
1505                                                           LPDIRECT3DRMMESH Mesh)
1506 {
1507     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1508
1509     FIXME("(%p)->(%p): stub\n", This, Mesh);
1510
1511     return E_NOTIMPL;
1512 }
1513
1514 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddMeshBuilder(IDirect3DRMMeshBuilder3* iface,
1515                                                                  LPDIRECT3DRMMESHBUILDER3 MeshBuilder,
1516                                                                  DWORD flags)
1517 {
1518     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1519
1520     FIXME("(%p)->(%p,%u): stub\n", This, MeshBuilder, flags);
1521
1522     return E_NOTIMPL;
1523 }
1524
1525 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFrame(IDirect3DRMMeshBuilder3* iface,
1526                                                            LPDIRECT3DRMFRAME3 Frame)
1527 {
1528     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1529
1530     FIXME("(%p)->(%p): stub\n", This, Frame);
1531
1532     return E_NOTIMPL;
1533 }
1534
1535 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFace(IDirect3DRMMeshBuilder3* iface,
1536                                                           LPDIRECT3DRMFACE2 Face)
1537 {
1538     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1539
1540     FIXME("(%p)->(%p): stub\n", This, Face);
1541
1542     return E_NOTIMPL;
1543 }
1544
1545 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFaces(IDirect3DRMMeshBuilder3* iface,
1546                                                            DWORD vcount, D3DVECTOR* vertices,
1547                                                            DWORD ncount, D3DVECTOR* normals,
1548                                                            DWORD* data,
1549                                                            LPDIRECT3DRMFACEARRAY* FaceArray)
1550 {
1551     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1552
1553     FIXME("(%p)->(%d,%p,%d,%p,%p,%p): stub\n", This, vcount, vertices, ncount,
1554           normals, data, FaceArray);
1555
1556     return E_NOTIMPL;
1557 }
1558
1559 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_ReserveSpace(IDirect3DRMMeshBuilder3* iface,
1560                                                                DWORD vertex_Count,
1561                                                                DWORD normal_count,
1562                                                                DWORD face_count)
1563 {
1564     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1565
1566     FIXME("(%p)->(%d,%d,%d): stub\n", This, vertex_Count, normal_count, face_count);
1567
1568     return E_NOTIMPL;
1569 }
1570
1571 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColorRGB(IDirect3DRMMeshBuilder3* iface,
1572                                                               D3DVALUE red, D3DVALUE green,
1573                                                               D3DVALUE blue)
1574 {
1575     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1576
1577     FIXME("(%p)->(%f,%f,%f): stub\n", This, red, green, blue);
1578
1579     return E_NOTIMPL;
1580 }
1581
1582 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetColor(IDirect3DRMMeshBuilder3* iface,
1583                                                            D3DCOLOR color)
1584 {
1585     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1586
1587     FIXME("(%p)->(%x): stub\n", This, color);
1588
1589     return E_NOTIMPL;
1590 }
1591
1592 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTexture(IDirect3DRMMeshBuilder3* iface,
1593                                                              LPDIRECT3DRMTEXTURE3 texture)
1594 {
1595     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1596
1597     FIXME("(%p)->(%p): stub\n", This, texture);
1598
1599     return E_NOTIMPL;
1600 }
1601
1602 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetMaterial(IDirect3DRMMeshBuilder3* iface,
1603                                                               LPDIRECT3DRMMATERIAL2 material)
1604 {
1605     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1606
1607     FIXME("(%p)->(%p): stub\n", This, material);
1608
1609     return E_NOTIMPL;
1610 }
1611
1612 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTextureTopology(IDirect3DRMMeshBuilder3* iface,
1613                                                                      BOOL wrap_u, BOOL wrap_v)
1614 {
1615     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1616
1617     FIXME("(%p)->(%d,%d): stub\n", This, wrap_u, wrap_v);
1618
1619     return E_NOTIMPL;
1620 }
1621
1622 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetQuality(IDirect3DRMMeshBuilder3* iface,
1623                                                              D3DRMRENDERQUALITY quality)
1624 {
1625     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1626
1627     FIXME("(%p)->(%d): stub\n", This, quality);
1628
1629     return E_NOTIMPL;
1630 }
1631
1632 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetPerspective(IDirect3DRMMeshBuilder3* iface,
1633                                                                  BOOL enable)
1634 {
1635     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1636
1637     FIXME("(%p)->(%d): stub\n", This, enable);
1638
1639     return E_NOTIMPL;
1640 }
1641
1642 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertex(IDirect3DRMMeshBuilder3* iface,
1643                                                             DWORD index,
1644                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
1645 {
1646     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1647
1648     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1649
1650     return E_NOTIMPL;
1651 }
1652
1653 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetNormal(IDirect3DRMMeshBuilder3* iface,
1654                                                             DWORD index,
1655                                                             D3DVALUE x, D3DVALUE y, D3DVALUE z)
1656 {
1657     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1658
1659     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1660
1661     return E_NOTIMPL;
1662 }
1663
1664 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetTextureCoordinates(IDirect3DRMMeshBuilder3* iface,
1665                                                                         DWORD index, D3DVALUE u,
1666                                                                         D3DVALUE v)
1667 {
1668     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1669
1670     TRACE("(%p)->(%u,%f,%f)\n", This, index, u, v);
1671
1672     if (index >= This->nb_coords2d)
1673         return D3DRMERR_BADVALUE;
1674
1675     This->pCoords2d[index].u = u;
1676     This->pCoords2d[index].v = v;
1677
1678     return D3DRM_OK;
1679 }
1680
1681 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertexColor(IDirect3DRMMeshBuilder3* iface,
1682                                                                  DWORD index, D3DCOLOR color)
1683 {
1684     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1685
1686     FIXME("(%p)->(%d,%x): stub\n", This, index, color);
1687
1688     return E_NOTIMPL;
1689 }
1690
1691 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertexColorRGB(IDirect3DRMMeshBuilder3* iface,
1692                                                                     DWORD index,
1693                                                                     D3DVALUE red, D3DVALUE green,
1694                                                                     D3DVALUE blue)
1695 {
1696     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1697
1698     FIXME("(%p)->(%d,%f,%f,%f): stub\n", This, index, red, green, blue);
1699
1700     return E_NOTIMPL;
1701 }
1702
1703 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetFaces(IDirect3DRMMeshBuilder3* iface,
1704                                                            LPDIRECT3DRMFACEARRAY* FaceArray)
1705 {
1706     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1707
1708     FIXME("(%p)->(%p): stub\n", This, FaceArray);
1709
1710     return E_NOTIMPL;
1711 }
1712
1713 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetGeometry(IDirect3DRMMeshBuilder3* iface,
1714                                                               DWORD* vcount, D3DVECTOR* vertices,
1715                                                               DWORD* ncount, D3DVECTOR* normals,
1716                                                               DWORD* face_data_size,
1717                                                               DWORD* face_data)
1718 {
1719     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1720
1721     FIXME("(%p)->(%p,%p,%p,%p,%p,%p): stub\n", This, vcount, vertices, ncount, normals,
1722           face_data_size, face_data);
1723
1724     return E_NOTIMPL;
1725 }
1726
1727 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetTextureCoordinates(IDirect3DRMMeshBuilder3* iface,
1728                                                                         DWORD index, D3DVALUE* u,
1729                                                                         D3DVALUE* v)
1730 {
1731     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1732
1733     TRACE("(%p)->(%d,%p,%p)\n", This, index, u, v);
1734
1735     if (index >= This->nb_coords2d)
1736         return D3DRMERR_BADVALUE;
1737
1738     *u = This->pCoords2d[index].u;
1739     *v = This->pCoords2d[index].v;
1740
1741     return D3DRM_OK;
1742 }
1743
1744
1745 static int WINAPI IDirect3DRMMeshBuilder3Impl_AddVertex(IDirect3DRMMeshBuilder3* iface,
1746                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
1747 {
1748     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1749
1750     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1751
1752     return 0;
1753 }
1754
1755 static int WINAPI IDirect3DRMMeshBuilder3Impl_AddNormal(IDirect3DRMMeshBuilder3* iface,
1756                                                         D3DVALUE x, D3DVALUE y, D3DVALUE z)
1757 {
1758     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1759
1760     FIXME("(%p)->(%f,%f,%f): stub\n", This, x, y, z);
1761
1762     return 0;
1763 }
1764
1765 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateFace(IDirect3DRMMeshBuilder3* iface,
1766                                                              LPDIRECT3DRMFACE2* Face)
1767 {
1768     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1769
1770     FIXME("(%p)->(%p): stub\n", This, Face);
1771
1772     return E_NOTIMPL;
1773 }
1774
1775 static D3DRMRENDERQUALITY WINAPI IDirect3DRMMeshBuilder3Impl_GetQuality(IDirect3DRMMeshBuilder3* iface)
1776 {
1777     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1778
1779     FIXME("(%p)->(): stub\n", This);
1780
1781     return 0;
1782 }
1783
1784 static BOOL WINAPI IDirect3DRMMeshBuilder3Impl_GetPerspective(IDirect3DRMMeshBuilder3* iface)
1785 {
1786     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1787
1788     FIXME("(%p)->(): stub\n", This);
1789
1790     return FALSE;
1791 }
1792
1793 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetFaceCount(IDirect3DRMMeshBuilder3* iface)
1794 {
1795     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1796
1797     TRACE("(%p)->()\n", This);
1798
1799     return This->nb_faces;
1800 }
1801
1802 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetVertexCount(IDirect3DRMMeshBuilder3* iface)
1803 {
1804     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1805
1806     TRACE("(%p)->()\n", This);
1807
1808     return This->nb_vertices;
1809 }
1810
1811 static D3DCOLOR WINAPI IDirect3DRMMeshBuilder3Impl_GetVertexColor(IDirect3DRMMeshBuilder3* iface,
1812                                                                   DWORD index)
1813 {
1814     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1815
1816     FIXME("(%p)->(%d): stub\n", This, index);
1817
1818     return 0;
1819 }
1820
1821 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateMesh(IDirect3DRMMeshBuilder3* iface,
1822                                                              LPDIRECT3DRMMESH* mesh)
1823 {
1824     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1825     HRESULT hr;
1826     D3DRMGROUPINDEX group;
1827
1828     TRACE("(%p)->(%p)\n", This, mesh);
1829
1830     if (!mesh)
1831         return E_POINTER;
1832
1833     hr = Direct3DRMMesh_create(mesh);
1834     if (FAILED(hr))
1835         return hr;
1836
1837     /* If there is mesh data, create a group and put data inside */
1838     if (This->nb_vertices)
1839     {
1840         unsigned* face_data;
1841         unsigned* out_ptr;
1842         DWORD* in_ptr = This->pFaceData;
1843         int i, j;
1844
1845         face_data = HeapAlloc(GetProcessHeap(), 0, This->face_data_size * sizeof(DWORD));
1846         if (!face_data)
1847         {
1848             IDirect3DRMMesh_Release(*mesh);
1849             return E_OUTOFMEMORY;
1850         }
1851         out_ptr = face_data;
1852
1853         /* Put only vertex indices */
1854         for (i = 0; i < This->nb_faces; i++)
1855         {
1856             DWORD nb_indices = *out_ptr++ = *in_ptr++;
1857             for (j = 0; j < nb_indices; j++)
1858             {
1859                 *out_ptr++ = *in_ptr++;
1860                 /* Skip normal index */
1861                 in_ptr++;
1862             }
1863         }
1864
1865         hr = IDirect3DRMMesh_AddGroup(*mesh, This->nb_vertices, This->nb_faces, 0, face_data, &group);
1866         HeapFree(GetProcessHeap(), 0, face_data);
1867         if (SUCCEEDED(hr))
1868         {
1869             D3DRMVERTEX* vertices;
1870
1871             vertices = HeapAlloc(GetProcessHeap(), 0, This->nb_vertices * sizeof(D3DRMVERTEX));
1872             if (vertices)
1873             {
1874                 for (i = 0; i < This->nb_vertices; i++)
1875                 {
1876                     vertices[i].position = This->pVertices[i];
1877                 }
1878                 hr = IDirect3DRMMesh_SetVertices(*mesh, 0, 0, This->nb_vertices, vertices);
1879                 HeapFree(GetProcessHeap(), 0, vertices);
1880             }
1881             else
1882             {
1883                 hr = E_OUTOFMEMORY;
1884             }
1885         }
1886         if (FAILED(hr))
1887             IDirect3DRMMesh_Release(*mesh);
1888     }
1889
1890     return hr;
1891 }
1892
1893 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetFace(IDirect3DRMMeshBuilder3* iface,
1894                                                           DWORD index, LPDIRECT3DRMFACE2* Face)
1895 {
1896     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1897
1898     FIXME("(%p)->(%u,%p): stub\n", This, index, Face);
1899
1900     return E_NOTIMPL;
1901 }
1902
1903 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetVertex(IDirect3DRMMeshBuilder3* iface,
1904                                                             DWORD index, LPD3DVECTOR vector)
1905 {
1906     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1907
1908     FIXME("(%p)->(%u,%p): stub\n", This, index, vector);
1909
1910     return E_NOTIMPL;
1911 }
1912
1913 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetNormal(IDirect3DRMMeshBuilder3* iface,
1914                                                             DWORD index, LPD3DVECTOR vector)
1915 {
1916     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1917
1918     FIXME("(%p)->(%u,%p): stub\n", This, index, vector);
1919
1920     return E_NOTIMPL;
1921 }
1922
1923 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteVertices(IDirect3DRMMeshBuilder3* iface,
1924                                                                  DWORD IndexFirst, DWORD count)
1925 {
1926     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1927
1928     FIXME("(%p)->(%u,%u): stub\n", This, IndexFirst, count);
1929
1930     return E_NOTIMPL;
1931 }
1932
1933 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteNormals(IDirect3DRMMeshBuilder3* iface,
1934                                                                 DWORD IndexFirst, DWORD count)
1935 {
1936     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1937
1938     FIXME("(%p)->(%u,%u): stub\n", This, IndexFirst, count);
1939
1940     return E_NOTIMPL;
1941 }
1942
1943 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteFace(IDirect3DRMMeshBuilder3* iface,
1944                                                              LPDIRECT3DRMFACE2 Face)
1945 {
1946     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1947
1948     FIXME("(%p)->(%p): stub\n", This, Face);
1949
1950     return E_NOTIMPL;
1951 }
1952
1953 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Empty(IDirect3DRMMeshBuilder3* iface, DWORD flags)
1954 {
1955     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1956
1957     FIXME("(%p)->(%u): stub\n", This, flags);
1958
1959     return E_NOTIMPL;
1960 }
1961
1962 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Optimize(IDirect3DRMMeshBuilder3* iface,
1963                                                            DWORD flags)
1964 {
1965     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1966
1967     FIXME("(%p)->(%u): stub\n", This, flags);
1968
1969     return E_NOTIMPL;
1970 }
1971
1972 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddFacesIndexed(IDirect3DRMMeshBuilder3* iface,
1973                                                                   DWORD flags, DWORD* indices,
1974                                                                   DWORD* IndexFirst, DWORD* count)
1975 {
1976     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1977
1978     FIXME("(%p)->(%u,%p,%p,%p): stub\n", This, flags, indices, IndexFirst, count);
1979
1980     return E_NOTIMPL;
1981 }
1982
1983 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_CreateSubMesh(IDirect3DRMMeshBuilder3* iface,
1984                                                                 LPUNKNOWN *unkwn)
1985 {
1986     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1987
1988     FIXME("(%p)->(%p): stub\n", This, unkwn);
1989
1990     return E_NOTIMPL;
1991 }
1992
1993 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetParentMesh(IDirect3DRMMeshBuilder3* iface,
1994                                                                 DWORD flags, LPUNKNOWN *unkwn)
1995 {
1996     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
1997
1998     FIXME("(%p)->(%u,%p): stub\n", This, flags, unkwn);
1999
2000     return E_NOTIMPL;
2001 }
2002
2003 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetSubMeshes(IDirect3DRMMeshBuilder3* iface,
2004                                                                LPDWORD count, LPUNKNOWN *unkwn)
2005 {
2006     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2007
2008     FIXME("(%p)->(%p,%p): stub\n", This, count, unkwn);
2009
2010     return E_NOTIMPL;
2011 }
2012
2013 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_DeleteSubMesh(IDirect3DRMMeshBuilder3* iface,
2014                                                                 LPUNKNOWN unkwn)
2015 {
2016     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2017
2018     FIXME("(%p)->(%p): stub\n", This, unkwn);
2019
2020     return E_NOTIMPL;
2021 }
2022
2023 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_Enable(IDirect3DRMMeshBuilder3* iface,
2024                                                          DWORD index)
2025 {
2026     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2027
2028     FIXME("(%p)->(%u): stub\n", This, index);
2029
2030     return E_NOTIMPL;
2031 }
2032
2033 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetEnable(IDirect3DRMMeshBuilder3* iface,
2034                                                             DWORD* indices)
2035 {
2036     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2037
2038     FIXME("(%p)->(%p): stub\n", This, indices);
2039
2040     return E_NOTIMPL;
2041 }
2042
2043 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_AddTriangles(IDirect3DRMMeshBuilder3* iface,
2044                                                                DWORD flags, DWORD format,
2045                                                                DWORD VertexCount, LPVOID data)
2046 {
2047     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2048
2049     FIXME("(%p)->(%u,%u,%u,%p): stub\n", This, flags, format, VertexCount, data);
2050
2051     return E_NOTIMPL;
2052 }
2053
2054 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetVertices(IDirect3DRMMeshBuilder3* iface,
2055                                                               DWORD IndexFirst, DWORD count,
2056                                                               LPD3DVECTOR vector)
2057 {
2058     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2059
2060     FIXME("(%p)->(%u,%u,%p): stub\n", This, IndexFirst, count, vector);
2061
2062     return E_NOTIMPL;
2063 }
2064
2065 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetVertices(IDirect3DRMMeshBuilder3* iface,
2066                                                               DWORD IndexFirst, LPDWORD vcount,
2067                                                               LPD3DVECTOR vertices)
2068 {
2069     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2070     DWORD count = This->nb_vertices - IndexFirst;
2071
2072     TRACE("(%p)->(%u,%p,%p)\n", This, IndexFirst, vcount, vertices);
2073
2074     if (vcount)
2075         *vcount = count;
2076     if (vertices && This->nb_vertices)
2077         memcpy(vertices, This->pVertices + IndexFirst, count * sizeof(D3DVECTOR));
2078
2079     return D3DRM_OK;
2080 }
2081
2082 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_SetNormals(IDirect3DRMMeshBuilder3* iface,
2083                                                              DWORD IndexFirst, DWORD count,
2084                                                              LPD3DVECTOR vector)
2085 {
2086     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2087
2088     FIXME("(%p)->(%u,%u,%p): stub\n", This, IndexFirst, count, vector);
2089
2090     return E_NOTIMPL;
2091 }
2092
2093 static HRESULT WINAPI IDirect3DRMMeshBuilder3Impl_GetNormals(IDirect3DRMMeshBuilder3* iface,
2094                                                              DWORD IndexFirst, LPDWORD count,
2095                                                              LPD3DVECTOR vector)
2096 {
2097     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2098
2099     FIXME("(%p)->(%u,%p,%p): stub\n", This, IndexFirst, count, vector);
2100
2101     return E_NOTIMPL;
2102 }
2103
2104 static int WINAPI IDirect3DRMMeshBuilder3Impl_GetNormalCount(IDirect3DRMMeshBuilder3* iface)
2105 {
2106     IDirect3DRMMeshBuilderImpl *This = impl_from_IDirect3DRMMeshBuilder3(iface);
2107
2108     FIXME("(%p)->(): stub\n", This);
2109
2110     return E_NOTIMPL;
2111 }
2112
2113 static const struct IDirect3DRMMeshBuilder3Vtbl Direct3DRMMeshBuilder3_Vtbl =
2114 {
2115     /*** IUnknown methods ***/
2116     IDirect3DRMMeshBuilder3Impl_QueryInterface,
2117     IDirect3DRMMeshBuilder3Impl_AddRef,
2118     IDirect3DRMMeshBuilder3Impl_Release,
2119     /*** IDirect3DRMObject methods ***/
2120     IDirect3DRMMeshBuilder3Impl_Clone,
2121     IDirect3DRMMeshBuilder3Impl_AddDestroyCallback,
2122     IDirect3DRMMeshBuilder3Impl_DeleteDestroyCallback,
2123     IDirect3DRMMeshBuilder3Impl_SetAppData,
2124     IDirect3DRMMeshBuilder3Impl_GetAppData,
2125     IDirect3DRMMeshBuilder3Impl_SetName,
2126     IDirect3DRMMeshBuilder3Impl_GetName,
2127     IDirect3DRMMeshBuilder3Impl_GetClassName,
2128     /*** IDirect3DRMMeshBuilder3 methods ***/
2129     IDirect3DRMMeshBuilder3Impl_Load,
2130     IDirect3DRMMeshBuilder3Impl_Save,
2131     IDirect3DRMMeshBuilder3Impl_Scale,
2132     IDirect3DRMMeshBuilder3Impl_Translate,
2133     IDirect3DRMMeshBuilder3Impl_SetColorSource,
2134     IDirect3DRMMeshBuilder3Impl_GetBox,
2135     IDirect3DRMMeshBuilder3Impl_GenerateNormals,
2136     IDirect3DRMMeshBuilder3Impl_GetColorSource,
2137     IDirect3DRMMeshBuilder3Impl_AddMesh,
2138     IDirect3DRMMeshBuilder3Impl_AddMeshBuilder,
2139     IDirect3DRMMeshBuilder3Impl_AddFrame,
2140     IDirect3DRMMeshBuilder3Impl_AddFace,
2141     IDirect3DRMMeshBuilder3Impl_AddFaces,
2142     IDirect3DRMMeshBuilder3Impl_ReserveSpace,
2143     IDirect3DRMMeshBuilder3Impl_SetColorRGB,
2144     IDirect3DRMMeshBuilder3Impl_SetColor,
2145     IDirect3DRMMeshBuilder3Impl_SetTexture,
2146     IDirect3DRMMeshBuilder3Impl_SetMaterial,
2147     IDirect3DRMMeshBuilder3Impl_SetTextureTopology,
2148     IDirect3DRMMeshBuilder3Impl_SetQuality,
2149     IDirect3DRMMeshBuilder3Impl_SetPerspective,
2150     IDirect3DRMMeshBuilder3Impl_SetVertex,
2151     IDirect3DRMMeshBuilder3Impl_SetNormal,
2152     IDirect3DRMMeshBuilder3Impl_SetTextureCoordinates,
2153     IDirect3DRMMeshBuilder3Impl_SetVertexColor,
2154     IDirect3DRMMeshBuilder3Impl_SetVertexColorRGB,
2155     IDirect3DRMMeshBuilder3Impl_GetFaces,
2156     IDirect3DRMMeshBuilder3Impl_GetGeometry,
2157     IDirect3DRMMeshBuilder3Impl_GetTextureCoordinates,
2158     IDirect3DRMMeshBuilder3Impl_AddVertex,
2159     IDirect3DRMMeshBuilder3Impl_AddNormal,
2160     IDirect3DRMMeshBuilder3Impl_CreateFace,
2161     IDirect3DRMMeshBuilder3Impl_GetQuality,
2162     IDirect3DRMMeshBuilder3Impl_GetPerspective,
2163     IDirect3DRMMeshBuilder3Impl_GetFaceCount,
2164     IDirect3DRMMeshBuilder3Impl_GetVertexCount,
2165     IDirect3DRMMeshBuilder3Impl_GetVertexColor,
2166     IDirect3DRMMeshBuilder3Impl_CreateMesh,
2167     IDirect3DRMMeshBuilder3Impl_GetFace,
2168     IDirect3DRMMeshBuilder3Impl_GetVertex,
2169     IDirect3DRMMeshBuilder3Impl_GetNormal,
2170     IDirect3DRMMeshBuilder3Impl_DeleteVertices,
2171     IDirect3DRMMeshBuilder3Impl_DeleteNormals,
2172     IDirect3DRMMeshBuilder3Impl_DeleteFace,
2173     IDirect3DRMMeshBuilder3Impl_Empty,
2174     IDirect3DRMMeshBuilder3Impl_Optimize,
2175     IDirect3DRMMeshBuilder3Impl_AddFacesIndexed,
2176     IDirect3DRMMeshBuilder3Impl_CreateSubMesh,
2177     IDirect3DRMMeshBuilder3Impl_GetParentMesh,
2178     IDirect3DRMMeshBuilder3Impl_GetSubMeshes,
2179     IDirect3DRMMeshBuilder3Impl_DeleteSubMesh,
2180     IDirect3DRMMeshBuilder3Impl_Enable,
2181     IDirect3DRMMeshBuilder3Impl_GetEnable,
2182     IDirect3DRMMeshBuilder3Impl_AddTriangles,
2183     IDirect3DRMMeshBuilder3Impl_SetVertices,
2184     IDirect3DRMMeshBuilder3Impl_GetVertices,
2185     IDirect3DRMMeshBuilder3Impl_SetNormals,
2186     IDirect3DRMMeshBuilder3Impl_GetNormals,
2187     IDirect3DRMMeshBuilder3Impl_GetNormalCount
2188 };
2189
2190 HRESULT Direct3DRMMeshBuilder_create(REFIID riid, IUnknown** ppObj)
2191 {
2192     IDirect3DRMMeshBuilderImpl* object;
2193
2194     TRACE("(%p)\n", ppObj);
2195
2196     object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirect3DRMMeshBuilderImpl));
2197     if (!object)
2198     {
2199         ERR("Out of memory\n");
2200         return E_OUTOFMEMORY;
2201     }
2202
2203     object->IDirect3DRMMeshBuilder2_iface.lpVtbl = &Direct3DRMMeshBuilder2_Vtbl;
2204     object->IDirect3DRMMeshBuilder3_iface.lpVtbl = &Direct3DRMMeshBuilder3_Vtbl;
2205     object->ref = 1;
2206
2207     if (IsEqualGUID(riid, &IID_IDirect3DRMMeshBuilder3))
2208         *ppObj = (IUnknown*)&object->IDirect3DRMMeshBuilder3_iface;
2209     else
2210         *ppObj = (IUnknown*)&object->IDirect3DRMMeshBuilder2_iface;
2211
2212     return S_OK;
2213 }
2214
2215 /*** IUnknown methods ***/
2216 static HRESULT WINAPI IDirect3DRMMeshImpl_QueryInterface(IDirect3DRMMesh* iface,
2217                                                          REFIID riid, void** ppvObject)
2218 {
2219     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2220
2221     TRACE("(%p)->(%s,%p)\n", This, debugstr_guid(riid), ppvObject);
2222
2223     *ppvObject = NULL;
2224
2225     if (IsEqualGUID(riid, &IID_IUnknown) ||
2226         IsEqualGUID(riid, &IID_IDirect3DRMMesh))
2227     {
2228         *ppvObject = &This->IDirect3DRMMesh_iface;
2229     }
2230     else
2231     {
2232         FIXME("interface %s not implemented\n", debugstr_guid(riid));
2233         return E_NOINTERFACE;
2234     }
2235
2236     IDirect3DRMMesh_AddRef(iface);
2237     return S_OK;
2238 }
2239
2240 static ULONG WINAPI IDirect3DRMMeshImpl_AddRef(IDirect3DRMMesh* iface)
2241 {
2242     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2243     ULONG ref = InterlockedIncrement(&This->ref);
2244
2245     TRACE("(%p)->(): new ref = %d\n", This, ref);
2246
2247     return ref;
2248 }
2249
2250 static ULONG WINAPI IDirect3DRMMeshImpl_Release(IDirect3DRMMesh* iface)
2251 {
2252     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2253     ULONG ref = InterlockedDecrement(&This->ref);
2254
2255     TRACE("(%p)->(): new ref = %d\n", This, ref);
2256
2257     if (!ref)
2258     {
2259         int i;
2260
2261         for (i = 0; i < This->nb_groups; i++)
2262             HeapFree(GetProcessHeap(), 0, This->groups[i].vertices);
2263         HeapFree(GetProcessHeap(), 0, This->groups);
2264         HeapFree(GetProcessHeap(), 0, This);
2265     }
2266
2267     return ref;
2268 }
2269
2270 /*** IDirect3DRMObject methods ***/
2271 static HRESULT WINAPI IDirect3DRMMeshImpl_Clone(IDirect3DRMMesh* iface,
2272                                                 LPUNKNOWN pUnkOuter, REFIID riid,
2273                                                 LPVOID *ppvObj)
2274 {
2275     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2276
2277     FIXME("(%p)->(%p,%s,%p): stub\n", This, pUnkOuter, debugstr_guid(riid), ppvObj);
2278
2279     return E_NOTIMPL;
2280 }
2281
2282 static HRESULT WINAPI IDirect3DRMMeshImpl_AddDestroyCallback(IDirect3DRMMesh* iface,
2283                                                              D3DRMOBJECTCALLBACK cb,
2284                                                              LPVOID argument)
2285 {
2286     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2287
2288     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
2289
2290     return E_NOTIMPL;
2291 }
2292
2293 static HRESULT WINAPI IDirect3DRMMeshImpl_DeleteDestroyCallback(IDirect3DRMMesh* iface,
2294                                                                  D3DRMOBJECTCALLBACK cb,
2295                                                                  LPVOID argument)
2296 {
2297     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2298
2299     FIXME("(%p)->(%p,%p): stub\n", This, cb, argument);
2300
2301     return E_NOTIMPL;
2302 }
2303
2304 static HRESULT WINAPI IDirect3DRMMeshImpl_SetAppData(IDirect3DRMMesh* iface,
2305                                                      DWORD data)
2306 {
2307     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2308
2309     FIXME("(%p)->(%u): stub\n", This, data);
2310
2311     return E_NOTIMPL;
2312 }
2313
2314 static DWORD WINAPI IDirect3DRMMeshImpl_GetAppData(IDirect3DRMMesh* iface)
2315 {
2316     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2317
2318     FIXME("(%p)->(): stub\n", This);
2319
2320     return 0;
2321 }
2322
2323 static HRESULT WINAPI IDirect3DRMMeshImpl_SetName(IDirect3DRMMesh* iface,
2324                                                   LPCSTR pName)
2325 {
2326     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2327
2328     FIXME("(%p)->(%s): stub\n", This, pName);
2329
2330     return E_NOTIMPL;
2331 }
2332
2333 static HRESULT WINAPI IDirect3DRMMeshImpl_GetName(IDirect3DRMMesh* iface,
2334                                                   LPDWORD lpdwSize, LPSTR lpName)
2335 {
2336     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2337
2338     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
2339
2340     return E_NOTIMPL;
2341 }
2342
2343 static HRESULT WINAPI IDirect3DRMMeshImpl_GetClassName(IDirect3DRMMesh* iface,
2344                                                        LPDWORD lpdwSize, LPSTR lpName)
2345 {
2346     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2347
2348     FIXME("(%p)->(%p,%p): stub\n", This, lpdwSize, lpName);
2349
2350     return E_NOTIMPL;
2351 }
2352
2353 /*** IDirect3DRMMesh methods ***/
2354 static HRESULT WINAPI IDirect3DRMMeshImpl_Scale(IDirect3DRMMesh* iface,
2355                                                 D3DVALUE sx, D3DVALUE sy, D3DVALUE sz)
2356 {
2357     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2358
2359     FIXME("(%p)->(%f,%f,%f): stub\n", This, sx, sy,sz);
2360
2361     return E_NOTIMPL;
2362 }
2363
2364 static HRESULT WINAPI IDirect3DRMMeshImpl_Translate(IDirect3DRMMesh* iface,
2365                                                     D3DVALUE tx, D3DVALUE ty, D3DVALUE tz)
2366 {
2367     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2368
2369     FIXME("(%p)->(%f,%f,%f): stub\n", This, tx, ty,tz);
2370
2371     return E_NOTIMPL;
2372 }
2373
2374 static HRESULT WINAPI IDirect3DRMMeshImpl_GetBox(IDirect3DRMMesh* iface,
2375                                                  D3DRMBOX * box)
2376 {
2377     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2378
2379     FIXME("(%p)->(%p): stub\n", This, box);
2380
2381     return E_NOTIMPL;
2382 }
2383
2384 static HRESULT WINAPI IDirect3DRMMeshImpl_AddGroup(IDirect3DRMMesh* iface,
2385                                                    unsigned vertex_count, unsigned face_count, unsigned vertex_per_face,
2386                                                    unsigned *face_data, D3DRMGROUPINDEX *return_id)
2387 {
2388     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2389     mesh_group* group;
2390
2391     TRACE("(%p)->(%u,%u,%u,%p,%p)\n", This, vertex_count, face_count, vertex_per_face, face_data, return_id);
2392
2393     if (!face_data || !return_id)
2394         return E_POINTER;
2395
2396     if ((This->nb_groups + 1) > This->groups_capacity)
2397     {
2398         ULONG new_capacity;
2399         mesh_group* groups;
2400
2401         if (!This->groups_capacity)
2402         {
2403             new_capacity = 16;
2404             groups = HeapAlloc(GetProcessHeap(), 0, new_capacity * sizeof(mesh_group));
2405         }
2406         else
2407         {
2408             new_capacity = This->groups_capacity * 2;
2409             groups = HeapReAlloc(GetProcessHeap(), 0, This->groups, new_capacity * sizeof(mesh_group));
2410         }
2411
2412         if (!groups)
2413             return E_OUTOFMEMORY;
2414
2415         This->groups_capacity = new_capacity;
2416         This->groups = groups;
2417     }
2418
2419     group = This->groups + This->nb_groups;
2420
2421     group->vertices = HeapAlloc(GetProcessHeap(), 0, vertex_count * sizeof(D3DRMVERTEX));
2422     if (!group->vertices)
2423         return E_OUTOFMEMORY;
2424     group->nb_vertices = vertex_count;
2425     group->nb_faces = face_count;
2426     group->vertex_per_face = vertex_per_face;
2427
2428     if (vertex_per_face)
2429     {
2430         group->face_data_size = face_count * vertex_per_face;
2431     }
2432     else
2433     {
2434         int i;
2435         unsigned nb_indices;
2436         unsigned* face_data_ptr = face_data;
2437         group->face_data_size = 0;
2438
2439         for (i = 0; i < face_count; i++)
2440         {
2441             nb_indices = *face_data_ptr;
2442             group->face_data_size += nb_indices + 1;
2443             face_data_ptr += nb_indices;
2444         }
2445     }
2446
2447     group->face_data = HeapAlloc(GetProcessHeap(), 0, group->face_data_size * sizeof(unsigned));
2448     if (!group->face_data)
2449     {
2450         HeapFree(GetProcessHeap(), 0 , group->vertices);
2451         return E_OUTOFMEMORY;
2452     }
2453
2454     memcpy(group->face_data, face_data, group->face_data_size * sizeof(unsigned));
2455
2456     group->material = NULL;
2457     group->texture = NULL;
2458
2459     *return_id = This->nb_groups++;
2460
2461     return D3DRM_OK;
2462 }
2463
2464 static HRESULT WINAPI IDirect3DRMMeshImpl_SetVertices(IDirect3DRMMesh* iface,
2465                                                       D3DRMGROUPINDEX id, unsigned index, unsigned count,
2466                                                       D3DRMVERTEX *values)
2467 {
2468     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2469
2470     TRACE("(%p)->(%u,%u,%u,%p)\n", This, id, index, count, values);
2471
2472     if (id >= This->nb_groups)
2473         return D3DRMERR_BADVALUE;
2474
2475     if ((index + count - 1) >= This->groups[id].nb_vertices)
2476         return D3DRMERR_BADVALUE;
2477
2478     if (!values)
2479         return E_POINTER;
2480
2481     memcpy(This->groups[id].vertices + index, values, count * sizeof(D3DRMVERTEX));
2482
2483     return D3DRM_OK;
2484 }
2485
2486 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupColor(IDirect3DRMMesh* iface,
2487                                                         D3DRMGROUPINDEX id, D3DCOLOR value)
2488 {
2489     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2490
2491     FIXME("(%p)->(%u,%x): stub\n", This, id, value);
2492
2493     return E_NOTIMPL;
2494 }
2495
2496 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupColorRGB(IDirect3DRMMesh* iface,
2497                                                            D3DRMGROUPINDEX id, D3DVALUE red, D3DVALUE green, D3DVALUE blue)
2498 {
2499     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2500
2501     FIXME("(%p)->(%u,%f,%f,%f): stub\n", This, id, red, green, blue);
2502
2503     return E_NOTIMPL;
2504 }
2505
2506 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupMapping(IDirect3DRMMesh* iface,
2507                                                           D3DRMGROUPINDEX id, D3DRMMAPPING value)
2508 {
2509     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2510
2511     FIXME("(%p)->(%u,%u): stub\n", This, id, value);
2512
2513     return E_NOTIMPL;
2514 }
2515
2516 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupQuality(IDirect3DRMMesh* iface,
2517                                                           D3DRMGROUPINDEX id, D3DRMRENDERQUALITY value)
2518 {
2519     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2520
2521     FIXME("(%p)->(%u,%u): stub\n", This, id, value);
2522
2523     return E_NOTIMPL;
2524 }
2525
2526 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupMaterial(IDirect3DRMMesh* iface,
2527                                                            D3DRMGROUPINDEX id, LPDIRECT3DRMMATERIAL value)
2528 {
2529     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2530
2531     FIXME("(%p)->(%u,%p): stub\n", This, id, value);
2532
2533     return E_NOTIMPL;
2534 }
2535
2536 static HRESULT WINAPI IDirect3DRMMeshImpl_SetGroupTexture(IDirect3DRMMesh* iface,
2537                                                           D3DRMGROUPINDEX id, LPDIRECT3DRMTEXTURE value)
2538 {
2539     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2540
2541     FIXME("(%p)->(%u,%p): stub\n", This, id, value);
2542
2543     return E_NOTIMPL;
2544 }
2545
2546 static DWORD WINAPI IDirect3DRMMeshImpl_GetGroupCount(IDirect3DRMMesh* iface)
2547 {
2548     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2549
2550     TRACE("(%p)->()\n", This);
2551
2552     return This->nb_groups;
2553 }
2554
2555 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroup(IDirect3DRMMesh* iface,
2556                                                    D3DRMGROUPINDEX id, unsigned *vertex_count, unsigned *face_count, unsigned *vertex_per_face,
2557                                                    DWORD *face_data_size, unsigned *face_data)
2558 {
2559     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2560
2561     TRACE("(%p)->(%u,%p,%p,%p,%p,%p)\n", This, id, vertex_count, face_count, vertex_per_face, face_data_size, face_data);
2562
2563     if (id >= This->nb_groups)
2564         return D3DRMERR_BADVALUE;
2565
2566     if (vertex_count)
2567         *vertex_count = This->groups[id].nb_vertices;
2568     if (face_count)
2569         *face_count = This->groups[id].nb_faces;
2570     if (vertex_per_face)
2571         *vertex_per_face = This->groups[id].vertex_per_face;
2572     if (face_data_size)
2573         *face_data_size = This->groups[id].face_data_size;
2574     if (face_data)
2575         memcpy(face_data, This->groups[id].face_data, This->groups[id].face_data_size);
2576
2577     return D3DRM_OK;
2578 }
2579
2580 static HRESULT WINAPI IDirect3DRMMeshImpl_GetVertices(IDirect3DRMMesh* iface,
2581                                                       D3DRMGROUPINDEX id, DWORD index, DWORD count, D3DRMVERTEX *return_ptr)
2582 {
2583     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2584
2585     TRACE("(%p)->(%u,%u,%u,%p)\n", This, id, index, count, return_ptr);
2586
2587     if (id >= This->nb_groups)
2588         return D3DRMERR_BADVALUE;
2589
2590     if ((index + count - 1) >= This->groups[id].nb_vertices)
2591         return D3DRMERR_BADVALUE;
2592
2593     if (!return_ptr)
2594         return E_POINTER;
2595
2596     memcpy(return_ptr, This->groups[id].vertices + index, count * sizeof(D3DRMVERTEX));
2597
2598     return D3DRM_OK;
2599 }
2600
2601 static D3DCOLOR WINAPI IDirect3DRMMeshImpl_GetGroupColor(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2602 {
2603     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2604
2605     FIXME("(%p)->(%u): stub\n", This, id);
2606
2607     return 0;
2608 }
2609
2610 static D3DRMMAPPING WINAPI IDirect3DRMMeshImpl_GetGroupMapping(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2611 {
2612     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2613
2614     FIXME("(%p)->(%u): stub\n", This, id);
2615
2616     return 0;
2617 }
2618 static D3DRMRENDERQUALITY WINAPI IDirect3DRMMeshImpl_GetGroupQuality(IDirect3DRMMesh* iface, D3DRMGROUPINDEX id)
2619 {
2620     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2621
2622     FIXME("(%p)->(%u): stub\n", This, id);
2623
2624     return 0;
2625 }
2626
2627 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroupMaterial(IDirect3DRMMesh* iface,
2628                                                            D3DRMGROUPINDEX id, LPDIRECT3DRMMATERIAL *material)
2629 {
2630     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2631
2632     TRACE("(%p)->(%u,%p)\n", This, id, material);
2633
2634     if (id >= This->nb_groups)
2635         return D3DRMERR_BADVALUE;
2636
2637     if (!material)
2638         return E_POINTER;
2639
2640     if (This->groups[id].material)
2641         IDirect3DRMTexture_QueryInterface(This->groups[id].material, &IID_IDirect3DRMMaterial, (void**)material);
2642     else
2643         *material = NULL;
2644
2645     return D3DRM_OK;
2646 }
2647
2648 static HRESULT WINAPI IDirect3DRMMeshImpl_GetGroupTexture(IDirect3DRMMesh* iface,
2649                                                           D3DRMGROUPINDEX id, LPDIRECT3DRMTEXTURE *texture)
2650 {
2651     IDirect3DRMMeshImpl *This = impl_from_IDirect3DRMMesh(iface);
2652
2653     TRACE("(%p)->(%u,%p)\n", This, id, texture);
2654
2655     if (id >= This->nb_groups)
2656         return D3DRMERR_BADVALUE;
2657
2658     if (!texture)
2659         return E_POINTER;
2660
2661     if (This->groups[id].texture)
2662         IDirect3DRMTexture_QueryInterface(This->groups[id].texture, &IID_IDirect3DRMTexture, (void**)texture);
2663     else
2664         *texture = NULL;
2665
2666     return D3DRM_OK;
2667 }
2668
2669 static const struct IDirect3DRMMeshVtbl Direct3DRMMesh_Vtbl =
2670 {
2671     /*** IUnknown methods ***/
2672     IDirect3DRMMeshImpl_QueryInterface,
2673     IDirect3DRMMeshImpl_AddRef,
2674     IDirect3DRMMeshImpl_Release,
2675     /*** IDirect3DRMObject methods ***/
2676     IDirect3DRMMeshImpl_Clone,
2677     IDirect3DRMMeshImpl_AddDestroyCallback,
2678     IDirect3DRMMeshImpl_DeleteDestroyCallback,
2679     IDirect3DRMMeshImpl_SetAppData,
2680     IDirect3DRMMeshImpl_GetAppData,
2681     IDirect3DRMMeshImpl_SetName,
2682     IDirect3DRMMeshImpl_GetName,
2683     IDirect3DRMMeshImpl_GetClassName,
2684     /*** IDirect3DRMMesh methods ***/
2685     IDirect3DRMMeshImpl_Scale,
2686     IDirect3DRMMeshImpl_Translate,
2687     IDirect3DRMMeshImpl_GetBox,
2688     IDirect3DRMMeshImpl_AddGroup,
2689     IDirect3DRMMeshImpl_SetVertices,
2690     IDirect3DRMMeshImpl_SetGroupColor,
2691     IDirect3DRMMeshImpl_SetGroupColorRGB,
2692     IDirect3DRMMeshImpl_SetGroupMapping,
2693     IDirect3DRMMeshImpl_SetGroupQuality,
2694     IDirect3DRMMeshImpl_SetGroupMaterial,
2695     IDirect3DRMMeshImpl_SetGroupTexture,
2696     IDirect3DRMMeshImpl_GetGroupCount,
2697     IDirect3DRMMeshImpl_GetGroup,
2698     IDirect3DRMMeshImpl_GetVertices,
2699     IDirect3DRMMeshImpl_GetGroupColor,
2700     IDirect3DRMMeshImpl_GetGroupMapping,
2701     IDirect3DRMMeshImpl_GetGroupQuality,
2702     IDirect3DRMMeshImpl_GetGroupMaterial,
2703     IDirect3DRMMeshImpl_GetGroupTexture
2704 };
2705
2706 static HRESULT Direct3DRMMesh_create(IDirect3DRMMesh** obj)
2707 {
2708     IDirect3DRMMeshImpl* object;
2709
2710     TRACE("(%p)\n", obj);
2711
2712     object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirect3DRMMeshImpl));
2713     if (!object)
2714     {
2715         ERR("Out of memory\n");
2716         return E_OUTOFMEMORY;
2717     }
2718
2719     object->IDirect3DRMMesh_iface.lpVtbl = &Direct3DRMMesh_Vtbl;
2720     object->ref = 1;
2721
2722     *obj = &object->IDirect3DRMMesh_iface;
2723
2724     return S_OK;
2725 }