wine/dlls/windows.media.speech/synthesizer.c
Bernhard Kölbl d81bb88366 windows.media.speech: Add stub ISpeechSynthesisStream iface.
Based on a patch by Connor McAdams <cmcadams@codeweavers.com>

Silences a warning about narrator being broken in Forza Horizon 5.

Signed-off-by: Bernhard Kölbl <besentv@gmail.com>
2022-06-14 22:04:55 +02:00

650 lines
20 KiB
C

/* WinRT Windows.Media.Speech implementation
*
* Copyright 2021 Rémi Bernon for CodeWeavers
*
* 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 "private.h"
#include "wine/debug.h"
WINE_DEFAULT_DEBUG_CHANNEL(speech);
/*
*
* IVectorView_VoiceInformation
*
*/
struct voice_information_vector
{
IVectorView_VoiceInformation IVectorView_VoiceInformation_iface;
LONG ref;
};
static inline struct voice_information_vector *impl_from_IVectorView_VoiceInformation( IVectorView_VoiceInformation *iface )
{
return CONTAINING_RECORD(iface, struct voice_information_vector, IVectorView_VoiceInformation_iface);
}
static HRESULT WINAPI vector_view_voice_information_QueryInterface( IVectorView_VoiceInformation *iface, REFIID iid, void **out )
{
struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface);
TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out);
if (IsEqualGUID(iid, &IID_IUnknown) ||
IsEqualGUID(iid, &IID_IInspectable) ||
IsEqualGUID(iid, &IID_IVectorView_VoiceInformation))
{
IInspectable_AddRef((*out = &impl->IVectorView_VoiceInformation_iface));
return S_OK;
}
WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
*out = NULL;
return E_NOINTERFACE;
}
static ULONG WINAPI vector_view_voice_information_AddRef( IVectorView_VoiceInformation *iface )
{
struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface);
ULONG ref = InterlockedIncrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
static ULONG WINAPI vector_view_voice_information_Release( IVectorView_VoiceInformation *iface )
{
struct voice_information_vector *impl = impl_from_IVectorView_VoiceInformation(iface);
ULONG ref = InterlockedDecrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
static HRESULT WINAPI vector_view_voice_information_GetIids( IVectorView_VoiceInformation *iface, ULONG *iid_count, IID **iids )
{
FIXME("iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids);
return E_NOTIMPL;
}
static HRESULT WINAPI vector_view_voice_information_GetRuntimeClassName( IVectorView_VoiceInformation *iface, HSTRING *class_name )
{
FIXME("iface %p, class_name %p stub!\n", iface, class_name);
return E_NOTIMPL;
}
static HRESULT WINAPI vector_view_voice_information_GetTrustLevel( IVectorView_VoiceInformation *iface, TrustLevel *trust_level )
{
FIXME("iface %p, trust_level %p stub!\n", iface, trust_level);
return E_NOTIMPL;
}
static HRESULT WINAPI vector_view_voice_information_GetAt( IVectorView_VoiceInformation *iface, UINT32 index, IVoiceInformation **value )
{
FIXME("iface %p, index %#x, value %p stub!\n", iface, index, value);
*value = NULL;
return E_BOUNDS;
}
static HRESULT WINAPI vector_view_voice_information_get_Size( IVectorView_VoiceInformation *iface, UINT32 *value )
{
FIXME("iface %p, value %p stub!\n", iface, value);
*value = 0;
return S_OK;
}
static HRESULT WINAPI vector_view_voice_information_IndexOf( IVectorView_VoiceInformation *iface,
IVoiceInformation *element, UINT32 *index, BOOLEAN *found )
{
FIXME("iface %p, element %p, index %p, found %p stub!\n", iface, element, index, found);
*index = 0;
*found = FALSE;
return S_OK;
}
static HRESULT WINAPI vector_view_voice_information_GetMany( IVectorView_VoiceInformation *iface, UINT32 start_index,
UINT32 items_size, IVoiceInformation **items, UINT *value )
{
FIXME("iface %p, start_index %#x, items %p, value %p stub!\n", iface, start_index, items, value);
*value = 0;
return S_OK;
}
static const struct IVectorView_VoiceInformationVtbl vector_view_voice_information_vtbl =
{
vector_view_voice_information_QueryInterface,
vector_view_voice_information_AddRef,
vector_view_voice_information_Release,
/* IInspectable methods */
vector_view_voice_information_GetIids,
vector_view_voice_information_GetRuntimeClassName,
vector_view_voice_information_GetTrustLevel,
/* IVectorView<VoiceInformation> methods */
vector_view_voice_information_GetAt,
vector_view_voice_information_get_Size,
vector_view_voice_information_IndexOf,
vector_view_voice_information_GetMany,
};
static struct voice_information_vector all_voices =
{
{&vector_view_voice_information_vtbl},
0
};
/*
*
* ISpeechSynthesisStream
*
*/
struct synthesis_stream
{
ISpeechSynthesisStream ISpeechSynthesisStream_iface;
LONG ref;
IVector_IMediaMarker *markers;
};
static inline struct synthesis_stream *impl_from_ISpeechSynthesisStream( ISpeechSynthesisStream *iface )
{
return CONTAINING_RECORD(iface, struct synthesis_stream, ISpeechSynthesisStream_iface);
}
HRESULT WINAPI synthesis_stream_QueryInterface( ISpeechSynthesisStream *iface, REFIID iid, void **out )
{
struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface);
TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out);
if (IsEqualGUID(iid, &IID_IUnknown) ||
IsEqualGUID(iid, &IID_IInspectable) ||
IsEqualGUID(iid, &IID_IAgileObject) ||
IsEqualGUID(iid, &IID_ISpeechSynthesisStream))
{
IInspectable_AddRef((*out = &impl->ISpeechSynthesisStream_iface));
return S_OK;
}
FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
*out = NULL;
return E_NOINTERFACE;
}
ULONG WINAPI synthesis_stream_AddRef( ISpeechSynthesisStream *iface )
{
struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface);
ULONG ref = InterlockedIncrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
ULONG WINAPI synthesis_stream_Release( ISpeechSynthesisStream *iface )
{
struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface);
ULONG ref = InterlockedDecrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
if (!ref)
free(impl);
return ref;
}
HRESULT WINAPI synthesis_stream_GetIids( ISpeechSynthesisStream *iface, ULONG *iid_count, IID **iids )
{
FIXME("iface %p, iid_count %p, iids %p stub.\n", iface, iid_count, iids);
return E_NOTIMPL;
}
HRESULT WINAPI synthesis_stream_GetRuntimeClassName( ISpeechSynthesisStream *iface, HSTRING *class_name )
{
FIXME("iface %p, class_name %p stub.\n", iface, class_name);
return E_NOTIMPL;
}
HRESULT WINAPI synthesis_stream_GetTrustLevel( ISpeechSynthesisStream *iface, TrustLevel *trust_level )
{
FIXME("iface %p, trust_level %p stub.\n", iface, trust_level);
return E_NOTIMPL;
}
HRESULT WINAPI synthesis_stream_get_Markers( ISpeechSynthesisStream *iface, IVectorView_IMediaMarker **value )
{
struct synthesis_stream *impl = impl_from_ISpeechSynthesisStream(iface);
FIXME("iface %p, value %p stub!\n", iface, value);
return IVector_IMediaMarker_GetView(impl->markers, value);
}
static const struct ISpeechSynthesisStreamVtbl synthesis_stream_vtbl =
{
/* IUnknown methods */
synthesis_stream_QueryInterface,
synthesis_stream_AddRef,
synthesis_stream_Release,
/* IInspectable methods */
synthesis_stream_GetIids,
synthesis_stream_GetRuntimeClassName,
synthesis_stream_GetTrustLevel,
/* ISpeechSynthesisStream methods */
synthesis_stream_get_Markers
};
static HRESULT synthesis_stream_create( ISpeechSynthesisStream **out )
{
struct synthesis_stream *impl;
struct vector_iids markers_iids =
{
.iterable = &IID_IIterable_IMediaMarker,
.iterator = &IID_IIterator_IMediaMarker,
.vector = &IID_IVector_IMediaMarker,
.view = &IID_IVectorView_IMediaMarker,
};
HRESULT hr;
TRACE("out %p.\n", out);
if (!(impl = calloc(1, sizeof(*impl))))
{
*out = NULL;
return E_OUTOFMEMORY;
}
impl->ISpeechSynthesisStream_iface.lpVtbl = &synthesis_stream_vtbl;
impl->ref = 1;
if (FAILED(hr = vector_inspectable_create(&markers_iids, (IVector_IInspectable**)&impl->markers)))
goto error;
TRACE("created ISpeechSynthesisStream %p.\n", impl);
*out = &impl->ISpeechSynthesisStream_iface;
return S_OK;
error:
free(impl);
return hr;
}
/*
*
* SpeechSynthesizer runtimeclass
*
*/
struct synthesizer
{
ISpeechSynthesizer ISpeechSynthesizer_iface;
ISpeechSynthesizer2 ISpeechSynthesizer2_iface;
IClosable IClosable_iface;
LONG ref;
};
/*
*
* ISpeechSynthesizer for SpeechSynthesizer runtimeclass
*
*/
static inline struct synthesizer *impl_from_ISpeechSynthesizer( ISpeechSynthesizer *iface )
{
return CONTAINING_RECORD(iface, struct synthesizer, ISpeechSynthesizer_iface);
}
static HRESULT WINAPI synthesizer_QueryInterface( ISpeechSynthesizer *iface, REFIID iid, void **out )
{
struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface);
TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out);
if (IsEqualGUID(iid, &IID_IUnknown) ||
IsEqualGUID(iid, &IID_IInspectable) ||
IsEqualGUID(iid, &IID_ISpeechSynthesizer))
{
IInspectable_AddRef((*out = &impl->ISpeechSynthesizer_iface));
return S_OK;
}
if (IsEqualGUID(iid, &IID_ISpeechSynthesizer2))
{
IInspectable_AddRef((*out = &impl->ISpeechSynthesizer2_iface));
return S_OK;
}
if (IsEqualGUID(iid, &IID_IClosable))
{
IInspectable_AddRef((*out = &impl->IClosable_iface));
return S_OK;
}
FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
*out = NULL;
return E_NOINTERFACE;
}
static ULONG WINAPI synthesizer_AddRef( ISpeechSynthesizer *iface )
{
struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface);
ULONG ref = InterlockedIncrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
static ULONG WINAPI synthesizer_Release( ISpeechSynthesizer *iface )
{
struct synthesizer *impl = impl_from_ISpeechSynthesizer(iface);
ULONG ref = InterlockedDecrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
if (!ref)
free(impl);
return ref;
}
static HRESULT WINAPI synthesizer_GetIids( ISpeechSynthesizer *iface, ULONG *iid_count, IID **iids )
{
FIXME("iface %p, iid_count %p, iids %p stub.\n", iface, iid_count, iids);
return E_NOTIMPL;
}
static HRESULT WINAPI synthesizer_GetRuntimeClassName( ISpeechSynthesizer *iface, HSTRING *class_name )
{
FIXME("iface %p, class_name %p stub.\n", iface, class_name);
return E_NOTIMPL;
}
static HRESULT WINAPI synthesizer_GetTrustLevel( ISpeechSynthesizer *iface, TrustLevel *trust_level )
{
FIXME("iface %p, trust_level %p stub.\n", iface, trust_level);
return E_NOTIMPL;
}
static HRESULT CALLBACK text_to_stream_operation( IInspectable *invoker, IInspectable **result )
{
return synthesis_stream_create((ISpeechSynthesisStream **)result);
}
static HRESULT WINAPI synthesizer_SynthesizeTextToStreamAsync( ISpeechSynthesizer *iface, HSTRING text,
IAsyncOperation_SpeechSynthesisStream **operation )
{
TRACE("iface %p, text %p, operation %p.\n", iface, text, operation);
return async_operation_inspectable_create(&IID_IAsyncOperation_SpeechSynthesisStream, NULL,
text_to_stream_operation, (IAsyncOperation_IInspectable **)operation);
}
static HRESULT CALLBACK ssml_to_stream_operation( IInspectable *invoker, IInspectable **result )
{
return synthesis_stream_create((ISpeechSynthesisStream **)result);
}
static HRESULT WINAPI synthesizer_SynthesizeSsmlToStreamAsync( ISpeechSynthesizer *iface, HSTRING ssml,
IAsyncOperation_SpeechSynthesisStream **operation )
{
TRACE("iface %p, ssml %p, operation %p.\n", iface, ssml, operation);
return async_operation_inspectable_create(&IID_IAsyncOperation_SpeechSynthesisStream, NULL,
ssml_to_stream_operation, (IAsyncOperation_IInspectable **)operation);
}
static HRESULT WINAPI synthesizer_put_Voice( ISpeechSynthesizer *iface, IVoiceInformation *value )
{
FIXME("iface %p, value %p stub.\n", iface, value);
return E_NOTIMPL;
}
static HRESULT WINAPI synthesizer_get_Voice( ISpeechSynthesizer *iface, IVoiceInformation **value )
{
FIXME("iface %p, value %p stub.\n", iface, value);
return E_NOTIMPL;
}
static const struct ISpeechSynthesizerVtbl synthesizer_vtbl =
{
/* IUnknown methods */
synthesizer_QueryInterface,
synthesizer_AddRef,
synthesizer_Release,
/* IInspectable methods */
synthesizer_GetIids,
synthesizer_GetRuntimeClassName,
synthesizer_GetTrustLevel,
/* ISpeechSynthesizer methods */
synthesizer_SynthesizeTextToStreamAsync,
synthesizer_SynthesizeSsmlToStreamAsync,
synthesizer_put_Voice,
synthesizer_get_Voice,
};
/*
*
* ISpeechSynthesizer2 for SpeechSynthesizer runtimeclass
*
*/
DEFINE_IINSPECTABLE(synthesizer2, ISpeechSynthesizer2, struct synthesizer, ISpeechSynthesizer_iface)
static HRESULT WINAPI synthesizer2_get_Options( ISpeechSynthesizer2 *iface, ISpeechSynthesizerOptions **value )
{
FIXME("iface %p, value %p stub.\n", iface, value);
return E_NOTIMPL;
}
static const struct ISpeechSynthesizer2Vtbl synthesizer2_vtbl =
{
/* IUnknown methods */
synthesizer2_QueryInterface,
synthesizer2_AddRef,
synthesizer2_Release,
/* IInspectable methods */
synthesizer2_GetIids,
synthesizer2_GetRuntimeClassName,
synthesizer2_GetTrustLevel,
/* ISpeechSynthesizer2 methods */
synthesizer2_get_Options,
};
/*
*
* IClosable for SpeechSynthesizer runtimeclass
*
*/
DEFINE_IINSPECTABLE(closable, IClosable, struct synthesizer, ISpeechSynthesizer_iface)
static HRESULT WINAPI closable_Close( IClosable *iface )
{
FIXME("iface %p stub.\n", iface);
return E_NOTIMPL;
}
static const struct IClosableVtbl closable_vtbl =
{
/* IUnknown methods */
closable_QueryInterface,
closable_AddRef,
closable_Release,
/* IInspectable methods */
closable_GetIids,
closable_GetRuntimeClassName,
closable_GetTrustLevel,
/* IClosable methods */
closable_Close,
};
/*
*
* Static interfaces for SpeechSynthesizer runtimeclass
*
*/
struct synthesizer_statics
{
IActivationFactory IActivationFactory_iface;
IInstalledVoicesStatic IInstalledVoicesStatic_iface;
LONG ref;
};
/*
*
* IActivationFactory for SpeechSynthesizer runtimeclass
*
*/
static inline struct synthesizer_statics *impl_from_IActivationFactory( IActivationFactory *iface )
{
return CONTAINING_RECORD(iface, struct synthesizer_statics, IActivationFactory_iface);
}
static HRESULT WINAPI factory_QueryInterface( IActivationFactory *iface, REFIID iid, void **out )
{
struct synthesizer_statics *impl = impl_from_IActivationFactory(iface);
TRACE("iface %p, iid %s, out %p stub!\n", iface, debugstr_guid(iid), out);
if (IsEqualGUID(iid, &IID_IUnknown) ||
IsEqualGUID(iid, &IID_IInspectable) ||
IsEqualGUID(iid, &IID_IAgileObject) ||
IsEqualGUID(iid, &IID_IActivationFactory))
{
IInspectable_AddRef((*out = &impl->IActivationFactory_iface));
return S_OK;
}
if (IsEqualGUID(iid, &IID_IInstalledVoicesStatic))
{
IInspectable_AddRef((*out = &impl->IInstalledVoicesStatic_iface));
return S_OK;
}
FIXME("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
*out = NULL;
return E_NOINTERFACE;
}
static ULONG WINAPI factory_AddRef( IActivationFactory *iface )
{
struct synthesizer_statics *impl = impl_from_IActivationFactory(iface);
ULONG ref = InterlockedIncrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
static ULONG WINAPI factory_Release( IActivationFactory *iface )
{
struct synthesizer_statics *impl = impl_from_IActivationFactory(iface);
ULONG ref = InterlockedDecrement(&impl->ref);
TRACE("iface %p, ref %lu.\n", iface, ref);
return ref;
}
static HRESULT WINAPI factory_GetIids( IActivationFactory *iface, ULONG *iid_count, IID **iids )
{
FIXME("iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids);
return E_NOTIMPL;
}
static HRESULT WINAPI factory_GetRuntimeClassName( IActivationFactory *iface, HSTRING *class_name )
{
FIXME("iface %p, class_name %p stub!\n", iface, class_name);
return E_NOTIMPL;
}
static HRESULT WINAPI factory_GetTrustLevel( IActivationFactory *iface, TrustLevel *trust_level )
{
FIXME("iface %p, trust_level %p stub!\n", iface, trust_level);
return E_NOTIMPL;
}
static HRESULT WINAPI factory_ActivateInstance( IActivationFactory *iface, IInspectable **instance )
{
struct synthesizer *impl;
TRACE("iface %p, instance %p.\n", iface, instance);
if (!(impl = calloc(1, sizeof(*impl))))
{
*instance = NULL;
return E_OUTOFMEMORY;
}
impl->ISpeechSynthesizer_iface.lpVtbl = &synthesizer_vtbl;
impl->ISpeechSynthesizer2_iface.lpVtbl = &synthesizer2_vtbl;
impl->IClosable_iface.lpVtbl = &closable_vtbl;
impl->ref = 1;
*instance = (IInspectable *)&impl->ISpeechSynthesizer_iface;
return S_OK;
}
static const struct IActivationFactoryVtbl factory_vtbl =
{
factory_QueryInterface,
factory_AddRef,
factory_Release,
/* IInspectable methods */
factory_GetIids,
factory_GetRuntimeClassName,
factory_GetTrustLevel,
/* IActivationFactory methods */
factory_ActivateInstance,
};
/*
*
* IInstalledVoicesStatic for SpeechSynthesizer runtimeclass
*
*/
DEFINE_IINSPECTABLE(installed_voices_static, IInstalledVoicesStatic, struct synthesizer_statics, IActivationFactory_iface)
static HRESULT WINAPI installed_voices_static_get_AllVoices( IInstalledVoicesStatic *iface, IVectorView_VoiceInformation **value )
{
TRACE("iface %p, value %p.\n", iface, value);
*value = &all_voices.IVectorView_VoiceInformation_iface;
IVectorView_VoiceInformation_AddRef(*value);
return S_OK;
}
static HRESULT WINAPI installed_voices_static_get_DefaultVoice( IInstalledVoicesStatic *iface, IVoiceInformation **value )
{
FIXME("iface %p, value %p stub!\n", iface, value);
return E_NOTIMPL;
}
static const struct IInstalledVoicesStaticVtbl installed_voices_static_vtbl =
{
installed_voices_static_QueryInterface,
installed_voices_static_AddRef,
installed_voices_static_Release,
/* IInspectable methods */
installed_voices_static_GetIids,
installed_voices_static_GetRuntimeClassName,
installed_voices_static_GetTrustLevel,
/* IInstalledVoicesStatic methods */
installed_voices_static_get_AllVoices,
installed_voices_static_get_DefaultVoice,
};
static struct synthesizer_statics synthesizer_statics =
{
.IActivationFactory_iface = {&factory_vtbl},
.IInstalledVoicesStatic_iface = {&installed_voices_static_vtbl},
.ref = 1
};
IActivationFactory *synthesizer_factory = &synthesizer_statics.IActivationFactory_iface;