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