/* * Unit tests for MultiMedia Stream functions * * Copyright (C) 2009, 2012 Christian Costa * * 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 #define NONAMELESSUNION #define WINE_NO_NAMELESS_EXTENSION #include "wine/test.h" #include "dshow.h" #include "amstream.h" #include "mmreg.h" #include "ks.h" #include "initguid.h" #include "ksmedia.h" #include "dvdmedia.h" #include "wmcodecdsp.h" #include "wine/strmbase.h" static const WAVEFORMATEX audio_format = { .wFormatTag = WAVE_FORMAT_PCM, .nChannels = 1, .nSamplesPerSec = 11025, .wBitsPerSample = 16, .nBlockAlign = 2, .nAvgBytesPerSec = 2 * 11025, }; static const AM_MEDIA_TYPE audio_mt = { /* MEDIATYPE_Audio, MEDIASUBTYPE_PCM, FORMAT_WaveFormatEx */ .majortype = {0x73647561, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0x00000001, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .formattype = {0x05589f81, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(WAVEFORMATEX), .pbFormat = (BYTE *)&audio_format, }; static const VIDEOINFO rgb8_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 8, .bmiHeader.biCompression = BI_RGB, }; static const VIDEOINFO rgb555_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 16, .bmiHeader.biCompression = BI_RGB, }; static const VIDEOINFO rgb565_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 16, .bmiHeader.biCompression = BI_BITFIELDS, .u.dwBitMasks = {0xf800, 0x07e0, 0x001f}, }; static const VIDEOINFO rgb24_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 24, .bmiHeader.biCompression = BI_RGB, }; static const VIDEOINFO rgb32_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 32, .bmiHeader.biCompression = BI_RGB, }; static const AM_MEDIA_TYPE rgb8_mt = { /* MEDIATYPE_Video, MEDIASUBTYPE_RGB8, FORMAT_VideoInfo */ .majortype = {0x73646976, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0xe436eb7a, 0x524f, 0x11ce, {0x9f, 0x53, 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70}}, .formattype = {0x05589f80, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(VIDEOINFO), .pbFormat = (BYTE *)&rgb8_video_info, }; static const AM_MEDIA_TYPE rgb555_mt = { /* MEDIATYPE_Video, MEDIASUBTYPE_RGB555, FORMAT_VideoInfo */ .majortype = {0x73646976, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0xe436eb7c, 0x524f, 0x11ce, {0x9f, 0x53, 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70}}, .formattype = {0x05589f80, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(VIDEOINFO), .pbFormat = (BYTE *)&rgb555_video_info, }; static const AM_MEDIA_TYPE rgb565_mt = { /* MEDIATYPE_Video, MEDIASUBTYPE_RGB565, FORMAT_VideoInfo */ .majortype = {0x73646976, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0xe436eb7b, 0x524f, 0x11ce, {0x9f, 0x53, 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70}}, .formattype = {0x05589f80, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(VIDEOINFO), .pbFormat = (BYTE *)&rgb565_video_info, }; static const AM_MEDIA_TYPE rgb24_mt = { /* MEDIATYPE_Video, MEDIASUBTYPE_RGB24, FORMAT_VideoInfo */ .majortype = {0x73646976, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0xe436eb7d, 0x524f, 0x11ce, {0x9f, 0x53, 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70}}, .formattype = {0x05589f80, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(VIDEOINFO), .pbFormat = (BYTE *)&rgb24_video_info, }; static const AM_MEDIA_TYPE rgb32_mt = { /* MEDIATYPE_Video, MEDIASUBTYPE_RGB32, FORMAT_VideoInfo */ .majortype = {0x73646976, 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}}, .subtype = {0xe436eb7e, 0x524f, 0x11ce, {0x9f, 0x53, 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70}}, .formattype = {0x05589f80, 0xc356, 0x11ce, {0xbf, 0x01, 0x00, 0xaa, 0x00, 0x55, 0x59, 0x5a}}, .cbFormat = sizeof(VIDEOINFO), .pbFormat = (BYTE *)&rgb32_video_info, }; static const DDSURFACEDESC rgb8_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXED8, .ddpfPixelFormat.u1.dwRGBBitCount = 8, }; static const DDSURFACEDESC rgb555_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB, .ddpfPixelFormat.u1.dwRGBBitCount = 16, .ddpfPixelFormat.u2.dwRBitMask = 0x7c00, .ddpfPixelFormat.u3.dwGBitMask = 0x03e0, .ddpfPixelFormat.u4.dwBBitMask = 0x001f, }; static const DDSURFACEDESC rgb565_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB, .ddpfPixelFormat.u1.dwRGBBitCount = 16, .ddpfPixelFormat.u2.dwRBitMask = 0xf800, .ddpfPixelFormat.u3.dwGBitMask = 0x07e0, .ddpfPixelFormat.u4.dwBBitMask = 0x001f, }; static const DDSURFACEDESC rgb24_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB, .ddpfPixelFormat.u1.dwRGBBitCount = 24, .ddpfPixelFormat.u2.dwRBitMask = 0xff0000, .ddpfPixelFormat.u3.dwGBitMask = 0x00ff00, .ddpfPixelFormat.u4.dwBBitMask = 0x0000ff, }; static const DDSURFACEDESC rgb32_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB, .ddpfPixelFormat.u1.dwRGBBitCount = 32, .ddpfPixelFormat.u2.dwRBitMask = 0xff0000, .ddpfPixelFormat.u3.dwGBitMask = 0x00ff00, .ddpfPixelFormat.u4.dwBBitMask = 0x0000ff, }; static const DDSURFACEDESC argb32_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_ALPHAPIXELS, .ddpfPixelFormat.u1.dwRGBBitCount = 32, .ddpfPixelFormat.u2.dwRBitMask = 0xff0000, .ddpfPixelFormat.u3.dwGBitMask = 0x00ff00, .ddpfPixelFormat.u4.dwBBitMask = 0x0000ff, .ddpfPixelFormat.u5.dwRGBAlphaBitMask = 0xff000000, }; static const DDSURFACEDESC yuy2_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_FOURCC, .ddpfPixelFormat.dwFourCC = MAKEFOURCC('Y', 'U', 'Y', '2'), .ddpfPixelFormat.u1.dwYUVBitCount = 16, }; static const DDSURFACEDESC yv12_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_FOURCC, .ddpfPixelFormat.dwFourCC = MAKEFOURCC('Y', 'V', '1', '2'), .ddpfPixelFormat.u1.dwYUVBitCount = 12, }; static const WCHAR primary_video_sink_id[] = L"I{A35FF56A-9FDA-11D0-8FDF-00C04FD9189D}"; static const WCHAR primary_audio_sink_id[] = L"I{A35FF56B-9FDA-11D0-8FDF-00C04FD9189D}"; static const WCHAR *load_resource(const WCHAR *name) { HMODULE module = GetModuleHandleA(NULL); HRSRC resource; DWORD written; HANDLE file; WCHAR *path; DWORD size; void *ptr; path = calloc(MAX_PATH + 1, sizeof(WCHAR)); ok(!!path, "Failed to allocate temp path string.\n"); GetTempPathW(MAX_PATH + 1, path); wcscat(path, name); file = CreateFileW(path, GENERIC_READ | GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, 0); ok(file != INVALID_HANDLE_VALUE, "Failed to create file %s, error %lu.\n", wine_dbgstr_w(path), GetLastError()); resource = FindResourceW(module, name, (const WCHAR *)RT_RCDATA); ok(!!resource, "Failed to find resource %s, error %lu.\n", wine_dbgstr_w(name), GetLastError()); size = SizeofResource(module, resource); ptr = LockResource(LoadResource(module, resource)); WriteFile(file, ptr, size, &written, NULL); ok(written == size, "Failed to write file %s.\n", wine_dbgstr_w(path)); CloseHandle(file); return path; } static void unload_resource(const WCHAR *path) { BOOL ret = DeleteFileW(path); ok(ret, "Failed to delete file %s.\n", wine_dbgstr_w(path)); free((void *)path); } #define EXPECT_REF(obj,ref) _expect_ref((IUnknown*)obj, ref, __LINE__) static void _expect_ref(IUnknown* obj, ULONG ref, int line) { ULONG rc; IUnknown_AddRef(obj); rc = IUnknown_Release(obj); ok_(__FILE__,line)(rc == ref, "expected refcount %ld, got %ld\n", ref, rc); } static IAMMultiMediaStream *create_ammultimediastream(void) { IAMMultiMediaStream *stream = NULL; HRESULT hr = CoCreateInstance(&CLSID_AMMultiMediaStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMultiMediaStream, (void **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); return stream; } static ULONG get_refcount(void *iface) { IUnknown *unknown = iface; IUnknown_AddRef(unknown); return IUnknown_Release(unknown); } #define check_interface(a, b, c) check_interface_(__LINE__, a, b, c) static void check_interface_(unsigned int line, void *iface_ptr, REFIID iid, BOOL supported) { IUnknown *iface = iface_ptr; HRESULT hr, expected_hr; IUnknown *unk; expected_hr = supported ? S_OK : E_NOINTERFACE; hr = IUnknown_QueryInterface(iface, iid, (void **)&unk); ok_(__FILE__, line)(hr == expected_hr, "Got hr %#lx, expected %#lx.\n", hr, expected_hr); if (SUCCEEDED(hr)) IUnknown_Release(unk); } static void test_interfaces(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; IMediaStream *stream; HRESULT hr; ULONG ref; check_interface(mmstream, &IID_IAMMultiMediaStream, TRUE); check_interface(mmstream, &IID_IMultiMediaStream, TRUE); check_interface(mmstream, &IID_IUnknown, TRUE); check_interface(mmstream, &IID_IAMMediaStream, FALSE); check_interface(mmstream, &IID_IAMMediaTypeStream, FALSE); check_interface(mmstream, &IID_IAudioMediaStream, FALSE); check_interface(mmstream, &IID_IBaseFilter, FALSE); check_interface(mmstream, &IID_IDirectDrawMediaStream, FALSE); check_interface(mmstream, &IID_IMediaFilter, FALSE); check_interface(mmstream, &IID_IMediaStream, FALSE); check_interface(mmstream, &IID_IMediaStreamFilter, FALSE); check_interface(mmstream, &IID_IPin, FALSE); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IBaseFilter, TRUE); check_interface(filter, &IID_IMediaFilter, TRUE); check_interface(filter, &IID_IMediaStreamFilter, TRUE); check_interface(filter, &IID_IPersist, TRUE); check_interface(filter, &IID_IUnknown, TRUE); check_interface(filter, &IID_IAMMediaStream, FALSE); check_interface(filter, &IID_IAMMediaTypeStream, FALSE); check_interface(filter, &IID_IAMMultiMediaStream, FALSE); check_interface(filter, &IID_IAudioMediaStream, FALSE); check_interface(filter, &IID_IDirectDrawMediaStream, FALSE); check_interface(filter, &IID_IMediaStream, FALSE); check_interface(filter, &IID_IMultiMediaStream, FALSE); check_interface(filter, &IID_IPin, FALSE); IMediaStreamFilter_Release(filter); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(stream, &IID_IAMMediaStream, TRUE); check_interface(stream, &IID_IAudioMediaStream, TRUE); check_interface(stream, &IID_IMediaStream, TRUE); check_interface(stream, &IID_IMemInputPin, TRUE); check_interface(stream, &IID_IPin, TRUE); check_interface(stream, &IID_IUnknown, TRUE); check_interface(stream, &IID_IAMMediaTypeStream, FALSE); check_interface(stream, &IID_IAMMultiMediaStream, FALSE); check_interface(stream, &IID_IBaseFilter, FALSE); check_interface(stream, &IID_IDirectDrawMediaStream, FALSE); check_interface(stream, &IID_IMediaFilter, FALSE); check_interface(stream, &IID_IMediaStreamFilter, FALSE); check_interface(stream, &IID_IMultiMediaStream, FALSE); check_interface(stream, &IID_IPersist, FALSE); IMediaStream_Release(stream); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(stream, &IID_IAMMediaStream, TRUE); check_interface(stream, &IID_IDirectDrawMediaStream, TRUE); check_interface(stream, &IID_IMediaStream, TRUE); check_interface(stream, &IID_IMemInputPin, TRUE); check_interface(stream, &IID_IPin, TRUE); check_interface(stream, &IID_IUnknown, TRUE); check_interface(stream, &IID_IAMMediaTypeStream, FALSE); check_interface(stream, &IID_IAMMultiMediaStream, FALSE); check_interface(stream, &IID_IAudioMediaStream, FALSE); check_interface(stream, &IID_IBaseFilter, FALSE); check_interface(stream, &IID_IDirectDraw, FALSE); check_interface(stream, &IID_IMediaFilter, FALSE); check_interface(stream, &IID_IMediaStreamFilter, FALSE); check_interface(stream, &IID_IMultiMediaStream, FALSE); check_interface(stream, &IID_IPersist, FALSE); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %lu.\n", ref); } static void test_openfile(const WCHAR *test_avi_path) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaControl *media_control; IMediaStreamFilter *filter; IMediaFilter *media_filter; IReferenceClock *clock; IGraphBuilder *graph; OAFilterState state; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!graph, "Expected NULL graph.\n"); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(!!filter, "Expected non-NULL filter.\n"); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_NORENDER); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); IGraphBuilder_Release(graph); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!graph, "Expected NULL graph.\n"); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(!!filter, "Expected non-NULL filter.\n"); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RENDERALLSTREAMS); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock = NULL; IMediaFilter_GetSyncSource(media_filter, &clock); ok(!!clock, "Expected non-NULL clock.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IReferenceClock_Release(clock); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!graph, "Expected NULL graph.\n"); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(!!filter, "Expected non-NULL filter.\n"); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RENDERTOEXISTING); ok(hr == VFW_E_CANNOT_CONNECT, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); IGraphBuilder_Release(graph); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RENDERTOEXISTING); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, TRUE); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock = NULL; IMediaFilter_GetSyncSource(media_filter, &clock); ok(!!clock, "Expected non-NULL clock.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IReferenceClock_Release(clock); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaControl, (void **)&media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); state = 0xdeadbeef; hr = IMediaControl_GetState(media_control, INFINITE, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == State_Running, "Got state %#lx.\n", state); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaControl_Release(media_control); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_NOCLOCK); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock = (IReferenceClock *)0xdeadbeef; IMediaFilter_GetSyncSource(media_filter, &clock); ok(!clock, "Got clock %p.\n", clock); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_mmstream_get_duration(const WCHAR *test_avi_path) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); HRESULT hr, audio_hr; LONGLONG duration; ULONG ref; duration = 0xdeadbeefdeadbeefULL; hr = IAMMultiMediaStream_GetDuration(mmstream, &duration); ok(hr == E_NOINTERFACE, "Got hr %#lx.\n", hr); ok(duration == 0xdeadbeefdeadbeefULL, "Got duration %s.\n", wine_dbgstr_longlong(duration)); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, AMMSF_ADDDEFAULTRENDERER, NULL); ok(hr == S_OK || hr == VFW_E_NO_AUDIO_HARDWARE, "Got hr %#lx.\n", hr); audio_hr = hr; hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RENDERTOEXISTING); ok(hr == S_OK, "Got hr %#lx.\n", hr); duration = 0xdeadbeefdeadbeefULL; hr = IAMMultiMediaStream_GetDuration(mmstream, &duration); if (audio_hr == S_OK) { ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(duration == 1000000LL, "Got duration %s.\n", wine_dbgstr_longlong(duration)); } else { ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(!duration, "Got duration %s.\n", wine_dbgstr_longlong(duration)); } ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); duration = 0xdeadbeefdeadbeefULL; hr = IAMMultiMediaStream_GetDuration(mmstream, &duration); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(duration == 0, "Got duration %s.\n", wine_dbgstr_longlong(duration)); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_RENDERTOEXISTING); ok(hr == VFW_E_CANNOT_CONNECT, "Got hr %#lx.\n", hr); duration = 0xdeadbeefdeadbeefULL; hr = IAMMultiMediaStream_GetDuration(mmstream, &duration); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(duration == 0, "Got duration %s.\n", wine_dbgstr_longlong(duration)); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_OpenFile(mmstream, test_avi_path, AMMSF_NORENDER); ok(hr == S_OK, "Got hr %#lx.\n", hr); duration = 0xdeadbeefdeadbeefULL; hr = IAMMultiMediaStream_GetDuration(mmstream, &duration); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(duration == 0, "Got duration %s.\n", wine_dbgstr_longlong(duration)); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static const GUID test_mspid = {0x88888888}; struct teststream { IAMMediaStream IAMMediaStream_iface; IPin IPin_iface; LONG refcount; GUID mspid; IAMMultiMediaStream *mmstream; IMediaStreamFilter *filter; IFilterGraph *graph; FILTER_STATE state; HRESULT set_state_result; }; static struct teststream *impl_from_IAMMediaStream(IAMMediaStream *iface) { return CONTAINING_RECORD(iface, struct teststream, IAMMediaStream_iface); } static struct teststream *impl_from_IPin(IPin *iface) { return CONTAINING_RECORD(iface, struct teststream, IPin_iface); } static HRESULT WINAPI pin_QueryInterface(IPin *iface, REFIID iid, void **out) { struct teststream *stream = impl_from_IPin(iface); return IAMMediaStream_QueryInterface(&stream->IAMMediaStream_iface, iid, out); } static ULONG WINAPI pin_AddRef(IPin *iface) { struct teststream *stream = impl_from_IPin(iface); return IAMMediaStream_AddRef(&stream->IAMMediaStream_iface); } static ULONG WINAPI pin_Release(IPin *iface) { struct teststream *stream = impl_from_IPin(iface); return IAMMediaStream_Release(&stream->IAMMediaStream_iface); } static HRESULT WINAPI pin_Connect(IPin *iface, IPin *peer, const AM_MEDIA_TYPE *mt) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_ReceiveConnection(IPin *iface, IPin *peer, const AM_MEDIA_TYPE *mt) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_Disconnect(IPin *iface) { if (winetest_debug > 1) trace("Disconnect\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_ConnectedTo(IPin *iface, IPin **peer) { if (winetest_debug > 1) trace("ConnectedTo\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_ConnectionMediaType(IPin *iface, AM_MEDIA_TYPE *mt) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_QueryPinInfo(IPin *iface, PIN_INFO *info) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_QueryDirection(IPin *iface, PIN_DIRECTION *dir) { if (winetest_debug > 1) trace("QueryDirection\n"); *dir = PINDIR_INPUT; return S_OK; } static HRESULT WINAPI pin_QueryId(IPin *iface, WCHAR **id) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_QueryAccept(IPin *iface, const AM_MEDIA_TYPE *mt) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_EnumMediaTypes(IPin *iface, IEnumMediaTypes **out) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_QueryInternalConnections(IPin *iface, IPin **pins, ULONG *count) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_EndOfStream(IPin *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_BeginFlush(IPin *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_EndFlush(IPin *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI pin_NewSegment(IPin *iface, REFERENCE_TIME start, REFERENCE_TIME stop, double rate) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static const IPinVtbl pin_vtbl = { pin_QueryInterface, pin_AddRef, pin_Release, pin_Connect, pin_ReceiveConnection, pin_Disconnect, pin_ConnectedTo, pin_ConnectionMediaType, pin_QueryPinInfo, pin_QueryDirection, pin_QueryId, pin_QueryAccept, pin_EnumMediaTypes, pin_QueryInternalConnections, pin_EndOfStream, pin_BeginFlush, pin_EndFlush, pin_NewSegment }; static HRESULT WINAPI stream_QueryInterface(IAMMediaStream *iface, REFIID iid, void **out) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("QueryInterface(%s)\n", wine_dbgstr_guid(iid)); if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IMediaStream) || IsEqualGUID(iid, &IID_IAMMediaStream)) { IAMMediaStream_AddRef(iface); *out = iface; return S_OK; } else if (IsEqualGUID(iid, &IID_IPin)) { IAMMediaStream_AddRef(iface); *out = &stream->IPin_iface; return S_OK; } ok(0, "Unexpected interface %s.\n", wine_dbgstr_guid(iid)); return E_NOINTERFACE; } static ULONG WINAPI stream_AddRef(IAMMediaStream *iface) { struct teststream *stream = impl_from_IAMMediaStream(iface); return InterlockedIncrement(&stream->refcount); } static ULONG WINAPI stream_Release(IAMMediaStream *iface) { struct teststream *stream = impl_from_IAMMediaStream(iface); return InterlockedDecrement(&stream->refcount); } static HRESULT WINAPI stream_GetMultiMediaStream(IAMMediaStream *iface, IMultiMediaStream **mmstream) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_GetInformation(IAMMediaStream *iface, MSPID *id, STREAM_TYPE *type) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("GetInformation(%p, %p)\n", id, type); if (id) *id = stream->mspid; if (type) *type = STREAMTYPE_READ; return S_OK; } static HRESULT WINAPI stream_SetSameFormat(IAMMediaStream *iface, IMediaStream *ref, DWORD flags) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_AllocateSample(IAMMediaStream *iface, DWORD flags, IStreamSample **sample) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_CreateSharedSample(IAMMediaStream *iface, IStreamSample *existing, DWORD flags, IStreamSample **out) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_SendEndOfStream(IAMMediaStream *iface, DWORD flags) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_Initialize(IAMMediaStream *iface, IUnknown *source, DWORD flags, REFMSPID id, const STREAM_TYPE type) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI stream_SetState(IAMMediaStream *iface, FILTER_STATE state) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("SetState(%#x)\n", state); if (SUCCEEDED(stream->set_state_result)) stream->state = state; return stream->set_state_result; } static HRESULT WINAPI stream_JoinAMMultiMediaStream(IAMMediaStream *iface, IAMMultiMediaStream *mmstream) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("JoinAMMultiMediaStream(%p)\n", mmstream); stream->mmstream = mmstream; return S_OK; } static HRESULT WINAPI stream_JoinFilter(IAMMediaStream *iface, IMediaStreamFilter *filter) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("JoinFilter(%p)\n", filter); stream->filter = filter; return S_OK; } static HRESULT WINAPI stream_JoinFilterGraph(IAMMediaStream *iface, IFilterGraph *graph) { struct teststream *stream = impl_from_IAMMediaStream(iface); if (winetest_debug > 1) trace("JoinFilterGraph(%p)\n", graph); stream->graph = graph; return S_OK; } static const IAMMediaStreamVtbl stream_vtbl = { stream_QueryInterface, stream_AddRef, stream_Release, stream_GetMultiMediaStream, stream_GetInformation, stream_SetSameFormat, stream_AllocateSample, stream_CreateSharedSample, stream_SendEndOfStream, stream_Initialize, stream_SetState, stream_JoinAMMultiMediaStream, stream_JoinFilter, stream_JoinFilterGraph, }; static void teststream_init(struct teststream *stream) { memset(stream, 0, sizeof(*stream)); stream->IAMMediaStream_iface.lpVtbl = &stream_vtbl; stream->IPin_iface.lpVtbl = &pin_vtbl; stream->refcount = 1; stream->mspid = test_mspid; stream->set_state_result = S_OK; } #define check_enum_stream(a,b,c,d) check_enum_stream_(__LINE__,a,b,c,d) static void check_enum_stream_(int line, IAMMultiMediaStream *mmstream, IMediaStreamFilter *filter, LONG index, IMediaStream *expect) { IMediaStream *stream = NULL, *stream2 = NULL; HRESULT hr; hr = IAMMultiMediaStream_EnumMediaStreams(mmstream, index, &stream); ok_(__FILE__, line)(hr == (expect ? S_OK : S_FALSE), "IAMMultiMediaStream::EnumMediaStreams() returned %#lx.\n", hr); hr = IMediaStreamFilter_EnumMediaStreams(filter, index, &stream2); ok_(__FILE__, line)(hr == (expect ? S_OK : S_FALSE), "IMediaStreamFilter::EnumMediaStreams() returned %#lx.\n", hr); if (hr == S_OK) { ok_(__FILE__, line)(stream == expect, "Expected stream %p, got %p.\n", expect, stream); ok_(__FILE__, line)(stream2 == expect, "Expected stream %p, got %p.\n", expect, stream2); IMediaStream_Release(stream); IMediaStream_Release(stream2); } } struct testfilter { struct strmbase_filter filter; struct strmbase_source source; IMediaSeeking IMediaSeeking_iface; LONGLONG current_position; LONGLONG stop_position; const AM_MEDIA_TYPE *preferred_mt; HANDLE wait_state_event; HRESULT get_duration_hr; HRESULT get_stop_position_hr; HRESULT set_positions_hr; HRESULT init_stream_hr; HRESULT cleanup_stream_hr; HRESULT wait_state_hr; HRESULT is_format_supported_hr; }; static inline struct testfilter *impl_from_BaseFilter(struct strmbase_filter *iface) { return CONTAINING_RECORD(iface, struct testfilter, filter); } static struct strmbase_pin *testfilter_get_pin(struct strmbase_filter *iface, unsigned int index) { struct testfilter *filter = impl_from_BaseFilter(iface); if (!index) return &filter->source.pin; return NULL; } static void testfilter_destroy(struct strmbase_filter *iface) { struct testfilter *filter = impl_from_BaseFilter(iface); CloseHandle(filter->wait_state_event); strmbase_source_cleanup(&filter->source); strmbase_filter_cleanup(&filter->filter); } static HRESULT testfilter_init_stream(struct strmbase_filter *iface) { struct testfilter *filter = impl_from_BaseFilter(iface); HRESULT hr; if (SUCCEEDED(filter->init_stream_hr) && filter->source.pin.peer) { hr = IMemAllocator_Commit(filter->source.pAllocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); } return filter->init_stream_hr; } static HRESULT testfilter_cleanup_stream(struct strmbase_filter *iface) { struct testfilter *filter = impl_from_BaseFilter(iface); HRESULT hr; if (SUCCEEDED(filter->cleanup_stream_hr) && filter->source.pin.peer) { hr = IMemAllocator_Decommit(filter->source.pAllocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); } return filter->cleanup_stream_hr; } static HRESULT testfilter_wait_state(struct strmbase_filter *iface, DWORD timeout) { struct testfilter *filter = impl_from_BaseFilter(iface); HRESULT hr; LeaveCriticalSection(&filter->filter.filter_cs); WaitForSingleObject(filter->wait_state_event, timeout); EnterCriticalSection(&filter->filter.filter_cs); hr = filter->wait_state_hr; return hr; } static const struct strmbase_filter_ops testfilter_ops = { .filter_get_pin = testfilter_get_pin, .filter_destroy = testfilter_destroy, .filter_init_stream = testfilter_init_stream, .filter_cleanup_stream = testfilter_cleanup_stream, .filter_wait_state = testfilter_wait_state, }; static inline struct testfilter *impl_from_base_pin(struct strmbase_pin *iface) { return CONTAINING_RECORD(iface, struct testfilter, source.pin); } static HRESULT testsource_get_media_type(struct strmbase_pin *iface, unsigned int index, AM_MEDIA_TYPE *mt) { struct testfilter *filter = impl_from_base_pin(iface); if (index < 1 && filter->preferred_mt) return CopyMediaType(mt, filter->preferred_mt); return VFW_S_NO_MORE_ITEMS; } static HRESULT testsource_query_interface(struct strmbase_pin *iface, REFIID iid, void **out) { struct testfilter *filter = impl_from_base_pin(iface); if (IsEqualGUID(iid, &IID_IMediaSeeking) && filter->IMediaSeeking_iface.lpVtbl) *out = &filter->IMediaSeeking_iface; else return E_NOINTERFACE; IUnknown_AddRef((IUnknown *)*out); return S_OK; } static HRESULT WINAPI testsource_DecideAllocator(struct strmbase_source *iface, IMemInputPin *pin, IMemAllocator **alloc) { ALLOCATOR_PROPERTIES props = {0}; HRESULT hr; /* AMDirectDrawStream tries to use it's custom allocator and * when it is able to do so it's behavior changes slightly * (e.g. it uses dynamic format change instead of reconnecting in SetFormat). * We don't yet implement the custom allocator so force the standard one for now. */ hr = CoCreateInstance(&CLSID_MemoryAllocator, NULL, CLSCTX_INPROC_SERVER, &IID_IMemAllocator, (void **)alloc); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMemInputPin_GetAllocatorRequirements(pin, &props); hr = iface->pFuncsTable->pfnDecideBufferSize(iface, *alloc, &props); ok(hr == S_OK, "Got hr %#lx.\n", hr); return IMemInputPin_NotifyAllocator(pin, *alloc, FALSE); } static HRESULT WINAPI testsource_DecideBufferSize(struct strmbase_source *iface, IMemAllocator *alloc, ALLOCATOR_PROPERTIES *requested) { ALLOCATOR_PROPERTIES actual; if (!requested->cbAlign) requested->cbAlign = 1; if (requested->cbBuffer < 4096) requested->cbBuffer = 4096; if (!requested->cBuffers) requested->cBuffers = 2; return IMemAllocator_SetProperties(alloc, requested, &actual); } static const struct strmbase_source_ops testsource_ops = { .base.pin_get_media_type = testsource_get_media_type, .base.pin_query_interface = testsource_query_interface, .pfnAttemptConnection = BaseOutputPinImpl_AttemptConnection, .pfnDecideBufferSize = testsource_DecideBufferSize, .pfnDecideAllocator = testsource_DecideAllocator, }; static void testfilter_init(struct testfilter *filter) { static const GUID clsid = {0xabacab}; memset(filter, 0, sizeof(*filter)); strmbase_filter_init(&filter->filter, NULL, &clsid, &testfilter_ops); strmbase_source_init(&filter->source, &filter->filter, L"", &testsource_ops); filter->stop_position = 0x8000000000000000ULL; filter->wait_state_event = CreateEventW(NULL, TRUE, TRUE, NULL); } static inline struct testfilter *impl_from_IMediaSeeking(IMediaSeeking *iface) { return CONTAINING_RECORD(iface, struct testfilter, IMediaSeeking_iface); } static HRESULT WINAPI testsource_seeking_QueryInterface(IMediaSeeking *iface, REFIID iid, void **out) { struct testfilter *filter = impl_from_IMediaSeeking(iface); return IBaseFilter_QueryInterface(&filter->filter.IBaseFilter_iface, iid, out); } static ULONG WINAPI testsource_seeking_AddRef(IMediaSeeking *iface) { struct testfilter *filter = impl_from_IMediaSeeking(iface); return IBaseFilter_AddRef(&filter->filter.IBaseFilter_iface); } static ULONG WINAPI testsource_seeking_Release(IMediaSeeking *iface) { struct testfilter *filter = impl_from_IMediaSeeking(iface); return IBaseFilter_Release(&filter->filter.IBaseFilter_iface); } static HRESULT WINAPI testsource_seeking_GetCapabilities(IMediaSeeking *iface, DWORD *capabilities) { return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_CheckCapabilities(IMediaSeeking *iface, DWORD *capabilities) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_IsFormatSupported(IMediaSeeking *iface, const GUID *format) { struct testfilter *filter = impl_from_IMediaSeeking(iface); ok(IsEqualGUID(format, &TIME_FORMAT_MEDIA_TIME), "Got format %s.\n", wine_dbgstr_guid(format)); return filter->is_format_supported_hr; } static HRESULT WINAPI testsource_seeking_QueryPreferredFormat(IMediaSeeking *iface, GUID *format) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_GetTimeFormat(IMediaSeeking *iface, GUID *format) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_IsUsingTimeFormat(IMediaSeeking *iface, const GUID *format) { return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_SetTimeFormat(IMediaSeeking *iface, const GUID *format) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_GetDuration(IMediaSeeking *iface, LONGLONG *duration) { struct testfilter *filter = impl_from_IMediaSeeking(iface); if (SUCCEEDED(filter->get_duration_hr)) *duration = 0x8000000000000000ULL; return filter->get_duration_hr; } static HRESULT WINAPI testsource_seeking_GetStopPosition(IMediaSeeking *iface, LONGLONG *stop) { struct testfilter *filter = impl_from_IMediaSeeking(iface); if (SUCCEEDED(filter->get_stop_position_hr)) *stop = 0x8000000000000000ULL; return filter->get_stop_position_hr; } static HRESULT WINAPI testsource_seeking_GetCurrentPosition(IMediaSeeking *iface, LONGLONG *current) { *current = 0; return S_OK; } static HRESULT WINAPI testsource_seeking_ConvertTimeFormat(IMediaSeeking *iface, LONGLONG *target, const GUID *target_format, LONGLONG source, const GUID *source_format) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_SetPositions(IMediaSeeking *iface, LONGLONG *current_ptr, DWORD current_flags, LONGLONG *stop_ptr, DWORD stop_flags) { struct testfilter *filter = impl_from_IMediaSeeking(iface); if (SUCCEEDED(filter->set_positions_hr)) { if (current_ptr) filter->current_position = *current_ptr; if (stop_ptr) filter->stop_position = *stop_ptr; } return filter->set_positions_hr; } static HRESULT WINAPI testsource_seeking_GetPositions(IMediaSeeking *iface, LONGLONG *current, LONGLONG *stop) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_GetAvailable(IMediaSeeking *iface, LONGLONG *earliest, LONGLONG *latest) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_SetRate(IMediaSeeking *iface, double rate) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_GetRate(IMediaSeeking *iface, double *rate) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testsource_seeking_GetPreroll(IMediaSeeking *iface, LONGLONG *preroll) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static const IMediaSeekingVtbl testsource_seeking_vtbl = { testsource_seeking_QueryInterface, testsource_seeking_AddRef, testsource_seeking_Release, testsource_seeking_GetCapabilities, testsource_seeking_CheckCapabilities, testsource_seeking_IsFormatSupported, testsource_seeking_QueryPreferredFormat, testsource_seeking_GetTimeFormat, testsource_seeking_IsUsingTimeFormat, testsource_seeking_SetTimeFormat, testsource_seeking_GetDuration, testsource_seeking_GetStopPosition, testsource_seeking_GetCurrentPosition, testsource_seeking_ConvertTimeFormat, testsource_seeking_SetPositions, testsource_seeking_GetPositions, testsource_seeking_GetAvailable, testsource_seeking_SetRate, testsource_seeking_GetRate, testsource_seeking_GetPreroll, }; #define check_get_stream(a,b,c,d) check_get_stream_(__LINE__,a,b,c,d) static void check_get_stream_(int line, IAMMultiMediaStream *mmstream, IMediaStreamFilter *filter, const GUID *mspid, IMediaStream *expect) { IMediaStream *stream = NULL, *stream2 = NULL; HRESULT hr; hr = IAMMultiMediaStream_GetMediaStream(mmstream, mspid, &stream); ok_(__FILE__, line)(hr == (expect ? S_OK : MS_E_NOSTREAM), "IAMMultiMediaStream::GetMediaStream() returned %#lx.\n", hr); hr = IMediaStreamFilter_GetMediaStream(filter, mspid, &stream2); ok_(__FILE__, line)(hr == (expect ? S_OK : MS_E_NOSTREAM), "IMediaStreamFilter::GetMediaStream() returned %#lx.\n", hr); if (hr == S_OK) { ok_(__FILE__, line)(stream == expect, "Expected stream %p, got %p.\n", expect, stream); ok_(__FILE__, line)(stream2 == expect, "Expected stream %p, got %p.\n", expect, stream2); } if (stream) IMediaStream_Release(stream); if (stream2) IMediaStream_Release(stream2); } static void test_add_stream(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStream *video_stream, *audio_stream, *stream; IDirectDrawMediaStream *ddraw_stream; IMediaStreamFilter *stream_filter; struct teststream teststream; IDirectDraw *ddraw, *ddraw2; IEnumFilters *enum_filters; IBaseFilter *filters[3]; IGraphBuilder *graph; FILTER_INFO info; ULONG ref, count; CLSID clsid; HRESULT hr; teststream_init(&teststream); hr = IAMMultiMediaStream_GetFilter(mmstream, &stream_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_EnumMediaStreams(mmstream, 0, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EnumMediaStreams(stream_filter, 0, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetMediaStream(mmstream, &MSPID_PrimaryAudio, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetMediaStream(stream_filter, &MSPID_PrimaryAudio, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetMediaStream(mmstream, &MSPID_PrimaryVideo, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetMediaStream(stream_filter, &MSPID_PrimaryVideo, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); check_enum_stream(mmstream, stream_filter, 0, NULL); check_get_stream(mmstream, stream_filter, NULL, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryAudio, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryVideo, NULL); check_get_stream(mmstream, stream_filter, &test_mspid, NULL); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &test_mspid, 0, &stream); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &video_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_EnumMediaStreams(mmstream, 0, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EnumMediaStreams(stream_filter, 0, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_EnumMediaStreams(mmstream, 1, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EnumMediaStreams(stream_filter, 1, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); check_enum_stream(mmstream, stream_filter, 0, video_stream); check_enum_stream(mmstream, stream_filter, 1, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryVideo, video_stream); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryAudio, NULL); check_get_stream(mmstream, stream_filter, &test_mspid, NULL); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_enum_stream(mmstream, stream_filter, 0, video_stream); check_enum_stream(mmstream, stream_filter, 1, audio_stream); check_enum_stream(mmstream, stream_filter, 2, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryVideo, video_stream); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryAudio, audio_stream); check_get_stream(mmstream, stream_filter, &test_mspid, NULL); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)&teststream, &IID_IUnknown, 0, &stream); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_QueryFilterInfo(stream_filter, &info); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)&teststream, &test_mspid, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(stream == (IMediaStream *)&teststream, "Streams didn't match.\n"); IMediaStream_Release(stream); ok(teststream.mmstream == mmstream, "IAMMultiMediaStream objects didn't match.\n"); ok(teststream.filter == stream_filter, "IMediaStreamFilter objects didn't match.\n"); ok(teststream.graph == info.pGraph, "IFilterGraph objects didn't match.\n"); IFilterGraph_Release(info.pGraph); check_enum_stream(mmstream, stream_filter, 0, video_stream); check_enum_stream(mmstream, stream_filter, 1, audio_stream); check_enum_stream(mmstream, stream_filter, 2, (IMediaStream *)&teststream); check_enum_stream(mmstream, stream_filter, 3, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryVideo, video_stream); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryAudio, audio_stream); check_get_stream(mmstream, stream_filter, &test_mspid, (IMediaStream *)&teststream); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected a non-NULL graph.\n"); hr = IGraphBuilder_EnumFilters(graph, &enum_filters); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumFilters_Next(enum_filters, 3, filters, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); ok(filters[0] == (IBaseFilter *)stream_filter, "Expected filter %p, got %p.\n", stream_filter, filters[0]); IBaseFilter_Release(filters[0]); IEnumFilters_Release(enum_filters); IGraphBuilder_Release(graph); IMediaStreamFilter_Release(stream_filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(video_stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(audio_stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ok(teststream.refcount == 1, "Got outstanding refcount %ld.\n", teststream.refcount); /* The return parameter is optional. */ mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetMediaStream(mmstream, &MSPID_PrimaryVideo, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); /* Test supplying a DirectDraw object with the primary video stream. */ hr = DirectDrawCreate(NULL, &ddraw, NULL); ok(hr == DD_OK, "Got hr %#lx.\n", hr); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)ddraw, &MSPID_PrimaryVideo, 0, &video_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(video_stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw2 == ddraw, "Expected IDirectDraw %p, got %p.\n", ddraw, ddraw2); IDirectDraw_Release(ddraw2); IDirectDrawMediaStream_Release(ddraw_stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(video_stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &video_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(video_stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!ddraw, "Expected a non-NULL IDirectDraw.\n"); IDirectDraw_Release(ddraw); IDirectDrawMediaStream_Release(ddraw_stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(video_stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); /* Test the AMMSF_ADDDEFAULTRENDERER flag. No stream is added; however, a * new filter will be added to the graph. */ mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_GetFilter(mmstream, &stream_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, AMMSF_ADDDEFAULTRENDERER, &video_stream); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, AMMSF_ADDDEFAULTRENDERER, NULL); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, AMMSF_ADDDEFAULTRENDERER, &audio_stream); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, AMMSF_ADDDEFAULTRENDERER, NULL); ok(hr == S_OK || hr == VFW_E_NO_AUDIO_HARDWARE, "Got hr %#lx.\n", hr); check_enum_stream(mmstream, stream_filter, 0, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryAudio, NULL); check_get_stream(mmstream, stream_filter, &MSPID_PrimaryVideo, NULL); if (hr == S_OK) { hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Got graph %p.\n", graph); hr = IGraphBuilder_EnumFilters(graph, &enum_filters); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumFilters_Next(enum_filters, 3, filters, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(count == 2, "Got count %lu.\n", count); ok(filters[1] == (IBaseFilter *)stream_filter, "Expected filter %p, got %p.\n", stream_filter, filters[1]); hr = IBaseFilter_GetClassID(filters[0], &clsid); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&clsid, &CLSID_DSoundRender), "Got unexpected filter %s.\n", wine_dbgstr_guid(&clsid)); IBaseFilter_Release(filters[0]); IBaseFilter_Release(filters[1]); IEnumFilters_Release(enum_filters); IGraphBuilder_Release(graph); } hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &test_mspid, AMMSF_ADDDEFAULTRENDERER, NULL); ok(hr == MS_E_PURPOSEID, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &test_mspid, AMMSF_ADDDEFAULTRENDERER, &audio_stream); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(stream_filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_media_streams(void) { IAMMultiMediaStream *pams; HRESULT hr; IMediaStream *video_stream = NULL; IMediaStream *audio_stream = NULL; IMediaStreamFilter* media_stream_filter = NULL; if (!(pams = create_ammultimediastream())) return; hr = IAMMultiMediaStream_Initialize(pams, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "IAMMultiMediaStream_Initialize returned: %lx\n", hr); /* Retrieve media stream filter */ hr = IAMMultiMediaStream_GetFilter(pams, NULL); ok(hr == E_POINTER, "IAMMultiMediaStream_GetFilter returned: %lx\n", hr); hr = IAMMultiMediaStream_GetFilter(pams, &media_stream_filter); ok(hr == S_OK, "IAMMultiMediaStream_GetFilter returned: %lx\n", hr); hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryVideo, 0, NULL); ok(hr == S_OK, "IAMMultiMediaStream_AddMediaStream returned: %lx\n", hr); hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryVideo, &video_stream); ok(hr == S_OK, "IAMMultiMediaStream_GetMediaStream returned: %lx\n", hr); /* Check interfaces and samples for video */ if (video_stream) { IAMMediaStream* am_media_stream; IMultiMediaStream *multi_media_stream; IPin *pin = NULL; IAudioMediaStream* audio_media_stream; IDirectDrawMediaStream *ddraw_stream = NULL; IDirectDrawStreamSample *ddraw_sample = NULL; hr = IMediaStream_QueryInterface(video_stream, &IID_IAMMediaStream, (LPVOID*)&am_media_stream); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); ok((void*)am_media_stream == (void*)video_stream, "Not same interface, got %p expected %p\n", am_media_stream, video_stream); hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, NULL); ok(hr == E_POINTER, "Expected E_POINTER, got %lx\n", hr); multi_media_stream = (void *)0xdeadbeef; hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, &multi_media_stream); ok(hr == S_OK, "IAMMediaStream_GetMultiMediaStream returned: %lx\n", hr); ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream); IMultiMediaStream_Release(multi_media_stream); IAMMediaStream_Release(am_media_stream); hr = IMediaStream_QueryInterface(video_stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); IPin_Release(pin); hr = IMediaStream_QueryInterface(video_stream, &IID_IAudioMediaStream, (LPVOID*)&audio_media_stream); ok(hr == E_NOINTERFACE, "IMediaStream_QueryInterface returned: %lx\n", hr); hr = IMediaStream_QueryInterface(video_stream, &IID_IDirectDrawMediaStream, (LPVOID*)&ddraw_stream); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); if (SUCCEEDED(hr)) { 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: %lx\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &ddraw_sample); ok(hr == S_OK, "IDirectDrawMediaStream_CreateSample returned: %lx\n", hr); hr = IDirectDrawMediaStream_GetMultiMediaStream(ddraw_stream, NULL); ok(hr == E_POINTER, "Expected E_POINTER, got %lx\n", hr); multi_media_stream = (void *)0xdeadbeef; hr = IDirectDrawMediaStream_GetMultiMediaStream(ddraw_stream, &multi_media_stream); ok(hr == S_OK, "IDirectDrawMediaStream_GetMultiMediaStream returned: %lx\n", hr); ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream); IMultiMediaStream_Release(multi_media_stream); } if (ddraw_sample) IDirectDrawStreamSample_Release(ddraw_sample); if (ddraw_stream) IDirectDrawMediaStream_Release(ddraw_stream); } hr = IAMMultiMediaStream_AddMediaStream(pams, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "IAMMultiMediaStream_AddMediaStream returned: %lx\n", hr); hr = IAMMultiMediaStream_GetMediaStream(pams, &MSPID_PrimaryAudio, &audio_stream); ok(hr == S_OK, "IAMMultiMediaStream_GetMediaStream returned: %lx\n", hr); /* Check interfaces and samples for audio */ if (audio_stream) { IAMMediaStream* am_media_stream; IMultiMediaStream *multi_media_stream; IPin *pin = NULL; IDirectDrawMediaStream* ddraw_stream = NULL; IAudioMediaStream* audio_media_stream = NULL; IAudioStreamSample *audio_sample = NULL; hr = IMediaStream_QueryInterface(audio_stream, &IID_IAMMediaStream, (LPVOID*)&am_media_stream); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); ok((void*)am_media_stream == (void*)audio_stream, "Not same interface, got %p expected %p\n", am_media_stream, audio_stream); hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, NULL); ok(hr == E_POINTER, "Expected E_POINTER, got %lx\n", hr); multi_media_stream = (void *)0xdeadbeef; hr = IAMMediaStream_GetMultiMediaStream(am_media_stream, &multi_media_stream); ok(hr == S_OK, "IAMMediaStream_GetMultiMediaStream returned: %lx\n", hr); ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream); IMultiMediaStream_Release(multi_media_stream); IAMMediaStream_Release(am_media_stream); hr = IMediaStream_QueryInterface(audio_stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); IPin_Release(pin); hr = IMediaStream_QueryInterface(audio_stream, &IID_IDirectDrawMediaStream, (LPVOID*)&ddraw_stream); ok(hr == E_NOINTERFACE, "IMediaStream_QueryInterface returned: %lx\n", hr); hr = IMediaStream_QueryInterface(audio_stream, &IID_IAudioMediaStream, (LPVOID*)&audio_media_stream); ok(hr == S_OK, "IMediaStream_QueryInterface returned: %lx\n", hr); if (SUCCEEDED(hr)) { IAudioData* audio_data = NULL; hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "CoCreateInstance returned: %lx\n", hr); hr = IAudioMediaStream_CreateSample(audio_media_stream, NULL, 0, &audio_sample); ok(hr == E_POINTER, "IAudioMediaStream_CreateSample returned: %lx\n", hr); EXPECT_REF(audio_stream, 3); EXPECT_REF(audio_data, 1); hr = IAudioMediaStream_CreateSample(audio_media_stream, audio_data, 0, &audio_sample); ok(hr == S_OK, "IAudioMediaStream_CreateSample returned: %lx\n", hr); EXPECT_REF(audio_stream, 4); EXPECT_REF(audio_data, 2); hr = IAudioMediaStream_GetMultiMediaStream(audio_media_stream, NULL); ok(hr == E_POINTER, "Expected E_POINTER, got %lx\n", hr); multi_media_stream = (void *)0xdeadbeef; hr = IAudioMediaStream_GetMultiMediaStream(audio_media_stream, &multi_media_stream); ok(hr == S_OK, "IAudioMediaStream_GetMultiMediaStream returned: %lx\n", hr); ok((void *)multi_media_stream == (void *)pams, "Expected %p, got %p\n", pams, multi_media_stream); IMultiMediaStream_Release(multi_media_stream); if (audio_data) IAudioData_Release(audio_data); if (audio_sample) IAudioStreamSample_Release(audio_sample); if (audio_media_stream) IAudioMediaStream_Release(audio_media_stream); } } if (media_stream_filter) { IEnumPins *enum_pins; hr = IMediaStreamFilter_EnumPins(media_stream_filter, &enum_pins); ok(hr == S_OK, "IBaseFilter_EnumPins returned: %lx\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: %lx\n", hr); ok(nb_pins == expected_nb_pins, "Number of pins is %lu instead of %lu\n", nb_pins, expected_nb_pins); for (i = 0; i < min(nb_pins, expected_nb_pins); i++) { IPin* pin; hr = IPin_ConnectedTo(pins[i], &pin); ok(hr == VFW_E_NOT_CONNECTED, "IPin_ConnectedTo returned: %lx\n", hr); IPin_Release(pins[i]); } IEnumPins_Release(enum_pins); } } /* Test open file with no filename */ hr = IAMMultiMediaStream_OpenFile(pams, NULL, AMMSF_RENDERTOEXISTING); ok(hr == E_POINTER, "IAMMultiMediaStream_OpenFile returned %lx instead of %lx\n", hr, E_POINTER); if (video_stream) IMediaStream_Release(video_stream); if (audio_stream) IMediaStream_Release(audio_stream); if (media_stream_filter) IMediaStreamFilter_Release(media_stream_filter); IAMMultiMediaStream_Release(pams); } static void test_enum_pins(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; IEnumPins *enum1, *enum2; IMediaStream *stream; IPin *pins[3], *pin; ULONG ref, count; HRESULT hr; hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(filter); ok(ref == 3, "Got unexpected refcount %ld.\n", ref); hr = IMediaStreamFilter_EnumPins(filter, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EnumPins(filter, &enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(filter); ok(ref == 3, "Got unexpected refcount %ld.\n", ref); ref = get_refcount(enum1); ok(ref == 1, "Got unexpected refcount %ld.\n", ref); hr = IEnumPins_Next(enum1, 1, NULL, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 1, pins, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IEnumPins_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Skip(enum1, 0); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IEnumPins_Skip(enum1, 1); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Reset() isn't enough; we have to call EnumPins() again to see the updated * pin count. */ hr = IEnumPins_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 1, pins, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); IEnumPins_Release(enum1); hr = IMediaStreamFilter_EnumPins(filter, &enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(filter); ok(ref == 4, "Got unexpected refcount %ld.\n", ref); ref = get_refcount(enum1); ok(ref == 1, "Got unexpected refcount %ld.\n", ref); ref = get_refcount(pin); ok(ref == 4, "Got unexpected refcount %ld.\n", ref); hr = IEnumPins_Next(enum1, 1, pins, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(pins[0] == pin, "Expected pin %p, got %p.\n", pin, pins[0]); ref = get_refcount(filter); ok(ref == 4, "Got unexpected refcount %ld.\n", ref); ref = get_refcount(enum1); ok(ref == 1, "Got unexpected refcount %ld.\n", ref); ref = get_refcount(pin); ok(ref == 5, "Got unexpected refcount %ld.\n", ref); IPin_Release(pins[0]); hr = IEnumPins_Next(enum1, 1, pins, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IEnumPins_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 1, pins, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); ok(pins[0] == pin, "Expected pin %p, got %p.\n", pin, pins[0]); IPin_Release(pins[0]); hr = IEnumPins_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 2, pins, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 2, pins, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); ok(pins[0] == pin, "Expected pin %p, got %p.\n", pin, pins[0]); IPin_Release(pins[0]); hr = IEnumPins_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Clone(enum1, &enum2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Skip(enum1, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumPins_Skip(enum1, 1); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum1, 1, pins, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); hr = IEnumPins_Next(enum2, 1, pins, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(pins[0] == pin, "Expected pin %p, got %p.\n", pin, pins[0]); IPin_Release(pins[0]); IEnumPins_Release(enum2); IEnumPins_Release(enum1); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaStream_Release(stream); ref = IPin_Release(pin); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_find_pin(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; IMediaStream *stream; IPin *pin, *pin2; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_FindPin(filter, primary_video_sink_id, &pin2); ok(hr == VFW_E_NOT_FOUND, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_FindPin(filter, primary_video_sink_id, &pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(pin2 == pin, "Expected pin %p, got %p.\n", pin, pin2); IPin_Release(pin2); IPin_Release(pin); IMediaStream_Release(stream); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_pin_info(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; IMediaStream *stream; PIN_DIRECTION dir; ULONG ref, count; PIN_INFO info; HRESULT hr; WCHAR *id; IPin *pin; hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_QueryPinInfo(pin, &info); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(info.pFilter == (IBaseFilter *)filter, "Expected filter %p, got %p.\n", filter, info.pFilter); ok(info.dir == PINDIR_INPUT, "Got direction %d.\n", info.dir); ok(!wcscmp(info.achName, primary_video_sink_id), "Got name %s.\n", wine_dbgstr_w(info.achName)); IBaseFilter_Release(info.pFilter); hr = IPin_QueryDirection(pin, &dir); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(dir == PINDIR_INPUT, "Got direction %d.\n", dir); hr = IPin_QueryId(pin, &id); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!wcscmp(id, primary_video_sink_id), "Got id %s.\n", wine_dbgstr_w(id)); CoTaskMemFree(id); hr = IPin_QueryInternalConnections(pin, NULL, &count); ok(hr == E_NOTIMPL, "Got hr %#lx.\n", hr); IPin_Release(pin); IMediaStream_Release(stream); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_QueryPinInfo(pin, &info); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(info.pFilter == (IBaseFilter *)filter, "Expected filter %p, got %p.\n", filter, info.pFilter); ok(info.dir == PINDIR_INPUT, "Got direction %d.\n", info.dir); ok(!wcscmp(info.achName, primary_audio_sink_id), "Got name %s.\n", wine_dbgstr_w(info.achName)); IBaseFilter_Release(info.pFilter); hr = IPin_QueryDirection(pin, &dir); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(dir == PINDIR_INPUT, "Got direction %d.\n", dir); hr = IPin_QueryId(pin, &id); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!wcscmp(id, primary_audio_sink_id), "Got id %s.\n", wine_dbgstr_w(id)); CoTaskMemFree(id); hr = IPin_QueryInternalConnections(pin, NULL, &count); ok(hr == E_NOTIMPL, "Got hr %#lx.\n", hr); IPin_Release(pin); IMediaStream_Release(stream); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } struct graph { IFilterGraph2 IFilterGraph2_iface; IMediaEventSink IMediaEventSink_iface; IUnknown *inner_unk; IFilterGraph2 *inner; IMediaEventSink *inner_event_sink; LONG refcount; unsigned int got_add_filter; IBaseFilter *filter; WCHAR filter_name[128]; unsigned int got_notify; LONG event_code; LONG_PTR event_param1; LONG_PTR event_param2; }; static struct graph *impl_from_IFilterGraph2(IFilterGraph2 *iface) { return CONTAINING_RECORD(iface, struct graph, IFilterGraph2_iface); } static struct graph *impl_from_IMediaEventSink(IMediaEventSink *iface) { return CONTAINING_RECORD(iface, struct graph, IMediaEventSink_iface); } static HRESULT WINAPI graph_QueryInterface(IFilterGraph2 *iface, REFIID iid, void **out) { struct graph *graph = impl_from_IFilterGraph2(iface); if (winetest_debug > 1) trace("QueryInterface(%s)\n", wine_dbgstr_guid(iid)); if (IsEqualGUID(iid, &IID_IFilterGraph2) || IsEqualGUID(iid, &IID_IGraphBuilder) || IsEqualGUID(iid, &IID_IFilterGraph) || IsEqualGUID(iid, &IID_IUnknown)) { *out = iface; IFilterGraph2_AddRef(iface); return S_OK; } else if (IsEqualGUID(iid, &IID_IMediaEventSink)) { *out = &graph->IMediaEventSink_iface; IFilterGraph2_AddRef(iface); return S_OK; } else { return IUnknown_QueryInterface(graph->inner_unk, iid, out); } return E_NOINTERFACE; } static ULONG WINAPI graph_AddRef(IFilterGraph2 *iface) { struct graph *graph = impl_from_IFilterGraph2(iface); return InterlockedIncrement(&graph->refcount); } static ULONG WINAPI graph_Release(IFilterGraph2 *iface) { struct graph *graph = impl_from_IFilterGraph2(iface); return InterlockedDecrement(&graph->refcount); } static HRESULT WINAPI graph_AddFilter(IFilterGraph2 *iface, IBaseFilter *filter, const WCHAR *name) { struct graph *graph = impl_from_IFilterGraph2(iface); if (winetest_debug > 1) trace("AddFilter(%p, %s)\n", filter, wine_dbgstr_w(name)); ++graph->got_add_filter; graph->filter = filter; if (name) wcscpy(graph->filter_name, name); else graph->filter_name[0] = 0; return IFilterGraph2_AddFilter(graph->inner, filter, name); } static HRESULT WINAPI graph_RemoveFilter(IFilterGraph2 *iface, IBaseFilter *filter) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_EnumFilters(IFilterGraph2 *iface, IEnumFilters **enumfilters) { struct graph *graph = impl_from_IFilterGraph2(iface); if (winetest_debug > 1) trace("EnumFilters()\n"); return IFilterGraph2_EnumFilters(graph->inner, enumfilters); } static HRESULT WINAPI graph_FindFilterByName(IFilterGraph2 *iface, const WCHAR *name, IBaseFilter **filter) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_ConnectDirect(IFilterGraph2 *iface, IPin *source, IPin *sink, const AM_MEDIA_TYPE *mt) { struct graph *graph = impl_from_IFilterGraph2(iface); return IFilterGraph2_ConnectDirect(graph->inner, source, sink, mt); } static HRESULT WINAPI graph_Reconnect(IFilterGraph2 *iface, IPin *pin) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_Disconnect(IFilterGraph2 *iface, IPin *pin) { struct graph *graph = impl_from_IFilterGraph2(iface); return IFilterGraph2_Disconnect(graph->inner, pin); } static HRESULT WINAPI graph_SetDefaultSyncSource(IFilterGraph2 *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_Connect(IFilterGraph2 *iface, IPin *source, IPin *sink) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_Render(IFilterGraph2 *iface, IPin *source) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_RenderFile(IFilterGraph2 *iface, const WCHAR *filename, const WCHAR *playlist) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_AddSourceFilter(IFilterGraph2 *iface, const WCHAR *filename, const WCHAR *filter_name, IBaseFilter **filter) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_SetLogFile(IFilterGraph2 *iface, DWORD_PTR file) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_Abort(IFilterGraph2 *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_ShouldOperationContinue(IFilterGraph2 *iface) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_AddSourceFilterForMoniker(IFilterGraph2 *iface, IMoniker *moniker, IBindCtx *bind_ctx, const WCHAR *filter_name, IBaseFilter **filter) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_ReconnectEx(IFilterGraph2 *iface, IPin *pin, const AM_MEDIA_TYPE *mt) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI graph_RenderEx(IFilterGraph2 *iface, IPin *pin, DWORD flags, DWORD *context) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static const IFilterGraph2Vtbl graph_vtbl = { graph_QueryInterface, graph_AddRef, graph_Release, graph_AddFilter, graph_RemoveFilter, graph_EnumFilters, graph_FindFilterByName, graph_ConnectDirect, graph_Reconnect, graph_Disconnect, graph_SetDefaultSyncSource, graph_Connect, graph_Render, graph_RenderFile, graph_AddSourceFilter, graph_SetLogFile, graph_Abort, graph_ShouldOperationContinue, graph_AddSourceFilterForMoniker, graph_ReconnectEx, graph_RenderEx, }; static HRESULT WINAPI event_sink_QueryInterface(IMediaEventSink *iface, REFIID iid, void **out) { struct graph *graph = impl_from_IMediaEventSink(iface); return IFilterGraph2_QueryInterface(&graph->IFilterGraph2_iface, iid, out); } static ULONG WINAPI event_sink_AddRef(IMediaEventSink *iface) { struct graph *graph = impl_from_IMediaEventSink(iface); return IFilterGraph2_AddRef(&graph->IFilterGraph2_iface); } static ULONG WINAPI event_sink_Release(IMediaEventSink *iface) { struct graph *graph = impl_from_IMediaEventSink(iface); return IFilterGraph2_Release(&graph->IFilterGraph2_iface); } static HRESULT WINAPI event_sink_Notify(IMediaEventSink *iface, LONG code, LONG_PTR param1, LONG_PTR param2) { struct graph *graph = impl_from_IMediaEventSink(iface); ++graph->got_notify; graph->event_code = code; graph->event_param1 = param1; graph->event_param2 = param2; return IMediaEventSink_Notify(graph->inner_event_sink, code, param1, param2); } static const IMediaEventSinkVtbl event_sink_vtbl = { event_sink_QueryInterface, event_sink_AddRef, event_sink_Release, event_sink_Notify, }; static void graph_init(struct graph *graph) { HRESULT hr; memset(graph, 0, sizeof(*graph)); graph->IFilterGraph2_iface.lpVtbl = &graph_vtbl; graph->IMediaEventSink_iface.lpVtbl = &event_sink_vtbl; graph->refcount = 1; hr = CoCreateInstance(&CLSID_FilterGraph, (IUnknown *)&graph->IFilterGraph2_iface, CLSCTX_INPROC_SERVER, &IID_IUnknown, (void **)&graph->inner_unk); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IUnknown_QueryInterface(graph->inner_unk, &IID_IFilterGraph2, (void **)&graph->inner); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IUnknown_QueryInterface(graph->inner_unk, &IID_IMediaEventSink, (void **)&graph->inner_event_sink); ok(hr == S_OK, "Got hr %#lx.\n", hr); } static void graph_destroy(struct graph *graph) { ULONG ref; IMediaEventSink_Release(graph->inner_event_sink); IFilterGraph2_Release(graph->inner); ref = IUnknown_Release(graph->inner_unk); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_initialize(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; IGraphBuilder *ret_graph; IMediaStream *stream; struct graph graph; STREAM_TYPE type; HRESULT hr; ULONG ref; ret_graph = (IGraphBuilder *)0xdeadbeef; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!ret_graph, "Got unexpected graph %p.\n", ret_graph); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected a non-NULL filter."); IMediaStreamFilter_Release(filter); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_WRITE, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); ret_graph = (IGraphBuilder *)0xdeadbeef; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!ret_graph, "Got unexpected graph %p.\n", ret_graph); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); type = 0xdeadbeef; hr = IMediaStream_GetInformation(stream, NULL, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(type == STREAMTYPE_READ, "Got type %u.\n", type); IMediaStream_Release(stream); ret_graph = NULL; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!ret_graph, "Got unexpected graph %p.\n", ret_graph); IGraphBuilder_Release(ret_graph); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_WRITE, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_WRITE, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); ret_graph = (IGraphBuilder *)0xdeadbeef; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!ret_graph, "Got unexpected graph %p.\n", ret_graph); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); type = 0xdeadbeef; hr = IMediaStream_GetInformation(stream, NULL, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(type == STREAMTYPE_WRITE, "Got type %u.\n", type); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_WRITE, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); ret_graph = (IGraphBuilder *)0xdeadbeef; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!ret_graph, "Got unexpected graph %p.\n", ret_graph); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); type = 0xdeadbeef; hr = IMediaStream_GetInformation(stream, NULL, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(type == STREAMTYPE_TRANSFORM, "Got type %u.\n", type); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); type = 0xdeadbeef; hr = IMediaStream_GetInformation(stream, NULL, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(type == STREAMTYPE_READ, "Got type %u.\n", type); ret_graph = NULL; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!ret_graph, "Got unexpected graph %p.\n", ret_graph); IGraphBuilder_Release(ret_graph); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); type = 0xdeadbeef; hr = IMediaStream_GetInformation(stream, NULL, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(type == STREAMTYPE_READ, "Got type %u.\n", type); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); /* Test with a custom filter graph. */ mmstream = create_ammultimediastream(); graph_init(&graph); ret_graph = (IGraphBuilder *)0xdeadbeef; hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!ret_graph, "Got unexpected graph %p.\n", ret_graph); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected a non-NULL filter."); graph.got_add_filter = 0; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph.IFilterGraph2_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_add_filter == 1, "Got %d calls to IGraphBuilder::AddFilter().\n", graph.got_add_filter); ok(graph.filter == (IBaseFilter *)filter, "Got filter %p.\n", filter); ok(!wcscmp(graph.filter_name, L"MediaStreamFilter"), "Got unexpected name %s.\n", wine_dbgstr_w(graph.filter_name)); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &ret_graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ret_graph == (IGraphBuilder *)&graph, "Got unexpected graph %p.\n", ret_graph); IGraphBuilder_Release(ret_graph); graph.got_add_filter = 0; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!graph.got_add_filter, "Got %d calls to IGraphBuilder::AddFilter().\n", graph.got_add_filter); graph.got_add_filter = 0; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!graph.got_add_filter, "Got %d calls to IGraphBuilder::AddFilter().\n", graph.got_add_filter); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_WRITE, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_TRANSFORM, 0, NULL); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); IMediaStreamFilter_Release(filter); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); graph_destroy(&graph); } static IAMMultiMediaStream *mmstream_mmstream; static STREAM_STATE mmstream_state; static DWORD CALLBACK mmstream_set_state(void *param) { HRESULT hr; hr = IAMMultiMediaStream_SetState(mmstream_mmstream, mmstream_state); ok(hr == S_OK, "Got hr %#lx.\n", hr); return 0; } static void test_set_state(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaControl *media_control; struct testfilter source; IGraphBuilder *graph; STREAM_STATE state; HANDLE thread; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaControl, (void **)&media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.init_stream_hr = E_FAIL; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); source.init_stream_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.init_stream_hr = S_FALSE; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.init_stream_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.wait_state_hr = E_FAIL; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.wait_state_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.wait_state_hr = VFW_S_STATE_INTERMEDIATE; ResetEvent(source.wait_state_event); mmstream_mmstream = mmstream; mmstream_state = STREAMSTATE_RUN; thread = CreateThread(NULL, 0, mmstream_set_state, NULL, 0, NULL); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "SetState returned prematurely.\n"); EnterCriticalSection(&source.filter.filter_cs); source.wait_state_hr = S_OK; SetEvent(source.wait_state_event); LeaveCriticalSection(&source.filter.filter_cs); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.init_stream_hr = E_FAIL; source.wait_state_hr = VFW_S_STATE_INTERMEDIATE; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); source.init_stream_hr = S_OK; source.wait_state_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.cleanup_stream_hr = E_FAIL; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); source.cleanup_stream_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.cleanup_stream_hr = S_FALSE; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); source.cleanup_stream_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.wait_state_hr = VFW_S_STATE_INTERMEDIATE; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.wait_state_hr = S_OK; /* Crashes on native. */ if (0) { hr = IAMMultiMediaStream_GetState(mmstream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } state = 0xdeadbeef; hr = IAMMultiMediaStream_GetState(mmstream, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == STREAMSTATE_STOP, "Got state %#x.\n", state); hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); state = 0xdeadbeef; hr = IAMMultiMediaStream_GetState(mmstream, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == STREAMSTATE_STOP, "Got state %#x.\n", state); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); state = 0xdeadbeef; hr = IAMMultiMediaStream_GetState(mmstream, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == STREAMSTATE_RUN, "Got state %#x.\n", state); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); state = 0xdeadbeef; hr = IAMMultiMediaStream_GetState(mmstream, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == STREAMSTATE_STOP, "Got state %#x.\n", state); source.init_stream_hr = E_FAIL; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); state = 0xdeadbeef; hr = IAMMultiMediaStream_GetState(mmstream, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == STREAMSTATE_STOP, "Got state %#x.\n", state); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaControl_Release(media_control); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_enum_media_types(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IEnumMediaTypes *enum1, *enum2; AM_MEDIA_TYPE *mts[2]; IMediaStream *stream; ULONG ref, count; HRESULT hr; IPin *pin; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EnumMediaTypes(pin, &enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 1, mts, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 0, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 2, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]); hr = IEnumMediaTypes_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Clone(enum1, &enum2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Skip(enum1, 2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Next(enum2, 1, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]); IEnumMediaTypes_Release(enum2); IEnumMediaTypes_Release(enum1); IPin_Release(pin); IMediaStream_Release(stream); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EnumMediaTypes(pin, &enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 1, mts, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 0, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]->pbFormat); CoTaskMemFree(mts[0]); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 2, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]->pbFormat); CoTaskMemFree(mts[0]); hr = IEnumMediaTypes_Reset(enum1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Clone(enum1, &enum2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Skip(enum1, 2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enum1, 1, mts, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(!count, "Got count %lu.\n", count); hr = IEnumMediaTypes_Next(enum2, 1, mts, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); CoTaskMemFree(mts[0]->pbFormat); CoTaskMemFree(mts[0]); IEnumMediaTypes_Release(enum2); IEnumMediaTypes_Release(enum1); IPin_Release(pin); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_media_types(void) { static const VIDEOINFOHEADER2 req_vih2; static const VIDEOINFOHEADER req_vih; static const MPEG2VIDEOINFO req_m2vi; static const MPEG1VIDEOINFO req_m1vi; static const WAVEFORMATEX req_wfx; static const WAVEFORMATEX expect_wfx = { .wFormatTag = WAVE_FORMAT_PCM, .nChannels = 1, .nSamplesPerSec = 11025, .nAvgBytesPerSec = 11025 * 2, .nBlockAlign = 2, .wBitsPerSample = 16, .cbSize = 0, }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IEnumMediaTypes *enummt; IMediaStream *stream; AM_MEDIA_TYPE *pmt; ULONG ref, count; unsigned int i; HRESULT hr; IPin *pin; static const struct { const GUID *type; BYTE *format; ULONG size; } tests[] = { {&GUID_NULL, NULL, 0 }, {&FORMAT_None, NULL, 0 }, {&FORMAT_WaveFormatEx, (BYTE *)&req_wfx, sizeof(WAVEFORMATEX)}, {&FORMAT_MPEG2Video, (BYTE *)&req_m2vi, sizeof(MPEG2VIDEOINFO)}, {&FORMAT_MPEGVideo, (BYTE *)&req_m1vi, sizeof(MPEG2VIDEOINFO)}, {&FORMAT_VideoInfo2, (BYTE *)&req_vih2, sizeof(VIDEOINFOHEADER2)}, {&FORMAT_VideoInfo, (BYTE *)&req_vih, sizeof(VIDEOINFOHEADER)}, }; static const GUID *rejected_subtypes[] = { &MEDIASUBTYPE_RGB1, &MEDIASUBTYPE_RGB4, &MEDIASUBTYPE_RGB565, &MEDIASUBTYPE_RGB555, &MEDIASUBTYPE_RGB24, &MEDIASUBTYPE_RGB32, &MEDIASUBTYPE_ARGB32, &MEDIASUBTYPE_ARGB1555, &MEDIASUBTYPE_ARGB4444, &MEDIASUBTYPE_Avi, &MEDIASUBTYPE_I420, &MEDIASUBTYPE_AYUV, &MEDIASUBTYPE_YV12, &MEDIASUBTYPE_YUY2, &MEDIASUBTYPE_UYVY, &MEDIASUBTYPE_YVYU, &MEDIASUBTYPE_NV12, &GUID_NULL, }; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EnumMediaTypes(pin, &enummt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); ok(IsEqualGUID(&pmt->majortype, &MEDIATYPE_Video), "Got major type %s\n", wine_dbgstr_guid(&pmt->majortype)); ok(IsEqualGUID(&pmt->subtype, &MEDIASUBTYPE_RGB8), "Got subtype %s\n", wine_dbgstr_guid(&pmt->subtype)); ok(pmt->bFixedSizeSamples == TRUE, "Got fixed size %d.\n", pmt->bFixedSizeSamples); ok(!pmt->bTemporalCompression, "Got temporal compression %d.\n", pmt->bTemporalCompression); ok(pmt->lSampleSize == 10000, "Got sample size %lu.\n", pmt->lSampleSize); ok(IsEqualGUID(&pmt->formattype, &GUID_NULL), "Got format type %s.\n", wine_dbgstr_guid(&pmt->formattype)); ok(!pmt->pUnk, "Got pUnk %p.\n", pmt->pUnk); for (i = 0; i < ARRAY_SIZE(tests); ++i) { pmt->formattype = *tests[i].type; pmt->cbFormat = tests[i].size; pmt->pbFormat = tests[i].format; hr = IPin_QueryAccept(pin, pmt); ok(hr == (i == 6 ? S_OK : VFW_E_TYPE_NOT_ACCEPTED), "Got hr %#lx.\n", hr); } pmt->bFixedSizeSamples = FALSE; pmt->bTemporalCompression = TRUE; pmt->lSampleSize = 123; hr = IPin_QueryAccept(pin, pmt); ok(hr == S_OK, "Got hr %#lx.\n", hr); pmt->majortype = MEDIATYPE_NULL; hr = IPin_QueryAccept(pin, pmt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); pmt->majortype = MEDIATYPE_Audio; hr = IPin_QueryAccept(pin, pmt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); pmt->majortype = MEDIATYPE_Stream; hr = IPin_QueryAccept(pin, pmt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); pmt->majortype = MEDIATYPE_Video; for (i = 0; i < ARRAY_SIZE(rejected_subtypes); ++i) { pmt->subtype = *rejected_subtypes[i]; hr = IPin_QueryAccept(pin, pmt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx for subtype %s.\n", hr, wine_dbgstr_guid(rejected_subtypes[i])); } CoTaskMemFree(pmt); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); IEnumMediaTypes_Release(enummt); IPin_Release(pin); IMediaStream_Release(stream); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EnumMediaTypes(pin, &enummt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, &count); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(count == 1, "Got count %lu.\n", count); ok(IsEqualGUID(&pmt->majortype, &MEDIATYPE_Audio), "Got major type %s\n", wine_dbgstr_guid(&pmt->majortype)); ok(IsEqualGUID(&pmt->subtype, &GUID_NULL), "Got subtype %s\n", wine_dbgstr_guid(&pmt->subtype)); ok(pmt->bFixedSizeSamples == TRUE, "Got fixed size %d.\n", pmt->bFixedSizeSamples); ok(!pmt->bTemporalCompression, "Got temporal compression %d.\n", pmt->bTemporalCompression); ok(pmt->lSampleSize == 2, "Got sample size %lu.\n", pmt->lSampleSize); ok(IsEqualGUID(&pmt->formattype, &FORMAT_WaveFormatEx), "Got format type %s.\n", wine_dbgstr_guid(&pmt->formattype)); ok(!pmt->pUnk, "Got pUnk %p.\n", pmt->pUnk); ok(pmt->cbFormat == sizeof(WAVEFORMATEX), "Got format size %lu.\n", pmt->cbFormat); ok(!memcmp(pmt->pbFormat, &expect_wfx, pmt->cbFormat), "Format blocks didn't match.\n"); hr = IPin_QueryAccept(pin, pmt); ok(hr == E_NOTIMPL, "Got hr %#lx.\n", hr); CoTaskMemFree(pmt->pbFormat); CoTaskMemFree(pmt); hr = IEnumMediaTypes_Next(enummt, 1, &pmt, &count); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); IEnumMediaTypes_Release(enummt); IPin_Release(pin); IMediaStream_Release(stream); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_get_end_of_stream_event_handle(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; struct testfilter source; IGraphBuilder *graph; IMediaStream *stream; HANDLE event; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); testfilter_init(&source); source.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetEndOfStreamEventHandle(mmstream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); event = INVALID_HANDLE_VALUE; hr = IAMMultiMediaStream_GetEndOfStreamEventHandle(mmstream, &event); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == WAIT_TIMEOUT, "Event should not be signaled.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == WAIT_TIMEOUT, "Event should not be signaled.\n"); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == 0, "Event should be signaled.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == 0, "Event should be signaled.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == WAIT_TIMEOUT, "Event should not be signaled.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static IUnknown *create_audio_data(void) { IUnknown *audio_data = NULL; HRESULT result = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IUnknown, (void **)&audio_data); ok(S_OK == result, "got 0x%08lx\n", result); return audio_data; } static void test_audiodata_query_interface(void) { IUnknown *unknown = create_audio_data(); IMemoryData *memory_data = NULL; IAudioData *audio_data = NULL; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IMemoryData, (void **)&memory_data); ok(E_NOINTERFACE == result, "got 0x%08lx\n", result); result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); ok(S_OK == result, "got 0x%08lx\n", result); if (S_OK == result) { result = IAudioData_QueryInterface(audio_data, &IID_IMemoryData, (void **)&memory_data); ok(E_NOINTERFACE == result, "got 0x%08lx\n", result); IAudioData_Release(audio_data); } IUnknown_Release(unknown); } static void test_audiodata_get_info(void) { IUnknown *unknown = create_audio_data(); IAudioData *audio_data = NULL; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); if (FAILED(result)) { /* test_audiodata_query_interface handles this case */ skip("No IAudioData\n"); goto out_unknown; } result = IAudioData_GetInfo(audio_data, NULL, NULL, NULL); ok(MS_E_NOTINIT == result, "got 0x%08lx\n", result); IAudioData_Release(audio_data); out_unknown: IUnknown_Release(unknown); } static void test_audiodata_set_buffer(void) { IUnknown *unknown = create_audio_data(); IAudioData *audio_data = NULL; BYTE buffer[100] = {0}; DWORD length = 0; BYTE *data = NULL; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); if (FAILED(result)) { /* test_audiodata_query_interface handles this case */ skip("No IAudioData\n"); goto out_unknown; } result = IAudioData_SetBuffer(audio_data, 100, NULL, 0); ok(S_OK == result, "got 0x%08lx\n", result); data = (BYTE *)0xdeadbeef; length = 0xdeadbeef; result = IAudioData_GetInfo(audio_data, &length, &data, NULL); ok(S_OK == result, "got 0x%08lx\n", result); ok(100 == length, "got %lu\n", length); ok(NULL != data, "got %p\n", data); result = IAudioData_SetBuffer(audio_data, 0, buffer, 0); ok(E_INVALIDARG == result, "got 0x%08lx\n", result); result = IAudioData_SetBuffer(audio_data, sizeof(buffer), buffer, 0); ok(S_OK == result, "got 0x%08lx\n", result); data = (BYTE *)0xdeadbeef; length = 0xdeadbeef; result = IAudioData_GetInfo(audio_data, &length, &data, NULL); ok(S_OK == result, "got 0x%08lx\n", result); ok(sizeof(buffer) == length, "got %lu\n", length); ok(buffer == data, "got %p\n", data); IAudioData_Release(audio_data); out_unknown: IUnknown_Release(unknown); } static void test_audiodata_set_actual(void) { IUnknown *unknown = create_audio_data(); IAudioData *audio_data = NULL; BYTE buffer[100] = {0}; DWORD actual_data = 0; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); if (FAILED(result)) { /* test_audiodata_query_interface handles this case */ skip("No IAudioData\n"); goto out_unknown; } result = IAudioData_SetActual(audio_data, 0); ok(S_OK == result, "got 0x%08lx\n", result); result = IAudioData_SetBuffer(audio_data, sizeof(buffer), buffer, 0); ok(S_OK == result, "got 0x%08lx\n", result); result = IAudioData_SetActual(audio_data, sizeof(buffer) + 1); ok(E_INVALIDARG == result, "got 0x%08lx\n", result); result = IAudioData_SetActual(audio_data, sizeof(buffer)); ok(S_OK == result, "got 0x%08lx\n", result); actual_data = 0xdeadbeef; result = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_data); ok(S_OK == result, "got 0x%08lx\n", result); ok(sizeof(buffer) == actual_data, "got %lu\n", actual_data); result = IAudioData_SetActual(audio_data, 0); ok(S_OK == result, "got 0x%08lx\n", result); actual_data = 0xdeadbeef; result = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_data); ok(S_OK == result, "got 0x%08lx\n", result); ok(0 == actual_data, "got %lu\n", actual_data); IAudioData_Release(audio_data); out_unknown: IUnknown_Release(unknown); } static void test_audiodata_get_format(void) { IUnknown *unknown = create_audio_data(); IAudioData *audio_data = NULL; WAVEFORMATEX wave_format = {0}; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); if (FAILED(result)) { /* test_audiodata_query_interface handles this case */ skip("No IAudioData\n"); goto out_unknown; } result = IAudioData_GetFormat(audio_data, NULL); ok(E_POINTER == result, "got 0x%08lx\n", result); wave_format.wFormatTag = 0xdead; wave_format.nChannels = 0xdead; wave_format.nSamplesPerSec = 0xdeadbeef; wave_format.nAvgBytesPerSec = 0xdeadbeef; wave_format.nBlockAlign = 0xdead; wave_format.wBitsPerSample = 0xdead; wave_format.cbSize = 0xdead; result = IAudioData_GetFormat(audio_data, &wave_format); ok(S_OK == result, "got 0x%08lx\n", result); ok(WAVE_FORMAT_PCM == wave_format.wFormatTag, "got %u\n", wave_format.wFormatTag); ok(1 == wave_format.nChannels, "got %u\n", wave_format.nChannels); ok(11025 == wave_format.nSamplesPerSec, "got %lu\n", wave_format.nSamplesPerSec); ok(22050 == wave_format.nAvgBytesPerSec, "got %lu\n", wave_format.nAvgBytesPerSec); ok(2 == wave_format.nBlockAlign, "got %u\n", wave_format.nBlockAlign); ok(16 == wave_format.wBitsPerSample, "got %u\n", wave_format.wBitsPerSample); ok(0 == wave_format.cbSize, "got %u\n", wave_format.cbSize); IAudioData_Release(audio_data); out_unknown: IUnknown_Release(unknown); } static void test_audiodata_set_format(void) { IUnknown *unknown = create_audio_data(); IAudioData *audio_data = NULL; WAVEFORMATPCMEX wave_format = {{0}}; HRESULT result; result = IUnknown_QueryInterface(unknown, &IID_IAudioData, (void **)&audio_data); if (FAILED(result)) { /* test_audiodata_query_interface handles this case */ skip("No IAudioData\n"); goto out_unknown; } result = IAudioData_SetFormat(audio_data, NULL); ok(E_POINTER == result, "got 0x%08lx\n", result); wave_format.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE; wave_format.Format.nChannels = 2; wave_format.Format.nSamplesPerSec = 44100; wave_format.Format.nAvgBytesPerSec = 176400; wave_format.Format.nBlockAlign = 4; wave_format.Format.wBitsPerSample = 16; wave_format.Format.cbSize = 22; wave_format.Samples.wValidBitsPerSample = 16; wave_format.dwChannelMask = KSAUDIO_SPEAKER_STEREO; wave_format.SubFormat = KSDATAFORMAT_SUBTYPE_PCM; result = IAudioData_SetFormat(audio_data, &wave_format.Format); ok(E_INVALIDARG == result, "got 0x%08lx\n", result); wave_format.Format.wFormatTag = WAVE_FORMAT_PCM; wave_format.Format.nChannels = 2; wave_format.Format.nSamplesPerSec = 44100; wave_format.Format.nAvgBytesPerSec = 176400; wave_format.Format.nBlockAlign = 4; wave_format.Format.wBitsPerSample = 16; wave_format.Format.cbSize = 0; result = IAudioData_SetFormat(audio_data, &wave_format.Format); ok(S_OK == result, "got 0x%08lx\n", result); wave_format.Format.wFormatTag = 0xdead; wave_format.Format.nChannels = 0xdead; wave_format.Format.nSamplesPerSec = 0xdeadbeef; wave_format.Format.nAvgBytesPerSec = 0xdeadbeef; wave_format.Format.nBlockAlign = 0xdead; wave_format.Format.wBitsPerSample = 0xdead; wave_format.Format.cbSize = 0xdead; result = IAudioData_GetFormat(audio_data, &wave_format.Format); ok(S_OK == result, "got 0x%08lx\n", result); ok(WAVE_FORMAT_PCM == wave_format.Format.wFormatTag, "got %u\n", wave_format.Format.wFormatTag); ok(2 == wave_format.Format.nChannels, "got %u\n", wave_format.Format.nChannels); ok(44100 == wave_format.Format.nSamplesPerSec, "got %lu\n", wave_format.Format.nSamplesPerSec); ok(176400 == wave_format.Format.nAvgBytesPerSec, "got %lu\n", wave_format.Format.nAvgBytesPerSec); ok(4 == wave_format.Format.nBlockAlign, "got %u\n", wave_format.Format.nBlockAlign); ok(16 == wave_format.Format.wBitsPerSample, "got %u\n", wave_format.Format.wBitsPerSample); ok(0 == wave_format.Format.cbSize, "got %u\n", wave_format.Format.cbSize); IAudioData_Release(audio_data); out_unknown: IUnknown_Release(unknown); } struct advise_time_cookie { LONGLONG base; LONGLONG offset; HANDLE event; HANDLE advise_time_called_event; BOOL unadvise_called; }; struct testclock { IReferenceClock IReferenceClock_iface; LONG refcount; LONGLONG time; struct advise_time_cookie *advise_time_cookie; HRESULT get_time_hr; }; static inline struct testclock *impl_from_IReferenceClock(IReferenceClock *iface) { return CONTAINING_RECORD(iface, struct testclock, IReferenceClock_iface); } static HRESULT WINAPI testclock_QueryInterface(IReferenceClock *iface, REFIID iid, void **out) { if (winetest_debug > 1) trace("QueryInterface(%s)\n", wine_dbgstr_guid(iid)); if (IsEqualGUID(iid, &IID_IReferenceClock) || IsEqualGUID(iid, &IID_IUnknown)) { *out = iface; IReferenceClock_AddRef(iface); return S_OK; } return E_NOINTERFACE; } static ULONG WINAPI testclock_AddRef(IReferenceClock *iface) { struct testclock *clock = impl_from_IReferenceClock(iface); return InterlockedIncrement(&clock->refcount); } static ULONG WINAPI testclock_Release(IReferenceClock *iface) { struct testclock *clock = impl_from_IReferenceClock(iface); return InterlockedDecrement(&clock->refcount); } static HRESULT WINAPI testclock_GetTime(IReferenceClock *iface, REFERENCE_TIME *time) { struct testclock *clock = impl_from_IReferenceClock(iface); if (SUCCEEDED(clock->get_time_hr)) *time = clock->time; return clock->get_time_hr; } static HRESULT WINAPI testclock_AdviseTime(IReferenceClock *iface, REFERENCE_TIME base, REFERENCE_TIME offset, HEVENT event, DWORD_PTR *cookie) { struct testclock *clock = impl_from_IReferenceClock(iface); if (clock->advise_time_cookie) { clock->advise_time_cookie->base = base; clock->advise_time_cookie->offset = offset; clock->advise_time_cookie->event = (HANDLE)event; SetEvent(clock->advise_time_cookie->advise_time_called_event); } else { SetEvent((HANDLE)event); } *cookie = (DWORD_PTR)clock->advise_time_cookie; return S_OK; } static HRESULT WINAPI testclock_AdvisePeriodic(IReferenceClock *iface, REFERENCE_TIME start, REFERENCE_TIME period, HSEMAPHORE semaphore, DWORD_PTR *cookie) { ok(0, "Unexpected call.\n"); return E_NOTIMPL; } static HRESULT WINAPI testclock_Unadvise(IReferenceClock *iface, DWORD_PTR cookie) { if (cookie) ((struct advise_time_cookie *)cookie)->unadvise_called = TRUE; return S_OK; } static IReferenceClockVtbl testclock_vtbl = { testclock_QueryInterface, testclock_AddRef, testclock_Release, testclock_GetTime, testclock_AdviseTime, testclock_AdvisePeriodic, testclock_Unadvise, }; static void testclock_init(struct testclock *clock) { memset(clock, 0, sizeof(*clock)); clock->IReferenceClock_iface.lpVtbl = &testclock_vtbl; } static void test_audiostream_get_format(void) { static const WAVEFORMATEX pin_format = { .wFormatTag = WAVE_FORMAT_PCM, .nChannels = 2, .nSamplesPerSec = 44100, .wBitsPerSample = 16, .nBlockAlign = 4, .nAvgBytesPerSec = 4 * 44100, }; AM_MEDIA_TYPE mt = { .majortype = MEDIATYPE_Audio, .subtype = MEDIASUBTYPE_PCM, .formattype = FORMAT_WaveFormatEx, .cbFormat = sizeof(WAVEFORMATEX), .pbFormat = (BYTE *)&pin_format, }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioMediaStream *audio_stream; struct testfilter source; IGraphBuilder *graph; IMediaStream *stream; WAVEFORMATEX format; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, L"source"); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_GetFormat(audio_stream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_GetFormat(audio_stream, &format); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); memset(&format, 0xcc, sizeof(format)); hr = IAudioMediaStream_GetFormat(audio_stream, &format); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(format.wFormatTag == WAVE_FORMAT_PCM, "Got tag %#x.\n", format.wFormatTag); ok(format.nChannels == 2, "Got %u channels.\n", format.nChannels); ok(format.nSamplesPerSec == 44100, "Got sample rate %lu.\n", format.nSamplesPerSec); ok(format.nAvgBytesPerSec == 176400, "Got %lu bytes/sec.\n", format.nAvgBytesPerSec); ok(format.nBlockAlign == 4, "Got alignment %u.\n", format.nBlockAlign); ok(format.wBitsPerSample == 16, "Got %u bits/sample.\n", format.wBitsPerSample); ok(!format.cbSize, "Got extra size %u.\n", format.cbSize); hr = IGraphBuilder_Disconnect(graph, pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_GetFormat(audio_stream, &format); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IBaseFilter_Release(&source.filter.IBaseFilter_iface); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static HRESULT set_audiostream_format(const WAVEFORMATEX *format) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioMediaStream *audio_stream; IMediaStream *stream; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_SetFormat(audio_stream, format); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); return hr; } static void test_audiostream_set_format(void) { static const WAVEFORMATEX valid_format = { .wFormatTag = WAVE_FORMAT_PCM, .nChannels = 2, .nSamplesPerSec = 44100, .wBitsPerSample = 16, .nBlockAlign = 4, .nAvgBytesPerSec = 4 * 44100, }; const AM_MEDIA_TYPE mt = { .majortype = MEDIATYPE_Audio, .subtype = MEDIASUBTYPE_PCM, .formattype = FORMAT_WaveFormatEx, .cbFormat = sizeof(WAVEFORMATEX), .pbFormat = (BYTE *)&valid_format, }; WAVEFORMATEXTENSIBLE extensible_format; IAudioMediaStream *audio_stream; IAMMultiMediaStream *mmstream; struct testfilter source; IGraphBuilder *graph; IMediaStream *stream; WAVEFORMATEX format; HRESULT hr; ULONG ref; IPin *pin; hr = set_audiostream_format(&valid_format); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = set_audiostream_format(NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); extensible_format.Format = valid_format; extensible_format.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE; extensible_format.Format.cbSize = sizeof(WAVEFORMATEXTENSIBLE) - sizeof(WAVEFORMATEX); extensible_format.Samples.wValidBitsPerSample = valid_format.wBitsPerSample; extensible_format.dwChannelMask = KSAUDIO_SPEAKER_STEREO; extensible_format.SubFormat = KSDATAFORMAT_SUBTYPE_PCM; hr = set_audiostream_format(&extensible_format.Format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.nBlockAlign = 1; hr = set_audiostream_format(&format); ok(hr == S_OK, "Got hr %#lx.\n", hr); format = valid_format; format.nAvgBytesPerSec = 1234; hr = set_audiostream_format(&format); ok(hr == S_OK, "Got hr %#lx.\n", hr); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_SetFormat(audio_stream, &valid_format); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_GetFormat(audio_stream, &format); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); format = valid_format; format.nChannels = 1; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.nSamplesPerSec = 11025; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.nAvgBytesPerSec = 1234; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.nBlockAlign = 1; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.wBitsPerSample = 8; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); format = valid_format; format.cbSize = 1; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_SetFormat(audio_stream, &valid_format); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); format = valid_format; format.nChannels = 1; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); format = valid_format; format.nChannels = 1; hr = IAudioMediaStream_SetFormat(audio_stream, &format); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IBaseFilter_Release(&source.filter.IBaseFilter_iface); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostream_receive_connection(void) { WAVEFORMATEXTENSIBLE extensible_format; IAudioMediaStream *audio_stream; IAMMultiMediaStream *mmstream; struct testfilter source; IGraphBuilder *graph; IMediaStream *stream; WAVEFORMATEX format; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-null graph\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); mt = audio_mt; mt.majortype = GUID_NULL; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); mt = audio_mt; mt.subtype = MEDIASUBTYPE_RGB24; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); mt = audio_mt; mt.formattype = GUID_NULL; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); mt = audio_mt; mt.cbFormat = sizeof(WAVEFORMATEX) - 1; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); extensible_format.Format = audio_format; extensible_format.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE; extensible_format.Format.cbSize = sizeof(WAVEFORMATEXTENSIBLE) - sizeof(WAVEFORMATEX); extensible_format.Samples.wValidBitsPerSample = audio_format.wBitsPerSample; extensible_format.dwChannelMask = KSAUDIO_SPEAKER_STEREO; extensible_format.SubFormat = KSDATAFORMAT_SUBTYPE_PCM; mt = audio_mt; mt.cbFormat = sizeof(extensible_format); mt.pbFormat = (BYTE *)&extensible_format; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_SetFormat(audio_stream, &audio_format); ok(hr == S_OK, "Got hr %#lx.\n", hr); format = audio_format; format.nChannels = 2; mt = audio_mt; mt.pbFormat = (BYTE *)&format; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IBaseFilter_Release(&source.filter.IBaseFilter_iface); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_receive_connection(void) { static const VIDEOINFOHEADER req_vih; IDirectDrawMediaStream *ddraw_stream; IAMMultiMediaStream *mmstream; struct testfilter source; DDSURFACEDESC format; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; int i; static const VIDEOINFO yuy2_video_info = { .bmiHeader.biSize = sizeof(BITMAPINFOHEADER), .bmiHeader.biWidth = 333, .bmiHeader.biHeight = -444, .bmiHeader.biPlanes = 1, .bmiHeader.biBitCount = 16, .bmiHeader.biCompression = MAKEFOURCC('Y', 'U', 'Y', '2'), }; const AM_MEDIA_TYPE yuy2_mt = { .majortype = MEDIATYPE_Video, .subtype = MEDIASUBTYPE_YUY2, .formattype = FORMAT_VideoInfo, .cbFormat = sizeof(VIDEOINFOHEADER), .pbFormat = (BYTE *)&yuy2_video_info, }; const AM_MEDIA_TYPE video_mt = { .majortype = MEDIATYPE_Video, .subtype = MEDIASUBTYPE_RGB8, .formattype = FORMAT_VideoInfo, .cbFormat = sizeof(VIDEOINFOHEADER), .pbFormat = (BYTE *)&req_vih, }; static const GUID *subtypes[] = { &MEDIASUBTYPE_RGB24, &MEDIASUBTYPE_RGB32, &MEDIASUBTYPE_RGB555, &MEDIASUBTYPE_RGB565, &MEDIASUBTYPE_RGB1, &MEDIASUBTYPE_RGB4, &MEDIASUBTYPE_ARGB32, &MEDIASUBTYPE_ARGB1555, &MEDIASUBTYPE_ARGB4444, &MEDIASUBTYPE_Avi, &MEDIASUBTYPE_I420, &MEDIASUBTYPE_AYUV, &MEDIASUBTYPE_YV12, &MEDIASUBTYPE_YUY2, &MEDIASUBTYPE_UYVY, &MEDIASUBTYPE_YVYU, &MEDIASUBTYPE_NV12, &GUID_NULL, }; mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); mt = video_mt; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < ARRAY_SIZE(subtypes); ++i) { mt = video_mt; mt.subtype = *subtypes[i]; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == (i < 4 ? S_OK : VFW_E_TYPE_NOT_ACCEPTED), "Got hr %#lx.\n", hr); if (hr == S_OK) { hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); } } format = rgb8_format; format.dwFlags = DDSD_WIDTH; format.dwWidth = 222; format.dwHeight = 555; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); format = rgb8_format; format.dwFlags = DDSD_HEIGHT; format.dwWidth = 333; format.dwHeight = 444; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb555_video_info; video_info.bmiHeader.biWidth = 333; video_info.bmiHeader.biHeight = 444; mt = rgb555_mt; mt.pbFormat = (BYTE *)&video_info; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 332; video_info.bmiHeader.biHeight = 444; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 333; video_info.bmiHeader.biHeight = 443; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb8_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb555_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb8_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb555_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb565_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb555_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb565_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb24_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb565_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb24_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb32_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb24_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb32_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb8_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &yuy2_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &yuy2_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); format = yuy2_format; format.ddpfPixelFormat.u2.dwRBitMask = 0xf800; format.ddpfPixelFormat.u3.dwGBitMask = 0x07e0; format.ddpfPixelFormat.u4.dwBBitMask = 0x001f; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb565_mt); ok(hr == VFW_E_TYPE_NOT_ACCEPTED, "Got hr %#lx.\n", hr); format = rgb8_format; format.dwFlags = 0; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_ReceiveConnection(pin, &source.source.pin.IPin_iface, &rgb565_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Disconnect(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IBaseFilter_Release(&source.filter.IBaseFilter_iface); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostream_receive(void) { ALLOCATOR_PROPERTIES properties = { .cBuffers = 3, .cbBuffer = 16, .cbAlign = 1, }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); ALLOCATOR_PROPERTIES actual; struct testfilter source; IMemAllocator *allocator; IGraphBuilder *graph; IMediaStream *stream; IMediaSample *sample1; IMediaSample *sample2; IMediaSample *sample3; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_MemoryAllocator, NULL, CLSCTX_INPROC_SERVER, &IID_IMemAllocator, (void **)&allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_SetProperties(allocator, &properties, &actual); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_Commit(allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(allocator, &sample1, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample1); ok(hr == VFW_E_WRONG_STATE, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(allocator, &sample1, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample1); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(sample1); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IMemAllocator_GetBuffer(allocator, &sample2, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(sample2); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(allocator, &sample3, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample3); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(sample3); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaSample_Release(sample2); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaSample_Release(sample3); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IMemAllocator_GetBuffer(allocator, &sample1, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample1); ok(hr == VFW_E_WRONG_STATE, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); hr = IMemAllocator_Decommit(allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMemAllocator_Release(allocator); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostream_initialize(void) { IAMMediaStream *stream; STREAM_TYPE type; MSPID mspid; HRESULT hr; ULONG ref; hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Crashes on native. */ if (0) { hr = IAMMediaStream_Initialize(stream, NULL, 0, NULL, STREAMTYPE_WRITE); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } hr = IAMMediaStream_Initialize(stream, NULL, 0, &test_mspid, STREAMTYPE_WRITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_GetInformation(stream, &mspid, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&mspid, &test_mspid), "Got mspid %s.\n", wine_dbgstr_guid(&mspid)); ok(type == STREAMTYPE_WRITE, "Got type %u.\n", type); hr = IAMMediaStream_Initialize(stream, NULL, 0, &MSPID_PrimaryAudio, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_GetInformation(stream, &mspid, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&mspid, &MSPID_PrimaryAudio), "Got mspid %s.\n", wine_dbgstr_guid(&mspid)); ok(type == STREAMTYPE_READ, "Got type %u.\n", type); ref = IAMMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostream_begin_flush_end_flush(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const MSPID mspid = {0x88888888, 1}; IAudioStreamSample *stream_sample; IAudioMediaStream *audio_stream; IMediaStreamFilter *filter; IMediaSample *media_sample; struct testfilter source2; struct testfilter source; IAMMediaStream *stream2; IAudioData *audio_data; IMediaStream *stream; struct graph graph; HRESULT hr; IPin *pin2; ULONG ref; IPin *pin; graph_init(&graph); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph.IFilterGraph2_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream2, NULL, 0, &mspid, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream2, &mspid, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream2, &IID_IPin, (void **)&pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); testfilter_init(&source2); source.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source2.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data, 16, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_ConnectDirect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_ConnectDirect(&graph.IFilterGraph2_iface, &source2.source.pin.IPin_iface, pin2, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(media_sample); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(ref == 1, "Got outstanding refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; graph.event_code = 0xcc; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; graph.event_code = 0xcc; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, pin); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, pin2); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, &source2.source.pin.IPin_iface); ref = IAudioStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); graph_destroy(&graph); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin2); ref = IAMMediaStream_Release(stream2); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static IMediaSample *ammediastream_allocate_sample(struct testfilter *source, const BYTE *input_data, DWORD input_length) { IMediaSample *sample; BYTE *sample_data; HRESULT hr; hr = IMemAllocator_GetBuffer(source->source.pAllocator, &sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaSample_GetPointer(sample, &sample_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaSample_SetActualDataLength(sample, input_length); ok(hr == S_OK, "Got hr %#lx.\n", hr); memcpy(sample_data, input_data, input_length); return sample; } static void test_audiostream_new_segment(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const BYTE test_data[8] = { 0 }; IAudioStreamSample *stream_sample; IAudioMediaStream *audio_stream; IMemInputPin *mem_input_pin; IMediaSample *media_sample; struct testfilter source; IAudioData *audio_data; STREAM_TIME start_time; STREAM_TIME end_time; IGraphBuilder *graph; IMediaStream *stream; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data, 5, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_NewSegment(pin, 11111111, 22222222, 1.0); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 5); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 23456789, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 23459057, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IPin_NewSegment(pin, 11111111, 22222222, 2.0); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 5); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 23456789, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 23459057, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IAudioStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void CALLBACK apc_func(ULONG_PTR param) { } static IPin *ammediastream_pin; static IMemInputPin *ammediastream_mem_input_pin; static IMediaSample *ammediastream_media_sample; static DWORD ammediastream_sleep_time; static HRESULT ammediastream_expected_hr; static DWORD CALLBACK ammediastream_end_of_stream(void *param) { HRESULT hr; Sleep(ammediastream_sleep_time); hr = IPin_EndOfStream(ammediastream_pin); ok(hr == ammediastream_expected_hr, "Got hr %#lx.\n", hr); return 0; } static DWORD CALLBACK ammediastream_receive(void *param) { HRESULT hr; Sleep(ammediastream_sleep_time); hr = IMemInputPin_Receive(ammediastream_mem_input_pin, ammediastream_media_sample); ok(hr == ammediastream_expected_hr, "Got hr %#lx.\n", hr); return 0; } struct ammediastream_receive_release_param { IMemInputPin *mem_input_pin; IMediaSample *media_sample; }; static DWORD CALLBACK ammediastream_receive_release(void *p) { struct ammediastream_receive_release_param *param = (struct ammediastream_receive_release_param *)p; HRESULT hr; ULONG ref; hr = IMemInputPin_Receive(param->mem_input_pin, param->media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(param->media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); free(p); return 0; } static HANDLE ammediastream_async_receive_time(struct testfilter *source, REFERENCE_TIME start_time, REFERENCE_TIME end_time, const BYTE *input_data, DWORD input_length) { struct ammediastream_receive_release_param *param; IMediaSample *sample; HRESULT hr; sample = ammediastream_allocate_sample(source, input_data, input_length); hr = IMediaSample_SetTime(sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); param = calloc(1, sizeof(*param)); param->mem_input_pin = source->source.pMemInputPin; param->media_sample = sample; return CreateThread(NULL, 0, ammediastream_receive_release, param, 0, NULL); } static IStreamSample *streamsample_sample; static DWORD streamsample_flags; static DWORD streamsample_timeout; static HRESULT streamsample_expected_hr; static DWORD CALLBACK streamsample_completion_status(void *param) { HRESULT hr; hr = IStreamSample_CompletionStatus(streamsample_sample, streamsample_flags, streamsample_timeout); ok(hr == streamsample_expected_hr, "Got hr %#lx.\n", hr); return 0; } static void test_audiostreamsample_update(void) { static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7 }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioStreamSample *stream_sample; IAudioMediaStream *audio_stream; IMediaControl *media_control; IMemInputPin *mem_input_pin; IMediaSample *media_sample1; IMediaSample *media_sample2; struct testfilter source; IAudioData *audio_data; IGraphBuilder *graph; IMediaStream *stream; DWORD actual_length; BYTE buffer[6]; HANDLE thread; HANDLE event; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaControl, (void **)&media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); event = CreateEventW(NULL, FALSE, FALSE, NULL); ok(event != NULL, "Expected non-NULL event."); hr = IAudioStreamSample_Update(stream_sample, 0, event, apc_func, 0); ok(hr == MS_E_NOTINIT, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_E_NOTINIT, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data, sizeof(buffer), buffer, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, event, apc_func, 0); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_E_NOTRUNNING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample1 = ammediastream_allocate_sample(&source, test_data, 8); hr = IMemInputPin_Receive(mem_input_pin, media_sample1); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(media_sample1); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_length); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(actual_length == 6, "Got actual length %lu.\n", actual_length); ok(memcmp(buffer, test_data, 6) == 0, "Sample data didn't match.\n"); ref = get_refcount(media_sample1); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); media_sample2 = ammediastream_allocate_sample(&source, test_data, 8); hr = IMemInputPin_Receive(mem_input_pin, media_sample2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(media_sample2); ok(ref == 2, "Got unexpected refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_length); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(actual_length == 6, "Got actual length %lu.\n", actual_length); ok(memcmp(buffer, &test_data[6], 2) == 0, "Sample data didn't match.\n"); ok(memcmp(&buffer[2], test_data, 4) == 0, "Sample data didn't match.\n"); ref = IMediaSample_Release(media_sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_length); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(actual_length == 4, "Got actual length %lu.\n", actual_length); ok(memcmp(buffer, &test_data[4], 4) == 0, "Sample data didn't match.\n"); ref = IMediaSample_Release(media_sample2); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaControl_Pause(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample1 = ammediastream_allocate_sample(&source, test_data, 6); hr = IMemInputPin_Receive(mem_input_pin, media_sample1); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample1); ok(ref == 1, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_E_NOTRUNNING, "Got hr %#lx.\n", hr); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample1 = ammediastream_allocate_sample(&source, test_data, 6); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample1; ammediastream_sleep_time = 100; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_GetInfo(audio_data, NULL, NULL, &actual_length); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(actual_length == 6, "Got actual length %lu.\n", actual_length); ok(memcmp(buffer, test_data, 6) == 0, "Sample data didn't match.\n"); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); ammediastream_pin = pin; ammediastream_sleep_time = 100; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_end_of_stream, NULL, 0, NULL); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); IAudioStreamSample_AddRef(stream_sample); ref = IAudioStreamSample_Release(stream_sample); ok(ref == 1, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_E_BUSY, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); CloseHandle(event); ref = IAudioStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaControl_Release(media_control); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } void test_audiostreamsample_completion_status(void) { static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioStreamSample *stream_sample1; IAudioStreamSample *stream_sample2; IAudioMediaStream *audio_stream; IMediaSample *media_sample; struct testfilter source; IAudioData *audio_data1; IAudioData *audio_data2; IGraphBuilder *graph; IMediaStream *stream; HANDLE event; HRESULT hr; ULONG ref; IPin *pin; event = CreateEventW(NULL, FALSE, FALSE, NULL); ok(event != NULL, "Expected non-NULL event."); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data1, 6, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data2, 6, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data1, 0, &stream_sample1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data2, 0, &stream_sample2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 6); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample2, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 12); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 6); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAudioStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IAudioStreamSample_Release(stream_sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioStreamSample_Release(stream_sample2); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data1); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data2); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); CloseHandle(event); } static void test_audiostreamsample_get_sample_times(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const BYTE test_data[8] = { 0 }; IAudioStreamSample *stream_sample; IMediaFilter *graph_media_filter; IAudioMediaStream *audio_stream; STREAM_TIME filter_start_time; IMemInputPin *mem_input_pin; IMediaStreamFilter *filter; IMediaSample *media_sample; struct testfilter source; STREAM_TIME current_time; struct testclock clock; IAudioData *audio_data; STREAM_TIME start_time; STREAM_TIME end_time; IGraphBuilder *graph; IMediaStream *stream; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&graph_media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioData_SetBuffer(audio_data, 5, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); clock.time = 12345678; current_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0, "Got current time %s.\n", wine_dbgstr_longlong(current_time)); hr = IMediaFilter_SetSyncSource(graph_media_filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); current_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0, "Got current time %s.\n", wine_dbgstr_longlong(current_time)); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &filter_start_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.get_time_hr = E_FAIL; current_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0xdeadbeefddf15da1 + filter_start_time, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(0xdeadbeefddf15da1 + filter_start_time), wine_dbgstr_longlong(current_time)); clock.get_time_hr = S_OK; current_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == filter_start_time, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(filter_start_time), wine_dbgstr_longlong(current_time)); clock.time = 23456789; current_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == filter_start_time + 11111111, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(filter_start_time + 11111111), wine_dbgstr_longlong(current_time)); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 0, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 0, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); media_sample = ammediastream_allocate_sample(&source, test_data, 8); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 12345678, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 12347946, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); media_sample = ammediastream_allocate_sample(&source, test_data, 6); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 12347946, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 12346585, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IAudioStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 12346585, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 12348399, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IAudioStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(graph_media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IAudioMediaStream_Release(audio_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_initialize(void) { IDirectDrawMediaStream *ddraw_stream; IAMMediaStream *stream; IDirectDraw *ddraw2; IDirectDraw *ddraw; STREAM_TYPE type; MSPID mspid; HRESULT hr; ULONG ref; hr = DirectDrawCreate(NULL, &ddraw, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMDirectDrawStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Crashes on native. */ if (0) { hr = IAMMediaStream_Initialize(stream, NULL, 0, NULL, STREAMTYPE_WRITE); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } hr = IAMMediaStream_Initialize(stream, NULL, 0, &test_mspid, STREAMTYPE_WRITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_GetInformation(stream, &mspid, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&mspid, &test_mspid), "Got mspid %s.\n", wine_dbgstr_guid(&mspid)); ok(type == STREAMTYPE_WRITE, "Got type %u.\n", type); hr = IAMMediaStream_Initialize(stream, (IUnknown *)ddraw, 0, &MSPID_PrimaryAudio, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_GetInformation(stream, &mspid, &type); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&mspid, &MSPID_PrimaryAudio), "Got mspid %s.\n", wine_dbgstr_guid(&mspid)); ok(type == STREAMTYPE_READ, "Got type %u.\n", type); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw2 == ddraw, "Expected ddraw %p, got %p.\n", ddraw, ddraw2); IDirectDrawMediaStream_Release(ddraw_stream); ref = IAMMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IDirectDraw_Release(ddraw2); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); } #define check_ddrawstream_get_format(a,b,c) check_ddrawstream_get_format_(__LINE__,a,b,c) static void check_ddrawstream_get_format_(int line, IDirectDrawMediaStream *stream, const AM_MEDIA_TYPE *mt, const DDSURFACEDESC *expected_format) { DDSURFACEDESC current_format; DDSURFACEDESC desired_format; struct testfilter source; FILTER_INFO filter_info; DDSURFACEDESC format; PIN_INFO pin_info; DWORD flags; HRESULT hr; IPin *pin; hr = IDirectDrawMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_QueryPinInfo(pin, &pin_info); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IBaseFilter_QueryFilterInfo(pin_info.pFilter, &filter_info); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IFilterGraph_AddFilter(filter_info.pGraph, &source.filter.IBaseFilter_iface, L"source"); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph_ConnectDirect(filter_info.pGraph, &source.source.pin.IPin_iface, pin, mt); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetFormat(stream, NULL, NULL, NULL, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); memset(¤t_format, 0xcc, sizeof(current_format)); current_format.dwSize = sizeof(current_format); memset(&desired_format, 0xcc, sizeof(desired_format)); desired_format.dwSize = sizeof(desired_format); flags = 0xdeadbeef; hr = IDirectDrawMediaStream_GetFormat(stream, ¤t_format, NULL, &desired_format, &flags); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); memset(&format, 0xcc, sizeof(format)); format.dwSize = sizeof(format); format.ddpfPixelFormat = expected_format->ddpfPixelFormat; format.dwFlags = DDSD_WIDTH | DDSD_HEIGHT | DDSD_CAPS; format.dwWidth = 333; format.dwHeight = 444; format.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN | DDSCAPS_SYSTEMMEMORY; ok_(__FILE__, line)(memcmp(¤t_format, &format, sizeof(DDSURFACEDESC)) == 0, "Current format didn't match.\n"); format.dwFlags = DDSD_WIDTH | DDSD_HEIGHT; ok_(__FILE__, line)(memcmp(&desired_format, &format, sizeof(DDSURFACEDESC)) == 0, "Desired format didn't match.\n"); hr = IFilterGraph_Disconnect(filter_info.pGraph, &source.source.pin.IPin_iface); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph_Disconnect(filter_info.pGraph, pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph_RemoveFilter(filter_info.pGraph, &source.filter.IBaseFilter_iface); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); IFilterGraph_Release(filter_info.pGraph); IBaseFilter_Release(pin_info.pFilter); IPin_Release(pin); } static void test_ddrawstream_get_format(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawMediaStream *ddraw_stream; DDSURFACEDESC current_format; DDSURFACEDESC desired_format; IDirectDrawPalette *palette; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; DWORD flags; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); current_format.dwSize = sizeof(current_format); desired_format.dwSize = sizeof(desired_format); hr = IDirectDrawMediaStream_GetFormat(ddraw_stream, ¤t_format, &palette, &desired_format, &flags); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.rcSource.right = 222; video_info.rcSource.bottom = 333; video_info.rcTarget.right = 444; video_info.rcTarget.bottom = 666; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; check_ddrawstream_get_format(ddraw_stream, &mt, &rgb32_format); video_info = rgb32_video_info; video_info.bmiHeader.biHeight = 444; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; check_ddrawstream_get_format(ddraw_stream, &mt, &rgb32_format); check_ddrawstream_get_format(ddraw_stream, &rgb8_mt, &rgb8_format); check_ddrawstream_get_format(ddraw_stream, &rgb555_mt, &rgb555_format); check_ddrawstream_get_format(ddraw_stream, &rgb565_mt, &rgb565_format); check_ddrawstream_get_format(ddraw_stream, &rgb24_mt, &rgb24_format); check_ddrawstream_get_format(ddraw_stream, &rgb32_mt, &rgb32_format); current_format.dwSize = sizeof(current_format); desired_format.dwSize = sizeof(desired_format); hr = IDirectDrawMediaStream_GetFormat(ddraw_stream, ¤t_format, &palette, &desired_format, &flags); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } #define check_ddrawstream_set_format(a,b,c,d) check_ddrawstream_set_format_(__LINE__,a,b,c,d) static void check_ddrawstream_set_format_(int line, IDirectDrawMediaStream *stream, const DDSURFACEDESC *format, const AM_MEDIA_TYPE *mt, HRESULT expected_hr) { struct testfilter source; FILTER_INFO filter_info; PIN_INFO pin_info; HRESULT hr; IPin *pin; hr = IDirectDrawMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_QueryPinInfo(pin, &pin_info); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IBaseFilter_QueryFilterInfo(pin_info.pFilter, &filter_info); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IFilterGraph_AddFilter(filter_info.pGraph, &source.filter.IBaseFilter_iface, L"source"); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(stream, format, NULL); ok_(__FILE__, line)(hr == expected_hr, "Got hr %#lx.\n", hr); if (mt) { DDSURFACEDESC current_format; DDSURFACEDESC desired_format; DWORD flags; hr = IFilterGraph_ConnectDirect(filter_info.pGraph, &source.source.pin.IPin_iface, pin, mt); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); memset(¤t_format, 0xcc, sizeof(current_format)); memset(&desired_format, 0xcc, sizeof(desired_format)); flags = 0xdeadbeef; current_format.dwSize = sizeof(current_format); desired_format.dwSize = sizeof(desired_format); hr = IDirectDrawMediaStream_GetFormat(stream, ¤t_format, NULL, &desired_format, &flags); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); if (format->dwFlags & DDSD_PIXELFORMAT) { ok_(__FILE__, line)(current_format.dwFlags == (DDSD_WIDTH | DDSD_HEIGHT | DDSD_CAPS | DDSD_PIXELFORMAT), "Got current format flags %#lx.\n", current_format.dwFlags); ok_(__FILE__, line)(memcmp(¤t_format.ddpfPixelFormat, &format->ddpfPixelFormat, sizeof(DDPIXELFORMAT)) == 0, "Current pixel format didn't match.\n"); ok_(__FILE__, line)(memcmp(&desired_format.ddpfPixelFormat, &format->ddpfPixelFormat, sizeof(DDPIXELFORMAT)) == 0, "Desired pixel format didn't match.\n"); } else { ok_(__FILE__, line)(current_format.dwFlags == (DDSD_WIDTH | DDSD_HEIGHT | DDSD_CAPS), "Got flags %#lx.\n", current_format.dwFlags); } ok_(__FILE__, line)(desired_format.dwFlags == (DDSD_WIDTH | DDSD_HEIGHT), "Got desired format flags %#lx.\n", desired_format.dwFlags); ok_(__FILE__, line)(current_format.ddsCaps.dwCaps == (DDSCAPS_OFFSCREENPLAIN | DDSCAPS_SYSTEMMEMORY), "Got current format caps %#lx.\n", current_format.ddsCaps.dwCaps); ok_(__FILE__, line)(desired_format.ddsCaps.dwCaps == (DDSCAPS_OFFSCREENPLAIN | DDSCAPS_SYSTEMMEMORY), "Got desired format caps %#lx.\n", desired_format.ddsCaps.dwCaps); ok_(__FILE__, line)(flags == 0, "Got flags %#lx.\n", flags); hr = IFilterGraph_Disconnect(filter_info.pGraph, &source.source.pin.IPin_iface); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph_Disconnect(filter_info.pGraph, pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); } hr = IFilterGraph_RemoveFilter(filter_info.pGraph, &source.filter.IBaseFilter_iface); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); IFilterGraph_Release(filter_info.pGraph); IBaseFilter_Release(pin_info.pFilter); IPin_Release(pin); } static void test_ddrawstream_set_format(void) { static const DDSURFACEDESC rgb1_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXED1, .ddpfPixelFormat.u1.dwRGBBitCount = 1, }; static const DDSURFACEDESC rgb2_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXED2, .ddpfPixelFormat.u1.dwRGBBitCount = 2, }; static const DDSURFACEDESC rgb4_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXED4, .ddpfPixelFormat.u1.dwRGBBitCount = 4, }; static const DDSURFACEDESC rgb4to8_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXEDTO8, .ddpfPixelFormat.u1.dwRGBBitCount = 4, }; static const DDSURFACEDESC rgb332_format = { .dwSize = sizeof(DDSURFACEDESC), .dwFlags = DDSD_PIXELFORMAT, .ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT), .ddpfPixelFormat.dwFlags = DDPF_RGB, .ddpfPixelFormat.u1.dwRGBBitCount = 8, .ddpfPixelFormat.u2.dwRBitMask = 0xe0, .ddpfPixelFormat.u3.dwGBitMask = 0x1c, .ddpfPixelFormat.u4.dwBBitMask = 0x03, }; IDirectDrawMediaStream *ddraw_stream; IAMMultiMediaStream *mmstream; DDSURFACEDESC current_format; DDSURFACEDESC desired_format; struct testfilter source; IGraphBuilder *graph; DDSURFACEDESC format; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_ddrawstream_set_format(ddraw_stream, &rgb8_format, &rgb8_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &rgb555_format, &rgb555_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &rgb565_format, &rgb565_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &rgb24_format, &rgb24_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &rgb32_format, &rgb32_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &argb32_format, &rgb32_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &yuy2_format, NULL, S_OK); check_ddrawstream_set_format(ddraw_stream, &yv12_format, NULL, S_OK); format = rgb32_format; format.ddpfPixelFormat.dwFlags |= DDPF_ALPHAPIXELS | DDPF_ALPHA | DDPF_COMPRESSED | DDPF_RGBTOYUV | DDPF_ZBUFFER | DDPF_ZPIXELS | DDPF_STENCILBUFFER | DDPF_ALPHAPREMULT | DDPF_LUMINANCE | DDPF_BUMPLUMINANCE | DDPF_BUMPDUDV; check_ddrawstream_set_format(ddraw_stream, &format, &rgb32_mt, S_OK); format = yuy2_format; format.ddpfPixelFormat.dwFlags |= DDPF_ALPHAPIXELS | DDPF_ALPHA | DDPF_PALETTEINDEXED4 | DDPF_PALETTEINDEXEDTO8 | DDPF_PALETTEINDEXED8 | DDPF_RGB | DDPF_COMPRESSED | DDPF_RGBTOYUV | DDPF_YUV | DDPF_ZBUFFER | DDPF_PALETTEINDEXED1 | DDPF_PALETTEINDEXED2 | DDPF_ZPIXELS | DDPF_STENCILBUFFER | DDPF_ALPHAPREMULT | DDPF_LUMINANCE | DDPF_BUMPLUMINANCE | DDPF_BUMPDUDV; check_ddrawstream_set_format(ddraw_stream, &format, NULL, S_OK); format = rgb32_format; format.dwFlags |= DDSD_CAPS; format.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN | DDSCAPS_VIDEOMEMORY; check_ddrawstream_set_format(ddraw_stream, &format, &rgb32_mt, S_OK); format = rgb8_format; format.dwFlags = 0; check_ddrawstream_set_format(ddraw_stream, &format, &rgb32_mt, S_OK); check_ddrawstream_set_format(ddraw_stream, &rgb1_format, NULL, DDERR_INVALIDSURFACETYPE); check_ddrawstream_set_format(ddraw_stream, &rgb2_format, NULL, DDERR_INVALIDSURFACETYPE); check_ddrawstream_set_format(ddraw_stream, &rgb4_format, NULL, DDERR_INVALIDSURFACETYPE); check_ddrawstream_set_format(ddraw_stream, &rgb4to8_format, NULL, DDERR_INVALIDSURFACETYPE); check_ddrawstream_set_format(ddraw_stream, &rgb332_format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb8_format; format.ddpfPixelFormat.dwFlags &= ~DDPF_RGB; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb8_format; format.ddpfPixelFormat.dwFlags |= DDPF_PALETTEINDEXED1; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb32_format; format.ddpfPixelFormat.dwFlags |= DDPF_PALETTEINDEXED8; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb32_format; format.ddpfPixelFormat.dwFlags |= DDPF_YUV; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb565_format; format.ddpfPixelFormat.u2.dwRBitMask = 0x001f; format.ddpfPixelFormat.u3.dwGBitMask = 0x07e0; format.ddpfPixelFormat.u4.dwBBitMask = 0xf800; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = rgb32_format; format.ddpfPixelFormat.u2.dwRBitMask = 0x00ff00; format.ddpfPixelFormat.u3.dwGBitMask = 0x0000ff; format.ddpfPixelFormat.u4.dwBBitMask = 0xff0000; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); format = yuy2_format; format.ddpfPixelFormat.u1.dwYUVBitCount = 0; check_ddrawstream_set_format(ddraw_stream, &format, NULL, E_INVALIDARG); format = rgb32_format; format.dwSize = sizeof(DDSURFACEDESC) + 1; check_ddrawstream_set_format(ddraw_stream, &format, NULL, E_INVALIDARG); format = rgb32_format; format.ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT) + 1; check_ddrawstream_set_format(ddraw_stream, &format, NULL, DDERR_INVALIDSURFACETYPE); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); mmstream = create_ammultimediastream(); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!graph, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, L"source"); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &rgb8_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); source.preferred_mt = NULL; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb555_format, NULL); ok(hr == DDERR_INVALIDSURFACETYPE, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&source.source.pin.mt.subtype, &MEDIASUBTYPE_RGB8), "Got subtype %s.\n", wine_dbgstr_guid(&source.source.pin.mt.subtype)); hr = IDirectDrawMediaStream_GetFormat(ddraw_stream, ¤t_format, NULL, &desired_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_format.ddpfPixelFormat.u1.dwRGBBitCount == 8, "Got rgb bit count %lu.\n", current_format.ddpfPixelFormat.u1.dwRGBBitCount); ok(desired_format.ddpfPixelFormat.u1.dwRGBBitCount == 8, "Got rgb bit count %lu.\n", desired_format.ddpfPixelFormat.u1.dwRGBBitCount); format = rgb555_format; format.dwFlags = 0; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&source.source.pin.mt.subtype, &MEDIASUBTYPE_RGB8), "Got subtype %s.\n", wine_dbgstr_guid(&source.source.pin.mt.subtype)); source.preferred_mt = &rgb555_mt; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb8_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &rgb555_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&source.source.pin.mt.subtype, &MEDIASUBTYPE_RGB555), "Got subtype %s.\n", wine_dbgstr_guid(&source.source.pin.mt.subtype)); hr = IDirectDrawMediaStream_GetFormat(ddraw_stream, ¤t_format, NULL, &desired_format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_format.ddpfPixelFormat.u1.dwRGBBitCount == 16, "Got rgb bit count %lu.\n", current_format.ddpfPixelFormat.u1.dwRGBBitCount); ok(desired_format.ddpfPixelFormat.u1.dwRGBBitCount == 16, "Got rgb bit count %lu.\n", desired_format.ddpfPixelFormat.u1.dwRGBBitCount); video_info = rgb555_video_info; video_info.bmiHeader.biWidth = 222; video_info.bmiHeader.biHeight = -555; mt = rgb555_mt; mt.pbFormat = (BYTE *)&video_info; source.preferred_mt = &mt; format = rgb555_format; format.dwFlags |= DDSD_WIDTH | DDSD_HEIGHT; format.dwWidth = 222; format.dwHeight = 555; hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, &format, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(IsEqualGUID(&source.source.pin.mt.subtype, &MEDIASUBTYPE_RGB555), "Got subtype %s.\n", wine_dbgstr_guid(&source.source.pin.mt.subtype)); ok(((VIDEOINFO *)source.source.pin.mt.pbFormat)->bmiHeader.biWidth == 222, "Got width %ld.\n", ((VIDEOINFO *)source.source.pin.mt.pbFormat)->bmiHeader.biWidth); ok(((VIDEOINFO *)source.source.pin.mt.pbFormat)->bmiHeader.biHeight == -555, "Got height %ld.\n", ((VIDEOINFO *)source.source.pin.mt.pbFormat)->bmiHeader.biHeight); hr = IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_receive(void) { ALLOCATOR_PROPERTIES properties = { .cBuffers = 1, .cbBuffer = 16, .cbAlign = 1, }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); ALLOCATOR_PROPERTIES actual; IMediaStreamFilter *filter; struct testfilter source; IMemAllocator *allocator; IGraphBuilder *graph; IMediaStream *stream; IMediaSample *sample; HANDLE thread; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_MemoryAllocator, NULL, CLSCTX_INPROC_SERVER, &IID_IMemAllocator, (void **)&allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_SetProperties(allocator, &properties, &actual); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_Commit(allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(allocator, &sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(allocator, &sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); ammediastream_mem_input_pin = source.source.pMemInputPin; ammediastream_media_sample = sample; ammediastream_sleep_time = 0; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IMemAllocator_GetBuffer(allocator, &sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); hr = IMemAllocator_Decommit(allocator); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMemAllocator_Release(allocator); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_begin_flush_end_flush(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const MSPID mspid = {0x88888888, 1}; IDirectDrawStreamSample *stream_sample; IDirectDrawMediaStream *ddraw_stream; IMediaStreamFilter *filter; IMediaSample *media_sample; IMediaFilter *media_filter; struct testfilter source2; struct testfilter source; IAMMediaStream *stream2; IMediaStream *stream; VIDEOINFO video_info; struct graph graph; AM_MEDIA_TYPE mt; HANDLE thread; HRESULT hr; IPin *pin2; ULONG ref; IPin *pin; graph_init(&graph); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph.IFilterGraph2_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMDirectDrawStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream2, NULL, 0, &mspid, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream2, &mspid, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream2, &IID_IPin, (void **)&pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_QueryInterface(&graph.IFilterGraph2_iface, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); testfilter_init(&source2); source.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source2.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaFilter_SetSyncSource(media_filter, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 3; video_info.bmiHeader.biHeight = 1; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IFilterGraph2_ConnectDirect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_ConnectDirect(&graph.IFilterGraph2_iface, &source2.source.pin.IPin_iface, pin2, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); ammediastream_mem_input_pin = source.source.pMemInputPin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 0; ammediastream_expected_hr = S_FALSE; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemAllocator_GetBuffer(source.source.pAllocator, &media_sample, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; graph.event_code = 0xcc; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; graph.event_code = 0xcc; hr = IPin_EndOfStream(pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, pin); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, pin2); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, &source2.source.pin.IPin_iface); ref = IDirectDrawStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); graph_destroy(&graph); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_new_segment(void) { static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawStreamSample *stream_sample; IDirectDrawMediaStream *ddraw_stream; IMemInputPin *mem_input_pin; IMediaSample *media_sample; IMediaFilter *media_filter; struct testfilter source; STREAM_TIME start_time; STREAM_TIME end_time; IGraphBuilder *graph; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaFilter_SetSyncSource(media_filter, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 3; video_info.bmiHeader.biHeight = 1; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_NewSegment(pin, 11111111, 22222222, 1.0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 23456789, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 34567900, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IPin_NewSegment(pin, 11111111, 22222222, 2.0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 23456789, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 34567900, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IDirectDrawStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_get_time_per_frame(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawMediaStream *ddraw_stream; struct testfilter source; STREAM_TIME frame_time; IGraphBuilder *graph; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetTimePerFrame(ddraw_stream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetTimePerFrame(ddraw_stream, &frame_time); ok(hr == MS_E_NOSTREAM, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.AvgTimePerFrame = 12345678; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); frame_time = 0xdeadbeefdeadbeef; hr = IDirectDrawMediaStream_GetTimePerFrame(ddraw_stream, &frame_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(frame_time == 12345678, "Got frame time %s.\n", wine_dbgstr_longlong(frame_time)); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void check_ammediastream_join_am_multi_media_stream(const CLSID *clsid) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMultiMediaStream *mmstream2; IAMMediaStream *stream; HRESULT hr; ULONG mmstream_ref; ULONG ref; hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); mmstream_ref = get_refcount(mmstream); hr = IAMMediaStream_JoinAMMultiMediaStream(stream, mmstream); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = get_refcount(mmstream); ok(ref == mmstream_ref, "Expected outstanding refcount %ld, got %ld.\n", mmstream_ref, ref); hr = IAMMediaStream_GetMultiMediaStream(stream, &mmstream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(mmstream2 == (IMultiMediaStream *)mmstream, "Expected mmstream %p, got %p.\n", mmstream, mmstream2); IMultiMediaStream_Release(mmstream2); hr = IAMMediaStream_JoinAMMultiMediaStream(stream, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_GetMultiMediaStream(stream, &mmstream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(mmstream2 == NULL, "Got mmstream %p.\n", mmstream2); ref = IAMMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ammediastream_join_am_multi_media_stream(void) { check_ammediastream_join_am_multi_media_stream(&CLSID_AMAudioStream); check_ammediastream_join_am_multi_media_stream(&CLSID_AMDirectDrawStream); } static void check_ammediastream_join_filter(const CLSID *clsid) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter, *filter2, *filter3; IAMMediaStream *stream; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); EXPECT_REF(filter, 3); hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(filter, 3); hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IMediaStreamFilter, (void **)&filter2); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(filter, 3); EXPECT_REF(filter2, 1); hr = IAMMediaStream_JoinFilter(stream, filter2); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(filter, 3); EXPECT_REF(filter2, 1); /* Crashes on native. */ if (0) { hr = IAMMediaStream_JoinFilter(stream, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); } hr = IAMMultiMediaStream_GetFilter(mmstream, &filter3); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(filter3 == filter, "Expected filter %p, got %p.\n", filter, filter3); EXPECT_REF(filter, 4); IMediaStreamFilter_Release(filter3); EXPECT_REF(filter, 3); ref = IMediaStreamFilter_Release(filter2); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); EXPECT_REF(filter, 3); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); EXPECT_REF(filter, 1); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ammediastream_join_filter(void) { check_ammediastream_join_filter(&CLSID_AMAudioStream); check_ammediastream_join_filter(&CLSID_AMDirectDrawStream); } static void check_ammediastream_join_filter_graph(const MSPID *id) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IGraphBuilder *builder, *builder2; IMediaStreamFilter *filter; IAMMediaStream *stream; IFilterGraph *graph; FILTER_INFO info; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, id, 0, (IMediaStream **)&stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &builder); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!builder, "Expected non-null graph.\n"); EXPECT_REF(builder, 4); hr = IMediaStreamFilter_QueryFilterInfo(filter, &info); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(info.pGraph == (IFilterGraph *)builder, "Expected graph %p, got %p.\n", (IFilterGraph *)builder, info.pGraph); EXPECT_REF(builder, 5); IFilterGraph_Release(info.pGraph); EXPECT_REF(builder, 4); hr = CoCreateInstance(&CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, &IID_IFilterGraph, (void **)&graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(builder, 4); EXPECT_REF(graph, 1); /* Crashes on native. */ if (0) { hr = IAMMediaStream_JoinFilterGraph(stream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } hr = IAMMediaStream_JoinFilterGraph(stream, graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(builder, 4); EXPECT_REF(graph, 1); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &builder2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(builder2 == builder, "Expected graph %p, got %p.\n", builder, builder2); EXPECT_REF(builder, 5); EXPECT_REF(graph, 1); IGraphBuilder_Release(builder2); EXPECT_REF(builder, 4); EXPECT_REF(graph, 1); hr = IMediaStreamFilter_QueryFilterInfo(filter, &info); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(info.pGraph == (IFilterGraph *)builder, "Expected graph %p, got %p.\n", (IFilterGraph *)builder, info.pGraph); EXPECT_REF(builder, 5); EXPECT_REF(graph, 1); IFilterGraph_Release(info.pGraph); EXPECT_REF(builder, 4); EXPECT_REF(graph, 1); ref = IFilterGraph_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(builder); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ammediastream_join_filter_graph(void) { check_ammediastream_join_filter_graph(&MSPID_PrimaryAudio); check_ammediastream_join_filter_graph(&MSPID_PrimaryVideo); } static void check_ammediastream_set_state(const MSPID *id) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAMMediaStream *am_stream; IMediaStream *stream; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, id, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAMMediaStream, (void **)&am_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_SetState(am_stream, 4); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_SetState(am_stream, State_Running); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_SetState(am_stream, State_Paused); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_SetState(am_stream, State_Stopped); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IAMMediaStream_Release(am_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ammediastream_set_state(void) { check_ammediastream_set_state(&MSPID_PrimaryAudio); check_ammediastream_set_state(&MSPID_PrimaryVideo); } static void check_ammediastream_end_of_stream(const CLSID *clsid, const MSPID *id, const AM_MEDIA_TYPE *mt) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; struct testfilter source; IMediaStream *stream; struct graph graph; HRESULT hr; ULONG ref; IPin *pin; graph_init(&graph); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph.IFilterGraph2_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, id, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); source.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_ConnectDirect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface, pin, mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; graph.event_code = 0xcc; hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); ok(graph.event_code == EC_COMPLETE, "Got event code %ld.\n", graph.event_code); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, pin); IFilterGraph2_Disconnect(&graph.IFilterGraph2_iface, &source.source.pin.IPin_iface); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); graph_destroy(&graph); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ammediastream_end_of_stream(void) { check_ammediastream_end_of_stream(&CLSID_AMAudioStream, &MSPID_PrimaryAudio, &audio_mt); check_ammediastream_end_of_stream(&CLSID_AMDirectDrawStream, &MSPID_PrimaryVideo, &rgb32_mt); } void test_mediastreamfilter_get_state(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IMediaStreamFilter *filter; FILTER_STATE state; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); /* Crashes on native. */ if (0) { hr = IMediaStreamFilter_GetState(filter, 0, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } state = 0xcc; hr = IMediaStreamFilter_GetState(filter, 0, &state); ok(state == State_Stopped, "Got state %#x.\n", state); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } void check_mediastreamfilter_state(FILTER_STATE expected_state, HRESULT (*set_state)(IMediaStreamFilter *), HRESULT (*reset_state)(IMediaStreamFilter *)) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); struct teststream teststream, teststream2; IMediaStreamFilter *filter; FILTER_STATE state; HRESULT hr; ULONG ref; teststream_init(&teststream); teststream_init(&teststream2); teststream2.mspid.Data2 = 1; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)&teststream, &teststream.mspid, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)&teststream2, &teststream2.mspid, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(filter != NULL, "Expected non-null filter\n"); hr = reset_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); teststream.state = 0xcc; teststream2.state = 0xcc; hr = set_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(teststream.state == expected_state, "Got state %#x.\n", teststream.state); ok(teststream2.state == expected_state, "Got state %#x.\n", teststream2.state); hr = IMediaStreamFilter_GetState(filter, 0, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == expected_state, "Got state %#x.\n", state); teststream.state = 0xcc; teststream2.state = 0xcc; hr = set_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(teststream.state == 0xcc, "Got state %#x.\n", teststream.state); ok(teststream2.state == 0xcc, "Got state %#x.\n", teststream2.state); hr = reset_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); teststream.set_state_result = E_FAIL; teststream.state = 0xcc; teststream2.state = 0xcc; hr = set_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(teststream.state == 0xcc, "Got state %#x.\n", teststream.state); ok(teststream2.state == expected_state, "Got state %#x.\n", teststream2.state); hr = IMediaStreamFilter_GetState(filter, 0, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == expected_state, "Got state %#x.\n", state); hr = reset_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); teststream.set_state_result = E_FAIL; teststream2.set_state_result = E_FAIL; teststream.state = 0xcc; teststream2.state = 0xcc; hr = set_state(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(teststream.state == 0xcc, "Got state %#x.\n", teststream.state); ok(teststream2.state == 0xcc, "Got state %#x.\n", teststream2.state); hr = IMediaStreamFilter_GetState(filter, 0, &state); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(state == expected_state, "Got state %#x.\n", state); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); ok(teststream.refcount == 1, "Got outstanding refcount %ld.\n", teststream.refcount); ok(teststream2.refcount == 1, "Got outstanding refcount %ld.\n", teststream2.refcount); } static HRESULT mediastreamfilter_stop(IMediaStreamFilter *filter) { return IMediaStreamFilter_Stop(filter); } static HRESULT mediastreamfilter_pause(IMediaStreamFilter *filter) { return IMediaStreamFilter_Pause(filter); } static HRESULT mediastreamfilter_run(IMediaStreamFilter *filter) { return IMediaStreamFilter_Run(filter, 0); } void test_mediastreamfilter_stop_pause_run(void) { check_mediastreamfilter_state(State_Stopped, mediastreamfilter_stop, mediastreamfilter_run); check_mediastreamfilter_state(State_Paused, mediastreamfilter_pause, mediastreamfilter_stop); check_mediastreamfilter_state(State_Running, mediastreamfilter_run, mediastreamfilter_stop); } static void test_mediastreamfilter_support_seeking(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const MSPID mspid1 = {0x88888888, 1}; static const MSPID mspid2 = {0x88888888, 2}; static const MSPID mspid3 = {0x88888888, 3}; struct testfilter source1, source2, source3; IAMMediaStream *stream1, *stream2, *stream3; IMediaStreamFilter *filter; IPin *pin1, *pin2, *pin3; ULONG ref, seeking_ref; IGraphBuilder *graph; HRESULT hr; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream3); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream1, NULL, 0, &mspid1, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream2, NULL, 0, &mspid2, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream3, NULL, 0, &mspid3, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream1, &mspid1, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream2, &mspid2, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream3, &mspid3, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream1, &IID_IPin, (void **)&pin1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream2, &IID_IPin, (void **)&pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream3, &IID_IPin, (void **)&pin3); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source1); testfilter_init(&source2); testfilter_init(&source3); source2.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; source3.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IGraphBuilder_AddFilter(graph, &source1.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_AddFilter(graph, &source2.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_AddFilter(graph, &source3.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == E_NOINTERFACE, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source1.source.pin.IPin_iface, pin1, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); source2.get_duration_hr = E_FAIL; hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == E_NOINTERFACE, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source2.source.pin.IPin_iface, pin2, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == E_NOINTERFACE, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source3.source.pin.IPin_iface, pin3, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, FALSE); seeking_ref = get_refcount(&source3.IMediaSeeking_iface); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); check_interface(filter, &IID_IMediaSeeking, TRUE); ref = get_refcount(&source3.IMediaSeeking_iface); ok(ref == seeking_ref, "Expected outstanding refcount %ld, got %ld.\n", seeking_ref, ref); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == HRESULT_FROM_WIN32(ERROR_ALREADY_INITIALIZED), "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin1); IGraphBuilder_Disconnect(graph, &source1.source.pin.IPin_iface); IGraphBuilder_Disconnect(graph, pin2); IGraphBuilder_Disconnect(graph, &source2.source.pin.IPin_iface); IGraphBuilder_Disconnect(graph, pin3); IGraphBuilder_Disconnect(graph, &source3.source.pin.IPin_iface); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin1); ref = IAMMediaStream_Release(stream1); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin2); ref = IAMMediaStream_Release(stream2); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin3); ref = IAMMediaStream_Release(stream3); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void check_mediastreamfilter_seeking(void (*check)(IMediaSeeking *seeking, struct testfilter *source1, struct testfilter *source2, struct testfilter *source3, HRESULT source2_hr, HRESULT expected_hr)) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); static const MSPID mspid1 = {0x88888888, 1}; static const MSPID mspid2 = {0x88888888, 2}; static const MSPID mspid3 = {0x88888888, 3}; IMediaStreamFilter *filter; struct testfilter source1; struct testfilter source2; struct testfilter source3; IAMMediaStream *stream1; IAMMediaStream *stream2; IAMMediaStream *stream3; IMediaSeeking *seeking; IGraphBuilder *graph; IPin *pin1; IPin *pin2; IPin *pin3; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioStream, NULL, CLSCTX_INPROC_SERVER, &IID_IAMMediaStream, (void **)&stream3); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream1, NULL, 0, &mspid1, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream2, NULL, 0, &mspid2, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_Initialize(stream3, NULL, 0, &mspid3, STREAMTYPE_READ); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream1, &mspid1, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream2, &mspid2, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)stream3, &mspid3, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream1, &IID_IPin, (void **)&pin1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream2, &IID_IPin, (void **)&pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMediaStream_QueryInterface(stream3, &IID_IPin, (void **)&pin3); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); testfilter_init(&source1); testfilter_init(&source2); testfilter_init(&source3); source1.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; source2.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; source3.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IGraphBuilder_AddFilter(graph, &source1.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_AddFilter(graph, &source2.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_AddFilter(graph, &source3.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source2.source.pin.IPin_iface, pin2, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source3.source.pin.IPin_iface, pin3, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_ConnectDirect(graph, &source1.source.pin.IPin_iface, pin1, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_QueryInterface(filter, &IID_IMediaSeeking, (void **)&seeking); ok(hr == S_OK, "Got hr %#lx.\n", hr); check(seeking, &source1, &source2, &source3, S_OK, S_OK); check(seeking, &source1, &source2, &source3, E_FAIL, E_FAIL); check(seeking, &source1, &source2, &source3, E_NOTIMPL, E_NOTIMPL); source2.IMediaSeeking_iface.lpVtbl = NULL; check(seeking, &source1, &source2, &source3, S_OK, E_NOTIMPL); source2.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; IGraphBuilder_Disconnect(graph, pin2); IGraphBuilder_Disconnect(graph, &source2.source.pin.IPin_iface); check(seeking, &source1, &source2, &source3, S_OK, E_NOTIMPL); IGraphBuilder_Disconnect(graph, pin2); IGraphBuilder_Disconnect(graph, &source2.source.pin.IPin_iface); IGraphBuilder_Disconnect(graph, pin3); IGraphBuilder_Disconnect(graph, &source3.source.pin.IPin_iface); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaSeeking_Release(seeking); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin1); ref = IAMMediaStream_Release(stream1); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin2); ref = IAMMediaStream_Release(stream2); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin3); ref = IAMMediaStream_Release(stream3); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void check_mediastreamfilter_set_positions(IMediaSeeking *seeking, struct testfilter *source1, struct testfilter *source2, struct testfilter *source3, HRESULT source2_hr, HRESULT expected_hr) { LONGLONG current_position = 12345678; LONGLONG stop_position = 87654321; HRESULT hr; source2->set_positions_hr = source2_hr; source1->current_position = 0xdeadbeefdeadbeefULL; source1->stop_position = 0xdeadbeefdeadbeefULL; source2->current_position = 0xdeadbeefdeadbeefULL; source2->stop_position = 0xdeadbeefdeadbeefULL; source3->current_position = 0xdeadbeefdeadbeefULL; source3->stop_position = 0xdeadbeefdeadbeefULL; hr = IMediaSeeking_SetPositions(seeking, ¤t_position, AM_SEEKING_AbsolutePositioning, &stop_position, AM_SEEKING_AbsolutePositioning); ok(hr == expected_hr, "Got hr %#lx.\n", hr); ok(source1->current_position == 0xdeadbeefdeadbeefULL, "Got current position %s.\n", wine_dbgstr_longlong(source1->current_position)); ok(source1->stop_position == 0xdeadbeefdeadbeefULL, "Got stop position %s.\n", wine_dbgstr_longlong(source1->stop_position)); if (SUCCEEDED(expected_hr)) { ok(source2->current_position == 12345678, "Got current position %s.\n", wine_dbgstr_longlong(source2->current_position)); ok(source2->stop_position == 87654321, "Got stop position %s.\n", wine_dbgstr_longlong(source2->stop_position)); } ok(source3->current_position == 0xdeadbeefdeadbeefULL, "Got current position %s.\n", wine_dbgstr_longlong(source3->current_position)); ok(source3->stop_position == 0xdeadbeefdeadbeefULL, "Got stop position %s.\n", wine_dbgstr_longlong(source3->stop_position)); } static void check_mediastreamfilter_get_duration(IMediaSeeking *seeking, struct testfilter *source1, struct testfilter *source2, struct testfilter *source3, HRESULT source2_hr, HRESULT expected_hr) { LONGLONG duration = 0xdeadbeefdeadbeefULL; HRESULT hr; source2->get_duration_hr = source2_hr; hr = IMediaSeeking_GetDuration(seeking, &duration); ok(hr == expected_hr, "Got hr %#lx.\n", hr); if (SUCCEEDED(expected_hr)) ok(duration == 0x8000000000000000ULL, "Got duration %s.\n", wine_dbgstr_longlong(duration)); else ok(duration == 0xdeadbeefdeadbeefULL, "Got duration %s.\n", wine_dbgstr_longlong(duration)); } static void check_mediastreamfilter_get_stop_position(IMediaSeeking *seeking, struct testfilter *source1, struct testfilter *source2, struct testfilter *source3, HRESULT source2_hr, HRESULT expected_hr) { LONGLONG stop = 0xdeadbeefdeadbeefULL; HRESULT hr; source2->get_stop_position_hr = source2_hr; hr = IMediaSeeking_GetStopPosition(seeking, &stop); ok(hr == expected_hr, "Got hr %#lx.\n", hr); if (SUCCEEDED(expected_hr)) ok(stop == 0x8000000000000000ULL, "Got stop position %s.\n", wine_dbgstr_longlong(stop)); else ok(stop == 0xdeadbeefdeadbeefULL, "Got stop position %s.\n", wine_dbgstr_longlong(stop)); } static void check_mediastreamfilter_is_format_supported(IMediaSeeking *seeking, struct testfilter *source1, struct testfilter *source2, struct testfilter *source3, HRESULT source2_hr, HRESULT expected_hr) { HRESULT hr; source2->is_format_supported_hr = source2_hr; hr = IMediaSeeking_IsFormatSupported(seeking, &TIME_FORMAT_MEDIA_TIME); ok(hr == expected_hr, "Got hr %#lx.\n", hr); } static void test_mediastreamfilter_seeking(void) { check_mediastreamfilter_seeking(check_mediastreamfilter_set_positions); check_mediastreamfilter_seeking(check_mediastreamfilter_get_duration); check_mediastreamfilter_seeking(check_mediastreamfilter_get_stop_position); check_mediastreamfilter_seeking(check_mediastreamfilter_is_format_supported); } static void test_mediastreamfilter_get_current_stream_time(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); STREAM_TIME filter_start_time; IMediaStreamFilter *filter; IMediaFilter *media_filter; struct testclock clock; IGraphBuilder *graph; STREAM_TIME time; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); /* Crashes on native. */ if (0) { hr = IMediaStreamFilter_GetCurrentStreamTime(filter, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetTime(mmstream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &time); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(time == 0, "Got time %s.\n", wine_dbgstr_longlong(time)); time = 0xdeadbeefdeadbeef; hr = IAMMultiMediaStream_GetTime(mmstream, &time); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(time == 0, "Got time %s.\n", wine_dbgstr_longlong(time)); hr = IMediaFilter_SetSyncSource(media_filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.get_time_hr = E_FAIL; time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &time); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(time == 0, "Got time %s.\n", wine_dbgstr_longlong(time)); time = 0xdeadbeefdeadbeef; hr = IAMMultiMediaStream_GetTime(mmstream, &time); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(time == 0, "Got time %s.\n", wine_dbgstr_longlong(time)); clock.time = 23456789; clock.get_time_hr = S_OK; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &filter_start_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.time = 34567890; time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(time == 11111101 + filter_start_time, "Got time %s.\n", wine_dbgstr_longlong(time)); time = 0xdeadbeefdeadbeef; hr = IAMMultiMediaStream_GetTime(mmstream, &time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(time == 11111101 + filter_start_time, "Got time %s.\n", wine_dbgstr_longlong(time)); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_mediastreamfilter_reference_time_to_stream_time(void) { IMediaStreamFilter *filter; struct testclock clock; REFERENCE_TIME time; HRESULT hr; ULONG ref; hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IMediaStreamFilter, (void **)&filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, &time); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); ok(time == 0xdeadbeefdeadbeef, "Got time %s.\n", wine_dbgstr_longlong(time)); hr = IMediaStreamFilter_SetSyncSource(filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.get_time_hr = E_FAIL; /* Crashes on native. */ if (0) { hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, NULL); ok(hr == S_FALSE, "Got hr %#lx.\n", hr); } time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, &time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(time == 0xdeadbeefdeadbeef, "Got time %s.\n", wine_dbgstr_longlong(time)); hr = IMediaStreamFilter_Run(filter, 23456789); ok(hr == S_OK, "Got hr %#lx.\n", hr); time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, &time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(time == 0xdeadbeefdd47d2da, "Got time %s.\n", wine_dbgstr_longlong(time)); clock.time = 34567890; clock.get_time_hr = S_OK; time = 0xdeadbeefdeadbeef; hr = IMediaStreamFilter_ReferenceTimeToStreamTime(filter, &time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(time == 0xdeadbeefdd47d2da, "Got time %s.\n", wine_dbgstr_longlong(time)); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } struct mediastreamfilter_wait_until_params { IMediaStreamFilter *filter; REFERENCE_TIME time; HRESULT expected_hr; }; static DWORD CALLBACK mediastreamfilter_wait_until(void *p) { struct mediastreamfilter_wait_until_params *params = (struct mediastreamfilter_wait_until_params *)p; HRESULT hr; hr = IMediaStreamFilter_WaitUntil(params->filter, params->time); ok(hr == params->expected_hr, "Got hr %#lx.\n", hr); return 0; } static void test_mediastreamfilter_wait_until(void) { struct mediastreamfilter_wait_until_params params1; struct mediastreamfilter_wait_until_params params2; struct advise_time_cookie cookie1 = { 0 }; struct advise_time_cookie cookie2 = { 0 }; IMediaStreamFilter *filter; struct testclock clock; HANDLE thread1; HANDLE thread2; HRESULT hr; ULONG ref; hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IMediaStreamFilter, (void **)&filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); cookie1.advise_time_called_event = CreateEventW(NULL, FALSE, FALSE, NULL); cookie2.advise_time_called_event = CreateEventW(NULL, FALSE, FALSE, NULL); hr = IMediaStreamFilter_Run(filter, 12345678); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_WaitUntil(filter, 23456789); ok(hr == E_FAIL, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_Stop(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_SetSyncSource(filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.advise_time_cookie = &cookie1; params1.filter = filter; params1.time = 23456789; params1.expected_hr = S_OK; thread1 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms1, 0, NULL); ok(!WaitForSingleObject(cookie1.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread1, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); ok(cookie1.base == 23456789, "Got base %s.\n", wine_dbgstr_longlong(cookie1.base)); ok(cookie1.offset == 12345678, "Got offset %s.\n", wine_dbgstr_longlong(cookie1.offset)); ok(!!cookie1.event, "Expected non-NULL event.\n"); SetEvent(cookie1.event); ok(!WaitForSingleObject(thread1, 2000), "Wait timed out.\n"); CloseHandle(thread1); ok(!cookie1.unadvise_called, "Unexpected Unadvise call.\n"); hr = IMediaStreamFilter_Run(filter, 12345678); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.time = 30000000; clock.advise_time_cookie = &cookie1; params1.filter = filter; params1.time = 23456789; params1.expected_hr = S_OK; thread1 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms1, 0, NULL); ok(!WaitForSingleObject(cookie1.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread1, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); ok(cookie1.base == 23456789, "Got base %s.\n", wine_dbgstr_longlong(cookie1.base)); ok(cookie1.offset == 12345678, "Got offset %s.\n", wine_dbgstr_longlong(cookie1.offset)); ok(!!cookie1.event, "Expected non-NULL event.\n"); clock.advise_time_cookie = &cookie2; params2.filter = filter; params2.time = 11111111; params2.expected_hr = S_OK; thread2 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms2, 0, NULL); ok(!WaitForSingleObject(cookie2.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread2, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); ok(cookie2.base == 11111111, "Got base %s.\n", wine_dbgstr_longlong(cookie2.base)); ok(cookie2.offset == 12345678, "Got offset %s.\n", wine_dbgstr_longlong(cookie2.offset)); ok(!!cookie2.event, "Expected non-NULL event.\n"); SetEvent(cookie1.event); ok(!WaitForSingleObject(thread1, 2000), "Wait timed out.\n"); CloseHandle(thread1); ok(WaitForSingleObject(thread2, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); SetEvent(cookie2.event); ok(!WaitForSingleObject(thread2, 2000), "Wait timed out.\n"); CloseHandle(thread2); clock.advise_time_cookie = &cookie1; params1.filter = filter; params1.time = 23456789; params1.expected_hr = S_FALSE; thread1 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms1, 0, NULL); ok(!WaitForSingleObject(cookie1.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread1, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); clock.advise_time_cookie = &cookie2; params2.filter = filter; params2.time = 23456789; params2.expected_hr = S_FALSE; thread2 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms2, 0, NULL); ok(!WaitForSingleObject(cookie2.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread2, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); hr = IMediaStreamFilter_Flush(filter, FALSE); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(cookie1.unadvise_called, "Expected Unadvise to be called.\n"); ok(cookie2.unadvise_called, "Expected Unadvise to be called.\n"); ok(!WaitForSingleObject(thread1, 2000), "Wait timed out.\n"); CloseHandle(thread1); ok(!WaitForSingleObject(thread2, 2000), "Wait timed out.\n"); CloseHandle(thread2); clock.advise_time_cookie = &cookie1; params1.filter = filter; params1.time = 23456789; params1.expected_hr = S_FALSE; thread1 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms1, 0, NULL); ok(!WaitForSingleObject(cookie1.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread1, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); clock.advise_time_cookie = &cookie2; params2.filter = filter; params2.time = 23456789; params2.expected_hr = S_FALSE; thread2 = CreateThread(NULL, 0, mediastreamfilter_wait_until, ¶ms2, 0, NULL); ok(!WaitForSingleObject(cookie2.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread2, 100) == WAIT_TIMEOUT, "WaitUntil returned prematurely.\n"); hr = IMediaStreamFilter_Stop(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(cookie1.unadvise_called, "Expected Unadvise to be called.\n"); ok(cookie2.unadvise_called, "Expected Unadvise to be called.\n"); ok(!WaitForSingleObject(thread1, 2000), "Wait timed out.\n"); CloseHandle(thread1); ok(!WaitForSingleObject(thread2, 2000), "Wait timed out.\n"); CloseHandle(thread2); CloseHandle(cookie1.advise_time_called_event); CloseHandle(cookie2.advise_time_called_event); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_mediastreamfilter_end_of_stream(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); struct testfilter source1, source2; IMediaStream *stream1, *stream2; IMediaControl *media_control; IMediaStreamFilter *filter; struct graph graph; IPin *pin1, *pin2; HRESULT hr; ULONG ref; graph_init(&graph); hr = IFilterGraph2_QueryInterface(&graph.IFilterGraph2_iface, &IID_IMediaControl, (void **)&media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, (IGraphBuilder *)&graph.IFilterGraph2_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream1, &IID_IPin, (void **)&pin1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream2, &IID_IPin, (void **)&pin2); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(filter != NULL, "Expected non-null filter\n"); testfilter_init(&source1); testfilter_init(&source2); hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source1.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IFilterGraph2_AddFilter(&graph.IFilterGraph2_iface, &source2.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Connect(&source1.source.pin.IPin_iface, pin1, &audio_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_Connect(&source2.source.pin.IPin_iface, pin2, &rgb32_mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Initially, EC_COMPLETE notifications are disabled. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Unsuccsessful call to SupportSeeking does not enable EC_COMPLETE notifications. */ hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == E_NOINTERFACE, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Successful call to SupportSeeking enables EC_COMPLETE notifications. */ source1.IMediaSeeking_iface.lpVtbl = &testsource_seeking_vtbl; hr = IMediaStreamFilter_SupportSeeking(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* EC_COMPLETE is sent on paused->running state transition * if EndOfStream has been called for all streams. */ graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Pause(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* EC_COMPLETE is sent on paused->running state transition * if EndOfStream has been called for all streams. */ graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Pause(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* EndOfStream count is reset on paused->stopped state transition. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* EOS count is not reset on running->paused state transition. */ hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaControl_Pause(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaControl_Run(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* EndOfStream count is not reset when Stop() is called on an already stopped filter. */ hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_Stop(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Flush with cancel_eos=TRUE decrements EOS count. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_Flush(filter, TRUE); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Flush followed by EndOfStream causes another EC_COMPLETE to be sent. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaStreamFilter_Flush(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 2, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Flush decrements EOS count event when it is 0. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_Flush(filter, TRUE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 0, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Flush with cancel_eos=FALSE does not decrement EOS count. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); graph.got_notify = 0; hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_Flush(filter, FALSE); hr = IMediaStreamFilter_EndOfStream(filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph.got_notify == 1, "Got %d calls to IMediaEventSink::Notify().\n", graph.got_notify); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaControl_Release(media_control); graph_destroy(&graph); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin1); ref = IMediaStream_Release(stream1); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin2); ref = IMediaStream_Release(stream2); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_getsetdirectdraw(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDraw *ddraw, *ddraw2, *ddraw3, *ddraw4; IDirectDrawMediaStream *ddraw_stream; IDirectDrawStreamSample *sample; IDirectDraw7 *ddraw7; IMediaStream *stream; HRESULT hr; ULONG ref; hr = DirectDrawCreate(NULL, &ddraw, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDraw_QueryInterface(ddraw, &IID_IDirectDraw7, (void **)&ddraw7); ok(hr == DD_OK, "Got hr %#lx.\n", hr); hr = IDirectDraw7_SetCooperativeLevel(ddraw7, GetDesktopWindow(), DDSCL_NORMAL); ok(hr == DD_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 1); EXPECT_REF(ddraw7, 1); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)ddraw7, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 2); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw2 == ddraw, "Expected ddraw %p, got %p.\n", ddraw, ddraw2); EXPECT_REF(ddraw, 3); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw3); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw3 == ddraw2, "Expected ddraw %p, got %p.\n", ddraw2, ddraw3); EXPECT_REF(ddraw, 4); IDirectDraw_Release(ddraw3); EXPECT_REF(ddraw, 3); /* The current ddraw is released when SetDirectDraw() is called. */ hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 2); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw3); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw3 == NULL, "Expected NULL, got %p.\n", ddraw3); hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, ddraw2); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 3); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw3); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw3 == ddraw2, "Expected ddraw %p, got %p.\n", ddraw2, ddraw3); EXPECT_REF(ddraw, 4); IDirectDraw_Release(ddraw3); EXPECT_REF(ddraw, 3); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* SetDirectDraw() doesn't take an extra reference to the ddraw object * if there are samples extant. */ hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, ddraw2); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 3); hr = DirectDrawCreate(NULL, &ddraw3, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDraw_SetCooperativeLevel(ddraw3, GetDesktopWindow(), DDSCL_NORMAL); ok(hr == DD_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw3, 1); hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, ddraw3); ok(hr == MS_E_SAMPLEALLOC, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw4); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw4 == ddraw2, "Expected ddraw %p, got %p.\n", ddraw2, ddraw4); EXPECT_REF(ddraw, 4); IDirectDraw_Release(ddraw4); EXPECT_REF(ddraw, 3); ref = IDirectDrawStreamSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, ddraw3); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw, 2); EXPECT_REF(ddraw3, 2); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw4); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(ddraw4 == ddraw3, "Expected ddraw %p, got %p.\n", ddraw3, ddraw4); EXPECT_REF(ddraw3, 3); IDirectDraw_Release(ddraw4); EXPECT_REF(ddraw3, 2); hr = IDirectDrawMediaStream_SetDirectDraw(ddraw_stream, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(ddraw3, 1); ref = IDirectDraw_Release(ddraw3); ok(!ref, "Got outstanding refcount %ld.\n", ref); EXPECT_REF(stream, 3); IDirectDrawMediaStream_Release(ddraw_stream); EXPECT_REF(stream, 2); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); EXPECT_REF(stream, 1); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDraw7_Release(ddraw7); ok(!ref, "Got outstanding refcount %ld.\n", ref); IDirectDraw_Release(ddraw2); EXPECT_REF(ddraw, 1); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostreamsample_get_media_stream(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioStreamSample *audio_sample; IAudioMediaStream *audio_stream; IMediaStream *stream, *stream2; IAudioData *audio_data; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &audio_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Crashes on native. */ if (0) { hr = IAudioStreamSample_GetMediaStream(audio_sample, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } EXPECT_REF(stream, 4); hr = IAudioStreamSample_GetMediaStream(audio_sample, &stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(stream2 == stream, "Expected stream %p, got %p.\n", stream, stream2); EXPECT_REF(stream, 5); IMediaStream_Release(stream2); IAudioMediaStream_Release(audio_stream); ref = IAudioStreamSample_Release(audio_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_audiostreamsample_get_audio_data(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IAudioData *audio_data, *audio_data2; IAudioStreamSample *audio_sample; IAudioMediaStream *audio_stream; IMediaStream *stream; HRESULT hr; ULONG ref; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryAudio, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IAudioMediaStream, (void **)&audio_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = CoCreateInstance(&CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER, &IID_IAudioData, (void **)&audio_data); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioMediaStream_CreateSample(audio_stream, audio_data, 0, &audio_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAudioStreamSample_GetAudioData(audio_sample, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); EXPECT_REF(audio_data, 2); hr = IAudioStreamSample_GetAudioData(audio_sample, &audio_data2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(audio_data2 == audio_data, "Expected audio data %p, got %p.\n", audio_data, audio_data2); EXPECT_REF(audio_data, 3); IAudioData_Release(audio_data2); IAudioMediaStream_Release(audio_stream); ref = IAudioStreamSample_Release(audio_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAudioData_Release(audio_data); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } #define get_ddrawstream_create_sample_desc(a,b,c,d) get_ddrawstream_create_sample_desc_(__LINE__,a,b,c,d) static void get_ddrawstream_create_sample_desc_(int line, const DDSURFACEDESC *format1, const DDSURFACEDESC *format2, const AM_MEDIA_TYPE *mt, DDSURFACEDESC *desc) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawMediaStream *ddraw_stream; IDirectDrawStreamSample *sample; IDirectDrawSurface *surface; struct testfilter source; IGraphBuilder *graph; IMediaStream *stream; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); if (format1) { hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, format1, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); } if (format2) { hr = IDirectDrawMediaStream_SetFormat(ddraw_stream, format2, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); } if (mt) { hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, mt); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_Disconnect(graph, pin); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); } hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &sample); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_GetSurface(sample, &surface, NULL); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); ok_(__FILE__, line)(!!surface, "Expected non-NULL sufrace.\n"); desc->dwSize = sizeof(*desc); hr = IDirectDrawSurface_GetSurfaceDesc(surface, desc); ok_(__FILE__, line)(hr == S_OK, "Got hr %#lx.\n", hr); ref = IDirectDrawStreamSample_Release(sample); ok_(__FILE__, line)(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawSurface_Release(surface); ok_(__FILE__, line)(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok_(__FILE__, line)(!ref, "Got outstanding refcount %ld.\n", ref); ref = IGraphBuilder_Release(graph); ok_(__FILE__, line)(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok_(__FILE__, line)(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstream_create_sample(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); DDSURFACEDESC desc2 = { sizeof(desc2) }; IDirectDrawSurface *surface, *surface2; DDSURFACEDESC desc = { sizeof(desc) }; IDirectDrawMediaStream *ddraw_stream; IDirectDrawStreamSample *sample; DDSURFACEDESC format1; DDSURFACEDESC format2; IMediaStream *stream; IDirectDraw *ddraw; HRESULT hr; RECT rect; ULONG ref; hr = DirectDrawCreate(NULL, &ddraw, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDraw_SetCooperativeLevel(ddraw, NULL, DDSCL_NORMAL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)ddraw, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Crashes on native. */ if (0) { hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } SetRectEmpty(&rect); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, &rect, 0, &sample); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); EXPECT_REF(mmstream, 1); EXPECT_REF(stream, 3); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(mmstream, 2); EXPECT_REF(stream, 4); hr = IDirectDrawStreamSample_GetSurface(sample, NULL, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_GetSurface(sample, NULL, &rect); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_GetSurface(sample, &surface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(surface != NULL, "Expected non-NULL surface.\n"); IDirectDrawSurface_Release(surface); surface = NULL; hr = IDirectDrawStreamSample_GetSurface(sample, &surface, &rect); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(surface != NULL, "Expected non-NULL surface.\n"); hr = IDirectDrawSurface_GetSurfaceDesc(surface, &desc); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(desc.dwWidth == 100, "Expected width 100, got %ld.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Expected height 100, got %ld.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Expected format flags DDPF_RGB, got %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount, "Expected non-zero RGB bit count.\n"); IDirectDrawSurface_Release(surface); IDirectDrawStreamSample_Release(sample); EXPECT_REF(stream, 3); memset(&desc, 0, sizeof(desc)); desc.dwSize = sizeof(desc); desc.dwFlags = DDSD_CAPS; desc.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE; hr = IDirectDraw_CreateSurface(ddraw, &desc, &surface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(surface, 1); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, surface, NULL, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); EXPECT_REF(surface, 2); surface2 = NULL; SetRectEmpty(&rect); hr = IDirectDrawStreamSample_GetSurface(sample, &surface2, &rect); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(surface2 == surface, "Expected surface %p, got %p.\n", surface, surface2); ok(rect.right > 0 && rect.bottom > 0, "Got rect %ld, %ld.\n", rect.right, rect.bottom); EXPECT_REF(surface, 3); IDirectDrawSurface_Release(surface2); EXPECT_REF(surface, 2); IDirectDrawStreamSample_Release(sample); EXPECT_REF(surface, 1); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, surface, &rect, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IDirectDrawStreamSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawSurface_Release(surface); ok(!ref, "Got outstanding refcount %ld.\n", ref); memset(&desc, 0, sizeof(desc)); desc.dwSize = sizeof(desc); desc.dwFlags = DDSD_WIDTH | DDSD_HEIGHT | DDSD_PIXELFORMAT | DDSD_CAPS; desc.dwWidth = 444; desc.dwHeight = 400; desc.ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT); desc.ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_ALPHAPIXELS; desc.ddpfPixelFormat.u1.dwRGBBitCount = 32; desc.ddpfPixelFormat.u2.dwRBitMask = 0xff0000; desc.ddpfPixelFormat.u3.dwGBitMask = 0x00ff00; desc.ddpfPixelFormat.u4.dwBBitMask = 0x0000ff; desc.ddpfPixelFormat.u5.dwRGBAlphaBitMask = 0xff000000; desc.ddsCaps.dwCaps = DDSCAPS_SYSTEMMEMORY | DDSCAPS_OFFSCREENPLAIN; hr = IDirectDraw_CreateSurface(ddraw, &desc, &surface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); SetRect(&rect, 111, 100, 333, 300); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, surface, &rect, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IDirectDrawStreamSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); surface2 = NULL; hr = IDirectDrawStreamSample_GetSurface(sample, &surface2, &rect); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_GetSurfaceDesc(surface, &desc); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_GetSurfaceDesc(surface2, &desc2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(desc2.dwWidth == 222, "Got width %lu.\n", desc2.dwWidth); ok(desc2.dwHeight == 200, "Got height %lu.\n", desc2.dwHeight); ok(memcmp(&desc2.ddpfPixelFormat, &desc.ddpfPixelFormat, sizeof(DDPIXELFORMAT)) == 0, "Pixel format didn't match.\n"); ref = IDirectDrawStreamSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawSurface_Release(surface); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawSurface_Release(surface2); ok(!ref, "Got outstanding refcount %ld.\n", ref); memset(&desc, 0, sizeof(desc)); desc.dwSize = sizeof(desc); desc.dwFlags = DDSD_WIDTH | DDSD_HEIGHT | DDSD_PIXELFORMAT | DDSD_CAPS; desc.dwWidth = 444; desc.dwHeight = 400; desc.ddpfPixelFormat.dwSize = sizeof(DDPIXELFORMAT); desc.ddpfPixelFormat.dwFlags = DDPF_RGB | DDPF_PALETTEINDEXED4; desc.ddpfPixelFormat.u1.dwRGBBitCount = 4; desc.ddsCaps.dwCaps = DDSCAPS_SYSTEMMEMORY | DDSCAPS_OFFSCREENPLAIN; hr = IDirectDraw_CreateSurface(ddraw, &desc, &surface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, surface, NULL, 0, &sample); ok(hr == DDERR_INVALIDSURFACETYPE, "Got hr %#lx.\n", hr); IDirectDrawMediaStream_Release(ddraw_stream); ref = IDirectDrawSurface_Release(surface); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); format1 = rgb8_format; format1.dwFlags = 0; format1.dwWidth = 333; format1.dwHeight = 444; get_ddrawstream_create_sample_desc(&format1, NULL, NULL, &desc); ok(desc.dwWidth == 100, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 32, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xff0000, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x00ff00, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x0000ff, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); format1 = rgb8_format; format1.dwFlags |= DDSD_WIDTH; format1.dwWidth = 333; format1.dwHeight = 444; format2 = rgb8_format; format2.dwFlags = 0; get_ddrawstream_create_sample_desc(&format1, &format2, NULL, &desc); ok(desc.dwWidth == 333, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 444, "Got height %lu.\n", desc.dwHeight); format1 = rgb8_format; format1.dwFlags |= DDSD_HEIGHT; format1.dwWidth = 333; format1.dwHeight = 444; format2 = rgb8_format; format2.dwFlags = 0; get_ddrawstream_create_sample_desc(&format1, &format2, NULL, &desc); ok(desc.dwWidth == 333, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 444, "Got height %lu.\n", desc.dwHeight); get_ddrawstream_create_sample_desc(NULL, NULL, &rgb8_mt, &desc); ok(desc.dwWidth == 333, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 444, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 32, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xff0000, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x00ff00, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x0000ff, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); get_ddrawstream_create_sample_desc(&rgb565_format, NULL, NULL, &desc); ok(desc.dwWidth == 100, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 16, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xf800, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x07e0, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x001f, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); get_ddrawstream_create_sample_desc(&argb32_format, NULL, NULL, &desc); ok(desc.dwWidth == 100, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == (DDPF_RGB | DDPF_ALPHAPIXELS), "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 32, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xff0000, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x00ff00, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x0000ff, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); ok(desc.ddpfPixelFormat.u5.dwRGBAlphaBitMask == 0xff000000, "Got alpha bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); format1 = rgb32_format; format1.dwFlags |= DDSD_CAPS; format1.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN | DDSCAPS_VIDEOMEMORY; get_ddrawstream_create_sample_desc(&format1, NULL, NULL, &desc); ok(desc.ddsCaps.dwCaps & DDSCAPS_OFFSCREENPLAIN, "Expected set DDSCAPS_OFFSCREENPLAIN.\n"); ok(desc.ddsCaps.dwCaps & DDSCAPS_SYSTEMMEMORY, "Expected set DDSCAPS_SYSTEMMEMORY.\n"); ok(!(desc.ddsCaps.dwCaps & DDSCAPS_VIDEOMEMORY), "Expected unset DDSCAPS_VIDEOMEMORY.\n"); ok(desc.dwWidth == 100, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 32, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xff0000, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x00ff00, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x0000ff, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); format1 = rgb32_format; format1.dwFlags |= DDSD_CKSRCBLT; format1.ddckCKSrcBlt.dwColorSpaceLowValue = 0xff00ff; format1.ddckCKSrcBlt.dwColorSpaceHighValue = 0xff00ff; get_ddrawstream_create_sample_desc(&format1, NULL, NULL, &desc); ok(!(desc.dwFlags & DDSD_CKSRCBLT), "Expected unset DDSD_CKSRCBLT.\n"); ok(desc.dwWidth == 100, "Got width %lu.\n", desc.dwWidth); ok(desc.dwHeight == 100, "Got height %lu.\n", desc.dwHeight); ok(desc.ddpfPixelFormat.dwFlags == DDPF_RGB, "Got flags %#lx.\n", desc.ddpfPixelFormat.dwFlags); ok(desc.ddpfPixelFormat.u1.dwRGBBitCount == 32, "Got rgb bit count %lu.\n", desc.ddpfPixelFormat.u1.dwRGBBitCount); ok(desc.ddpfPixelFormat.u2.dwRBitMask == 0xff0000, "Got r bit mask %#lx.\n", desc.ddpfPixelFormat.u2.dwRBitMask); ok(desc.ddpfPixelFormat.u3.dwGBitMask == 0x00ff00, "Got g bit mask %#lx.\n", desc.ddpfPixelFormat.u3.dwGBitMask); ok(desc.ddpfPixelFormat.u4.dwBBitMask == 0x0000ff, "Got b bit mask %#lx.\n", desc.ddpfPixelFormat.u4.dwBBitMask); ok(desc.ddckCKSrcBlt.dwColorSpaceLowValue == 0, "Got color key low value %#lx.\n", desc.ddckCKSrcBlt.dwColorSpaceLowValue); ok(desc.ddckCKSrcBlt.dwColorSpaceHighValue == 0, "Got color key high value %#lx.\n", desc.ddckCKSrcBlt.dwColorSpaceHighValue); } static void test_ddrawstreamsample_get_media_stream(void) { IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawMediaStream *ddraw_stream; IDirectDrawStreamSample *sample; IMediaStream *stream, *stream2; IDirectDraw *ddraw; HRESULT hr; ULONG ref; hr = DirectDrawCreate(NULL, &ddraw, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDraw_SetCooperativeLevel(ddraw, NULL, DDSCL_NORMAL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, (IUnknown *)ddraw, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Crashes on native. */ if (0) { hr = IDirectDrawStreamSample_GetMediaStream(sample, NULL); ok(hr == E_POINTER, "Got hr %#lx.\n", hr); } EXPECT_REF(stream, 4); hr = IDirectDrawStreamSample_GetMediaStream(sample, &stream2); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(stream2 == stream, "Expected stream %p, got %p.\n", stream, stream2); EXPECT_REF(stream, 5); IMediaStream_Release(stream2); EXPECT_REF(stream, 4); IDirectDrawMediaStream_Release(ddraw_stream); ref = IDirectDrawStreamSample_Release(sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstreamsample_update(void) { static const BYTE initial_data[] = { 0xde, 0xad, 0xbe, 0xef, 0xde, 0xad, 0xbe, 0xef, 0xde, 0xad, 0xbe, 0xef, }; static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawStreamSample *stream_sample; struct advise_time_cookie cookie = {0}; IDirectDrawMediaStream *ddraw_stream; IMediaControl *media_control; IDirectDrawSurface *surface; IMemInputPin *mem_input_pin; IMediaSample *media_sample; IMediaFilter *media_filter; REFERENCE_TIME start_time; REFERENCE_TIME end_time; struct testfilter source; struct testclock clock; IGraphBuilder *graph; IMediaStream *stream; VIDEOINFO video_info; DDSURFACEDESC desc; IDirectDraw *ddraw; AM_MEDIA_TYPE mt; HANDLE thread; HANDLE event; HRESULT hr; ULONG ref; IPin *pin; RECT rect; int i; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_GetDirectDraw(ddraw_stream, &ddraw); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaControl, (void **)&media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); event = CreateEventW(NULL, FALSE, FALSE, NULL); ok(event != NULL, "Expected non-NULL event."); cookie.advise_time_called_event = CreateEventW(NULL, FALSE, FALSE, NULL); ok(cookie.advise_time_called_event != NULL, "Expected non-NULL event."); hr = IMediaFilter_SetSyncSource(media_filter, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); desc = rgb24_format; desc.dwFlags |= DDSD_WIDTH | DDSD_HEIGHT | DDSD_CAPS; desc.dwWidth = 4; desc.dwHeight = 5; desc.ddsCaps.dwCaps = DDSCAPS_SYSTEMMEMORY | DDSCAPS_OFFSCREENPLAIN; hr = IDirectDraw_CreateSurface(ddraw, &desc, &surface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Make the rect width equal to the surface width, as the native * implementation incorrectly handles rects that are not full-width * when the ddraw stream's custom allocator is not used. */ SetRect(&rect, 0, 1, 4, 3); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, surface, &rect, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, event, apc_func, 0); ok(hr == E_INVALIDARG, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_E_NOTRUNNING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb24_video_info; video_info.bmiHeader.biWidth = 4; video_info.bmiHeader.biHeight = -2; mt = rgb24_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 0; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); Sleep(100); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); video_info = rgb24_video_info; video_info.bmiHeader.biWidth = 4; video_info.bmiHeader.biHeight = 2; mt = rgb24_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 0; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); Sleep(100); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaControl_Pause(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_E_NOTRUNNING, "Got hr %#lx.\n", hr); hr = IMediaControl_Stop(media_control); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 100; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ammediastream_pin = pin; ammediastream_sleep_time = 100; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_end_of_stream, NULL, 0, NULL); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 100; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, event, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(event, 0) == WAIT_TIMEOUT, "Event should not be signaled.\n"); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ok(WaitForSingleObject(event, 0) == 0, "Event should be signaled.\n"); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); EXPECT_REF(stream_sample, 1); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_E_BUSY, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaFilter_SetSyncSource(media_filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) memcpy((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_E_BUSY, "Got hr %#lx.\n", hr); clock.advise_time_cookie = &cookie; media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); start_time = 11111111; end_time = 11111111; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ammediastream_mem_input_pin = mem_input_pin; ammediastream_media_sample = media_sample; ammediastream_sleep_time = 0; ammediastream_expected_hr = S_OK; thread = CreateThread(NULL, 0, ammediastream_receive, NULL, 0, NULL); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); for (i = 0; i < 5; ++i) ok(memcmp((BYTE *)desc.lpSurface + i * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); SetEvent(cookie.event); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawSurface_Lock(surface, NULL, &desc, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(memcmp((BYTE *)desc.lpSurface + 0 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 1 * desc.u1.lPitch, &test_data[12], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 2 * desc.u1.lPitch, &test_data[0], 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 3 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); ok(memcmp((BYTE *)desc.lpSurface + 4 * desc.u1.lPitch, initial_data, 12) == 0, "Sample data didn't match.\n"); hr = IDirectDrawSurface_Unlock(surface, desc.lpSurface); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample, 0, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); CloseHandle(cookie.advise_time_called_event); CloseHandle(event); ref = IDirectDrawStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawSurface_Release(surface); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaControl_Release(media_control); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDraw_Release(ddraw); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstreamsample_completion_status(void) { static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); struct advise_time_cookie cookie = { 0 }; IDirectDrawStreamSample *stream_sample1; IDirectDrawStreamSample *stream_sample2; IDirectDrawMediaStream *ddraw_stream; STREAM_TIME filter_start_time; IMediaStreamFilter *filter; IMediaSample *media_sample; IMediaFilter *media_filter; struct testfilter source; struct testclock clock; VIDEOINFO video_info; IGraphBuilder *graph; IMediaStream *stream; AM_MEDIA_TYPE mt; HANDLE thread; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); cookie.advise_time_called_event = CreateEventW(NULL, FALSE, FALSE, NULL); hr = IMediaFilter_SetSyncSource(media_filter, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 3; video_info.bmiHeader.biHeight = 1; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &stream_sample1); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &stream_sample2); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Initial status is S_OK. */ hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Update changes the status to MS_S_PENDING. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, 100); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); /* Each Receive call changes the status of one queued sample to S_OK in the same order Update was called. */ hr = IDirectDrawStreamSample_Update(stream_sample2, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* COMPSTAT_NOUPDATEOK removes the sample from the queue and changes the status to MS_S_NOUPDATE. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample2, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* COMPSTAT_ABORT removes the sample from the queue and changes the status to MS_S_NOUPDATE. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample2, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_ABORT, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* COMPSTAT_WAIT has no effect when combined with COMPSTAT_NOUPDATEOK. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK | COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); /* COMPSTAT_WAIT has no effect when combined with COMPSTAT_ABORT. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_ABORT | COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); /* EndOfStream changes the status of the queued samples to MS_S_ENDOFSTREAM. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); /* Update after EndOfStream changes the status to MS_S_ENDOFSTREAM. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Continuous update can be canceled by COMPSTAT_NOUPDATEOK. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); /* Continuous update can be canceled by COMPSTAT_ABORT. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_ABORT, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); /* If a sample is in countinuous update mode, when Receive is called it's status remains MS_S_PENDING * and the sample is moved to the end of the queue. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_E_BUSY, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_E_BUSY, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample2, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample2, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* In continuous update mode, flushing does not affect the status. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_BeginFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IPin_EndFlush(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); /* In continuous update mode, stopping and running the stream does not affect the status. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); /* In continuous update mode, EndOfStream changes the status to MS_S_ENDOFSTREAM. */ hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_ENDOFSTREAM, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* COMPSTAT_WAIT resets the sample to the non-continuous update mode. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* In continuous update mode, CompletionStatus with COMPSTAT_WAIT returns when Receive is called. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC | SSUPDATE_CONTINUOUS, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); streamsample_sample = (IStreamSample *)stream_sample1; streamsample_flags = COMPSTAT_WAIT; streamsample_timeout = INFINITE; streamsample_expected_hr = S_OK; thread = CreateThread(NULL, 0, streamsample_completion_status, NULL, 0, NULL); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "CompletionStatus returned prematurely.\n"); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* CompletionStatus with COMPSTAT_WAIT returns when Receive is called. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); streamsample_sample = (IStreamSample *)stream_sample1; streamsample_flags = COMPSTAT_WAIT; streamsample_timeout = INFINITE; streamsample_expected_hr = S_OK; thread = CreateThread(NULL, 0, streamsample_completion_status, NULL, 0, NULL); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "CompletionStatus returned prematurely.\n"); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); /* CompletionStatus with COMPSTAT_WAIT returns when EndOfStream is called. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); streamsample_sample = (IStreamSample *)stream_sample1; streamsample_flags = COMPSTAT_WAIT; streamsample_timeout = INFINITE; streamsample_expected_hr = MS_S_ENDOFSTREAM; thread = CreateThread(NULL, 0, streamsample_completion_status, NULL, 0, NULL); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "CompletionStatus returned prematurely.\n"); hr = IPin_EndOfStream(pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* Stopping and running the stream does not affect the status and does not remove the sample from the queue. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, 6); hr = IMemInputPin_Receive(source.source.pMemInputPin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); ref = IMediaSample_Release(media_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* When the stream is stopped Update does not change the status. */ hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_E_NOTRUNNING, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_WAIT, INFINITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* When the wait time is less than 1ms the sample is updated immediately. */ hr = IMediaFilter_SetSyncSource(media_filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.time = 12345678; hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &filter_start_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.time = 12345678 - filter_start_time + 11111111; clock.advise_time_cookie = &cookie; hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 9999, 11111111 + 9999, test_data, sizeof(test_data)); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, INFINITE); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* When the wait time is 1ms or greater AdviseTime is called * with base equal to the sample start time and offset equal to the filter start time. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 10000, 11111111 + 10000, test_data, sizeof(test_data)); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); ok(cookie.base == 11111111 + 10000, "Got base %s.\n", wine_dbgstr_longlong(cookie.base)); ok(cookie.offset == 12345678 - filter_start_time, "Got offset %s.\n", wine_dbgstr_longlong(cookie.offset)); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); clock.time = 12345678 - filter_start_time + 11111111 + 10000; SetEvent(cookie.event); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); /* NewSegment does not affect the values passed to AdviseTime. */ hr = IPin_NewSegment(pin, 22222222, 33333333, 1.0); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 20000, 11111111 + 20000, test_data, sizeof(test_data)); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); ok(cookie.base == 11111111 + 20000, "Got base %s.\n", wine_dbgstr_longlong(cookie.base)); ok(cookie.offset == 12345678 - filter_start_time, "Got offset %s.\n", wine_dbgstr_longlong(cookie.offset)); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); clock.time = 12345678 - filter_start_time + 11111111 + 20000; SetEvent(cookie.event); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); /* COMPSTAT_NOUPDATEOK does not cause Receive to return. * Receive waits for the next sample to be queued and updates it. */ hr = IDirectDrawStreamSample_Update(stream_sample1, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 30000, 11111111 + 30000, test_data, sizeof(test_data)); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); ok(cookie.base == 11111111 + 30000, "Got base %s.\n", wine_dbgstr_longlong(cookie.base)); ok(cookie.offset == 12345678 - filter_start_time, "Got offset %s.\n", wine_dbgstr_longlong(cookie.offset)); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_NOUPDATEOK | COMPSTAT_WAIT, INFINITE); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); clock.time = 12345678 - filter_start_time + 11111111 + 30000; SetEvent(cookie.event); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); hr = IDirectDrawStreamSample_Update(stream_sample1, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); /* COMPSTAT_ABORT does not cause Receive to return. * Receive waits for the next sample to be queued and updates it. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 40000, 11111111 + 40000, test_data, sizeof(test_data)); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); ok(cookie.base == 11111111 + 40000, "Got base %s.\n", wine_dbgstr_longlong(cookie.base)); ok(cookie.offset == 12345678 - filter_start_time, "Got offset %s.\n", wine_dbgstr_longlong(cookie.offset)); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, COMPSTAT_ABORT, 0); ok(hr == MS_S_NOUPDATE, "Got hr %#lx.\n", hr); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); clock.time = 12345678 - filter_start_time + 11111111 + 40000; SetEvent(cookie.event); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); hr = IDirectDrawStreamSample_Update(stream_sample1, 0, NULL, NULL, 0); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); /* Stopping the stream causes Receive to return and leaves the sample with MS_S_PENDING status. */ hr = IDirectDrawStreamSample_Update(stream_sample1, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); thread = ammediastream_async_receive_time(&source, 11111111 + 50000, 11111111 + 50000, test_data, sizeof(test_data)); ok(!WaitForSingleObject(cookie.advise_time_called_event, 2000), "Expected AdviseTime to be called.\n"); ok(WaitForSingleObject(thread, 100) == WAIT_TIMEOUT, "Receive returned prematurely.\n"); ok(cookie.base == 11111111 + 50000, "Got base %s.\n", wine_dbgstr_longlong(cookie.base)); ok(cookie.offset == 12345678 - filter_start_time, "Got offset %s.\n", wine_dbgstr_longlong(cookie.offset)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!WaitForSingleObject(thread, 2000), "Wait timed out.\n"); CloseHandle(thread); hr = IDirectDrawStreamSample_CompletionStatus(stream_sample1, 0, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); CloseHandle(cookie.advise_time_called_event); ref = IDirectDrawStreamSample_Release(stream_sample1); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IDirectDrawStreamSample_Release(stream_sample2); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } static void test_ddrawstreamsample_get_sample_times(void) { static const BYTE test_data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; IAMMultiMediaStream *mmstream = create_ammultimediastream(); IDirectDrawStreamSample *stream_sample; IMediaFilter *graph_media_filter; IDirectDrawMediaStream *ddraw_stream; STREAM_TIME filter_start_time; IMemInputPin *mem_input_pin; IMediaStreamFilter *filter; IMediaSample *media_sample; struct testfilter source; STREAM_TIME current_time; struct testclock clock; STREAM_TIME start_time; STREAM_TIME end_time; IGraphBuilder *graph; IMediaStream *stream; VIDEOINFO video_info; AM_MEDIA_TYPE mt; HRESULT hr; ULONG ref; IPin *pin; hr = IAMMultiMediaStream_Initialize(mmstream, STREAMTYPE_READ, 0, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilter(mmstream, &filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(!!filter, "Expected non-null filter.\n"); hr = IAMMultiMediaStream_AddMediaStream(mmstream, NULL, &MSPID_PrimaryVideo, 0, &stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IDirectDrawMediaStream, (void **)&ddraw_stream); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IPin, (void **)&pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStream_QueryInterface(stream, &IID_IMemInputPin, (void **)&mem_input_pin); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IAMMultiMediaStream_GetFilterGraph(mmstream, &graph); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(graph != NULL, "Expected non-NULL graph.\n"); hr = IGraphBuilder_QueryInterface(graph, &IID_IMediaFilter, (void **)&graph_media_filter); ok(hr == S_OK, "Got hr %#lx.\n", hr); testfilter_init(&source); hr = IGraphBuilder_AddFilter(graph, &source.filter.IBaseFilter_iface, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); testclock_init(&clock); video_info = rgb32_video_info; video_info.bmiHeader.biWidth = 3; video_info.bmiHeader.biHeight = 1; mt = rgb32_mt; mt.pbFormat = (BYTE *)&video_info; hr = IGraphBuilder_ConnectDirect(graph, &source.source.pin.IPin_iface, pin, &mt); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IDirectDrawMediaStream_CreateSample(ddraw_stream, NULL, NULL, 0, &stream_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.time = 12345678; current_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0, "Got current time %s.\n", wine_dbgstr_longlong(current_time)); hr = IMediaFilter_SetSyncSource(graph_media_filter, &clock.IReferenceClock_iface); ok(hr == S_OK, "Got hr %#lx.\n", hr); current_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0, "Got current time %s.\n", wine_dbgstr_longlong(current_time)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_RUN); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMediaStreamFilter_GetCurrentStreamTime(filter, &filter_start_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); clock.get_time_hr = E_FAIL; current_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == 0xdeadbeefddf15da1 + filter_start_time, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(0xdeadbeefddf15da1 + filter_start_time), wine_dbgstr_longlong(current_time)); clock.get_time_hr = S_OK; current_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == filter_start_time, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(filter_start_time), wine_dbgstr_longlong(current_time)); clock.time = 23456789; current_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, NULL, NULL, ¤t_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(current_time == filter_start_time + 11111111, "Expected current time %s, got %s.\n", wine_dbgstr_longlong(filter_start_time + 11111111), wine_dbgstr_longlong(current_time)); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 0, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 0, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IDirectDrawStreamSample_Update(stream_sample, SSUPDATE_ASYNC, NULL, NULL, 0); ok(hr == MS_S_PENDING, "Got hr %#lx.\n", hr); media_sample = ammediastream_allocate_sample(&source, test_data, sizeof(test_data)); start_time = 12345678; end_time = 23456789; hr = IMediaSample_SetTime(media_sample, &start_time, &end_time); ok(hr == S_OK, "Got hr %#lx.\n", hr); hr = IMemInputPin_Receive(mem_input_pin, media_sample); ok(hr == S_OK, "Got hr %#lx.\n", hr); IMediaSample_Release(media_sample); start_time = 0xdeadbeefdeadbeef; end_time = 0xdeadbeefdeadbeef; hr = IDirectDrawStreamSample_GetSampleTimes(stream_sample, &start_time, &end_time, NULL); ok(hr == S_OK, "Got hr %#lx.\n", hr); ok(start_time == 12345678, "Got start time %s.\n", wine_dbgstr_longlong(start_time)); ok(end_time == 23456789, "Got end time %s.\n", wine_dbgstr_longlong(end_time)); hr = IAMMultiMediaStream_SetState(mmstream, STREAMSTATE_STOP); ok(hr == S_OK, "Got hr %#lx.\n", hr); IGraphBuilder_Disconnect(graph, pin); IGraphBuilder_Disconnect(graph, &source.source.pin.IPin_iface); ref = IDirectDrawStreamSample_Release(stream_sample); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IAMMultiMediaStream_Release(mmstream); ok(!ref, "Got outstanding refcount %ld.\n", ref); IMediaFilter_Release(graph_media_filter); ref = IGraphBuilder_Release(graph); ok(!ref, "Got outstanding refcount %ld.\n", ref); ref = IMediaStreamFilter_Release(filter); ok(!ref, "Got outstanding refcount %ld.\n", ref); IPin_Release(pin); IMemInputPin_Release(mem_input_pin); IDirectDrawMediaStream_Release(ddraw_stream); ref = IMediaStream_Release(stream); ok(!ref, "Got outstanding refcount %ld.\n", ref); } START_TEST(amstream) { const WCHAR *test_avi_path; CoInitializeEx(NULL, COINIT_MULTITHREADED); test_interfaces(); test_add_stream(); test_media_streams(); test_enum_pins(); test_find_pin(); test_pin_info(); test_initialize(); test_set_state(); test_enum_media_types(); test_media_types(); test_get_end_of_stream_event_handle(); test_avi_path = load_resource(L"test.avi"); test_openfile(test_avi_path); test_mmstream_get_duration(test_avi_path); unload_resource(test_avi_path); test_audiodata_query_interface(); test_audiodata_get_info(); test_audiodata_set_buffer(); test_audiodata_set_actual(); test_audiodata_get_format(); test_audiodata_set_format(); test_audiostream_get_format(); test_audiostream_set_format(); test_audiostream_receive_connection(); test_audiostream_receive(); test_audiostream_initialize(); test_audiostream_begin_flush_end_flush(); test_audiostream_new_segment(); test_audiostreamsample_update(); test_audiostreamsample_completion_status(); test_audiostreamsample_get_sample_times(); test_audiostreamsample_get_media_stream(); test_audiostreamsample_get_audio_data(); test_ddrawstream_initialize(); test_ddrawstream_getsetdirectdraw(); test_ddrawstream_receive_connection(); test_ddrawstream_create_sample(); test_ddrawstream_get_format(); test_ddrawstream_set_format(); test_ddrawstream_receive(); test_ddrawstream_begin_flush_end_flush(); test_ddrawstream_new_segment(); test_ddrawstream_get_time_per_frame(); test_ddrawstreamsample_get_media_stream(); test_ddrawstreamsample_update(); test_ddrawstreamsample_completion_status(); test_ddrawstreamsample_get_sample_times(); test_ammediastream_join_am_multi_media_stream(); test_ammediastream_join_filter(); test_ammediastream_join_filter_graph(); test_ammediastream_set_state(); test_ammediastream_end_of_stream(); test_mediastreamfilter_get_state(); test_mediastreamfilter_stop_pause_run(); test_mediastreamfilter_support_seeking(); test_mediastreamfilter_seeking(); test_mediastreamfilter_get_current_stream_time(); test_mediastreamfilter_reference_time_to_stream_time(); test_mediastreamfilter_wait_until(); test_mediastreamfilter_end_of_stream(); CoUninitialize(); }