2017-09-20 14:03:36 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Fabian Maurer
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define COBJMACROS
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "dshow.h"
|
|
|
|
#include "evr.h"
|
|
|
|
#include "d3d9.h"
|
|
|
|
#include "initguid.h"
|
|
|
|
#include "dxva2api.h"
|
|
|
|
|
|
|
|
#include "wine/test.h"
|
|
|
|
|
|
|
|
#define QI_SUCCEED(iface, riid, ppv) hr = IUnknown_QueryInterface(iface, &riid, (LPVOID*)&ppv); \
|
|
|
|
ok(hr == S_OK, "%s: run %d: IUnknown_QueryInterface returned %x\n", testid, testrun, hr); \
|
|
|
|
ok(ppv != NULL, "%s: run %d: Pointer is NULL\n", testid, testrun);
|
|
|
|
|
|
|
|
#define QI_FAIL(iface, riid, ppv) hr = IUnknown_QueryInterface(iface, &riid, (LPVOID*)&ppv); \
|
|
|
|
ok(hr == E_NOINTERFACE, "%s: run %d: IUnknown_QueryInterface returned %x\n", testid, testrun, hr); \
|
|
|
|
ok(ppv == NULL, "%s: run %d: Pointer is %p\n", testid, testrun, ppv);
|
|
|
|
|
|
|
|
#define ADDREF_EXPECT(iface, num) if (iface) { \
|
|
|
|
refCount = IUnknown_AddRef(iface); \
|
|
|
|
ok(refCount == num, "%s: run %d: IUnknown_AddRef should return %d, got %d\n", testid, testrun, num, refCount); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define RELEASE_EXPECT(iface, num) if (iface) { \
|
|
|
|
refCount = IUnknown_Release(iface); \
|
|
|
|
ok(refCount == num, "%s: run %d: IUnknown_Release should return %d, got %d\n", testid, testrun, num, refCount); \
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_aggregation(const CLSID clsid_inner, const IID iid_inner, const char *testid, int testrun)
|
|
|
|
{
|
|
|
|
const CLSID clsid_outer = CLSID_SystemClock;
|
|
|
|
const IID iid_outer = IID_IReferenceClock;
|
|
|
|
HRESULT hr;
|
|
|
|
ULONG refCount;
|
|
|
|
IUnknown *unk_outer = NULL;
|
|
|
|
IUnknown *unk_inner = NULL;
|
|
|
|
IUnknown *unk_inner_fail = NULL;
|
|
|
|
IUnknown *unk_outer_test = NULL;
|
|
|
|
IUnknown *unk_inner_test = NULL;
|
|
|
|
IUnknown *unk_aggregatee = NULL;
|
|
|
|
IUnknown *unk_aggregator = NULL;
|
|
|
|
IUnknown *unk_test = NULL;
|
|
|
|
|
|
|
|
hr = CoCreateInstance(&clsid_outer, NULL, CLSCTX_INPROC_SERVER, &IID_IUnknown, (LPVOID*)&unk_outer);
|
|
|
|
ok(hr == S_OK, "%s: run %d: First CoCreateInstance failed with %x\n", testid, testrun, hr);
|
|
|
|
ok(unk_outer != NULL, "%s: run %d: unk_outer is NULL\n", testid, testrun);
|
|
|
|
|
|
|
|
if (!unk_outer)
|
|
|
|
{
|
|
|
|
skip("%s: run %d: unk_outer is NULL\n", testid, testrun);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* for aggregation, we should only be able to request IUnknown */
|
|
|
|
hr = CoCreateInstance(&clsid_inner, unk_outer, CLSCTX_INPROC_SERVER, &iid_inner, (LPVOID*)&unk_inner_fail);
|
|
|
|
if (hr == REGDB_E_CLASSNOTREG)
|
|
|
|
{
|
|
|
|
skip("%s: run %d: Class not registered\n", testid, testrun);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ok(hr == E_NOINTERFACE, "%s: run %d: Second CoCreateInstance returned %x\n", testid, testrun, hr);
|
|
|
|
ok(unk_inner_fail == NULL, "%s: run %d: unk_inner_fail is not NULL\n", testid, testrun);
|
|
|
|
|
|
|
|
/* aggregation, request IUnknown */
|
|
|
|
hr = CoCreateInstance(&clsid_inner, unk_outer, CLSCTX_INPROC_SERVER, &IID_IUnknown, (LPVOID*)&unk_inner);
|
2017-09-23 17:21:16 +02:00
|
|
|
todo_wine ok(hr == S_OK, "%s: run %d: Third CoCreateInstance returned %x\n", testid, testrun, hr);
|
|
|
|
todo_wine ok(unk_inner != NULL, "%s: run %d: unk_inner is NULL\n", testid, testrun);
|
2017-09-20 14:03:36 +02:00
|
|
|
|
|
|
|
if (!unk_inner)
|
|
|
|
{
|
|
|
|
skip("%s: run %d: unk_inner is NULL\n", testid, testrun);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ADDREF_EXPECT(unk_outer, 2);
|
|
|
|
ADDREF_EXPECT(unk_inner, 2);
|
|
|
|
RELEASE_EXPECT(unk_outer, 1);
|
|
|
|
RELEASE_EXPECT(unk_inner, 1);
|
|
|
|
|
|
|
|
QI_FAIL(unk_outer, iid_inner, unk_aggregatee);
|
|
|
|
QI_FAIL(unk_inner, iid_outer, unk_aggregator);
|
|
|
|
|
|
|
|
QI_SUCCEED(unk_outer, iid_outer, unk_aggregator);
|
|
|
|
QI_SUCCEED(unk_outer, IID_IUnknown, unk_outer_test);
|
|
|
|
QI_SUCCEED(unk_inner, iid_inner, unk_aggregatee);
|
|
|
|
QI_SUCCEED(unk_inner, IID_IUnknown, unk_inner_test);
|
|
|
|
|
|
|
|
if (!unk_aggregator || !unk_outer_test || !unk_aggregatee || !unk_inner_test)
|
|
|
|
{
|
|
|
|
skip("%s: run %d: One of the required interfaces is NULL\n", testid, testrun);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ADDREF_EXPECT(unk_aggregator, 5);
|
|
|
|
ADDREF_EXPECT(unk_outer_test, 6);
|
|
|
|
ADDREF_EXPECT(unk_aggregatee, 7);
|
|
|
|
ADDREF_EXPECT(unk_inner_test, 3);
|
|
|
|
RELEASE_EXPECT(unk_aggregator, 6);
|
|
|
|
RELEASE_EXPECT(unk_outer_test, 5);
|
|
|
|
RELEASE_EXPECT(unk_aggregatee, 4);
|
|
|
|
RELEASE_EXPECT(unk_inner_test, 2);
|
|
|
|
|
|
|
|
QI_SUCCEED(unk_aggregator, IID_IUnknown, unk_test);
|
|
|
|
QI_SUCCEED(unk_outer_test, IID_IUnknown, unk_test);
|
|
|
|
QI_SUCCEED(unk_aggregatee, IID_IUnknown, unk_test);
|
|
|
|
QI_SUCCEED(unk_inner_test, IID_IUnknown, unk_test);
|
|
|
|
|
|
|
|
QI_FAIL(unk_aggregator, iid_inner, unk_test);
|
|
|
|
QI_FAIL(unk_outer_test, iid_inner, unk_test);
|
|
|
|
QI_FAIL(unk_aggregatee, iid_inner, unk_test);
|
|
|
|
QI_SUCCEED(unk_inner_test, iid_inner, unk_test);
|
|
|
|
|
|
|
|
QI_SUCCEED(unk_aggregator, iid_outer, unk_test);
|
|
|
|
QI_SUCCEED(unk_outer_test, iid_outer, unk_test);
|
|
|
|
QI_SUCCEED(unk_aggregatee, iid_outer, unk_test);
|
|
|
|
QI_FAIL(unk_inner_test, iid_outer, unk_test);
|
|
|
|
|
|
|
|
RELEASE_EXPECT(unk_aggregator, 10);
|
|
|
|
RELEASE_EXPECT(unk_outer_test, 9);
|
|
|
|
RELEASE_EXPECT(unk_aggregatee, 8);
|
|
|
|
RELEASE_EXPECT(unk_inner_test, 2);
|
|
|
|
RELEASE_EXPECT(unk_outer, 7);
|
|
|
|
RELEASE_EXPECT(unk_inner, 1);
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
refCount = IUnknown_Release(unk_inner);
|
|
|
|
} while (refCount);
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
refCount = IUnknown_Release(unk_outer);
|
|
|
|
} while (refCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_evr_filter_aggregations(void)
|
|
|
|
{
|
|
|
|
const IID * iids[] = {
|
|
|
|
&IID_IAMCertifiedOutputProtection, &IID_IAMFilterMiscFlags, &IID_IBaseFilter,
|
|
|
|
&IID_IKsPropertySet, &IID_IMediaEventSink, &IID_IMediaSeeking, &IID_IQualityControl,
|
|
|
|
&IID_IQualProp, &IID_IEVRFilterConfig, &IID_IMFGetService, &IID_IMFVideoPositionMapper,
|
|
|
|
&IID_IMFVideoRenderer, &IID_IQualityControl
|
|
|
|
};
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < sizeof(iids) / sizeof(iids[0]); i++)
|
|
|
|
{
|
|
|
|
test_aggregation(CLSID_EnhancedVideoRenderer, *iids[i], "filter", i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
START_TEST(filter)
|
|
|
|
{
|
|
|
|
CoInitialize(NULL);
|
|
|
|
|
|
|
|
test_evr_filter_aggregations();
|
|
|
|
|
|
|
|
CoUninitialize();
|
|
|
|
}
|