2009-04-13 10:07:35 +02:00
|
|
|
/*
|
|
|
|
* Unit tests for MultiMedia Stream functions
|
|
|
|
*
|
2012-03-26 10:09:33 +02:00
|
|
|
* Copyright (C) 2009, 2012 Christian Costa
|
2009-04-13 10:07:35 +02:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define COBJMACROS
|
|
|
|
|
|
|
|
#include "wine/test.h"
|
2009-04-15 00:16:03 +02:00
|
|
|
#include "initguid.h"
|
2012-03-27 23:46:11 +02:00
|
|
|
#include "uuids.h"
|
2009-04-13 10:07:35 +02:00
|
|
|
#include "amstream.h"
|
2012-03-26 10:09:41 +02:00
|
|
|
#include "vfwmsgs.h"
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2015-06-22 21:39:41 +02:00
|
|
|
#define EXPECT_REF(obj,ref) _expect_ref((IUnknown*)obj, ref, __LINE__)
|
|
|
|
static void _expect_ref(IUnknown* obj, ULONG ref, int line)
|
|
|
|
{
|
2016-05-16 19:54:31 +02:00
|
|
|
ULONG rc;
|
|
|
|
IUnknown_AddRef(obj);
|
|
|
|
rc = IUnknown_Release(obj);
|
|
|
|
ok_(__FILE__,line)(rc == ref, "expected refcount %d, got %d\n", ref, rc);
|
2015-06-22 21:39:41 +02:00
|
|
|
}
|
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
static const WCHAR filenameW[] = {'t','e','s','t','.','a','v','i',0};
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2010-10-01 13:41:42 +02:00
|
|
|
static IDirectDraw7* pdd7;
|
|
|
|
static IDirectDrawSurface7* pdds7;
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2015-06-20 23:26:50 +02:00
|
|
|
static IAMMultiMediaStream *create_ammultimediastream(void)
|
2009-04-13 10:07:35 +02:00
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream *stream = NULL;
|
|
|
|
CoCreateInstance(&CLSID_AMMultiMediaStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMultiMediaStream,
|
|
|
|
(void**)&stream);
|
|
|
|
return stream;
|
2009-04-13 10:07:35 +02:00
|
|
|
}
|
|
|
|
|
2009-04-19 22:35:41 +02:00
|
|
|
static int create_directdraw(void)
|
2009-04-13 10:07:35 +02:00
|
|
|
{
|
2009-04-19 22:35:41 +02:00
|
|
|
HRESULT hr;
|
|
|
|
IDirectDraw* pdd = NULL;
|
|
|
|
DDSURFACEDESC2 ddsd;
|
|
|
|
|
|
|
|
hr = DirectDrawCreate(NULL, &pdd, NULL);
|
|
|
|
ok(hr==DD_OK, "DirectDrawCreate returned: %x\n", hr);
|
2009-04-22 10:22:05 +02:00
|
|
|
if (hr != DD_OK)
|
|
|
|
goto error;
|
2009-04-19 22:35:41 +02:00
|
|
|
|
|
|
|
hr = IDirectDraw_QueryInterface(pdd, &IID_IDirectDraw7, (LPVOID*)&pdd7);
|
|
|
|
ok(hr==DD_OK, "QueryInterface returned: %x\n", hr);
|
|
|
|
if (hr != DD_OK) goto error;
|
|
|
|
|
|
|
|
hr = IDirectDraw7_SetCooperativeLevel(pdd7, GetDesktopWindow(), DDSCL_NORMAL);
|
|
|
|
ok(hr==DD_OK, "SetCooperativeLevel returned: %x\n", hr);
|
|
|
|
|
|
|
|
ZeroMemory(&ddsd, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
ddsd.dwFlags = DDSD_CAPS;
|
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
|
|
|
|
hr = IDirectDraw7_CreateSurface(pdd7, &ddsd, &pdds7, NULL);
|
|
|
|
ok(hr==DD_OK, "CreateSurface returned: %x\n", hr);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (pdds7)
|
|
|
|
IDirectDrawSurface7_Release(pdds7);
|
|
|
|
if (pdd7)
|
|
|
|
IDirectDraw7_Release(pdd7);
|
|
|
|
if (pdd)
|
|
|
|
IDirectDraw_Release(pdd);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void release_directdraw(void)
|
|
|
|
{
|
|
|
|
IDirectDrawSurface7_Release(pdds7);
|
|
|
|
IDirectDraw7_Release(pdd7);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_openfile(void)
|
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream *pams;
|
2009-04-13 10:07:35 +02:00
|
|
|
HRESULT hr;
|
|
|
|
IGraphBuilder* pgraph;
|
|
|
|
|
2015-06-20 23:26:50 +02:00
|
|
|
if (!(pams = create_ammultimediastream()))
|
2009-04-19 22:35:41 +02:00
|
|
|
return;
|
|
|
|
|
2009-04-13 10:07:35 +02:00
|
|
|
hr = IAMMultiMediaStream_GetFilterGraph(pams, &pgraph);
|
2009-04-19 22:35:41 +02:00
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_GetFilterGraph returned: %x\n", hr);
|
2009-04-13 10:07:35 +02:00
|
|
|
ok(pgraph==NULL, "Filtergraph should not be created yet\n");
|
|
|
|
|
|
|
|
if (pgraph)
|
|
|
|
IGraphBuilder_Release(pgraph);
|
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
hr = IAMMultiMediaStream_OpenFile(pams, filenameW, 0);
|
2009-04-19 22:35:41 +02:00
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_OpenFile returned: %x\n", hr);
|
2009-04-13 10:07:35 +02:00
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_GetFilterGraph(pams, &pgraph);
|
2009-04-19 22:35:41 +02:00
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_GetFilterGraph returned: %x\n", hr);
|
2009-04-13 10:07:35 +02:00
|
|
|
ok(pgraph!=NULL, "Filtergraph should be created\n");
|
|
|
|
|
|
|
|
if (pgraph)
|
|
|
|
IGraphBuilder_Release(pgraph);
|
2009-04-19 22:35:41 +02:00
|
|
|
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream_Release(pams);
|
2009-04-19 22:35:41 +02:00
|
|
|
}
|
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
static void test_renderfile(void)
|
2009-04-19 22:35:41 +02:00
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream *pams;
|
2009-04-19 22:35:41 +02:00
|
|
|
HRESULT hr;
|
|
|
|
IMediaStream *pvidstream = NULL;
|
|
|
|
IDirectDrawMediaStream *pddstream = NULL;
|
|
|
|
IDirectDrawStreamSample *pddsample = NULL;
|
2015-06-22 21:39:41 +02:00
|
|
|
IDirectDrawSurface *surface;
|
|
|
|
RECT rect;
|
2009-04-19 22:35:41 +02:00
|
|
|
|
2015-06-20 23:26:50 +02:00
|
|
|
if (!(pams = create_ammultimediastream()))
|
2012-03-26 10:09:33 +02:00
|
|
|
return;
|
2009-04-19 22:35:41 +02:00
|
|
|
if (!create_directdraw())
|
2012-03-26 10:09:33 +02:00
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream_Release(pams);
|
2009-04-19 22:35:41 +02:00
|
|
|
return;
|
2012-03-26 10:09:33 +02:00
|
|
|
}
|
2009-04-19 22:35:41 +02:00
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_Initialize(pams, STREAMTYPE_READ, 0, NULL);
|
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_Initialize returned: %x\n", hr);
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, (IUnknown*)pdd7, &MSPID_PrimaryVideo, 0, NULL);
|
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryAudio, AMMSF_ADDDEFAULTRENDERER, NULL);
|
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
hr = IAMMultiMediaStream_OpenFile(pams, filenameW, 0);
|
2009-04-19 22:35:41 +02:00
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_OpenFile returned: %x\n", hr);
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &pvidstream);
|
|
|
|
ok(hr==S_OK, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
if (FAILED(hr)) goto error;
|
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(pvidstream, &IID_IDirectDrawMediaStream, (LPVOID*)&pddstream);
|
|
|
|
ok(hr==S_OK, "IMediaStream_QueryInterface returned: %x\n", hr);
|
|
|
|
if (FAILED(hr)) goto error;
|
|
|
|
|
|
|
|
hr = IDirectDrawMediaStream_CreateSample(pddstream, NULL, NULL, 0, &pddsample);
|
2012-04-26 07:58:05 +02:00
|
|
|
ok(hr == S_OK, "IDirectDrawMediaStream_CreateSample returned: %x\n", hr);
|
2009-04-19 22:35:41 +02:00
|
|
|
|
2015-06-22 21:39:41 +02:00
|
|
|
surface = NULL;
|
|
|
|
hr = IDirectDrawStreamSample_GetSurface(pddsample, &surface, &rect);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
ok(surface == NULL, "got %p\n", surface);
|
|
|
|
IDirectDrawStreamSample_Release(pddsample);
|
|
|
|
|
|
|
|
hr = IDirectDrawSurface7_QueryInterface(pdds7, &IID_IDirectDrawSurface, (void**)&surface);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
EXPECT_REF(surface, 1);
|
|
|
|
hr = IDirectDrawMediaStream_CreateSample(pddstream, surface, NULL, 0, &pddsample);
|
|
|
|
ok(hr == S_OK, "IDirectDrawMediaStream_CreateSample returned: %x\n", hr);
|
|
|
|
EXPECT_REF(surface, 2);
|
|
|
|
IDirectDrawStreamSample_Release(pddsample);
|
|
|
|
IDirectDrawSurface_Release(surface);
|
|
|
|
|
2009-04-19 22:35:41 +02:00
|
|
|
error:
|
|
|
|
if (pddstream)
|
|
|
|
IDirectDrawMediaStream_Release(pddstream);
|
|
|
|
if (pvidstream)
|
|
|
|
IMediaStream_Release(pvidstream);
|
|
|
|
|
|
|
|
release_directdraw();
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream_Release(pams);
|
2009-04-13 10:07:35 +02:00
|
|
|
}
|
|
|
|
|
2012-03-26 10:09:41 +02:00
|
|
|
static void test_media_streams(void)
|
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream *pams;
|
2012-03-26 10:09:41 +02:00
|
|
|
HRESULT hr;
|
|
|
|
IMediaStream *video_stream = NULL;
|
|
|
|
IMediaStream *audio_stream = NULL;
|
|
|
|
IMediaStream *dummy_stream;
|
2012-03-26 10:10:13 +02:00
|
|
|
IMediaStreamFilter* media_stream_filter = NULL;
|
2012-03-26 10:09:41 +02:00
|
|
|
|
2015-06-20 23:26:50 +02:00
|
|
|
if (!(pams = create_ammultimediastream()))
|
2012-03-26 10:09:41 +02:00
|
|
|
return;
|
|
|
|
if (!create_directdraw())
|
|
|
|
{
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream_Release(pams);
|
2012-03-26 10:09:41 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_Initialize(pams, STREAMTYPE_READ, 0, NULL);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_Initialize returned: %x\n", hr);
|
|
|
|
|
2012-05-15 10:15:32 +02:00
|
|
|
/* Retrieve media stream filter */
|
2012-03-26 10:10:13 +02:00
|
|
|
hr = IAMMultiMediaStream_GetFilter(pams, NULL);
|
|
|
|
ok(hr == E_POINTER, "IAMMultiMediaStream_GetFilter returned: %x\n", hr);
|
|
|
|
hr = IAMMultiMediaStream_GetFilter(pams, &media_stream_filter);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_GetFilter returned: %x\n", hr);
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
/* Verify behaviour with invalid purpose id */
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &IID_IUnknown, &dummy_stream);
|
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &IID_IUnknown, 0, NULL);
|
|
|
|
ok(hr == MS_E_PURPOSEID, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
|
|
|
|
2012-03-26 10:09:41 +02:00
|
|
|
/* Verify there is no video media stream */
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &video_stream);
|
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
|
|
|
|
/* Verify there is no default renderer for video stream */
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, (IUnknown*)pdd7, &MSPID_PrimaryVideo, AMMSF_ADDDEFAULTRENDERER, NULL);
|
2012-03-26 10:09:57 +02:00
|
|
|
ok(hr == MS_E_PURPOSEID, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &video_stream);
|
2012-03-26 10:09:57 +02:00
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryVideo, AMMSF_ADDDEFAULTRENDERER, NULL);
|
2012-03-26 10:09:57 +02:00
|
|
|
ok(hr == MS_E_PURPOSEID, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &video_stream);
|
2012-03-26 10:09:57 +02:00
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
|
|
|
|
/* Verify normal case for video stream */
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryVideo, 0, NULL);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &video_stream);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
|
2012-03-26 10:10:21 +02:00
|
|
|
/* Verify the video stream has been added to the media stream filter */
|
|
|
|
if (media_stream_filter)
|
|
|
|
{
|
|
|
|
hr = IMediaStreamFilter_GetMediaStream(media_stream_filter, &MSPID_PrimaryVideo, &dummy_stream);
|
2012-03-27 23:46:03 +02:00
|
|
|
ok(hr == S_OK, "IMediaStreamFilter_GetMediaStream returned: %x\n", hr);
|
|
|
|
ok(dummy_stream == video_stream, "Got wrong returned pointer %p, expected %p\n", dummy_stream, video_stream);
|
2012-03-26 10:10:21 +02:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
IMediaStream_Release(dummy_stream);
|
|
|
|
}
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
/* Check interfaces and samples for video */
|
|
|
|
if (video_stream)
|
|
|
|
{
|
|
|
|
IAMMediaStream* am_media_stream;
|
2016-04-30 03:56:00 +02:00
|
|
|
IMultiMediaStream *multi_media_stream;
|
2012-03-27 23:46:19 +02:00
|
|
|
IAudioMediaStream* audio_media_stream;
|
|
|
|
IDirectDrawMediaStream *ddraw_stream = NULL;
|
|
|
|
IDirectDrawStreamSample *ddraw_sample = NULL;
|
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(video_stream, &IID_IAMMediaStream, (LPVOID*)&am_media_stream);
|
2012-09-23 20:43:40 +02:00
|
|
|
ok(hr == S_OK, "IMediaStream_QueryInterface returned: %x\n", hr);
|
|
|
|
ok((void*)am_media_stream == (void*)video_stream, "Not same interface, got %p expected %p\n", am_media_stream, video_stream);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, NULL);
|
2016-04-30 03:56:38 +02:00
|
|
|
ok(hr == E_POINTER, "Expected E_POINTER, got %x\n", hr);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
multi_media_stream = (void *)0xdeadbeef;
|
|
|
|
hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, &multi_media_stream);
|
2016-04-30 03:56:38 +02:00
|
|
|
ok(hr == S_OK, "IAMMediaStream_GetMultiMediaStream returned: %x\n", hr);
|
|
|
|
ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream);
|
|
|
|
IMultiMediaStream_Release(multi_media_stream);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
IAMMediaStream_Release(am_media_stream);
|
2012-03-27 23:46:19 +02:00
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(video_stream, &IID_IAudioMediaStream, (LPVOID*)&audio_media_stream);
|
|
|
|
ok(hr == E_NOINTERFACE, "IMediaStream_QueryInterface returned: %x\n", hr);
|
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(video_stream, &IID_IDirectDrawMediaStream, (LPVOID*)&ddraw_stream);
|
|
|
|
ok(hr == S_OK, "IMediaStream_QueryInterface returned: %x\n", hr);
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
2012-05-01 10:21:20 +02:00
|
|
|
DDSURFACEDESC current_format, desired_format;
|
|
|
|
IDirectDrawPalette *palette;
|
|
|
|
DWORD flags;
|
|
|
|
|
|
|
|
hr = IDirectDrawMediaStream_GetFormat(ddraw_stream, ¤t_format, &palette, &desired_format, &flags);
|
|
|
|
ok(hr == MS_E_NOSTREAM, "IDirectDrawoMediaStream_GetFormat returned: %x\n", hr);
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &ddraw_sample);
|
2012-04-26 07:58:05 +02:00
|
|
|
ok(hr == S_OK, "IDirectDrawMediaStream_CreateSample returned: %x\n", hr);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
hr = IDirectDrawMediaStream_GetMultiMediaStream(ddraw_stream, NULL);
|
2016-04-30 03:57:50 +02:00
|
|
|
ok(hr == E_POINTER, "Expected E_POINTER, got %x\n", hr);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
multi_media_stream = (void *)0xdeadbeef;
|
|
|
|
hr = IDirectDrawMediaStream_GetMultiMediaStream(ddraw_stream, &multi_media_stream);
|
2016-04-30 03:57:50 +02:00
|
|
|
ok(hr == S_OK, "IDirectDrawMediaStream_GetMultiMediaStream returned: %x\n", hr);
|
|
|
|
ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream);
|
|
|
|
IMultiMediaStream_Release(multi_media_stream);
|
2012-03-27 23:46:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ddraw_sample)
|
2012-08-19 17:11:05 +02:00
|
|
|
IDirectDrawStreamSample_Release(ddraw_sample);
|
2012-03-27 23:46:19 +02:00
|
|
|
if (ddraw_stream)
|
|
|
|
IDirectDrawMediaStream_Release(ddraw_stream);
|
|
|
|
}
|
|
|
|
|
2012-03-26 10:09:41 +02:00
|
|
|
/* Verify there is no audio media stream */
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryAudio, &audio_stream);
|
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
|
|
|
|
/* Verify no stream is created when using the default renderer for audio stream */
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryAudio, AMMSF_ADDDEFAULTRENDERER, NULL);
|
|
|
|
ok((hr == S_OK) || (hr == VFW_E_NO_AUDIO_HARDWARE), "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
2012-03-27 23:46:11 +02:00
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
IGraphBuilder* filtergraph = NULL;
|
|
|
|
IBaseFilter* filter = NULL;
|
|
|
|
const WCHAR name[] = {'0','0','0','1',0};
|
|
|
|
CLSID clsid;
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_GetFilterGraph(pams, &filtergraph);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_GetFilterGraph returned: %x\n", hr);
|
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
hr = IGraphBuilder_FindFilterByName(filtergraph, name, &filter);
|
|
|
|
ok(hr == S_OK, "IGraphBuilder_FindFilterByName returned: %x\n", hr);
|
|
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
hr = IBaseFilter_GetClassID(filter, &clsid);
|
|
|
|
ok(hr == S_OK, "IGraphBuilder_FindFilterByName returned: %x\n", hr);
|
|
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
|
|
ok(IsEqualGUID(&clsid, &CLSID_DSoundRender), "Got wrong CLSID\n");
|
|
|
|
if (filter)
|
|
|
|
IBaseFilter_Release(filter);
|
|
|
|
if (filtergraph)
|
|
|
|
IGraphBuilder_Release(filtergraph);
|
|
|
|
}
|
2012-03-26 10:09:41 +02:00
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryAudio, &audio_stream);
|
2012-03-26 10:10:05 +02:00
|
|
|
ok(hr == MS_E_NOSTREAM, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
|
|
|
|
/* Verify a stream is created when no default renderer is used */
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryAudio, 0, NULL);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_AddMediaStream returned: %x\n", hr);
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryAudio, &audio_stream);
|
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
|
2012-03-26 10:10:21 +02:00
|
|
|
/* verify the audio stream has been added to the media stream filter */
|
|
|
|
if (media_stream_filter)
|
|
|
|
{
|
|
|
|
hr = IMediaStreamFilter_GetMediaStream(media_stream_filter, &MSPID_PrimaryAudio, &dummy_stream);
|
2012-03-27 23:46:03 +02:00
|
|
|
ok(hr == S_OK, "IAMMultiMediaStream_GetMediaStream returned: %x\n", hr);
|
|
|
|
ok(dummy_stream == audio_stream, "Got wrong returned pointer %p, expected %p\n", dummy_stream, audio_stream);
|
2012-03-26 10:10:21 +02:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
IMediaStream_Release(dummy_stream);
|
|
|
|
}
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
/* Check interfaces and samples for audio */
|
|
|
|
if (audio_stream)
|
2012-03-26 10:09:41 +02:00
|
|
|
{
|
2012-03-27 23:46:19 +02:00
|
|
|
IAMMediaStream* am_media_stream;
|
2016-04-30 03:56:00 +02:00
|
|
|
IMultiMediaStream *multi_media_stream;
|
2012-03-27 23:46:19 +02:00
|
|
|
IDirectDrawMediaStream* ddraw_stream = NULL;
|
|
|
|
IAudioMediaStream* audio_media_stream = NULL;
|
|
|
|
IAudioStreamSample *audio_sample = NULL;
|
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(audio_stream, &IID_IAMMediaStream, (LPVOID*)&am_media_stream);
|
2012-09-23 20:43:48 +02:00
|
|
|
ok(hr == S_OK, "IMediaStream_QueryInterface returned: %x\n", hr);
|
|
|
|
ok((void*)am_media_stream == (void*)audio_stream, "Not same interface, got %p expected %p\n", am_media_stream, audio_stream);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, NULL);
|
2016-04-30 03:58:55 +02:00
|
|
|
ok(hr == E_POINTER, "Expected E_POINTER, got %x\n", hr);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
multi_media_stream = (void *)0xdeadbeef;
|
|
|
|
hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, &multi_media_stream);
|
2016-04-30 03:58:55 +02:00
|
|
|
ok(hr == S_OK, "IAMMediaStream_GetMultiMediaStream returned: %x\n", hr);
|
|
|
|
ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream);
|
|
|
|
IMultiMediaStream_Release(multi_media_stream);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
IAMMediaStream_Release(am_media_stream);
|
2012-03-27 23:46:19 +02:00
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(audio_stream, &IID_IDirectDrawMediaStream, (LPVOID*)&ddraw_stream);
|
2012-04-23 21:44:28 +02:00
|
|
|
ok(hr == E_NOINTERFACE, "IMediaStream_QueryInterface returned: %x\n", hr);
|
2012-03-27 23:46:19 +02:00
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(audio_stream, &IID_IAudioMediaStream, (LPVOID*)&audio_media_stream);
|
2012-04-23 21:44:28 +02:00
|
|
|
ok(hr == S_OK, "IMediaStream_QueryInterface returned: %x\n", hr);
|
2012-03-26 10:09:41 +02:00
|
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
2012-03-27 23:46:19 +02:00
|
|
|
IAudioData* audio_data = NULL;
|
2012-05-01 10:21:20 +02:00
|
|
|
WAVEFORMATEX format;
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data);
|
2012-04-03 17:51:07 +02:00
|
|
|
ok(hr == S_OK, "CoCreateInstance returned: %x\n", hr);
|
2012-03-27 23:46:19 +02:00
|
|
|
|
2012-05-01 10:21:20 +02:00
|
|
|
hr = IAudioMediaStream_GetFormat(audio_media_stream, NULL);
|
|
|
|
ok(hr == E_POINTER, "IAudioMediaStream_GetFormat returned: %x\n", hr);
|
|
|
|
hr = IAudioMediaStream_GetFormat(audio_media_stream, &format);
|
|
|
|
ok(hr == MS_E_NOSTREAM, "IAudioMediaStream_GetFormat returned: %x\n", hr);
|
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
hr = IAudioMediaStream_CreateSample(audio_media_stream, NULL, 0, &audio_sample);
|
2012-04-26 07:58:14 +02:00
|
|
|
ok(hr == E_POINTER, "IAudioMediaStream_CreateSample returned: %x\n", hr);
|
2012-03-27 23:46:19 +02:00
|
|
|
hr = IAudioMediaStream_CreateSample(audio_media_stream, audio_data, 0, &audio_sample);
|
2012-04-26 07:58:14 +02:00
|
|
|
ok(hr == S_OK, "IAudioMediaStream_CreateSample returned: %x\n", hr);
|
2012-03-27 23:46:19 +02:00
|
|
|
|
2016-04-30 03:56:00 +02:00
|
|
|
hr = IAudioMediaStream_GetMultiMediaStream(audio_media_stream, NULL);
|
2016-04-30 03:59:42 +02:00
|
|
|
ok(hr == E_POINTER, "Expected E_POINTER, got %x\n", hr);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
|
|
|
multi_media_stream = (void *)0xdeadbeef;
|
|
|
|
hr = IAudioMediaStream_GetMultiMediaStream(audio_media_stream, &multi_media_stream);
|
2016-04-30 03:59:42 +02:00
|
|
|
ok(hr == S_OK, "IAudioMediaStream_GetMultiMediaStream returned: %x\n", hr);
|
|
|
|
ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream);
|
|
|
|
IMultiMediaStream_Release(multi_media_stream);
|
2016-04-30 03:56:00 +02:00
|
|
|
|
2012-03-27 23:46:19 +02:00
|
|
|
if (audio_data)
|
|
|
|
IAudioData_Release(audio_data);
|
|
|
|
if (audio_sample)
|
|
|
|
IAudioStreamSample_Release(audio_sample);
|
|
|
|
if (audio_media_stream)
|
|
|
|
IAudioMediaStream_Release(audio_media_stream);
|
2012-03-26 10:09:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-01 10:21:28 +02:00
|
|
|
if (media_stream_filter)
|
|
|
|
{
|
|
|
|
IEnumPins *enum_pins;
|
|
|
|
|
|
|
|
hr = IMediaStreamFilter_EnumPins(media_stream_filter, &enum_pins);
|
|
|
|
ok(hr == S_OK, "IBaseFilter_EnumPins returned: %x\n", hr);
|
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
IPin* pins[3] = { NULL, NULL, NULL };
|
|
|
|
ULONG nb_pins;
|
|
|
|
ULONG expected_nb_pins = audio_stream ? 2 : 1;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
hr = IEnumPins_Next(enum_pins, 3, pins, &nb_pins);
|
|
|
|
ok(SUCCEEDED(hr), "IEnumPins_Next returned: %x\n", hr);
|
|
|
|
ok(nb_pins == expected_nb_pins, "Number of pins is %u instead of %u\n", nb_pins, expected_nb_pins);
|
|
|
|
for (i = 0; i < min(nb_pins, expected_nb_pins); i++)
|
|
|
|
{
|
|
|
|
IEnumMediaTypes* enum_media_types;
|
|
|
|
AM_MEDIA_TYPE* media_types[10];
|
|
|
|
ULONG nb_media_types;
|
|
|
|
IPin* pin;
|
|
|
|
PIN_INFO info;
|
|
|
|
WCHAR id[40];
|
|
|
|
|
|
|
|
/* Pin name is "I{guid MSPID_PrimaryVideo or MSPID_PrimaryAudio}" */
|
|
|
|
id[0] = 'I';
|
|
|
|
StringFromGUID2(i ? &MSPID_PrimaryAudio : &MSPID_PrimaryVideo, id + 1, 40);
|
|
|
|
|
|
|
|
hr = IPin_ConnectedTo(pins[i], &pin);
|
|
|
|
ok(hr == VFW_E_NOT_CONNECTED, "IPin_ConnectedTo returned: %x\n", hr);
|
|
|
|
hr = IPin_QueryPinInfo(pins[i], &info);
|
|
|
|
ok(hr == S_OK, "IPin_QueryPinInfo returned: %x\n", hr);
|
|
|
|
IBaseFilter_Release(info.pFilter);
|
|
|
|
ok(info.dir == PINDIR_INPUT, "Pin direction is %u instead of %u\n", info.dir, PINDIR_INPUT);
|
|
|
|
ok(!lstrcmpW(info.achName, id), "Pin name is %s instead of %s\n", wine_dbgstr_w(info.achName), wine_dbgstr_w(id));
|
|
|
|
hr = IPin_EnumMediaTypes(pins[i], &enum_media_types);
|
|
|
|
ok(hr == S_OK, "IPin_EnumMediaTypes returned: %x\n", hr);
|
2015-10-12 21:15:56 +02:00
|
|
|
hr = IEnumMediaTypes_Next(enum_media_types, sizeof(media_types) / sizeof(media_types[0]), media_types, &nb_media_types);
|
2012-05-01 10:21:28 +02:00
|
|
|
ok(SUCCEEDED(hr), "IEnumMediaTypes_Next returned: %x\n", hr);
|
2015-10-12 21:15:56 +02:00
|
|
|
ok(nb_media_types > 0, "nb_media_types should be >0\n");
|
2012-05-01 10:21:28 +02:00
|
|
|
IEnumMediaTypes_Release(enum_media_types);
|
|
|
|
IPin_Release(pins[i]);
|
|
|
|
}
|
|
|
|
IEnumPins_Release(enum_pins);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-12 07:15:58 +02:00
|
|
|
/* Test open file with no filename */
|
|
|
|
hr = IAMMultiMediaStream_OpenFile(pams, NULL, 0);
|
|
|
|
ok(hr == E_POINTER, "IAMMultiMediaStream_OpenFile returned %x instead of %x\n", hr, E_POINTER);
|
|
|
|
|
2012-03-26 10:09:41 +02:00
|
|
|
if (video_stream)
|
|
|
|
IMediaStream_Release(video_stream);
|
|
|
|
if (audio_stream)
|
|
|
|
IMediaStream_Release(audio_stream);
|
2012-03-26 10:10:13 +02:00
|
|
|
if (media_stream_filter)
|
|
|
|
IMediaStreamFilter_Release(media_stream_filter);
|
2012-03-26 10:09:41 +02:00
|
|
|
|
|
|
|
release_directdraw();
|
2015-06-20 23:26:50 +02:00
|
|
|
IAMMultiMediaStream_Release(pams);
|
2012-03-26 10:09:41 +02:00
|
|
|
}
|
|
|
|
|
2015-06-22 21:39:41 +02:00
|
|
|
static void test_IDirectDrawStreamSample(void)
|
|
|
|
{
|
2015-06-23 22:24:46 +02:00
|
|
|
DDSURFACEDESC desc = { sizeof(desc) };
|
2015-06-22 21:39:41 +02:00
|
|
|
IAMMultiMediaStream *pams;
|
|
|
|
HRESULT hr;
|
|
|
|
IMediaStream *pvidstream = NULL;
|
|
|
|
IDirectDrawMediaStream *pddstream = NULL;
|
|
|
|
IDirectDrawStreamSample *pddsample = NULL;
|
2015-06-23 22:24:46 +02:00
|
|
|
IDirectDrawSurface7 *surface7;
|
2015-06-22 21:39:41 +02:00
|
|
|
IDirectDrawSurface *surface, *surface2;
|
2015-06-22 22:52:08 +02:00
|
|
|
IDirectDraw *ddraw, *ddraw2;
|
|
|
|
IDirectDraw7 *ddraw7;
|
2015-06-22 21:39:41 +02:00
|
|
|
RECT rect;
|
|
|
|
|
|
|
|
if (!(pams = create_ammultimediastream()))
|
|
|
|
return;
|
|
|
|
if (!create_directdraw())
|
|
|
|
{
|
|
|
|
IAMMultiMediaStream_Release(pams);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_Initialize(pams, STREAMTYPE_READ, 0, NULL);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_AddMediaStream(pams, (IUnknown*)pdd7, &MSPID_PrimaryVideo, 0, NULL);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &pvidstream);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
if (FAILED(hr)) goto error;
|
|
|
|
|
|
|
|
hr = IMediaStream_QueryInterface(pvidstream, &IID_IDirectDrawMediaStream, (LPVOID*)&pddstream);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
if (FAILED(hr)) goto error;
|
|
|
|
|
2015-06-22 22:52:08 +02:00
|
|
|
hr = IDirectDrawMediaStream_GetDirectDraw(pddstream, &ddraw);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IDirectDrawMediaStream_GetDirectDraw(pddstream, &ddraw2);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
ok(ddraw == ddraw2, "got %p, %p\n", ddraw, ddraw2);
|
|
|
|
|
|
|
|
hr = IDirectDraw_QueryInterface(ddraw, &IID_IDirectDraw7, (void**)&ddraw7);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
IDirectDraw7_Release(ddraw7);
|
|
|
|
|
|
|
|
IDirectDraw_Release(ddraw2);
|
|
|
|
IDirectDraw_Release(ddraw);
|
|
|
|
|
2015-06-22 21:39:41 +02:00
|
|
|
hr = IDirectDrawMediaStream_CreateSample(pddstream, NULL, NULL, 0, &pddsample);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
surface = NULL;
|
|
|
|
hr = IDirectDrawStreamSample_GetSurface(pddsample, &surface, &rect);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
ok(surface != NULL, "got %p\n", surface);
|
2015-06-23 22:24:46 +02:00
|
|
|
|
|
|
|
hr = IDirectDrawSurface_QueryInterface(surface, &IID_IDirectDrawSurface7, (void**)&surface7);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
IDirectDrawSurface7_Release(surface7);
|
|
|
|
|
|
|
|
hr = IDirectDrawSurface_GetSurfaceDesc(surface, &desc);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
ok(desc.dwWidth == 100, "width %d\n", desc.dwWidth);
|
|
|
|
ok(desc.dwHeight == 100, "height %d\n", desc.dwHeight);
|
|
|
|
ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "format flags %08x\n", desc.ddpfPixelFormat.dwFlags);
|
|
|
|
ok(desc.ddpfPixelFormat.dwRGBBitCount, "dwRGBBitCount %d\n", desc.ddpfPixelFormat.dwRGBBitCount);
|
2015-06-22 21:39:41 +02:00
|
|
|
IDirectDrawSurface_Release(surface);
|
|
|
|
IDirectDrawStreamSample_Release(pddsample);
|
|
|
|
|
|
|
|
hr = IDirectDrawSurface7_QueryInterface(pdds7, &IID_IDirectDrawSurface, (void**)&surface);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
EXPECT_REF(surface, 1);
|
|
|
|
hr = IDirectDrawMediaStream_CreateSample(pddstream, surface, NULL, 0, &pddsample);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
EXPECT_REF(surface, 2);
|
|
|
|
|
|
|
|
surface2 = NULL;
|
|
|
|
memset(&rect, 0, sizeof(rect));
|
|
|
|
hr = IDirectDrawStreamSample_GetSurface(pddsample, &surface2, &rect);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
ok(surface == surface2, "got %p\n", surface2);
|
|
|
|
ok(rect.right > 0 && rect.bottom > 0, "got %d, %d\n", rect.right, rect.bottom);
|
|
|
|
EXPECT_REF(surface, 3);
|
|
|
|
IDirectDrawSurface_Release(surface2);
|
|
|
|
|
|
|
|
hr = IDirectDrawStreamSample_GetSurface(pddsample, NULL, NULL);
|
|
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
|
|
|
|
IDirectDrawStreamSample_Release(pddsample);
|
|
|
|
IDirectDrawSurface_Release(surface);
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (pddstream)
|
|
|
|
IDirectDrawMediaStream_Release(pddstream);
|
|
|
|
if (pvidstream)
|
|
|
|
IMediaStream_Release(pvidstream);
|
|
|
|
|
|
|
|
release_directdraw();
|
|
|
|
IAMMultiMediaStream_Release(pams);
|
|
|
|
}
|
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
START_TEST(amstream)
|
2009-04-13 10:07:35 +02:00
|
|
|
{
|
2012-03-26 10:09:33 +02:00
|
|
|
HANDLE file;
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
CoInitializeEx(NULL, COINIT_MULTITHREADED);
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2012-03-26 10:09:41 +02:00
|
|
|
test_media_streams();
|
2015-06-22 21:39:41 +02:00
|
|
|
test_IDirectDrawStreamSample();
|
2012-03-26 10:09:41 +02:00
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
file = CreateFileW(filenameW, 0, 0, NULL, OPEN_EXISTING, 0, NULL);
|
|
|
|
if (file != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
CloseHandle(file);
|
2009-04-13 10:07:35 +02:00
|
|
|
|
2012-03-26 10:09:33 +02:00
|
|
|
test_openfile();
|
|
|
|
test_renderfile();
|
|
|
|
}
|
2009-04-13 10:07:35 +02:00
|
|
|
|
|
|
|
CoUninitialize();
|
|
|
|
}
|