585 lines
16 KiB
C
585 lines
16 KiB
C
/* GStreamer Audio Converter
|
|
*
|
|
* Copyright 2020 Derek Lesho
|
|
*
|
|
* This library 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 library 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 library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "gst_private.h"
|
|
|
|
#include "mfapi.h"
|
|
#include "mferror.h"
|
|
#include "ks.h"
|
|
#include "ksmedia.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mfplat);
|
|
|
|
struct audio_converter
|
|
{
|
|
IMFTransform IMFTransform_iface;
|
|
LONG refcount;
|
|
IMFMediaType *input_type;
|
|
IMFMediaType *output_type;
|
|
CRITICAL_SECTION cs;
|
|
};
|
|
|
|
static struct audio_converter *impl_audio_converter_from_IMFTransform(IMFTransform *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct audio_converter, IMFTransform_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_QueryInterface(IMFTransform *iface, REFIID riid, void **obj)
|
|
{
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), obj);
|
|
|
|
if (IsEqualIID(riid, &IID_IMFTransform) ||
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
{
|
|
*obj = iface;
|
|
IMFTransform_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("Unsupported %s.\n", debugstr_guid(riid));
|
|
*obj = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI audio_converter_AddRef(IMFTransform *iface)
|
|
{
|
|
struct audio_converter *transform = impl_audio_converter_from_IMFTransform(iface);
|
|
ULONG refcount = InterlockedIncrement(&transform->refcount);
|
|
|
|
TRACE("%p, refcount %lu.\n", iface, refcount);
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static ULONG WINAPI audio_converter_Release(IMFTransform *iface)
|
|
{
|
|
struct audio_converter *transform = impl_audio_converter_from_IMFTransform(iface);
|
|
ULONG refcount = InterlockedDecrement(&transform->refcount);
|
|
|
|
TRACE("%p, refcount %lu.\n", iface, refcount);
|
|
|
|
if (!refcount)
|
|
{
|
|
transform->cs.DebugInfo->Spare[0] = 0;
|
|
DeleteCriticalSection(&transform->cs);
|
|
free(transform);
|
|
}
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetStreamLimits(IMFTransform *iface, DWORD *input_minimum, DWORD *input_maximum,
|
|
DWORD *output_minimum, DWORD *output_maximum)
|
|
{
|
|
TRACE("%p, %p, %p, %p, %p.\n", iface, input_minimum, input_maximum, output_minimum, output_maximum);
|
|
|
|
*input_minimum = *input_maximum = *output_minimum = *output_maximum = 1;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetStreamCount(IMFTransform *iface, DWORD *inputs, DWORD *outputs)
|
|
{
|
|
TRACE("%p, %p, %p.\n", iface, inputs, outputs);
|
|
|
|
*inputs = *outputs = 1;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetStreamIDs(IMFTransform *iface, DWORD input_size, DWORD *inputs,
|
|
DWORD output_size, DWORD *outputs)
|
|
{
|
|
TRACE("%p, %lu, %p, %lu, %p.\n", iface, input_size, inputs, output_size, outputs);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetInputStreamInfo(IMFTransform *iface, DWORD id, MFT_INPUT_STREAM_INFO *info)
|
|
{
|
|
FIXME("%p, %lu, %p.\n", iface, id, info);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetOutputStreamInfo(IMFTransform *iface, DWORD id, MFT_OUTPUT_STREAM_INFO *info)
|
|
{
|
|
FIXME("%p. %lu, %p.\n", iface, id, info);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetAttributes(IMFTransform *iface, IMFAttributes **attributes)
|
|
{
|
|
FIXME("%p, %p.\n", iface, attributes);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetInputStreamAttributes(IMFTransform *iface, DWORD id,
|
|
IMFAttributes **attributes)
|
|
{
|
|
FIXME("%p, %lu, %p.\n", iface, id, attributes);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetOutputStreamAttributes(IMFTransform *iface, DWORD id,
|
|
IMFAttributes **attributes)
|
|
{
|
|
FIXME("%p, %lu, %p.\n", iface, id, attributes);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_DeleteInputStream(IMFTransform *iface, DWORD id)
|
|
{
|
|
TRACE("%p, %lu.\n", iface, id);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_AddInputStreams(IMFTransform *iface, DWORD streams, DWORD *ids)
|
|
{
|
|
TRACE("%p, %lu, %p.\n", iface, streams, ids);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetInputAvailableType(IMFTransform *iface, DWORD id, DWORD index,
|
|
IMFMediaType **type)
|
|
{
|
|
IMFMediaType *ret;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %lu, %lu, %p.\n", iface, id, index, type);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (index >= 2)
|
|
return MF_E_NO_MORE_TYPES;
|
|
|
|
if (FAILED(hr = MFCreateMediaType(&ret)))
|
|
return hr;
|
|
|
|
if (FAILED(hr = IMFMediaType_SetGUID(ret, &MF_MT_MAJOR_TYPE, &MFMediaType_Audio)))
|
|
{
|
|
IMFMediaType_Release(ret);
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED(hr = IMFMediaType_SetGUID(ret, &MF_MT_SUBTYPE, index ? &MFAudioFormat_Float : &MFAudioFormat_PCM)))
|
|
{
|
|
IMFMediaType_Release(ret);
|
|
return hr;
|
|
}
|
|
|
|
*type = ret;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetOutputAvailableType(IMFTransform *iface, DWORD id, DWORD index,
|
|
IMFMediaType **type)
|
|
{
|
|
IMFMediaType *output_type;
|
|
HRESULT hr;
|
|
|
|
static const struct
|
|
{
|
|
const GUID *subtype;
|
|
DWORD depth;
|
|
}
|
|
formats[] =
|
|
{
|
|
{&MFAudioFormat_PCM, 16},
|
|
{&MFAudioFormat_PCM, 24},
|
|
{&MFAudioFormat_PCM, 32},
|
|
{&MFAudioFormat_Float, 32},
|
|
};
|
|
|
|
static const DWORD rates[] = {44100, 48000};
|
|
static const DWORD channel_cnts[] = {1, 2, 6};
|
|
const GUID *subtype;
|
|
DWORD rate, channels, bps;
|
|
|
|
TRACE("%p, %lu, %lu, %p.\n", iface, id, index, type);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (index >= ARRAY_SIZE(formats) * 2/*rates*/ * 3/*layouts*/)
|
|
return MF_E_NO_MORE_TYPES;
|
|
|
|
if (FAILED(hr = MFCreateMediaType(&output_type)))
|
|
return hr;
|
|
|
|
subtype = formats[index / 6].subtype;
|
|
bps = formats[index / 6].depth;
|
|
rate = rates[index % 2];
|
|
channels = channel_cnts[(index / 2) % 3];
|
|
|
|
if (FAILED(hr = IMFMediaType_SetGUID(output_type, &MF_MT_MAJOR_TYPE, &MFMediaType_Audio)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetGUID(output_type, &MF_MT_SUBTYPE, subtype)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_SAMPLES_PER_SECOND, rate)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_NUM_CHANNELS, channels)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_BITS_PER_SAMPLE, bps)))
|
|
goto fail;
|
|
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_BLOCK_ALIGNMENT, channels * bps / 8)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_AVG_BYTES_PER_SECOND, rate * channels * bps / 8)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_AUDIO_CHANNEL_MASK,
|
|
channels == 1 ? KSAUDIO_SPEAKER_MONO :
|
|
channels == 2 ? KSAUDIO_SPEAKER_STEREO :
|
|
/*channels == 6*/ KSAUDIO_SPEAKER_5POINT1)))
|
|
goto fail;
|
|
if (FAILED(hr = IMFMediaType_SetUINT32(output_type, &MF_MT_ALL_SAMPLES_INDEPENDENT, TRUE)))
|
|
goto fail;
|
|
|
|
*type = output_type;
|
|
|
|
return S_OK;
|
|
fail:
|
|
IMFMediaType_Release(output_type);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_SetInputType(IMFTransform *iface, DWORD id, IMFMediaType *type, DWORD flags)
|
|
{
|
|
GUID major_type, subtype;
|
|
UINT32 unused;
|
|
HRESULT hr;
|
|
|
|
struct audio_converter *converter = impl_audio_converter_from_IMFTransform(iface);
|
|
|
|
TRACE("%p, %lu, %p, %#lx.\n", iface, id, type, flags);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (!type)
|
|
{
|
|
if (flags & MFT_SET_TYPE_TEST_ONLY)
|
|
return S_OK;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
if (converter->input_type)
|
|
{
|
|
IMFMediaType_Release(converter->input_type);
|
|
converter->input_type = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
if (FAILED(IMFMediaType_GetGUID(type, &MF_MT_MAJOR_TYPE, &major_type)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetGUID(type, &MF_MT_SUBTYPE, &subtype)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_SAMPLES_PER_SECOND, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_NUM_CHANNELS, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (IsEqualGUID(&subtype, &MFAudioFormat_PCM) && FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_BITS_PER_SAMPLE, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (!(IsEqualGUID(&major_type, &MFMediaType_Audio)))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (!IsEqualGUID(&subtype, &MFAudioFormat_PCM) && !IsEqualGUID(&subtype, &MFAudioFormat_Float))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (flags & MFT_SET_TYPE_TEST_ONLY)
|
|
return S_OK;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
hr = S_OK;
|
|
|
|
if (!converter->input_type)
|
|
hr = MFCreateMediaType(&converter->input_type);
|
|
|
|
if (SUCCEEDED(hr))
|
|
hr = IMFMediaType_CopyAllItems(type, (IMFAttributes *) converter->input_type);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
IMFMediaType_Release(converter->input_type);
|
|
converter->input_type = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_SetOutputType(IMFTransform *iface, DWORD id, IMFMediaType *type, DWORD flags)
|
|
{
|
|
struct audio_converter *converter = impl_audio_converter_from_IMFTransform(iface);
|
|
GUID major_type, subtype;
|
|
UINT32 unused;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %lu, %p, %#lx.\n", iface, id, type, flags);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (!converter->input_type)
|
|
return MF_E_TRANSFORM_TYPE_NOT_SET;
|
|
|
|
if (!type)
|
|
{
|
|
if (flags & MFT_SET_TYPE_TEST_ONLY)
|
|
return S_OK;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
if (converter->output_type)
|
|
{
|
|
IMFMediaType_Release(converter->output_type);
|
|
converter->output_type = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
if (FAILED(IMFMediaType_GetGUID(type, &MF_MT_MAJOR_TYPE, &major_type)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetGUID(type, &MF_MT_SUBTYPE, &subtype)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_NUM_CHANNELS, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (IsEqualGUID(&subtype, &MFAudioFormat_PCM) && FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_BITS_PER_SAMPLE, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
if (FAILED(IMFMediaType_GetUINT32(type, &MF_MT_AUDIO_SAMPLES_PER_SECOND, &unused)))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (!(IsEqualGUID(&major_type, &MFMediaType_Audio)))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (!IsEqualGUID(&subtype, &MFAudioFormat_PCM) && !IsEqualGUID(&subtype, &MFAudioFormat_Float))
|
|
return MF_E_INVALIDTYPE;
|
|
|
|
if (flags & MFT_SET_TYPE_TEST_ONLY)
|
|
return S_OK;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
hr = S_OK;
|
|
|
|
if (!converter->output_type)
|
|
hr = MFCreateMediaType(&converter->output_type);
|
|
|
|
if (SUCCEEDED(hr))
|
|
hr = IMFMediaType_CopyAllItems(type, (IMFAttributes *) converter->output_type);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
IMFMediaType_Release(converter->output_type);
|
|
converter->output_type = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetInputCurrentType(IMFTransform *iface, DWORD id, IMFMediaType **type)
|
|
{
|
|
struct audio_converter *converter = impl_audio_converter_from_IMFTransform(iface);
|
|
IMFMediaType *ret;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %lu, %p.\n", converter, id, type);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (FAILED(hr = MFCreateMediaType(&ret)))
|
|
return hr;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
if (converter->input_type)
|
|
hr = IMFMediaType_CopyAllItems(converter->input_type, (IMFAttributes *)ret);
|
|
else
|
|
hr = MF_E_TRANSFORM_TYPE_NOT_SET;
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
if (SUCCEEDED(hr))
|
|
*type = ret;
|
|
else
|
|
IMFMediaType_Release(ret);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetOutputCurrentType(IMFTransform *iface, DWORD id, IMFMediaType **type)
|
|
{
|
|
struct audio_converter *converter = impl_audio_converter_from_IMFTransform(iface);
|
|
IMFMediaType *ret;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %lu, %p.\n", converter, id, type);
|
|
|
|
if (id != 0)
|
|
return MF_E_INVALIDSTREAMNUMBER;
|
|
|
|
if (FAILED(hr = MFCreateMediaType(&ret)))
|
|
return hr;
|
|
|
|
EnterCriticalSection(&converter->cs);
|
|
|
|
if (converter->output_type)
|
|
hr = IMFMediaType_CopyAllItems(converter->output_type, (IMFAttributes *)ret);
|
|
else
|
|
hr = MF_E_TRANSFORM_TYPE_NOT_SET;
|
|
|
|
LeaveCriticalSection(&converter->cs);
|
|
|
|
if (SUCCEEDED(hr))
|
|
*type = ret;
|
|
else
|
|
IMFMediaType_Release(ret);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetInputStatus(IMFTransform *iface, DWORD id, DWORD *flags)
|
|
{
|
|
FIXME("%p, %lu, %p.\n", iface, id, flags);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_GetOutputStatus(IMFTransform *iface, DWORD *flags)
|
|
{
|
|
FIXME("%p, %p.\n", iface, flags);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_SetOutputBounds(IMFTransform *iface, LONGLONG lower, LONGLONG upper)
|
|
{
|
|
FIXME("%p, %s, %s.\n", iface, wine_dbgstr_longlong(lower), wine_dbgstr_longlong(upper));
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_ProcessEvent(IMFTransform *iface, DWORD id, IMFMediaEvent *event)
|
|
{
|
|
TRACE("%p, %lu, %p.\n", iface, id, event);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_ProcessMessage(IMFTransform *iface, MFT_MESSAGE_TYPE message, ULONG_PTR param)
|
|
{
|
|
TRACE("%p, %u, %Iu.\n", iface, message, param);
|
|
|
|
switch(message)
|
|
{
|
|
case MFT_MESSAGE_NOTIFY_BEGIN_STREAMING:
|
|
return S_OK;
|
|
default:
|
|
FIXME("Unhandled message type %x.\n", message);
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_ProcessInput(IMFTransform *iface, DWORD id, IMFSample *sample, DWORD flags)
|
|
{
|
|
FIXME("%p, %lu, %p, %#lx.\n", iface, id, sample, flags);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI audio_converter_ProcessOutput(IMFTransform *iface, DWORD flags, DWORD count,
|
|
MFT_OUTPUT_DATA_BUFFER *samples, DWORD *status)
|
|
{
|
|
FIXME("%p, %#lx, %lu, %p, %p.\n", iface, flags, count, samples, status);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IMFTransformVtbl audio_converter_vtbl =
|
|
{
|
|
audio_converter_QueryInterface,
|
|
audio_converter_AddRef,
|
|
audio_converter_Release,
|
|
audio_converter_GetStreamLimits,
|
|
audio_converter_GetStreamCount,
|
|
audio_converter_GetStreamIDs,
|
|
audio_converter_GetInputStreamInfo,
|
|
audio_converter_GetOutputStreamInfo,
|
|
audio_converter_GetAttributes,
|
|
audio_converter_GetInputStreamAttributes,
|
|
audio_converter_GetOutputStreamAttributes,
|
|
audio_converter_DeleteInputStream,
|
|
audio_converter_AddInputStreams,
|
|
audio_converter_GetInputAvailableType,
|
|
audio_converter_GetOutputAvailableType,
|
|
audio_converter_SetInputType,
|
|
audio_converter_SetOutputType,
|
|
audio_converter_GetInputCurrentType,
|
|
audio_converter_GetOutputCurrentType,
|
|
audio_converter_GetInputStatus,
|
|
audio_converter_GetOutputStatus,
|
|
audio_converter_SetOutputBounds,
|
|
audio_converter_ProcessEvent,
|
|
audio_converter_ProcessMessage,
|
|
audio_converter_ProcessInput,
|
|
audio_converter_ProcessOutput,
|
|
};
|
|
|
|
HRESULT audio_converter_create(REFIID riid, void **ret)
|
|
{
|
|
struct audio_converter *object;
|
|
|
|
TRACE("%s %p\n", debugstr_guid(riid), ret);
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
object->IMFTransform_iface.lpVtbl = &audio_converter_vtbl;
|
|
object->refcount = 1;
|
|
|
|
InitializeCriticalSection(&object->cs);
|
|
object->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": audio_converter_lock");
|
|
|
|
*ret = &object->IMFTransform_iface;
|
|
return S_OK;
|
|
}
|