/* IDirectMusicBand Implementation * * Copyright (C) 2003 Rok Mandeljc * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Library General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include #include "windef.h" #include "winbase.h" #include "winuser.h" #include "wingdi.h" #include "wine/debug.h" #include "dmband_private.h" WINE_DEFAULT_DEBUG_CHANNEL(dmband); WINE_DECLARE_DEBUG_CHANNEL(dmfile); /***************************************************************************** * IDirectMusicBandImpl implementation */ /* IDirectMusicBand IUnknown part: */ HRESULT WINAPI IDirectMusicBandImpl_QueryInterface (LPDIRECTMUSICBAND iface, REFIID riid, LPVOID *ppobj) { ICOM_THIS(IDirectMusicBandImpl,iface); if (IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IDirectMusicBand)) { IDirectMusicBandImpl_AddRef(iface); *ppobj = This; return S_OK; } WARN("(%p)->(%s,%p),not found\n", This, debugstr_guid(riid), ppobj); return E_NOINTERFACE; } ULONG WINAPI IDirectMusicBandImpl_AddRef (LPDIRECTMUSICBAND iface) { ICOM_THIS(IDirectMusicBandImpl,iface); TRACE("(%p) : AddRef from %ld\n", This, This->ref); return ++(This->ref); } ULONG WINAPI IDirectMusicBandImpl_Release (LPDIRECTMUSICBAND iface) { ICOM_THIS(IDirectMusicBandImpl,iface); ULONG ref = --This->ref; TRACE("(%p) : ReleaseRef to %ld\n", This, This->ref); if (ref == 0) { HeapFree(GetProcessHeap(), 0, This); } return ref; } /* IDirectMusicBand IDirectMusicBand part: */ HRESULT WINAPI IDirectMusicBandImpl_CreateSegment (LPDIRECTMUSICBAND iface, IDirectMusicSegment** ppSegment) { ICOM_THIS(IDirectMusicBandImpl,iface); FIXME("(%p, %p): stub\n", This, ppSegment); return S_OK; } HRESULT WINAPI IDirectMusicBandImpl_Download (LPDIRECTMUSICBAND iface, IDirectMusicPerformance* pPerformance) { ICOM_THIS(IDirectMusicBandImpl,iface); FIXME("(%p, %p): stub\n", This, pPerformance); return S_OK; } HRESULT WINAPI IDirectMusicBandImpl_Unload (LPDIRECTMUSICBAND iface, IDirectMusicPerformance* pPerformance) { ICOM_THIS(IDirectMusicBandImpl,iface); FIXME("(%p, %p): stub\n", This, pPerformance); return S_OK; } ICOM_VTABLE(IDirectMusicBand) DirectMusicBand_Vtbl = { ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE IDirectMusicBandImpl_QueryInterface, IDirectMusicBandImpl_AddRef, IDirectMusicBandImpl_Release, IDirectMusicBandImpl_CreateSegment, IDirectMusicBandImpl_Download, IDirectMusicBandImpl_Unload }; /* for ClassFactory */ HRESULT WINAPI DMUSIC_CreateDirectMusicBand (LPCGUID lpcGUID, LPDIRECTMUSICBAND* ppDMBand, LPUNKNOWN pUnkOuter) { IDirectMusicBandImpl* dmband; if (IsEqualGUID (lpcGUID, &IID_IDirectMusicBand)) { dmband = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicBandImpl)); if (NULL == dmband) { *ppDMBand = (LPDIRECTMUSICBAND) NULL; return E_OUTOFMEMORY; } dmband->lpVtbl = &DirectMusicBand_Vtbl; dmband->ref = 1; *ppDMBand = (LPDIRECTMUSICBAND) dmband; return S_OK; } WARN("No interface found\n"); return E_NOINTERFACE; } /***************************************************************************** * IDirectMusicBandObject implementation */ /* IDirectMusicBandObject IUnknown part: */ HRESULT WINAPI IDirectMusicBandObject_QueryInterface (LPDIRECTMUSICOBJECT iface, REFIID riid, LPVOID *ppobj) { ICOM_THIS(IDirectMusicBandObject,iface); if (IsEqualGUID (riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IDirectMusicObject)) { IDirectMusicBandObject_AddRef(iface); *ppobj = This; return S_OK; } else if (IsEqualGUID (riid, &IID_IPersistStream)) { IPersistStream_AddRef ((LPPERSISTSTREAM)This->pStream); *ppobj = (LPPERSISTSTREAM)This->pStream; return S_OK; } else if (IsEqualGUID (riid, &IID_IDirectMusicBand)) { IDirectMusicBand_AddRef ((LPDIRECTMUSICBAND)This->pBand); *ppobj = (LPDIRECTMUSICBAND)This->pBand; return S_OK; } WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppobj); return E_NOINTERFACE; } ULONG WINAPI IDirectMusicBandObject_AddRef (LPDIRECTMUSICOBJECT iface) { ICOM_THIS(IDirectMusicBandObject,iface); TRACE("(%p) : AddRef from %ld\n", This, This->ref); return ++(This->ref); } ULONG WINAPI IDirectMusicBandObject_Release (LPDIRECTMUSICOBJECT iface) { ICOM_THIS(IDirectMusicBandObject,iface); ULONG ref = --This->ref; TRACE("(%p) : ReleaseRef to %ld\n", This, This->ref); if (ref == 0) { HeapFree(GetProcessHeap(), 0, This); } return ref; } /* IDirectMusicBandObject IDirectMusicObject part: */ HRESULT WINAPI IDirectMusicBandObject_GetDescriptor (LPDIRECTMUSICOBJECT iface, LPDMUS_OBJECTDESC pDesc) { ICOM_THIS(IDirectMusicBandObject,iface); TRACE("(%p, %p)\n", This, pDesc); pDesc = This->pDesc; return S_OK; } HRESULT WINAPI IDirectMusicBandObject_SetDescriptor (LPDIRECTMUSICOBJECT iface, LPDMUS_OBJECTDESC pDesc) { ICOM_THIS(IDirectMusicBandObject,iface); TRACE("(%p, %p)\n", This, pDesc); This->pDesc = pDesc; return S_OK; } HRESULT WINAPI IDirectMusicBandObject_ParseDescriptor (LPDIRECTMUSICOBJECT iface, LPSTREAM pStream, LPDMUS_OBJECTDESC pDesc) { ICOM_THIS(IDirectMusicBandObject,iface); FIXME("(%p, %p, %p): stub\n", This, pStream, pDesc); return S_OK; } ICOM_VTABLE(IDirectMusicObject) DirectMusicBandObject_Vtbl = { ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE IDirectMusicBandObject_QueryInterface, IDirectMusicBandObject_AddRef, IDirectMusicBandObject_Release, IDirectMusicBandObject_GetDescriptor, IDirectMusicBandObject_SetDescriptor, IDirectMusicBandObject_ParseDescriptor }; /* for ClassFactory */ HRESULT WINAPI DMUSIC_CreateDirectMusicBandObject (LPCGUID lpcGUID, LPDIRECTMUSICOBJECT* ppObject, LPUNKNOWN pUnkOuter) { IDirectMusicBandObject *obj; TRACE("(%p,%p,%p)\n", lpcGUID, ppObject, pUnkOuter); if (IsEqualGUID (lpcGUID, &IID_IDirectMusicObject)) { obj = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicBandObject)); if (NULL == obj) { *ppObject = (LPDIRECTMUSICOBJECT) NULL; return E_OUTOFMEMORY; } obj->lpVtbl = &DirectMusicBandObject_Vtbl; obj->ref = 1; /* prepare IPersistStream */ obj->pStream = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, sizeof(IDirectMusicBandObjectStream)); obj->pStream->lpVtbl = &DirectMusicBandObjectStream_Vtbl; obj->pStream->ref = 1; obj->pStream->pParentObject = obj; /* prepare IDirectMusicBand */ DMUSIC_CreateDirectMusicBand (&IID_IDirectMusicBand, (LPDIRECTMUSICBAND*)&obj->pBand, NULL); obj->pBand->pObject = obj; *ppObject = (LPDIRECTMUSICOBJECT) obj; return S_OK; } WARN("No interface found\n"); return E_NOINTERFACE; } /***************************************************************************** * IDirectMusicBandObjectStream implementation */ /* IDirectMusicBandObjectStream IUnknown part: */ HRESULT WINAPI IDirectMusicBandObjectStream_QueryInterface (LPPERSISTSTREAM iface, REFIID riid, LPVOID *ppobj) { ICOM_THIS(IDirectMusicBandObjectStream,iface); if (IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IPersistStream)) { IDirectMusicBandObjectStream_AddRef(iface); *ppobj = This; return S_OK; } WARN("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppobj); return E_NOINTERFACE; } ULONG WINAPI IDirectMusicBandObjectStream_AddRef (LPPERSISTSTREAM iface) { ICOM_THIS(IDirectMusicBandObjectStream,iface); TRACE("(%p) : AddRef from %ld\n", This, This->ref); return ++(This->ref); } ULONG WINAPI IDirectMusicBandObjectStream_Release (LPPERSISTSTREAM iface) { ICOM_THIS(IDirectMusicBandObjectStream,iface); ULONG ref = --This->ref; TRACE("(%p) : ReleaseRef to %ld\n", This, This->ref); if (ref == 0) { HeapFree(GetProcessHeap(), 0, This); } return ref; } /* IDirectMusicBandObjectStream IPersist part: */ HRESULT WINAPI IDirectMusicBandObjectStream_GetClassID (LPPERSISTSTREAM iface, CLSID* pClassID) { return E_NOTIMPL; } /* IDirectMusicBandObjectStream IPersistStream part: */ HRESULT WINAPI IDirectMusicBandObjectStream_IsDirty (LPPERSISTSTREAM iface) { return E_NOTIMPL; } HRESULT WINAPI IDirectMusicBandObjectStream_Load (LPPERSISTSTREAM iface, IStream* pStm) { ICOM_THIS(IDirectMusicBandObjectStream,iface); FOURCC chunkID; DWORD chunkSize, StreamSize, StreamCount, ListSize[3], ListCount[3]; LARGE_INTEGER liMove; /* used when skipping chunks */ DMUS_IO_REFERENCE tempReferenceHeader; DMUS_OBJECTDESC ObjDesc; IDirectMusicBandImpl* pBand = This->pParentObject->pBand; /* that's where we load data to */ LPDIRECTMUSICLOADER pLoader; LPDIRECTMUSICGETLOADER pGetLoader; IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(DWORD), NULL); TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case FOURCC_RIFF: { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); TRACE_(dmfile)(": RIFF chunk of type %s", debugstr_fourcc(chunkID)); StreamSize = chunkSize - sizeof(FOURCC); StreamCount = 0; switch (chunkID) { case DMUS_FOURCC_BAND_FORM: { TRACE_(dmfile)(": band form\n"); do { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(FOURCC), NULL); StreamCount += sizeof(FOURCC) + sizeof(DWORD) + chunkSize; TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case DMUS_FOURCC_GUID_CHUNK: { TRACE_(dmfile)(": GUID chunk\n"); IStream_Read (pStm, &pBand->vVersion, chunkSize, NULL); break; } case DMUS_FOURCC_VERSION_CHUNK: { TRACE_(dmfile)(": version chunk\n"); IStream_Read (pStm, &pBand->guidID, chunkSize, NULL); break; } case FOURCC_LIST: { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); TRACE_(dmfile)(": LIST chunk of type %s", debugstr_fourcc(chunkID)); ListSize[0] = chunkSize - sizeof(FOURCC); ListCount[0] = 0; switch (chunkID) { case DMUS_FOURCC_UNFO_LIST: { TRACE_(dmfile)(": UNFO list\n"); do { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(FOURCC), NULL); ListCount[0] += sizeof(FOURCC) + sizeof(DWORD) + chunkSize; TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case DMUS_FOURCC_UNAM_CHUNK: { TRACE_(dmfile)(": name chunk\n"); pBand->wszName = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, chunkSize); IStream_Read (pStm, pBand->wszName, chunkSize, NULL); break; } case DMUS_FOURCC_UART_CHUNK: { TRACE_(dmfile)(": artist chunk\n"); pBand->wszArtist = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, chunkSize); IStream_Read (pStm, pBand->wszArtist, chunkSize, NULL); break; } case DMUS_FOURCC_UCOP_CHUNK: { TRACE_(dmfile)(": copyright chunk\n"); pBand->wszCopyright = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, chunkSize); IStream_Read (pStm, pBand->wszCopyright, chunkSize, NULL); break; } case DMUS_FOURCC_USBJ_CHUNK: { TRACE_(dmfile)(": subject chunk\n"); pBand->wszSubject = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, chunkSize); IStream_Read (pStm, pBand->wszSubject, chunkSize, NULL); break; } case DMUS_FOURCC_UCMT_CHUNK: { TRACE_(dmfile)(": comment chunk\n"); pBand->wszComment = HeapAlloc (GetProcessHeap (), HEAP_ZERO_MEMORY, chunkSize); IStream_Read (pStm, pBand->wszComment, chunkSize, NULL); break; } default: { TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); break; } } TRACE_(dmfile)(": ListCount[0] = %ld < ListSize[0] = %ld\n", ListCount[0], ListSize[0]); } while (ListCount[0] < ListSize[0]); break; } case DMUS_FOURCC_INSTRUMENTS_LIST: { TRACE_(dmfile)(": instruments list\n"); do { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(FOURCC), NULL); ListCount[0] += sizeof(FOURCC) + sizeof(DWORD) + chunkSize; TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case FOURCC_LIST: { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); TRACE_(dmfile)(": LIST chunk of type %s", debugstr_fourcc(chunkID)); ListSize[1] = chunkSize - sizeof(FOURCC); ListCount[1] = 0; switch (chunkID) { case DMUS_FOURCC_INSTRUMENT_LIST: { TRACE_(dmfile)(": instrument list\n"); do { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(FOURCC), NULL); ListCount[1] += sizeof(FOURCC) + sizeof(DWORD) + chunkSize; TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case DMUS_FOURCC_INSTRUMENT_CHUNK: { TRACE_(dmfile)(": band instrument header\n"); IStream_Read (pStm, &pBand->pInstruments[pBand->dwInstruments], chunkSize, NULL); break; } case FOURCC_LIST: { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); TRACE_(dmfile)(": LIST chunk of type %s", debugstr_fourcc(chunkID)); ListSize[2] = chunkSize - sizeof(FOURCC); ListCount[2] = 0; switch (chunkID) { case DMUS_FOURCC_REF_LIST: { TRACE_(dmfile)(": reference list\n"); ZeroMemory ((LPVOID)&ObjDesc, sizeof(DMUS_OBJECTDESC)); do { IStream_Read (pStm, &chunkID, sizeof(FOURCC), NULL); IStream_Read (pStm, &chunkSize, sizeof(FOURCC), NULL); ListCount[2] += sizeof(FOURCC) + sizeof(DWORD) + chunkSize; TRACE_(dmfile)(": %s chunk (size = %ld)", debugstr_fourcc (chunkID), chunkSize); switch (chunkID) { case DMUS_FOURCC_REF_CHUNK: { TRACE_(dmfile)(": reference header chunk\n"); IStream_Read (pStm, &tempReferenceHeader, chunkSize, NULL); /* copy retrieved data to DMUS_OBJECTDESC */ ObjDesc.dwSize = sizeof(DMUS_OBJECTDESC); ObjDesc.guidClass = tempReferenceHeader.guidClassID; ObjDesc.dwValidData = tempReferenceHeader.dwValidData; break; } case DMUS_FOURCC_GUID_CHUNK: { TRACE_(dmfile)(": guid chunk\n"); IStream_Read (pStm, &ObjDesc.guidObject, chunkSize, NULL); break; } case DMUS_FOURCC_DATE_CHUNK: { TRACE_(dmfile)(": file date chunk\n"); IStream_Read (pStm, &ObjDesc.ftDate, chunkSize, NULL); break; } case DMUS_FOURCC_NAME_CHUNK: { TRACE_(dmfile)(": name chunk\n"); IStream_Read (pStm, &ObjDesc.wszName, chunkSize, NULL); break; } case DMUS_FOURCC_FILE_CHUNK: { TRACE_(dmfile)(": file name chunk\n"); IStream_Read (pStm, &ObjDesc.wszFileName, chunkSize, NULL); break; } case DMUS_FOURCC_CATEGORY_CHUNK: { TRACE_(dmfile)(": category chunk\n"); IStream_Read (pStm, &ObjDesc.wszCategory, chunkSize, NULL); break; } case DMUS_FOURCC_VERSION_CHUNK: { TRACE_(dmfile)(": version chunk\n"); IStream_Read (pStm, &ObjDesc.vVersion, chunkSize, NULL); break; } default: { TRACE_(dmfile)(": unknown chunk (skipping)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip this chunk */ break; } } TRACE_(dmfile)(": ListCount[2] = %ld < ListSize[2] = %ld\n", ListCount[2], ListSize[2]); } while (ListCount[2] < ListSize[2]); /* let's see what we have */ TRACE_(dmfile)(": (READ): reference: dwSize = %ld; dwValidData = %ld; guidObject = %s; guidClass = %s; \ vVersion = %08lx,%08lx; wszName = %s; wszCategory = %s; wszFileName = %s\n", ObjDesc.dwSize, ObjDesc.dwValidData, debugstr_guid(&ObjDesc.guidObject), debugstr_guid(&ObjDesc.guidClass), ObjDesc.vVersion.dwVersionMS, ObjDesc.vVersion.dwVersionLS, debugstr_w(ObjDesc.wszName), debugstr_w(ObjDesc.wszCategory), debugstr_w(ObjDesc.wszFileName)); /* now, let's convience loader to load reference */ if (IStream_QueryInterface (pStm, &IID_IDirectMusicGetLoader, (LPVOID*)&pGetLoader) == S_OK) { if (IDirectMusicGetLoader_GetLoader (pGetLoader, &pLoader) == S_OK) { /* load referenced object */ IDirectMusicObject* pObject; if(FAILED(IDirectMusicLoader_GetObject (pLoader, &ObjDesc, &IID_IDirectMusicObject, (LPVOID*)&pObject))) /* acquire collection from loaded referenced object */ if(FAILED(IDirectMusicObject_QueryInterface (pObject, &IID_IDirectMusicCollection, (LPVOID*)&pBand->ppReferenceCollection[pBand->dwInstruments]))) IDirectMusicLoader_Release (pLoader); } IDirectMusicGetLoader_Release (pGetLoader); } else { ERR("Could not get IDirectMusicGetLoader... reference will not be loaded :(\n"); /* E_FAIL */ } break; } default: { TRACE_(dmfile)(": unexpected chunk; loading failed)\n"); return E_FAIL; } } break; } default: { TRACE_(dmfile)(": unknown chunk (skipping)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip this chunk */ break; } } TRACE_(dmfile)(": ListCount[1] = %ld < ListSize[1] = %ld\n", ListCount[1], ListSize[1]); } while (ListCount[1] < ListSize[1]); /* causes crash :( */ #if 0 /* hmm... in dxdiag segment's band there aren't any references, but loader still desperatly loads default collection... does that mean that if there is no reference, use default? */ if (!pBand->ppReferenceCollection[pBand->dwInstruments]) { TRACE(": (READ): loading default collection (as no specific reference was made)\n"); ZeroMemory ((LPVOID)&ObjDesc, sizeof(DMUS_OBJECTDESC)); ObjDesc.dwSize = sizeof(DMUS_OBJECTDESC); ObjDesc.dwValidData = DMUS_OBJ_CLASS | DMUS_OBJ_OBJECT; ObjDesc.guidObject = GUID_DefaultGMCollection; ObjDesc.guidClass = CLSID_DirectMusicCollection; if (SUCCEEDED(IStream_QueryInterface (pStm, &IID_IDirectMusicGetLoader, (LPVOID*)&pGetLoader))) { if (SUCCEEDED(IDirectMusicGetLoader_GetLoader (pGetLoader, &pLoader))) { IDirectMusicObject* pObject; if (SUCCEEDED(IDirectMusicLoader_GetObject (pLoader, &ObjDesc, &IID_IDirectMusicObject, (LPVOID*)&pObject))) { IDirectMusicObject_QueryInterface (pObject, &IID_IDirectMusicCollection, (LPVOID*)&pBand->ppReferenceCollection[pBand->dwInstruments]); IDirectMusicLoader_Release (pLoader); } } IDirectMusicGetLoader_Release (pGetLoader); } else { ERR("Could not get IDirectMusicGetLoader... reference will not be loaded :(\n"); /* E_FAIL */ } } #endif pBand->dwInstruments++; /* add count */ break; } default: { TRACE_(dmfile)(": unexpected chunk; loading failed)\n"); return E_FAIL; } } break; } default: { TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); break; } } TRACE_(dmfile)(": ListCount[0] = %ld < ListSize[0] = %ld\n", ListCount[0], ListSize[0]); } while (ListCount[0] < ListSize[0]); break; } default: { TRACE_(dmfile)(": unknown (skipping)\n"); liMove.QuadPart = chunkSize - sizeof(FOURCC); IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); break; } } break; } default: { TRACE_(dmfile)(": unknown chunk (irrevelant & skipping)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); break; } } TRACE_(dmfile)(": StreamCount[0] = %ld < StreamSize[0] = %ld\n", StreamCount, StreamSize); } while (StreamCount < StreamSize); break; } default: { TRACE_(dmfile)(": unexpected chunk; loading failed)\n"); liMove.QuadPart = StreamSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */ return E_FAIL; } } TRACE_(dmfile)(": reading finished\n"); break; } default: { TRACE_(dmfile)(": unexpected chunk; loading failed)\n"); liMove.QuadPart = chunkSize; IStream_Seek (pStm, liMove, STREAM_SEEK_CUR, NULL); /* skip the rest of the chunk */ return E_FAIL; } } return S_OK; } HRESULT WINAPI IDirectMusicBandObjectStream_Save (LPPERSISTSTREAM iface, IStream* pStm, BOOL fClearDirty) { return E_NOTIMPL; } HRESULT WINAPI IDirectMusicBandObjectStream_GetSizeMax (LPPERSISTSTREAM iface, ULARGE_INTEGER* pcbSize) { return E_NOTIMPL; } ICOM_VTABLE(IPersistStream) DirectMusicBandObjectStream_Vtbl = { ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE IDirectMusicBandObjectStream_QueryInterface, IDirectMusicBandObjectStream_AddRef, IDirectMusicBandObjectStream_Release, IDirectMusicBandObjectStream_GetClassID, IDirectMusicBandObjectStream_IsDirty, IDirectMusicBandObjectStream_Load, IDirectMusicBandObjectStream_Save, IDirectMusicBandObjectStream_GetSizeMax };