2 * IDirectMusicInstrument Implementation
4 * Copyright (C) 2003-2004 Rok Mandeljc
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "dmusic_private.h"
23 WINE_DEFAULT_DEBUG_CHANNEL(dmusic);
25 static const GUID IID_IDirectMusicInstrumentPRIVATE = { 0xbcb20080, 0xa40c, 0x11d1, { 0x86, 0xbc, 0x00, 0xc0, 0x4f, 0xbf, 0x8f, 0xef } };
27 /* IDirectMusicInstrument IUnknown part: */
28 static HRESULT WINAPI IDirectMusicInstrumentImpl_QueryInterface(LPDIRECTMUSICINSTRUMENT iface, REFIID riid, LPVOID *ret_iface)
30 TRACE("(%p)->(%s, %p)\n", iface, debugstr_dmguid(riid), ret_iface);
32 if (IsEqualIID(riid, &IID_IUnknown) ||
33 IsEqualIID(riid, &IID_IDirectMusicInstrument))
36 IDirectMusicInstrument_AddRef(iface);
39 else if (IsEqualIID(riid, &IID_IDirectMusicInstrumentPRIVATE))
41 /* it seems to me that this interface is only basic IUnknown, without any
42 * other inherited functions... *sigh* this is the worst scenario, since it means
43 * that whoever calls it knows the layout of original implementation table and therefore
44 * tries to get data by direct access... expect crashes
46 FIXME("*sigh*... requested private/unspecified interface\n");
49 IDirectMusicInstrument_AddRef(iface);
53 WARN("(%p)->(%s, %p): not found\n", iface, debugstr_dmguid(riid), ret_iface);
58 static ULONG WINAPI IDirectMusicInstrumentImpl_AddRef(LPDIRECTMUSICINSTRUMENT iface)
60 IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
61 ULONG ref = InterlockedIncrement(&This->ref);
63 TRACE("(%p)->(): new ref = %u\n", iface, ref);
70 static ULONG WINAPI IDirectMusicInstrumentImpl_Release(LPDIRECTMUSICINSTRUMENT iface)
72 IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
73 ULONG ref = InterlockedDecrement(&This->ref);
75 TRACE("(%p)->(): new ref = %u\n", iface, ref);
79 HeapFree(GetProcessHeap(), 0, This->regions);
80 HeapFree(GetProcessHeap(), 0, This);
83 DMUSIC_UnlockModule();
88 /* IDirectMusicInstrumentImpl IDirectMusicInstrument part: */
89 static HRESULT WINAPI IDirectMusicInstrumentImpl_GetPatch(LPDIRECTMUSICINSTRUMENT iface, DWORD* pdwPatch)
91 IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
93 TRACE("(%p)->(%p)\n", This, pdwPatch);
95 *pdwPatch = MIDILOCALE2Patch(&This->header.Locale);
100 static HRESULT WINAPI IDirectMusicInstrumentImpl_SetPatch(LPDIRECTMUSICINSTRUMENT iface, DWORD dwPatch)
102 IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
104 TRACE("(%p)->(%d): stub\n", This, dwPatch);
106 Patch2MIDILOCALE(dwPatch, &This->header.Locale);
111 static const IDirectMusicInstrumentVtbl DirectMusicInstrument_Vtbl =
113 IDirectMusicInstrumentImpl_QueryInterface,
114 IDirectMusicInstrumentImpl_AddRef,
115 IDirectMusicInstrumentImpl_Release,
116 IDirectMusicInstrumentImpl_GetPatch,
117 IDirectMusicInstrumentImpl_SetPatch
120 /* for ClassFactory */
121 HRESULT DMUSIC_CreateDirectMusicInstrumentImpl (LPCGUID lpcGUID, LPVOID* ppobj, LPUNKNOWN pUnkOuter) {
122 IDirectMusicInstrumentImpl* dminst;
124 dminst = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicInstrumentImpl));
125 if (NULL == dminst) {
127 return E_OUTOFMEMORY;
129 dminst->IDirectMusicInstrument_iface.lpVtbl = &DirectMusicInstrument_Vtbl;
130 dminst->ref = 0; /* will be inited by QueryInterface */
132 return IDirectMusicInstrument_QueryInterface(&dminst->IDirectMusicInstrument_iface, lpcGUID, ppobj);
135 static HRESULT read_from_stream(IStream *stream, void *data, ULONG size)
140 hr = IStream_Read(stream, data, size, &bytes_read);
142 TRACE("IStream_Read failed: %08x\n", hr);
145 if (bytes_read < size) {
146 TRACE("Didn't read full chunk: %u < %u\n", bytes_read, size);
153 static inline DWORD subtract_bytes(DWORD len, DWORD bytes)
156 TRACE("Apparent mismatch in chunk lengths? %u bytes remaining, %u bytes read\n", len, bytes);
162 static inline HRESULT advance_stream(IStream *stream, ULONG bytes)
167 move.QuadPart = bytes;
169 ret = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL);
171 WARN("IStream_Seek failed: %08x\n", ret);
176 static HRESULT load_region(IDirectMusicInstrumentImpl *This, IStream *stream, instrument_region *region, ULONG length)
179 DMUS_PRIVATE_CHUNK chunk;
181 TRACE("(%p, %p, %p, %u)\n", This, stream, region, length);
185 ret = read_from_stream(stream, &chunk, sizeof(chunk));
189 length = subtract_bytes(length, sizeof(chunk));
194 TRACE("RGNH chunk (region header): %u bytes\n", chunk.dwSize);
196 ret = read_from_stream(stream, ®ion->header, sizeof(region->header));
200 length = subtract_bytes(length, sizeof(region->header));
204 TRACE("WSMP chunk (wave sample): %u bytes\n", chunk.dwSize);
206 ret = read_from_stream(stream, ®ion->wave_sample, sizeof(region->wave_sample));
209 length = subtract_bytes(length, sizeof(region->wave_sample));
211 if (!(region->loop_present = (chunk.dwSize != sizeof(region->wave_sample))))
214 ret = read_from_stream(stream, ®ion->wave_loop, sizeof(region->wave_loop));
218 length = subtract_bytes(length, sizeof(region->wave_loop));
222 TRACE("WLNK chunk (wave link): %u bytes\n", chunk.dwSize);
224 ret = read_from_stream(stream, ®ion->wave_link, sizeof(region->wave_link));
228 length = subtract_bytes(length, sizeof(region->wave_link));
232 TRACE("Unknown chunk %s (skipping): %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
234 ret = advance_stream(stream, chunk.dwSize);
238 length = subtract_bytes(length, chunk.dwSize);
246 /* Function that loads all instrument data and which is called from IDirectMusicCollection_GetInstrument as in native */
247 HRESULT IDirectMusicInstrumentImpl_CustomLoad(IDirectMusicInstrument *iface, IStream *stream)
249 IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
251 DMUS_PRIVATE_CHUNK chunk;
253 ULONG length = This->length;
255 TRACE("(%p, %p): offset = 0x%s, length = %u)\n", This, stream, wine_dbgstr_longlong(This->liInstrumentPosition.QuadPart), This->length);
260 hr = IStream_Seek(stream, This->liInstrumentPosition, STREAM_SEEK_SET, NULL);
263 WARN("IStream_Seek failed: %08x\n", hr);
264 return DMUS_E_UNSUPPORTED_STREAM;
267 This->regions = HeapAlloc(GetProcessHeap(), 0, sizeof(*This->regions) * This->header.cRegions);
269 return E_OUTOFMEMORY;
273 hr = read_from_stream(stream, &chunk, sizeof(chunk));
277 length = subtract_bytes(length, sizeof(chunk) + chunk.dwSize);
283 TRACE("Chunk %s: %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
285 /* Instrument header and id are already set so just skip */
286 hr = advance_stream(stream, chunk.dwSize);
293 DWORD size = chunk.dwSize;
295 TRACE("LIST chunk: %u bytes\n", chunk.dwSize);
297 hr = read_from_stream(stream, &chunk.fccID, sizeof(chunk.fccID));
301 size = subtract_bytes(size, sizeof(chunk.fccID));
306 TRACE("LRGN chunk (regions list): %u bytes\n", size);
310 hr = read_from_stream(stream, &chunk, sizeof(chunk));
314 if (chunk.fccID != FOURCC_LIST)
316 TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
320 hr = read_from_stream(stream, &chunk.fccID, sizeof(chunk.fccID));
324 if (chunk.fccID == FOURCC_RGN)
326 TRACE("RGN chunk (region list): %u bytes\n", chunk.dwSize);
327 hr = load_region(This, stream, &This->regions[i++], chunk.dwSize - sizeof(chunk.fccID));
331 TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
332 hr = advance_stream(stream, chunk.dwSize - sizeof(chunk.fccID));
337 size = subtract_bytes(size, chunk.dwSize + sizeof(chunk));
342 TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
344 hr = advance_stream(stream, chunk.dwSize - sizeof(chunk.fccID));
348 size = subtract_bytes(size, chunk.dwSize - sizeof(chunk.fccID));
355 TRACE("Unknown chunk %s: %u bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
357 hr = advance_stream(stream, chunk.dwSize);
370 HeapFree(GetProcessHeap(), 0, This->regions);
371 This->regions = NULL;
373 return DMUS_E_UNSUPPORTED_STREAM;