Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-cw
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-cw
Commits
3adf8116
Commit
3adf8116
authored
Jul 04, 2008
by
Maarten Lankhorst
Committed by
Alexandre Julliard
Jul 09, 2008
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
quartz: Add tests for avi splitter.
parent
19360d41
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
464 additions
and
0 deletions
+464
-0
Makefile.in
dlls/quartz/tests/Makefile.in
+1
-0
avisplitter.c
dlls/quartz/tests/avisplitter.c
+463
-0
No files found.
dlls/quartz/tests/Makefile.in
View file @
3adf8116
...
...
@@ -6,6 +6,7 @@ TESTDLL = quartz.dll
IMPORTS
=
oleaut32 ole32 advapi32 kernel32
CTESTS
=
\
avisplitter.c
\
filtergraph.c
\
filtermapper.c
\
memallocator.c
\
...
...
dlls/quartz/tests/avisplitter.c
0 → 100644
View file @
3adf8116
/*
* Unit tests for the avi splitter functions
*
* Copyright (C) 2007 Google (Lei Zhang)
* Copyright (C) 2008 Google (Maarten Lankhorst)
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#define COBJMACROS
#include "wine/test.h"
#include "dshow.h"
#include "tlhelp32.h"
static
IUnknown
*
pAviSplitter
=
NULL
;
static
int
count_threads
(
void
)
{
THREADENTRY32
te
;
int
threads
;
HANDLE
h
;
h
=
CreateToolhelp32Snapshot
(
TH32CS_SNAPTHREAD
,
0
);
te
.
dwSize
=
sizeof
(
te
);
if
(
h
==
INVALID_HANDLE_VALUE
)
return
-
1
;
Thread32First
(
h
,
&
te
);
if
(
te
.
th32OwnerProcessID
==
GetCurrentProcessId
())
threads
=
1
;
else
threads
=
0
;
while
(
Thread32Next
(
h
,
&
te
))
if
(
te
.
th32OwnerProcessID
==
GetCurrentProcessId
())
++
threads
;
CloseHandle
(
h
);
return
threads
;
}
static
int
create_avisplitter
(
void
)
{
HRESULT
hr
;
hr
=
CoCreateInstance
(
&
CLSID_AviSplitter
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IUnknown
,
(
LPVOID
*
)
&
pAviSplitter
);
return
(
hr
==
S_OK
&&
pAviSplitter
!=
NULL
);
}
static
void
release_avisplitter
(
void
)
{
HRESULT
hr
;
Sleep
(
1000
);
hr
=
IUnknown_Release
(
pAviSplitter
);
/* Looks like wine has a reference leak somewhere on test_threads tests,
* it passes in windows
*/
ok
(
hr
==
0
,
"IUnknown_Release failed with %d
\n
"
,
(
INT
)
hr
);
while
(
hr
>
0
)
hr
=
IUnknown_Release
(
pAviSplitter
);
pAviSplitter
=
NULL
;
}
static
void
test_query_interface
(
void
)
{
HRESULT
hr
;
ULONG
ref
;
IUnknown
*
iface
=
NULL
;
hr
=
IUnknown_QueryInterface
(
pAviSplitter
,
&
IID_IBaseFilter
,
(
void
**
)
&
iface
);
ok
(
hr
==
S_OK
,
"IID_IBaseFilter should exist, got %08x!
\n
"
,
GetLastError
());
if
(
hr
==
S_OK
)
{
ref
=
IUnknown_Release
(
iface
);
iface
=
NULL
;
ok
(
ref
==
1
,
"Reference is %u, expected 1
\n
"
,
ref
);
}
hr
=
IUnknown_QueryInterface
(
pAviSplitter
,
&
IID_IMediaSeeking
,
(
void
**
)
&
iface
);
if
(
hr
==
S_OK
)
ref
=
IUnknown_Release
(
iface
);
iface
=
NULL
;
todo_wine
ok
(
hr
==
E_NOINTERFACE
,
"Query for IMediaSeeking returned: %08x
\n
"
,
hr
);
/* These interfaces should not be present:
IID_IKsPropertySet, IID_IMediaPosition, IID_IQualityControl, IID_IQualProp
*/
}
static
void
test_pin
(
IPin
*
pin
)
{
IMemInputPin
*
mpin
=
NULL
;
IPin_QueryInterface
(
pin
,
&
IID_IMemInputPin
,
(
void
**
)
&
mpin
);
ok
(
mpin
==
NULL
,
"IMemInputPin found!
\n
"
);
if
(
mpin
)
IMemInputPin_Release
(
mpin
);
/* TODO */
}
static
void
test_basefilter
(
void
)
{
IEnumPins
*
pin_enum
=
NULL
;
IBaseFilter
*
base
=
NULL
;
IPin
*
pins
[
2
];
ULONG
ref
;
HRESULT
hr
;
IUnknown_QueryInterface
(
pAviSplitter
,
&
IID_IBaseFilter
,
(
void
*
)
&
base
);
if
(
base
==
NULL
)
{
/* test_query_interface handles this case */
skip
(
"No IBaseFilter
\n
"
);
return
;
}
hr
=
IBaseFilter_EnumPins
(
base
,
NULL
);
ok
(
hr
==
E_POINTER
,
"hr = %08x and not E_POINTER
\n
"
,
hr
);
hr
=
IBaseFilter_EnumPins
(
base
,
&
pin_enum
);
ok
(
hr
==
S_OK
,
"hr = %08x and not S_OK
\n
"
,
hr
);
hr
=
IEnumPins_Next
(
pin_enum
,
1
,
NULL
,
NULL
);
ok
(
hr
==
E_POINTER
,
"hr = %08x and not E_POINTER
\n
"
,
hr
);
hr
=
IEnumPins_Next
(
pin_enum
,
2
,
pins
,
NULL
);
ok
(
hr
==
E_INVALIDARG
,
"hr = %08x and not E_INVALIDARG
\n
"
,
hr
);
pins
[
0
]
=
(
void
*
)
0xdead
;
pins
[
1
]
=
(
void
*
)
0xdeed
;
hr
=
IEnumPins_Next
(
pin_enum
,
2
,
pins
,
&
ref
);
ok
(
hr
==
S_FALSE
,
"hr = %08x instead of S_FALSE
\n
"
,
hr
);
ok
(
pins
[
0
]
!=
(
void
*
)
0xdead
&&
pins
[
0
]
!=
NULL
,
"pins[0] = %p
\n
"
,
pins
[
0
]);
if
(
pins
[
0
]
!=
(
void
*
)
0xdead
&&
pins
[
0
]
!=
NULL
)
{
test_pin
(
pins
[
0
]);
IPin_Release
(
pins
[
0
]);
}
ok
(
pins
[
1
]
==
(
void
*
)
0xdeed
,
"pins[1] = %p
\n
"
,
pins
[
1
]);
ref
=
IEnumPins_Release
(
pin_enum
);
ok
(
ref
==
0
,
"ref is %u and not 0!
\n
"
,
ref
);
IBaseFilter_Release
(
base
);
}
static
const
WCHAR
wfile
[]
=
{
't'
,
'e'
,
's'
,
't'
,
'.'
,
'a'
,
'v'
,
'i'
,
0
};
static
const
char
afile
[]
=
"test.avi"
;
/* This test doesn't use the quartz filtergraph because it makes it impossible
* to be certain that a thread is really one owned by the avi splitter
* A lot of the decoder filters will also have their own thread, and windows'
* filtergraph has a seperate thread for start/stop/seeking requests.
* By avoiding the filtergraph all together and connecting streams directly to
* the null renderer I am sure that this is not the case here.
*/
static
void
test_threads
()
{
IFileSourceFilter
*
pfile
=
NULL
;
IBaseFilter
*
preader
=
NULL
,
*
pavi
=
NULL
;
IEnumPins
*
enumpins
=
NULL
;
IPin
*
filepin
=
NULL
,
*
avipin
=
NULL
;
HRESULT
hr
;
int
baselevel
,
curlevel
,
expected
;
HANDLE
file
=
NULL
;
PIN_DIRECTION
dir
=
PINDIR_OUTPUT
;
char
buffer
[
13
];
DWORD
readbytes
;
FILTER_STATE
state
;
/* Before doing anything */
baselevel
=
count_threads
();
expected
=
1
;
ok
(
baselevel
==
expected
,
"Basic amount of threads should be %d, not %d!
\n
"
,
expected
,
baselevel
);
file
=
CreateFileW
(
wfile
,
GENERIC_READ
,
FILE_SHARE_READ
|
FILE_SHARE_WRITE
,
NULL
,
OPEN_EXISTING
,
0
,
NULL
);
if
(
!
file
)
{
skip
(
"Could not read test file
\"
%s
\"
, skipping test
\n
"
,
afile
);
return
;
}
memset
(
buffer
,
0
,
13
);
readbytes
=
12
;
ReadFile
(
file
,
buffer
,
readbytes
,
&
readbytes
,
NULL
);
CloseHandle
(
file
);
if
(
strncmp
(
buffer
,
"RIFF"
,
4
)
||
strcmp
(
buffer
+
8
,
"AVI "
))
{
skip
(
"%s is not an avi riff file, not doing the avi splitter test
\n
"
,
afile
);
return
;
}
hr
=
IUnknown_QueryInterface
(
pAviSplitter
,
&
IID_IFileSourceFilter
,
(
void
**
)
&
pfile
);
ok
(
hr
==
E_NOINTERFACE
,
"Avi splitter returns unexpected error: %08x
\n
"
,
hr
);
if
(
pfile
)
IUnknown_Release
(
pfile
);
pfile
=
NULL
;
hr
=
CoCreateInstance
(
&
CLSID_AsyncReader
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IBaseFilter
,
(
LPVOID
*
)
&
preader
);
ok
(
hr
==
S_OK
,
"Could not create asynchronous reader: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
hr
=
IUnknown_QueryInterface
(
preader
,
&
IID_IFileSourceFilter
,
(
void
**
)
&
pfile
);
ok
(
hr
==
S_OK
,
"Could not get IFileSourceFilter: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
hr
=
IUnknown_QueryInterface
(
pAviSplitter
,
&
IID_IBaseFilter
,
(
void
**
)
&
pavi
);
ok
(
hr
==
S_OK
,
"Could not get base filter: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
hr
=
IFileSourceFilter_Load
(
pfile
,
wfile
,
NULL
);
if
(
hr
!=
S_OK
)
{
trace
(
"Could not load file
\n
"
);
goto
fail
;
}
hr
=
IBaseFilter_EnumPins
(
preader
,
&
enumpins
);
ok
(
hr
==
S_OK
,
"No enumpins: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
hr
=
IEnumPins_Next
(
enumpins
,
1
,
&
filepin
,
NULL
);
ok
(
hr
==
S_OK
,
"No pin: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
IUnknown_Release
(
enumpins
);
enumpins
=
NULL
;
hr
=
IBaseFilter_EnumPins
(
pavi
,
&
enumpins
);
ok
(
hr
==
S_OK
,
"No enumpins: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
hr
=
IEnumPins_Next
(
enumpins
,
1
,
&
avipin
,
NULL
);
ok
(
hr
==
S_OK
,
"No pin: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
curlevel
=
count_threads
();
ok
(
curlevel
==
baselevel
,
"Amount of threads should be %d not %d
\n
"
,
baselevel
,
curlevel
);
hr
=
IPin_Connect
(
filepin
,
avipin
,
NULL
);
ok
(
hr
==
S_OK
,
"Could not connect: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
goto
fail
;
expected
=
1
+
baselevel
;
curlevel
=
count_threads
();
ok
(
curlevel
==
expected
,
"Amount of threads should be %d not %d
\n
"
,
expected
,
curlevel
);
IUnknown_Release
(
avipin
);
avipin
=
NULL
;
IEnumPins_Reset
(
enumpins
);
/* Windows puts the pins in the order: Outputpins - Inputpin,
* wine does the reverse, just don't test it for now
* Hate to admit it, but windows way makes more sense
*/
while
(
IEnumPins_Next
(
enumpins
,
1
,
&
avipin
,
NULL
)
==
S_OK
)
{
ok
(
hr
==
S_OK
,
"hr: %08x
\n
"
,
hr
);
IPin_QueryDirection
(
avipin
,
&
dir
);
if
(
dir
==
PINDIR_OUTPUT
)
{
/* Well, connect it to a null renderer! */
IBaseFilter
*
pnull
=
NULL
;
IEnumPins
*
nullenum
=
NULL
;
IPin
*
nullpin
=
NULL
;
hr
=
CoCreateInstance
(
&
CLSID_NullRenderer
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IBaseFilter
,
(
LPVOID
*
)
&
pnull
);
ok
(
hr
==
S_OK
,
"Could not create null renderer: %08x
\n
"
,
hr
);
if
(
hr
!=
S_OK
)
break
;
IBaseFilter_EnumPins
(
pnull
,
&
nullenum
);
IEnumPins_Next
(
nullenum
,
1
,
&
nullpin
,
NULL
);
IEnumPins_Release
(
nullenum
);
IPin_QueryDirection
(
nullpin
,
&
dir
);
hr
=
IPin_Connect
(
avipin
,
nullpin
,
NULL
);
ok
(
hr
==
S_OK
,
"Failed to connect output pin: %08x
\n
"
,
hr
);
IPin_Release
(
nullpin
);
if
(
hr
!=
S_OK
)
{
IBaseFilter_Release
(
pnull
);
break
;
}
IBaseFilter_Run
(
pnull
,
0
);
++
expected
;
}
IUnknown_Release
(
avipin
);
avipin
=
NULL
;
}
if
(
avipin
)
IUnknown_Release
(
avipin
);
avipin
=
NULL
;
if
(
hr
!=
S_OK
)
goto
fail2
;
/* At this point there is a minimalistic connected avi splitter that can
* Be used for all sorts of source filter tests, however that still needs
* to be written at a later time.
*
* Interesting tests:
* - Can you disconnect an output pin while running?
* Expecting: Yes
* - Can you disconnect the pullpin while running?
* Expecting: No
* - Is the reference count incremented during playback or when connected?
* Does this happen once for every output pin? Or is there something else
* going on.
* Expecting: You tell me
*/
IBaseFilter_Run
(
preader
,
0
);
IBaseFilter_Run
(
pavi
,
0
);
IBaseFilter_GetState
(
pavi
,
INFINITE
,
&
state
);
curlevel
=
count_threads
();
/* On a 2 stream filter, there are 4 or 5 threads (seems to be 5)
* One is the thread we are in. That leaves 3 or 4 for other dark purposes
* Wine is 1 thread short!
*/
ok
(
curlevel
==
expected
||
curlevel
==
expected
+
1
,
"Amount of threads should be %d not %d
\n
"
,
expected
,
curlevel
);
IBaseFilter_Pause
(
pavi
);
IBaseFilter_Pause
(
preader
);
IBaseFilter_Stop
(
pavi
);
IBaseFilter_Stop
(
preader
);
IBaseFilter_GetState
(
pavi
,
INFINITE
,
&
state
);
IBaseFilter_GetState
(
preader
,
INFINITE
,
&
state
);
fail2:
IEnumPins_Reset
(
enumpins
);
while
(
IEnumPins_Next
(
enumpins
,
1
,
&
avipin
,
NULL
)
==
S_OK
)
{
IPin
*
to
=
NULL
;
IPin_QueryDirection
(
avipin
,
&
dir
);
IPin_ConnectedTo
(
avipin
,
&
to
);
if
(
to
)
{
IPin_Release
(
to
);
if
(
dir
==
PINDIR_OUTPUT
)
{
PIN_INFO
info
;
IPin_QueryPinInfo
(
to
,
&
info
);
/* Release twice: Once normal, second from the
* previous while loop
*/
IBaseFilter_Stop
(
info
.
pFilter
);
IPin_Disconnect
(
to
);
IPin_Disconnect
(
avipin
);
IBaseFilter_Release
(
info
.
pFilter
);
IBaseFilter_Release
(
info
.
pFilter
);
}
else
{
IPin_Disconnect
(
to
);
IPin_Disconnect
(
avipin
);
}
}
IPin_Release
(
avipin
);
avipin
=
NULL
;
}
fail:
if
(
hr
!=
S_OK
)
skip
(
"Prerequisites not matched, skipping remainder of test
\n
"
);
if
(
enumpins
)
IUnknown_Release
(
enumpins
);
if
(
avipin
)
IUnknown_Release
(
avipin
);
if
(
filepin
)
{
IPin
*
to
=
NULL
;
IPin_ConnectedTo
(
filepin
,
&
to
);
if
(
to
)
{
IPin_Disconnect
(
filepin
);
IPin_Disconnect
(
to
);
}
IUnknown_Release
(
filepin
);
}
if
(
preader
)
IUnknown_Release
(
preader
);
if
(
pavi
)
IUnknown_Release
(
pavi
);
if
(
pfile
)
IUnknown_Release
(
pfile
);
ok
(
baselevel
==
1
,
"Basic amount of threads should be %d, not %d!
\n
"
,
1
,
baselevel
);
}
START_TEST
(
avisplitter
)
{
CoInitialize
(
NULL
);
if
(
!
create_avisplitter
())
{
skip
(
"Could not create avisplitter
\n
"
);
return
;
}
test_query_interface
();
test_basefilter
();
test_threads
();
release_avisplitter
();
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment