/* IDirectMusicInstrument Implementation * * Copyright (C) 2003-2004 Rok Mandeljc * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "dmusic_private.h" WINE_DEFAULT_DEBUG_CHANNEL(dmusic); static const GUID IID_IDirectMusicInstrumentPRIVATE = {0xbcb20080,0xa40c,0x11d1,{0x86,0xbc,0x00,0xc0,0x4f,0xbf,0x8f,0xef}}; static ULONG WINAPI IDirectMusicInstrumentImpl_IUnknown_AddRef (LPUNKNOWN iface); static ULONG WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_AddRef (LPDIRECTMUSICINSTRUMENT iface); /* IDirectMusicInstrument IUnknown part: */ static HRESULT WINAPI IDirectMusicInstrumentImpl_IUnknown_QueryInterface (LPUNKNOWN iface, REFIID riid, LPVOID *ppobj) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, UnknownVtbl, iface); TRACE("(%p, %s, %p)\n", This, debugstr_dmguid(riid), ppobj); if (IsEqualIID (riid, &IID_IUnknown)) { *ppobj = &This->UnknownVtbl; IDirectMusicInstrumentImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl); return S_OK; } else if (IsEqualIID (riid, &IID_IDirectMusicInstrument)) { *ppobj = &This->InstrumentVtbl; IDirectMusicInstrumentImpl_IDirectMusicInstrument_AddRef ((LPDIRECTMUSICINSTRUMENT)&This->InstrumentVtbl); return S_OK; } else if (IsEqualIID (riid, &IID_IDirectMusicInstrumentPRIVATE)) { /* it seems to me that this interface is only basic IUnknown, without any other inherited functions... *sigh* this is the worst scenario, since it means that whoever calls it knows the layout of original implementation table and therefore tries to get data by direct access... expect crashes */ FIXME("*sigh*... requested private/unspecified interface\n"); *ppobj = &This->UnknownVtbl; IDirectMusicInstrumentImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl); return S_OK; } WARN("(%p, %s, %p): not found\n", This, debugstr_dmguid(riid), ppobj); return E_NOINTERFACE; } static ULONG WINAPI IDirectMusicInstrumentImpl_IUnknown_AddRef (LPUNKNOWN iface) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, UnknownVtbl, iface); ULONG refCount = InterlockedIncrement(&This->ref); TRACE("(%p)->(ref before=%u)\n", This, refCount - 1); DMUSIC_LockModule(); return refCount; } static ULONG WINAPI IDirectMusicInstrumentImpl_IUnknown_Release (LPUNKNOWN iface) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, UnknownVtbl, iface); ULONG refCount = InterlockedDecrement(&This->ref); TRACE("(%p)->(ref before=%u)\n", This, refCount + 1); if (!refCount) { HeapFree(GetProcessHeap(), 0, This); } DMUSIC_UnlockModule(); return refCount; } static const IUnknownVtbl DirectMusicInstrument_Unknown_Vtbl = { IDirectMusicInstrumentImpl_IUnknown_QueryInterface, IDirectMusicInstrumentImpl_IUnknown_AddRef, IDirectMusicInstrumentImpl_IUnknown_Release }; /* IDirectMusicInstrumentImpl IDirectMusicInstrument part: */ static HRESULT WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_QueryInterface (LPDIRECTMUSICINSTRUMENT iface, REFIID riid, LPVOID *ppobj) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); return IDirectMusicInstrumentImpl_IUnknown_QueryInterface ((LPUNKNOWN)&This->UnknownVtbl, riid, ppobj); } static ULONG WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_AddRef (LPDIRECTMUSICINSTRUMENT iface) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); return IDirectMusicInstrumentImpl_IUnknown_AddRef ((LPUNKNOWN)&This->UnknownVtbl); } static ULONG WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_Release (LPDIRECTMUSICINSTRUMENT iface) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); return IDirectMusicInstrumentImpl_IUnknown_Release ((LPUNKNOWN)&This->UnknownVtbl); } static HRESULT WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_GetPatch (LPDIRECTMUSICINSTRUMENT iface, DWORD* pdwPatch) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); TRACE("(%p, %p)\n", This, pdwPatch); *pdwPatch = MIDILOCALE2Patch(&This->pHeader->Locale); return S_OK; } static HRESULT WINAPI IDirectMusicInstrumentImpl_IDirectMusicInstrument_SetPatch (LPDIRECTMUSICINSTRUMENT iface, DWORD dwPatch) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); TRACE("(%p, %d): stub\n", This, dwPatch); Patch2MIDILOCALE(dwPatch, &This->pHeader->Locale); return S_OK; } static const IDirectMusicInstrumentVtbl DirectMusicInstrument_Instrument_Vtbl = { IDirectMusicInstrumentImpl_IDirectMusicInstrument_QueryInterface, IDirectMusicInstrumentImpl_IDirectMusicInstrument_AddRef, IDirectMusicInstrumentImpl_IDirectMusicInstrument_Release, IDirectMusicInstrumentImpl_IDirectMusicInstrument_GetPatch, IDirectMusicInstrumentImpl_IDirectMusicInstrument_SetPatch }; /* for ClassFactory */ HRESULT WINAPI DMUSIC_CreateDirectMusicInstrumentImpl (LPCGUID lpcGUID, LPVOID* ppobj, LPUNKNOWN pUnkOuter) { IDirectMusicInstrumentImpl* dminst; dminst = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicInstrumentImpl)); if (NULL == dminst) { *ppobj = NULL; return E_OUTOFMEMORY; } dminst->UnknownVtbl = &DirectMusicInstrument_Unknown_Vtbl; dminst->InstrumentVtbl = &DirectMusicInstrument_Instrument_Vtbl; dminst->ref = 0; /* will be inited by QueryInterface */ return IDirectMusicInstrumentImpl_IUnknown_QueryInterface ((LPUNKNOWN)&dminst->UnknownVtbl, lpcGUID, ppobj); } static HRESULT read_from_stream(IStream *stream, void *data, ULONG size) { ULONG readed; HRESULT hr; hr = IStream_Read(stream, data, size, &readed); if(FAILED(hr)){ TRACE("IStream_Read failed: %08x\n", hr); return hr; } if(readed < size){ TRACE("Didn't read full chunk: %u < %u\n", readed, size); return E_FAIL; } return S_OK; } static inline DWORD subtract_bytes(DWORD len, DWORD bytes) { if(bytes > len){ TRACE("Apparent mismatch in chunk lengths? %u bytes remaining, %u bytes read\n", len, bytes); return 0; } return len - bytes; } static HRESULT load_instrument(IDirectMusicInstrumentImpl *This, IStream *stream, DWORD length) { HRESULT hr; FOURCC fourcc; DWORD bytes; LARGE_INTEGER move; while(length){ hr = read_from_stream(stream, &fourcc, sizeof(fourcc)); if(FAILED(hr)) return hr; hr = read_from_stream(stream, &bytes, sizeof(bytes)); if(FAILED(hr)) return hr; length = subtract_bytes(length, sizeof(fourcc) + sizeof(bytes)); switch(fourcc){ case FOURCC_INSH: TRACE("INSH chunk: %u bytes\n", bytes); hr = read_from_stream(stream, This->pHeader, sizeof(*This->pHeader)); if(FAILED(hr)) return hr; move.QuadPart = bytes - sizeof(*This->pHeader); hr = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL); if(FAILED(hr)){ WARN("IStream_Seek failed: %08x\n", hr); return hr; } length = subtract_bytes(length, bytes); break; case FOURCC_DLID: TRACE("DLID chunk: %u bytes\n", bytes); hr = read_from_stream(stream, This->pInstrumentID, sizeof(*This->pInstrumentID)); if(FAILED(hr)) return hr; move.QuadPart = bytes - sizeof(*This->pInstrumentID); hr = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL); if(FAILED(hr)){ WARN("IStream_Seek failed: %08x\n", hr); return hr; } length = subtract_bytes(length, bytes); break; default: TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(fourcc), bytes); move.QuadPart = bytes; hr = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL); if(FAILED(hr)){ WARN("IStream_Seek failed: %08x\n", hr); return hr; } length = subtract_bytes(length, bytes); break; } } return S_OK; } /* aux. function that completely loads instrument; my tests indicate that it's called somewhere around IDirectMusicCollection_GetInstrument */ HRESULT WINAPI IDirectMusicInstrumentImpl_Custom_Load (LPDIRECTMUSICINSTRUMENT iface, LPSTREAM stream) { ICOM_THIS_MULTI(IDirectMusicInstrumentImpl, InstrumentVtbl, iface); LARGE_INTEGER move; FOURCC fourcc; DWORD bytes; HRESULT hr; TRACE("(%p, %p, offset = %s)\n", This, stream, wine_dbgstr_longlong(This->liInstrumentPosition.QuadPart)); hr = IStream_Seek(stream, This->liInstrumentPosition, STREAM_SEEK_SET, NULL); if(FAILED(hr)){ WARN("IStream_Seek failed: %08x\n", hr); goto load_failure; } hr = read_from_stream(stream, &fourcc, sizeof(fourcc)); if(FAILED(hr)) goto load_failure; if(fourcc != FOURCC_LIST){ WARN("Loading failed: Expected LIST chunk, got: %s\n", debugstr_fourcc(fourcc)); goto load_failure; } hr = read_from_stream(stream, &bytes, sizeof(bytes)); if(FAILED(hr)) goto load_failure; TRACE("LIST chunk: %u bytes\n", bytes); while(1){ hr = read_from_stream(stream, &fourcc, sizeof(fourcc)); if(FAILED(hr)) goto load_failure; switch(fourcc){ case FOURCC_INS: TRACE("INS chunk: (no byte count)\n"); hr = load_instrument(This, stream, bytes - sizeof(FOURCC)); if(FAILED(hr)) goto load_failure; break; default: hr = read_from_stream(stream, &bytes, sizeof(bytes)); if(FAILED(hr)) goto load_failure; TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(fourcc), bytes); move.QuadPart = bytes; hr = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL); if(FAILED(hr)){ WARN("IStream_Seek failed: %08x\n", hr); return hr; } break; } } return S_OK; load_failure: return DMUS_E_UNSUPPORTED_STREAM; }