446 lines
15 KiB
C
446 lines
15 KiB
C
/*
|
|
* 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"
|
|
#include "dmobject.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(dmusic);
|
|
|
|
static const GUID IID_IDirectMusicInstrumentPRIVATE = { 0xbcb20080, 0xa40c, 0x11d1, { 0x86, 0xbc, 0x00, 0xc0, 0x4f, 0xbf, 0x8f, 0xef } };
|
|
|
|
/* IDirectMusicInstrument IUnknown part: */
|
|
static HRESULT WINAPI IDirectMusicInstrumentImpl_QueryInterface(LPDIRECTMUSICINSTRUMENT iface, REFIID riid, LPVOID *ret_iface)
|
|
{
|
|
TRACE("(%p)->(%s, %p)\n", iface, debugstr_dmguid(riid), ret_iface);
|
|
|
|
if (IsEqualIID(riid, &IID_IUnknown) ||
|
|
IsEqualIID(riid, &IID_IDirectMusicInstrument))
|
|
{
|
|
*ret_iface = iface;
|
|
IDirectMusicInstrument_AddRef(iface);
|
|
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");
|
|
|
|
*ret_iface = iface;
|
|
IDirectMusicInstrument_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("(%p)->(%s, %p): not found\n", iface, debugstr_dmguid(riid), ret_iface);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI IDirectMusicInstrumentImpl_AddRef(LPDIRECTMUSICINSTRUMENT iface)
|
|
{
|
|
IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
|
|
ULONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p): new ref = %lu\n", iface, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI IDirectMusicInstrumentImpl_Release(LPDIRECTMUSICINSTRUMENT iface)
|
|
{
|
|
IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
|
|
ULONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p): new ref = %lu\n", iface, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
ULONG i;
|
|
|
|
HeapFree(GetProcessHeap(), 0, This->regions);
|
|
for (i = 0; i < This->nb_articulations; i++)
|
|
HeapFree(GetProcessHeap(), 0, This->articulations->connections);
|
|
HeapFree(GetProcessHeap(), 0, This->articulations);
|
|
HeapFree(GetProcessHeap(), 0, This);
|
|
DMUSIC_UnlockModule();
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
/* IDirectMusicInstrumentImpl IDirectMusicInstrument part: */
|
|
static HRESULT WINAPI IDirectMusicInstrumentImpl_GetPatch(LPDIRECTMUSICINSTRUMENT iface, DWORD* pdwPatch)
|
|
{
|
|
IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pdwPatch);
|
|
|
|
*pdwPatch = MIDILOCALE2Patch(&This->header.Locale);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI IDirectMusicInstrumentImpl_SetPatch(LPDIRECTMUSICINSTRUMENT iface, DWORD dwPatch)
|
|
{
|
|
IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
|
|
|
|
TRACE("(%p, %ld): stub\n", This, dwPatch);
|
|
|
|
Patch2MIDILOCALE(dwPatch, &This->header.Locale);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IDirectMusicInstrumentVtbl DirectMusicInstrument_Vtbl =
|
|
{
|
|
IDirectMusicInstrumentImpl_QueryInterface,
|
|
IDirectMusicInstrumentImpl_AddRef,
|
|
IDirectMusicInstrumentImpl_Release,
|
|
IDirectMusicInstrumentImpl_GetPatch,
|
|
IDirectMusicInstrumentImpl_SetPatch
|
|
};
|
|
|
|
/* for ClassFactory */
|
|
HRESULT DMUSIC_CreateDirectMusicInstrumentImpl (LPCGUID lpcGUID, LPVOID* ppobj, LPUNKNOWN pUnkOuter) {
|
|
IDirectMusicInstrumentImpl* dminst;
|
|
HRESULT hr;
|
|
|
|
dminst = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirectMusicInstrumentImpl));
|
|
if (NULL == dminst) {
|
|
*ppobj = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
dminst->IDirectMusicInstrument_iface.lpVtbl = &DirectMusicInstrument_Vtbl;
|
|
dminst->ref = 1;
|
|
|
|
DMUSIC_LockModule();
|
|
hr = IDirectMusicInstrument_QueryInterface(&dminst->IDirectMusicInstrument_iface, lpcGUID,
|
|
ppobj);
|
|
IDirectMusicInstrument_Release(&dminst->IDirectMusicInstrument_iface);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT read_from_stream(IStream *stream, void *data, ULONG size)
|
|
{
|
|
ULONG bytes_read;
|
|
HRESULT hr;
|
|
|
|
hr = IStream_Read(stream, data, size, &bytes_read);
|
|
if(FAILED(hr)){
|
|
TRACE("IStream_Read failed: %08lx\n", hr);
|
|
return hr;
|
|
}
|
|
if (bytes_read < size) {
|
|
TRACE("Didn't read full chunk: %lu < %lu\n", bytes_read, 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? %lu bytes remaining, %lu bytes read\n", len, bytes);
|
|
return 0;
|
|
}
|
|
return len - bytes;
|
|
}
|
|
|
|
static inline HRESULT advance_stream(IStream *stream, ULONG bytes)
|
|
{
|
|
LARGE_INTEGER move;
|
|
HRESULT ret;
|
|
|
|
move.QuadPart = bytes;
|
|
|
|
ret = IStream_Seek(stream, move, STREAM_SEEK_CUR, NULL);
|
|
if (FAILED(ret))
|
|
WARN("IStream_Seek failed: %08lx\n", ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static HRESULT load_region(IDirectMusicInstrumentImpl *This, IStream *stream, instrument_region *region, ULONG length)
|
|
{
|
|
HRESULT ret;
|
|
DMUS_PRIVATE_CHUNK chunk;
|
|
|
|
TRACE("(%p, %p, %p, %lu)\n", This, stream, region, length);
|
|
|
|
while (length)
|
|
{
|
|
ret = read_from_stream(stream, &chunk, sizeof(chunk));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
length = subtract_bytes(length, sizeof(chunk));
|
|
|
|
switch (chunk.fccID)
|
|
{
|
|
case FOURCC_RGNH:
|
|
TRACE("RGNH chunk (region header): %lu bytes\n", chunk.dwSize);
|
|
|
|
ret = read_from_stream(stream, ®ion->header, sizeof(region->header));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
length = subtract_bytes(length, sizeof(region->header));
|
|
break;
|
|
|
|
case FOURCC_WSMP:
|
|
TRACE("WSMP chunk (wave sample): %lu bytes\n", chunk.dwSize);
|
|
|
|
ret = read_from_stream(stream, ®ion->wave_sample, sizeof(region->wave_sample));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
length = subtract_bytes(length, sizeof(region->wave_sample));
|
|
|
|
if (!(region->loop_present = (chunk.dwSize != sizeof(region->wave_sample))))
|
|
break;
|
|
|
|
ret = read_from_stream(stream, ®ion->wave_loop, sizeof(region->wave_loop));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
length = subtract_bytes(length, sizeof(region->wave_loop));
|
|
break;
|
|
|
|
case FOURCC_WLNK:
|
|
TRACE("WLNK chunk (wave link): %lu bytes\n", chunk.dwSize);
|
|
|
|
ret = read_from_stream(stream, ®ion->wave_link, sizeof(region->wave_link));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
length = subtract_bytes(length, sizeof(region->wave_link));
|
|
break;
|
|
|
|
default:
|
|
TRACE("Unknown chunk %s (skipping): %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
|
|
ret = advance_stream(stream, chunk.dwSize);
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
length = subtract_bytes(length, chunk.dwSize);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT load_articulation(IDirectMusicInstrumentImpl *This, IStream *stream, ULONG length)
|
|
{
|
|
HRESULT ret;
|
|
instrument_articulation *articulation;
|
|
|
|
if (!This->articulations)
|
|
This->articulations = HeapAlloc(GetProcessHeap(), 0, sizeof(*This->articulations));
|
|
else
|
|
This->articulations = HeapReAlloc(GetProcessHeap(), 0, This->articulations, sizeof(*This->articulations) * (This->nb_articulations + 1));
|
|
if (!This->articulations)
|
|
return E_OUTOFMEMORY;
|
|
|
|
articulation = &This->articulations[This->nb_articulations];
|
|
|
|
ret = read_from_stream(stream, &articulation->connections_list, sizeof(CONNECTIONLIST));
|
|
if (FAILED(ret))
|
|
return ret;
|
|
|
|
articulation->connections = HeapAlloc(GetProcessHeap(), 0, sizeof(CONNECTION) * articulation->connections_list.cConnections);
|
|
if (!articulation->connections)
|
|
return E_OUTOFMEMORY;
|
|
|
|
ret = read_from_stream(stream, articulation->connections, sizeof(CONNECTION) * articulation->connections_list.cConnections);
|
|
if (FAILED(ret))
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, articulation->connections);
|
|
return ret;
|
|
}
|
|
|
|
subtract_bytes(length, sizeof(CONNECTIONLIST) + sizeof(CONNECTION) * articulation->connections_list.cConnections);
|
|
|
|
This->nb_articulations++;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/* Function that loads all instrument data and which is called from IDirectMusicCollection_GetInstrument as in native */
|
|
HRESULT IDirectMusicInstrumentImpl_CustomLoad(IDirectMusicInstrument *iface, IStream *stream)
|
|
{
|
|
IDirectMusicInstrumentImpl *This = impl_from_IDirectMusicInstrument(iface);
|
|
HRESULT hr;
|
|
DMUS_PRIVATE_CHUNK chunk;
|
|
ULONG i = 0;
|
|
ULONG length = This->length;
|
|
|
|
TRACE("(%p, %p): offset = 0x%s, length = %lu)\n", This, stream, wine_dbgstr_longlong(This->liInstrumentPosition.QuadPart), This->length);
|
|
|
|
if (This->loaded)
|
|
return S_OK;
|
|
|
|
hr = IStream_Seek(stream, This->liInstrumentPosition, STREAM_SEEK_SET, NULL);
|
|
if (FAILED(hr))
|
|
{
|
|
WARN("IStream_Seek failed: %08lx\n", hr);
|
|
return DMUS_E_UNSUPPORTED_STREAM;
|
|
}
|
|
|
|
This->regions = HeapAlloc(GetProcessHeap(), 0, sizeof(*This->regions) * This->header.cRegions);
|
|
if (!This->regions)
|
|
return E_OUTOFMEMORY;
|
|
|
|
while (length)
|
|
{
|
|
hr = read_from_stream(stream, &chunk, sizeof(chunk));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
length = subtract_bytes(length, sizeof(chunk) + chunk.dwSize);
|
|
|
|
switch (chunk.fccID)
|
|
{
|
|
case FOURCC_INSH:
|
|
case FOURCC_DLID:
|
|
TRACE("Chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
|
|
/* Instrument header and id are already set so just skip */
|
|
hr = advance_stream(stream, chunk.dwSize);
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
break;
|
|
|
|
case FOURCC_LIST: {
|
|
DWORD size = chunk.dwSize;
|
|
|
|
TRACE("LIST chunk: %lu bytes\n", chunk.dwSize);
|
|
|
|
hr = read_from_stream(stream, &chunk.fccID, sizeof(chunk.fccID));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
size = subtract_bytes(size, sizeof(chunk.fccID));
|
|
|
|
switch (chunk.fccID)
|
|
{
|
|
case FOURCC_LRGN:
|
|
TRACE("LRGN chunk (regions list): %lu bytes\n", size);
|
|
|
|
while (size)
|
|
{
|
|
hr = read_from_stream(stream, &chunk, sizeof(chunk));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
if (chunk.fccID != FOURCC_LIST)
|
|
{
|
|
TRACE("Unknown chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
goto error;
|
|
}
|
|
|
|
hr = read_from_stream(stream, &chunk.fccID, sizeof(chunk.fccID));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
if (chunk.fccID == FOURCC_RGN)
|
|
{
|
|
TRACE("RGN chunk (region): %lu bytes\n", chunk.dwSize);
|
|
hr = load_region(This, stream, &This->regions[i++], chunk.dwSize - sizeof(chunk.fccID));
|
|
}
|
|
else
|
|
{
|
|
TRACE("Unknown chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
hr = advance_stream(stream, chunk.dwSize - sizeof(chunk.fccID));
|
|
}
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
size = subtract_bytes(size, chunk.dwSize + sizeof(chunk));
|
|
}
|
|
break;
|
|
|
|
case FOURCC_LART:
|
|
TRACE("LART chunk (articulations list): %lu bytes\n", size);
|
|
|
|
while (size)
|
|
{
|
|
hr = read_from_stream(stream, &chunk, sizeof(chunk));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
if (chunk.fccID == FOURCC_ART1)
|
|
{
|
|
TRACE("ART1 chunk (level 1 articulation): %lu bytes\n", chunk.dwSize);
|
|
hr = load_articulation(This, stream, chunk.dwSize);
|
|
}
|
|
else
|
|
{
|
|
TRACE("Unknown chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
hr = advance_stream(stream, chunk.dwSize);
|
|
}
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
size = subtract_bytes(size, chunk.dwSize + sizeof(chunk));
|
|
}
|
|
break;
|
|
|
|
default:
|
|
TRACE("Unknown chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
|
|
hr = advance_stream(stream, chunk.dwSize - sizeof(chunk.fccID));
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
size = subtract_bytes(size, chunk.dwSize - sizeof(chunk.fccID));
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
TRACE("Unknown chunk %s: %lu bytes\n", debugstr_fourcc(chunk.fccID), chunk.dwSize);
|
|
|
|
hr = advance_stream(stream, chunk.dwSize);
|
|
if (FAILED(hr))
|
|
goto error;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
This->loaded = TRUE;
|
|
|
|
return S_OK;
|
|
|
|
error:
|
|
HeapFree(GetProcessHeap(), 0, This->regions);
|
|
This->regions = NULL;
|
|
|
|
return DMUS_E_UNSUPPORTED_STREAM;
|
|
}
|