d3dxof: Add support for DXFILELOAD_FROMRESOURCE source.
[wine/hacks.git] / dlls / d3dxof / d3dxof.c
blob6a27cd5cc1bf19fff09df1b430ff414cd12c1535
1 /*
2 * Implementation of DirectX File Interfaces
4 * Copyright 2004, 2008 Christian Costa
6 * This file contains the (internal) driver registration functions,
7 * driver enumeration APIs and DirectDraw creation functions.
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24 #include "config.h"
25 #include "wine/debug.h"
27 #define COBJMACROS
29 #include "winbase.h"
30 #include "wingdi.h"
32 #include "d3dxof_private.h"
33 #include "dxfile.h"
35 #include <stdio.h>
37 WINE_DEFAULT_DEBUG_CHANNEL(d3dxof);
39 #define MAKEFOUR(a,b,c,d) ((DWORD)a + ((DWORD)b << 8) + ((DWORD)c << 16) + ((DWORD)d << 24))
40 #define XOFFILE_FORMAT_MAGIC MAKEFOUR('x','o','f',' ')
41 #define XOFFILE_FORMAT_VERSION_302 MAKEFOUR('0','3','0','2')
42 #define XOFFILE_FORMAT_VERSION_303 MAKEFOUR('0','3','0','3')
43 #define XOFFILE_FORMAT_BINARY MAKEFOUR('b','i','n',' ')
44 #define XOFFILE_FORMAT_TEXT MAKEFOUR('t','x','t',' ')
45 #define XOFFILE_FORMAT_COMPRESSED MAKEFOUR('c','m','p',' ')
46 #define XOFFILE_FORMAT_FLOAT_BITS_32 MAKEFOUR('0','0','3','2')
47 #define XOFFILE_FORMAT_FLOAT_BITS_64 MAKEFOUR('0','0','6','4')
49 static const struct IDirectXFileVtbl IDirectXFile_Vtbl;
50 static const struct IDirectXFileBinaryVtbl IDirectXFileBinary_Vtbl;
51 static const struct IDirectXFileDataVtbl IDirectXFileData_Vtbl;
52 static const struct IDirectXFileDataReferenceVtbl IDirectXFileDataReference_Vtbl;
53 static const struct IDirectXFileEnumObjectVtbl IDirectXFileEnumObject_Vtbl;
54 static const struct IDirectXFileObjectVtbl IDirectXFileObject_Vtbl;
55 static const struct IDirectXFileSaveObjectVtbl IDirectXFileSaveObject_Vtbl;
57 static HRESULT IDirectXFileDataReferenceImpl_Create(IDirectXFileDataReferenceImpl** ppObj);
58 static HRESULT IDirectXFileEnumObjectImpl_Create(IDirectXFileEnumObjectImpl** ppObj);
59 static HRESULT IDirectXFileSaveObjectImpl_Create(IDirectXFileSaveObjectImpl** ppObj);
61 HRESULT IDirectXFileImpl_Create(IUnknown* pUnkOuter, LPVOID* ppObj)
63 IDirectXFileImpl* object;
65 TRACE("(%p,%p)\n", pUnkOuter, ppObj);
67 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileImpl));
68 if (!object)
70 ERR("Out of memory\n");
71 return DXFILEERR_BADALLOC;
74 object->lpVtbl.lpVtbl = &IDirectXFile_Vtbl;
75 object->ref = 1;
77 *ppObj = object;
79 return S_OK;
82 /*** IUnknown methods ***/
83 static HRESULT WINAPI IDirectXFileImpl_QueryInterface(IDirectXFile* iface, REFIID riid, void** ppvObject)
85 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
87 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
89 if (IsEqualGUID(riid, &IID_IUnknown)
90 || IsEqualGUID(riid, &IID_IDirectXFile))
92 IClassFactory_AddRef(iface);
93 *ppvObject = This;
94 return S_OK;
97 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
98 return E_NOINTERFACE;
101 static ULONG WINAPI IDirectXFileImpl_AddRef(IDirectXFile* iface)
103 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
104 ULONG ref = InterlockedIncrement(&This->ref);
106 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
108 return ref;
111 static ULONG WINAPI IDirectXFileImpl_Release(IDirectXFile* iface)
113 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
114 ULONG ref = InterlockedDecrement(&This->ref);
116 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
118 if (!ref)
119 HeapFree(GetProcessHeap(), 0, This);
121 return ref;
124 /*** IDirectXFile methods ***/
125 static HRESULT WINAPI IDirectXFileImpl_CreateEnumObject(IDirectXFile* iface, LPVOID pvSource, DXFILELOADOPTIONS dwLoadOptions, LPDIRECTXFILEENUMOBJECT* ppEnumObj)
127 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
128 IDirectXFileEnumObjectImpl* object;
129 HRESULT hr;
130 DWORD* header;
131 HANDLE hFile = INVALID_HANDLE_VALUE;
132 HANDLE file_mapping = 0;
133 LPBYTE buffer = NULL;
134 HGLOBAL resource_data = 0;
135 LPBYTE file_buffer;
136 DWORD file_size;
138 LPDXFILELOADMEMORY lpdxflm = NULL;
140 TRACE("(%p/%p)->(%p,%x,%p)\n", This, iface, pvSource, dwLoadOptions, ppEnumObj);
142 if (!ppEnumObj)
143 return DXFILEERR_BADVALUE;
145 if (dwLoadOptions == DXFILELOAD_FROMFILE)
147 TRACE("Open source file '%s'\n", (char*)pvSource);
149 hFile = CreateFileA(pvSource, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
150 if (hFile == INVALID_HANDLE_VALUE)
152 TRACE("File '%s' not found\n", (char*)pvSource);
153 return DXFILEERR_FILENOTFOUND;
156 file_size = GetFileSize(hFile, NULL);
158 file_mapping = CreateFileMappingA(hFile, NULL, PAGE_READONLY, 0, 0, NULL);
159 if (!file_mapping)
161 hr = DXFILEERR_BADFILETYPE;
162 goto error;
165 buffer = MapViewOfFile(file_mapping, FILE_MAP_READ, 0, 0, 0);
166 if (!buffer)
168 hr = DXFILEERR_BADFILETYPE;
169 goto error;
171 file_buffer = buffer;
173 else if (dwLoadOptions == DXFILELOAD_FROMRESOURCE)
175 HRSRC resource_info;
176 LPDXFILELOADRESOURCE lpdxflr = pvSource;
178 TRACE("Source in resource (module = %p, name = %s, type = %s\n", lpdxflr->hModule, debugstr_a(lpdxflr->lpName), debugstr_a(lpdxflr->lpType));
180 resource_info = FindResourceA(lpdxflr->hModule, lpdxflr->lpName, lpdxflr->lpType);
181 if (!resource_info)
183 hr = DXFILEERR_RESOURCENOTFOUND;
184 goto error;
187 file_size = SizeofResource(lpdxflr->hModule, resource_info);
189 resource_data = LoadResource(lpdxflr->hModule, resource_info);
190 if (!resource_data)
192 hr = DXFILEERR_BADRESOURCE;
193 goto error;
196 file_buffer = LockResource(resource_data);
197 if (!file_buffer)
199 hr = DXFILEERR_BADRESOURCE;
200 goto error;
203 else if (dwLoadOptions == DXFILELOAD_FROMMEMORY)
205 lpdxflm = pvSource;
207 TRACE("Source in memory at %p with size %d\n", lpdxflm->lpMemory, lpdxflm->dSize);
209 file_buffer = lpdxflm->lpMemory;
210 file_size = lpdxflm->dSize;
212 else
214 FIXME("Source type %d is not handled yet\n", dwLoadOptions);
215 hr = DXFILEERR_NOTDONEYET;
216 goto error;
219 header = (DWORD*)file_buffer;
221 if (TRACE_ON(d3dxof))
223 char string[17];
224 memcpy(string, header, 16);
225 string[16] = 0;
226 TRACE("header = '%s'\n", string);
229 if (file_size < 16)
231 hr = DXFILEERR_BADFILETYPE;
232 goto error;
235 if (header[0] != XOFFILE_FORMAT_MAGIC)
237 hr = DXFILEERR_BADFILETYPE;
238 goto error;
241 if ((header[1] != XOFFILE_FORMAT_VERSION_302) && (header[1] != XOFFILE_FORMAT_VERSION_303))
243 hr = DXFILEERR_BADFILEVERSION;
244 goto error;
247 if ((header[2] != XOFFILE_FORMAT_BINARY) && (header[2] != XOFFILE_FORMAT_TEXT) && (header[2] != XOFFILE_FORMAT_COMPRESSED))
249 hr = DXFILEERR_BADFILETYPE;
250 goto error;
253 if (header[2] == XOFFILE_FORMAT_COMPRESSED)
255 FIXME("Compressed formats not supported yet\n");
256 hr = DXFILEERR_BADVALUE;
257 goto error;
260 if ((header[3] != XOFFILE_FORMAT_FLOAT_BITS_32) && (header[3] != XOFFILE_FORMAT_FLOAT_BITS_64))
262 hr = DXFILEERR_BADFILEFLOATSIZE;
263 goto error;
266 TRACE("Header is correct\n");
268 hr = IDirectXFileEnumObjectImpl_Create(&object);
269 if (FAILED(hr))
270 goto error;
272 object->source = dwLoadOptions;
273 object->hFile = hFile;
274 object->file_mapping = file_mapping;
275 object->buffer = buffer;
276 object->pDirectXFile = This;
277 object->buf.pdxf = This;
278 object->buf.txt = (header[2] == XOFFILE_FORMAT_TEXT);
279 object->buf.token_present = FALSE;
280 object->buf.cur_subobject = 0;
282 TRACE("File size is %d bytes\n", file_size);
284 /* Go to data after header */
285 object->buf.buffer = file_buffer + 16;
286 object->buf.rem_bytes = file_size - 16;
288 *ppEnumObj = (LPDIRECTXFILEENUMOBJECT)object;
290 while (object->buf.rem_bytes && is_template_available(&object->buf))
292 if (!parse_template(&object->buf))
294 TRACE("Template is not correct\n");
295 hr = DXFILEERR_BADVALUE;
296 goto error;
298 else
300 TRACE("Template successfully parsed:\n");
301 if (TRACE_ON(d3dxof))
302 dump_template(This->xtemplates, &This->xtemplates[This->nb_xtemplates - 1]);
306 if (TRACE_ON(d3dxof))
308 int i;
309 TRACE("Registered templates (%d):\n", This->nb_xtemplates);
310 for (i = 0; i < This->nb_xtemplates; i++)
311 DPRINTF("%s - %s\n", This->xtemplates[i].name, debugstr_guid(&This->xtemplates[i].class_id));
314 return DXFILE_OK;
316 error:
317 if (buffer)
318 UnmapViewOfFile(buffer);
319 if (file_mapping)
320 CloseHandle(file_mapping);
321 if (hFile != INVALID_HANDLE_VALUE)
322 CloseHandle(hFile);
323 if (resource_data)
324 FreeResource(resource_data);
325 *ppEnumObj = NULL;
327 return hr;
330 static HRESULT WINAPI IDirectXFileImpl_CreateSaveObject(IDirectXFile* iface, LPCSTR szFileName, DXFILEFORMAT dwFileFormat, LPDIRECTXFILESAVEOBJECT* ppSaveObj)
332 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
334 FIXME("(%p/%p)->(%s,%x,%p) partial stub!\n", This, iface, szFileName, dwFileFormat, ppSaveObj);
336 if (!szFileName || !ppSaveObj)
337 return E_POINTER;
339 return IDirectXFileSaveObjectImpl_Create((IDirectXFileSaveObjectImpl**)ppSaveObj);
342 static HRESULT WINAPI IDirectXFileImpl_RegisterTemplates(IDirectXFile* iface, LPVOID pvData, DWORD cbSize)
344 IDirectXFileImpl *This = (IDirectXFileImpl *)iface;
345 DWORD token_header;
346 parse_buffer buf;
348 buf.buffer = pvData;
349 buf.rem_bytes = cbSize;
350 buf.txt = FALSE;
351 buf.token_present = FALSE;
352 buf.pdxf = This;
354 TRACE("(%p/%p)->(%p,%d)\n", This, iface, pvData, cbSize);
356 if (!pvData)
357 return DXFILEERR_BADVALUE;
359 if (cbSize < 16)
360 return DXFILEERR_BADFILETYPE;
362 if (TRACE_ON(d3dxof))
364 char string[17];
365 memcpy(string, pvData, 16);
366 string[16] = 0;
367 TRACE("header = '%s'\n", string);
370 read_bytes(&buf, &token_header, 4);
372 if (token_header != XOFFILE_FORMAT_MAGIC)
373 return DXFILEERR_BADFILETYPE;
375 read_bytes(&buf, &token_header, 4);
377 if ((token_header != XOFFILE_FORMAT_VERSION_302) && (token_header != XOFFILE_FORMAT_VERSION_303))
378 return DXFILEERR_BADFILEVERSION;
380 read_bytes(&buf, &token_header, 4);
382 if ((token_header != XOFFILE_FORMAT_BINARY) && (token_header != XOFFILE_FORMAT_TEXT) && (token_header != XOFFILE_FORMAT_COMPRESSED))
383 return DXFILEERR_BADFILETYPE;
385 if (token_header == XOFFILE_FORMAT_TEXT)
387 buf.txt = TRUE;
390 if (token_header == XOFFILE_FORMAT_COMPRESSED)
392 FIXME("Compressed formats not supported yet\n");
393 return DXFILEERR_BADVALUE;
396 read_bytes(&buf, &token_header, 4);
398 if ((token_header != XOFFILE_FORMAT_FLOAT_BITS_32) && (token_header != XOFFILE_FORMAT_FLOAT_BITS_64))
399 return DXFILEERR_BADFILEFLOATSIZE;
401 TRACE("Header is correct\n");
403 while (buf.rem_bytes)
405 if (!parse_template(&buf))
407 TRACE("Template is not correct\n");
408 return DXFILEERR_BADVALUE;
410 else
412 TRACE("Template successfully parsed:\n");
413 if (TRACE_ON(d3dxof))
414 dump_template(This->xtemplates, &This->xtemplates[This->nb_xtemplates - 1]);
418 if (TRACE_ON(d3dxof))
420 int i;
421 TRACE("Registered templates (%d):\n", This->nb_xtemplates);
422 for (i = 0; i < This->nb_xtemplates; i++)
423 DPRINTF("%s - %s\n", This->xtemplates[i].name, debugstr_guid(&This->xtemplates[i].class_id));
426 return DXFILE_OK;
429 static const IDirectXFileVtbl IDirectXFile_Vtbl =
431 IDirectXFileImpl_QueryInterface,
432 IDirectXFileImpl_AddRef,
433 IDirectXFileImpl_Release,
434 IDirectXFileImpl_CreateEnumObject,
435 IDirectXFileImpl_CreateSaveObject,
436 IDirectXFileImpl_RegisterTemplates
439 static HRESULT IDirectXFileBinaryImpl_Create(IDirectXFileBinaryImpl** ppObj)
441 IDirectXFileBinaryImpl* object;
443 TRACE("(%p)\n", ppObj);
445 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileBinaryImpl));
446 if (!object)
448 ERR("Out of memory\n");
449 return DXFILEERR_BADALLOC;
452 object->lpVtbl.lpVtbl = &IDirectXFileBinary_Vtbl;
453 object->ref = 1;
455 *ppObj = object;
457 return DXFILE_OK;
460 /*** IUnknown methods ***/
461 static HRESULT WINAPI IDirectXFileBinaryImpl_QueryInterface(IDirectXFileBinary* iface, REFIID riid, void** ppvObject)
463 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
465 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
467 if (IsEqualGUID(riid, &IID_IUnknown)
468 || IsEqualGUID(riid, &IID_IDirectXFileObject)
469 || IsEqualGUID(riid, &IID_IDirectXFileBinary))
471 IClassFactory_AddRef(iface);
472 *ppvObject = This;
473 return S_OK;
476 /* Do not print an error for interfaces that can be queried to retrieve the type of the object */
477 if (!IsEqualGUID(riid, &IID_IDirectXFileData)
478 && !IsEqualGUID(riid, &IID_IDirectXFileDataReference))
479 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
481 return E_NOINTERFACE;
484 static ULONG WINAPI IDirectXFileBinaryImpl_AddRef(IDirectXFileBinary* iface)
486 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
487 ULONG ref = InterlockedIncrement(&This->ref);
489 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
491 return ref;
494 static ULONG WINAPI IDirectXFileBinaryImpl_Release(IDirectXFileBinary* iface)
496 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
497 ULONG ref = InterlockedDecrement(&This->ref);
499 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
501 if (!ref)
502 HeapFree(GetProcessHeap(), 0, This);
504 return ref;
507 /*** IDirectXFileObject methods ***/
508 static HRESULT WINAPI IDirectXFileBinaryImpl_GetName(IDirectXFileBinary* iface, LPSTR pstrNameBuf, LPDWORD pdwBufLen)
511 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
513 FIXME("(%p/%p)->(%p,%p) stub!\n", This, iface, pstrNameBuf, pdwBufLen);
515 return DXFILEERR_BADVALUE;
518 static HRESULT WINAPI IDirectXFileBinaryImpl_GetId(IDirectXFileBinary* iface, LPGUID pGuid)
520 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
522 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pGuid);
524 return DXFILEERR_BADVALUE;
527 /*** IDirectXFileBinary methods ***/
528 static HRESULT WINAPI IDirectXFileBinaryImpl_GetSize(IDirectXFileBinary* iface, DWORD* pcbSize)
530 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
532 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pcbSize);
534 return DXFILEERR_BADVALUE;
537 static HRESULT WINAPI IDirectXFileBinaryImpl_GetMimeType(IDirectXFileBinary* iface, LPCSTR* pszMimeType)
539 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
541 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pszMimeType);
543 return DXFILEERR_BADVALUE;
546 static HRESULT WINAPI IDirectXFileBinaryImpl_Read(IDirectXFileBinary* iface, LPVOID pvData, DWORD cbSize, LPDWORD pcbRead)
548 IDirectXFileBinaryImpl *This = (IDirectXFileBinaryImpl *)iface;
550 FIXME("(%p/%p)->(%p, %d, %p) stub!\n", This, iface, pvData, cbSize, pcbRead);
552 return DXFILEERR_BADVALUE;
555 static const IDirectXFileBinaryVtbl IDirectXFileBinary_Vtbl =
557 IDirectXFileBinaryImpl_QueryInterface,
558 IDirectXFileBinaryImpl_AddRef,
559 IDirectXFileBinaryImpl_Release,
560 IDirectXFileBinaryImpl_GetName,
561 IDirectXFileBinaryImpl_GetId,
562 IDirectXFileBinaryImpl_GetSize,
563 IDirectXFileBinaryImpl_GetMimeType,
564 IDirectXFileBinaryImpl_Read
567 static HRESULT IDirectXFileDataImpl_Create(IDirectXFileDataImpl** ppObj)
569 IDirectXFileDataImpl* object;
571 TRACE("(%p)\n", ppObj);
573 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileDataImpl));
574 if (!object)
576 ERR("Out of memory\n");
577 return DXFILEERR_BADALLOC;
580 object->lpVtbl.lpVtbl = &IDirectXFileData_Vtbl;
581 object->ref = 1;
583 *ppObj = object;
585 return S_OK;
588 /*** IUnknown methods ***/
589 static HRESULT WINAPI IDirectXFileDataImpl_QueryInterface(IDirectXFileData* iface, REFIID riid, void** ppvObject)
591 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
593 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
595 if (IsEqualGUID(riid, &IID_IUnknown)
596 || IsEqualGUID(riid, &IID_IDirectXFileObject)
597 || IsEqualGUID(riid, &IID_IDirectXFileData))
599 IClassFactory_AddRef(iface);
600 *ppvObject = This;
601 return S_OK;
604 /* Do not print an error for interfaces that can be queried to retrieve the type of the object */
605 if (!IsEqualGUID(riid, &IID_IDirectXFileBinary)
606 && !IsEqualGUID(riid, &IID_IDirectXFileDataReference))
607 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
609 return E_NOINTERFACE;
612 static ULONG WINAPI IDirectXFileDataImpl_AddRef(IDirectXFileData* iface)
614 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
615 ULONG ref = InterlockedIncrement(&This->ref);
617 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
619 return ref;
622 static ULONG WINAPI IDirectXFileDataImpl_Release(IDirectXFileData* iface)
624 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
625 ULONG ref = InterlockedDecrement(&This->ref);
627 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
629 if (!ref)
631 if (!This->level)
633 HeapFree(GetProcessHeap(), 0, This->pdata);
634 HeapFree(GetProcessHeap(), 0, This->pstrings);
636 HeapFree(GetProcessHeap(), 0, This);
639 return ref;
642 /*** IDirectXFileObject methods ***/
643 static HRESULT WINAPI IDirectXFileDataImpl_GetName(IDirectXFileData* iface, LPSTR pstrNameBuf, LPDWORD pdwBufLen)
646 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
648 TRACE("(%p/%p)->(%p,%p)\n", This, iface, pstrNameBuf, pdwBufLen);
650 if (!pstrNameBuf)
651 return DXFILEERR_BADVALUE;
653 strcpy(pstrNameBuf, This->pobj->name);
655 return DXFILE_OK;
658 static HRESULT WINAPI IDirectXFileDataImpl_GetId(IDirectXFileData* iface, LPGUID pGuid)
660 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
662 TRACE("(%p/%p)->(%p)\n", This, iface, pGuid);
664 if (!pGuid)
665 return DXFILEERR_BADVALUE;
667 memcpy(pGuid, &This->pobj->class_id, 16);
669 return DXFILE_OK;
672 /*** IDirectXFileData methods ***/
673 static HRESULT WINAPI IDirectXFileDataImpl_GetData(IDirectXFileData* iface, LPCSTR szMember, DWORD* pcbSize, void** ppvData)
675 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
677 TRACE("(%p/%p)->(%s,%p,%p)\n", This, iface, szMember, pcbSize, ppvData);
679 if (!pcbSize || !ppvData)
680 return DXFILEERR_BADVALUE;
682 if (szMember)
684 FIXME("Specifying a member is not supported yet!\n");
685 return DXFILEERR_BADVALUE;
688 *pcbSize = This->pobj->size;
689 *ppvData = This->pobj->pdata;
691 return DXFILE_OK;
694 static HRESULT WINAPI IDirectXFileDataImpl_GetType(IDirectXFileData* iface, const GUID** pguid)
696 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
697 static GUID guid;
699 TRACE("(%p/%p)->(%p)\n", This, iface, pguid);
701 if (!pguid)
702 return DXFILEERR_BADVALUE;
704 memcpy(&guid, &This->pobj->type, 16);
705 *pguid = &guid;
707 return DXFILE_OK;
710 static HRESULT WINAPI IDirectXFileDataImpl_GetNextObject(IDirectXFileData* iface, LPDIRECTXFILEOBJECT* ppChildObj)
712 HRESULT hr;
713 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
715 TRACE("(%p/%p)->(%p)\n", This, iface, ppChildObj);
717 if (This->cur_enum_object >= This->pobj->nb_childs)
718 return DXFILEERR_NOMOREOBJECTS;
720 if (This->from_ref && (This->level >= 1))
722 /* Only 2 levels can enumerated if the object is obtained from a reference */
723 return DXFILEERR_NOMOREOBJECTS;
726 if (This->pobj->childs[This->cur_enum_object]->binary)
728 IDirectXFileBinaryImpl *object;
730 hr = IDirectXFileBinaryImpl_Create(&object);
731 if (FAILED(hr))
732 return hr;
734 *ppChildObj = (LPDIRECTXFILEOBJECT)object;
736 else if (This->pobj->childs[This->cur_enum_object]->ptarget)
738 IDirectXFileDataReferenceImpl *object;
740 hr = IDirectXFileDataReferenceImpl_Create(&object);
741 if (FAILED(hr))
742 return hr;
744 object->ptarget = This->pobj->childs[This->cur_enum_object++]->ptarget;
746 *ppChildObj = (LPDIRECTXFILEOBJECT)object;
748 else
750 IDirectXFileDataImpl *object;
752 hr = IDirectXFileDataImpl_Create(&object);
753 if (FAILED(hr))
754 return hr;
756 object->pobj = This->pobj->childs[This->cur_enum_object++];
757 object->cur_enum_object = 0;
758 object->from_ref = This->from_ref;
759 object->level = This->level + 1;
761 *ppChildObj = (LPDIRECTXFILEOBJECT)object;
764 return DXFILE_OK;
767 static HRESULT WINAPI IDirectXFileDataImpl_AddDataObject(IDirectXFileData* iface, LPDIRECTXFILEDATA pDataObj)
769 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
771 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pDataObj);
773 return DXFILEERR_BADVALUE;
776 static HRESULT WINAPI IDirectXFileDataImpl_AddDataReference(IDirectXFileData* iface, LPCSTR szRef, const GUID* pguidRef)
778 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
780 FIXME("(%p/%p)->(%s,%p) stub!\n", This, iface, szRef, pguidRef);
782 return DXFILEERR_BADVALUE;
785 static HRESULT WINAPI IDirectXFileDataImpl_AddBinaryObject(IDirectXFileData* iface, LPCSTR szName, const GUID* pguid, LPCSTR szMimeType, LPVOID pvData, DWORD cbSize)
787 IDirectXFileDataImpl *This = (IDirectXFileDataImpl *)iface;
789 FIXME("(%p/%p)->(%s,%p,%s,%p,%d) stub!\n", This, iface, szName, pguid, szMimeType, pvData, cbSize);
791 return DXFILEERR_BADVALUE;
794 static const IDirectXFileDataVtbl IDirectXFileData_Vtbl =
796 IDirectXFileDataImpl_QueryInterface,
797 IDirectXFileDataImpl_AddRef,
798 IDirectXFileDataImpl_Release,
799 IDirectXFileDataImpl_GetName,
800 IDirectXFileDataImpl_GetId,
801 IDirectXFileDataImpl_GetData,
802 IDirectXFileDataImpl_GetType,
803 IDirectXFileDataImpl_GetNextObject,
804 IDirectXFileDataImpl_AddDataObject,
805 IDirectXFileDataImpl_AddDataReference,
806 IDirectXFileDataImpl_AddBinaryObject
809 static HRESULT IDirectXFileDataReferenceImpl_Create(IDirectXFileDataReferenceImpl** ppObj)
811 IDirectXFileDataReferenceImpl* object;
813 TRACE("(%p)\n", ppObj);
815 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileDataReferenceImpl));
816 if (!object)
818 ERR("Out of memory\n");
819 return DXFILEERR_BADALLOC;
822 object->lpVtbl.lpVtbl = &IDirectXFileDataReference_Vtbl;
823 object->ref = 1;
825 *ppObj = object;
827 return S_OK;
830 /*** IUnknown methods ***/
831 static HRESULT WINAPI IDirectXFileDataReferenceImpl_QueryInterface(IDirectXFileDataReference* iface, REFIID riid, void** ppvObject)
833 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
835 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
837 if (IsEqualGUID(riid, &IID_IUnknown)
838 || IsEqualGUID(riid, &IID_IDirectXFileObject)
839 || IsEqualGUID(riid, &IID_IDirectXFileDataReference))
841 IClassFactory_AddRef(iface);
842 *ppvObject = This;
843 return S_OK;
846 /* Do not print an error for interfaces that can be queried to retrieve the type of the object */
847 if (!IsEqualGUID(riid, &IID_IDirectXFileData)
848 && !IsEqualGUID(riid, &IID_IDirectXFileBinary))
849 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
851 return E_NOINTERFACE;
854 static ULONG WINAPI IDirectXFileDataReferenceImpl_AddRef(IDirectXFileDataReference* iface)
856 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
857 ULONG ref = InterlockedIncrement(&This->ref);
859 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
861 return ref;
864 static ULONG WINAPI IDirectXFileDataReferenceImpl_Release(IDirectXFileDataReference* iface)
866 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
867 ULONG ref = InterlockedDecrement(&This->ref);
869 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
871 if (!ref)
872 HeapFree(GetProcessHeap(), 0, This);
874 return ref;
877 /*** IDirectXFileObject methods ***/
878 static HRESULT WINAPI IDirectXFileDataReferenceImpl_GetName(IDirectXFileDataReference* iface, LPSTR pstrNameBuf, LPDWORD pdwBufLen)
880 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
882 TRACE("(%p/%p)->(%p,%p)\n", This, iface, pstrNameBuf, pdwBufLen);
884 if (!pstrNameBuf)
885 return DXFILEERR_BADVALUE;
887 strcpy(pstrNameBuf, This->ptarget->name);
889 return DXFILEERR_BADVALUE;
892 static HRESULT WINAPI IDirectXFileDataReferenceImpl_GetId(IDirectXFileDataReference* iface, LPGUID pGuid)
894 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
896 TRACE("(%p/%p)->(%p)\n", This, iface, pGuid);
898 if (!pGuid)
899 return DXFILEERR_BADVALUE;
901 memcpy(pGuid, &This->ptarget->class_id, 16);
903 return DXFILE_OK;
906 /*** IDirectXFileDataReference ***/
907 static HRESULT WINAPI IDirectXFileDataReferenceImpl_Resolve(IDirectXFileDataReference* iface, LPDIRECTXFILEDATA* ppDataObj)
909 IDirectXFileDataReferenceImpl *This = (IDirectXFileDataReferenceImpl *)iface;
910 IDirectXFileDataImpl *object;
911 HRESULT hr;
913 TRACE("(%p/%p)->(%p)\n", This, iface, ppDataObj);
915 if (!ppDataObj)
916 return DXFILEERR_BADVALUE;
918 hr = IDirectXFileDataImpl_Create(&object);
919 if (FAILED(hr))
920 return hr;
922 object->pobj = This->ptarget;
923 object->cur_enum_object = 0;
924 object->level = 0;
925 object->from_ref = TRUE;
927 *ppDataObj = (LPDIRECTXFILEDATA)object;
929 return DXFILE_OK;
932 static const IDirectXFileDataReferenceVtbl IDirectXFileDataReference_Vtbl =
934 IDirectXFileDataReferenceImpl_QueryInterface,
935 IDirectXFileDataReferenceImpl_AddRef,
936 IDirectXFileDataReferenceImpl_Release,
937 IDirectXFileDataReferenceImpl_GetName,
938 IDirectXFileDataReferenceImpl_GetId,
939 IDirectXFileDataReferenceImpl_Resolve
942 static HRESULT IDirectXFileEnumObjectImpl_Create(IDirectXFileEnumObjectImpl** ppObj)
944 IDirectXFileEnumObjectImpl* object;
946 TRACE("(%p)\n", ppObj);
948 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileEnumObjectImpl));
949 if (!object)
951 ERR("Out of memory\n");
952 return DXFILEERR_BADALLOC;
955 object->lpVtbl.lpVtbl = &IDirectXFileEnumObject_Vtbl;
956 object->ref = 1;
958 *ppObj = object;
960 return S_OK;
963 /*** IUnknown methods ***/
964 static HRESULT WINAPI IDirectXFileEnumObjectImpl_QueryInterface(IDirectXFileEnumObject* iface, REFIID riid, void** ppvObject)
966 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
968 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
970 if (IsEqualGUID(riid, &IID_IUnknown)
971 || IsEqualGUID(riid, &IID_IDirectXFileEnumObject))
973 IClassFactory_AddRef(iface);
974 *ppvObject = This;
975 return S_OK;
978 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
979 return E_NOINTERFACE;
982 static ULONG WINAPI IDirectXFileEnumObjectImpl_AddRef(IDirectXFileEnumObject* iface)
984 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
985 ULONG ref = InterlockedIncrement(&This->ref);
987 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
989 return ref;
992 static ULONG WINAPI IDirectXFileEnumObjectImpl_Release(IDirectXFileEnumObject* iface)
994 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
995 ULONG ref = InterlockedDecrement(&This->ref);
997 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
999 if (!ref)
1001 int i;
1002 for (i = 0; i < This->nb_xobjects; i++)
1004 IDirectXFileData_Release(This->pRefObjects[i]);
1005 HeapFree(GetProcessHeap(), 0, This->xobjects[i]);
1007 if (This->source == DXFILELOAD_FROMFILE)
1009 UnmapViewOfFile(This->buffer);
1010 CloseHandle(This->file_mapping);
1011 CloseHandle(This->hFile);
1013 else if (This->source == DXFILELOAD_FROMRESOURCE)
1014 FreeResource(This->resource_data);
1015 HeapFree(GetProcessHeap(), 0, This);
1018 return ref;
1021 /*** IDirectXFileEnumObject methods ***/
1022 static HRESULT WINAPI IDirectXFileEnumObjectImpl_GetNextDataObject(IDirectXFileEnumObject* iface, LPDIRECTXFILEDATA* ppDataObj)
1024 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
1025 IDirectXFileDataImpl* object;
1026 HRESULT hr;
1027 LPBYTE pdata = NULL;
1028 LPBYTE pstrings = NULL;
1030 TRACE("(%p/%p)->(%p)\n", This, iface, ppDataObj);
1032 if (This->nb_xobjects >= MAX_OBJECTS)
1034 ERR("Too many objects\n");
1035 return DXFILEERR_NOMOREOBJECTS;
1038 if (!This->buf.rem_bytes)
1039 return DXFILEERR_NOMOREOBJECTS;
1041 hr = IDirectXFileDataImpl_Create(&object);
1042 if (FAILED(hr))
1043 return hr;
1045 This->buf.pxo_globals = This->xobjects;
1046 This->buf.nb_pxo_globals = This->nb_xobjects;
1047 This->buf.cur_subobject = 1;
1048 This->buf.level = 0;
1050 This->buf.pxo_tab = HeapAlloc(GetProcessHeap(), 0, sizeof(xobject)*MAX_SUBOBJECTS);
1051 if (!This->buf.pxo_tab)
1053 ERR("Out of memory\n");
1054 hr = DXFILEERR_BADALLOC;
1055 goto error;
1057 This->buf.pxo = This->xobjects[This->nb_xobjects] = This->buf.pxo_tab;
1059 pdata = HeapAlloc(GetProcessHeap(), 0, MAX_DATA_SIZE);
1060 if (!pdata)
1062 ERR("Out of memory\n");
1063 hr = DXFILEERR_BADALLOC;
1064 goto error;
1066 This->buf.cur_pdata = This->buf.pdata = object->pdata = pdata;
1068 pstrings = HeapAlloc(GetProcessHeap(), 0, MAX_STRINGS_BUFFER);
1069 if (!pstrings)
1071 ERR("Out of memory\n");
1072 hr = DXFILEERR_BADALLOC;
1073 goto error;
1075 This->buf.cur_pstrings = This->buf.pstrings = object->pstrings = pstrings;
1077 if (!parse_object(&This->buf))
1079 TRACE("Object is not correct\n");
1080 hr = DXFILEERR_PARSEERROR;
1081 goto error;
1084 This->buf.pxo->nb_subobjects = This->buf.cur_subobject;
1085 if (This->buf.cur_subobject > MAX_SUBOBJECTS)
1087 FIXME("Too many suobjects %d\n", This->buf.cur_subobject);
1088 hr = DXFILEERR_BADALLOC;
1089 goto error;
1092 object->pstrings = pstrings;
1093 object->pobj = This->buf.pxo;
1094 object->cur_enum_object = 0;
1095 object->level = 0;
1096 object->from_ref = FALSE;
1098 *ppDataObj = (LPDIRECTXFILEDATA)object;
1100 /* Get a reference to created object */
1101 This->pRefObjects[This->nb_xobjects] = (LPDIRECTXFILEDATA)object;
1102 IDirectXFileData_AddRef(This->pRefObjects[This->nb_xobjects]);
1104 This->nb_xobjects++;
1106 return DXFILE_OK;
1108 error:
1110 HeapFree(GetProcessHeap(), 0, This->buf.pxo_tab);
1111 HeapFree(GetProcessHeap(), 0, pdata);
1112 HeapFree(GetProcessHeap(), 0, pstrings);
1114 return hr;
1117 static HRESULT WINAPI IDirectXFileEnumObjectImpl_GetDataObjectById(IDirectXFileEnumObject* iface, REFGUID rguid, LPDIRECTXFILEDATA* ppDataObj)
1119 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
1121 FIXME("(%p/%p)->(%p,%p) stub!\n", This, iface, rguid, ppDataObj);
1123 return DXFILEERR_BADVALUE;
1126 static HRESULT WINAPI IDirectXFileEnumObjectImpl_GetDataObjectByName(IDirectXFileEnumObject* iface, LPCSTR szName, LPDIRECTXFILEDATA* ppDataObj)
1128 IDirectXFileEnumObjectImpl *This = (IDirectXFileEnumObjectImpl *)iface;
1130 FIXME("(%p/%p)->(%s,%p) stub!\n", This, iface, szName, ppDataObj);
1132 return DXFILEERR_BADVALUE;
1135 static const IDirectXFileEnumObjectVtbl IDirectXFileEnumObject_Vtbl =
1137 IDirectXFileEnumObjectImpl_QueryInterface,
1138 IDirectXFileEnumObjectImpl_AddRef,
1139 IDirectXFileEnumObjectImpl_Release,
1140 IDirectXFileEnumObjectImpl_GetNextDataObject,
1141 IDirectXFileEnumObjectImpl_GetDataObjectById,
1142 IDirectXFileEnumObjectImpl_GetDataObjectByName
1145 static HRESULT IDirectXFileSaveObjectImpl_Create(IDirectXFileSaveObjectImpl** ppObj)
1147 IDirectXFileSaveObjectImpl* object;
1149 TRACE("(%p)\n", ppObj);
1151 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectXFileSaveObjectImpl));
1152 if (!object)
1154 ERR("Out of memory\n");
1155 return DXFILEERR_BADALLOC;
1158 object->lpVtbl.lpVtbl = &IDirectXFileSaveObject_Vtbl;
1159 object->ref = 1;
1161 *ppObj = object;
1163 return S_OK;
1166 /*** IUnknown methods ***/
1167 static HRESULT WINAPI IDirectXFileSaveObjectImpl_QueryInterface(IDirectXFileSaveObject* iface, REFIID riid, void** ppvObject)
1169 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1171 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
1173 if (IsEqualGUID(riid, &IID_IUnknown)
1174 || IsEqualGUID(riid, &IID_IDirectXFileSaveObject))
1176 IClassFactory_AddRef(iface);
1177 *ppvObject = This;
1178 return S_OK;
1181 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
1182 return E_NOINTERFACE;
1185 static ULONG WINAPI IDirectXFileSaveObjectImpl_AddRef(IDirectXFileSaveObject* iface)
1187 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1188 ULONG ref = InterlockedIncrement(&This->ref);
1190 TRACE("(%p/%p): AddRef from %d\n", iface, This, ref - 1);
1192 return ref;
1195 static ULONG WINAPI IDirectXFileSaveObjectImpl_Release(IDirectXFileSaveObject* iface)
1197 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1198 ULONG ref = InterlockedDecrement(&This->ref);
1200 TRACE("(%p/%p): ReleaseRef to %d\n", iface, This, ref);
1202 if (!ref)
1203 HeapFree(GetProcessHeap(), 0, This);
1205 return ref;
1208 static HRESULT WINAPI IDirectXFileSaveObjectImpl_SaveTemplates(IDirectXFileSaveObject* iface, DWORD cTemplates, const GUID** ppguidTemplates)
1210 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1212 FIXME("(%p/%p)->(%d,%p) stub!\n", This, iface, cTemplates, ppguidTemplates);
1214 return DXFILEERR_BADVALUE;
1217 static HRESULT WINAPI IDirectXFileSaveObjectImpl_CreateDataObject(IDirectXFileSaveObject* iface, REFGUID rguidTemplate, LPCSTR szName, const GUID* pguid, DWORD cbSize, LPVOID pvData, LPDIRECTXFILEDATA* ppDataObj)
1219 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1221 FIXME("(%p/%p)->(%p,%s,%p,%d,%p,%p) stub!\n", This, iface, rguidTemplate, szName, pguid, cbSize, pvData, ppDataObj);
1223 return DXFILEERR_BADVALUE;
1226 static HRESULT WINAPI IDirectXFileSaveObjectImpl_SaveData(IDirectXFileSaveObject* iface, LPDIRECTXFILEDATA ppDataObj)
1228 IDirectXFileSaveObjectImpl *This = (IDirectXFileSaveObjectImpl *)iface;
1230 FIXME("(%p/%p)->(%p) stub!\n", This, iface, ppDataObj);
1232 return DXFILEERR_BADVALUE;
1235 static const IDirectXFileSaveObjectVtbl IDirectXFileSaveObject_Vtbl =
1237 IDirectXFileSaveObjectImpl_QueryInterface,
1238 IDirectXFileSaveObjectImpl_AddRef,
1239 IDirectXFileSaveObjectImpl_Release,
1240 IDirectXFileSaveObjectImpl_SaveTemplates,
1241 IDirectXFileSaveObjectImpl_CreateDataObject,
1242 IDirectXFileSaveObjectImpl_SaveData