2011-05-21 14:24:50 +02:00
/*
* Directshow capture interface
* Copyright ( c ) 2010 Ramiro Polla
*
* This file is part of FFmpeg .
*
* FFmpeg 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 .
*
* FFmpeg 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 FFmpeg ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
2014-11-25 03:49:09 +01:00
# include "dshow_capture.h"
2011-09-09 05:12:42 +02:00
# include "libavutil/parseutils.h"
2012-09-07 01:41:19 +02:00
# include "libavutil/pixdesc.h"
2011-09-09 05:09:23 +02:00
# include "libavutil/opt.h"
2011-12-03 00:45:46 +01:00
# include "libavformat/internal.h"
2013-01-15 23:01:43 +01:00
# include "libavformat/riff.h"
2011-05-26 19:11:25 +02:00
# include "avdevice.h"
2013-01-16 01:07:54 +01:00
# include "libavcodec/raw.h"
2015-04-22 18:38:38 +02:00
# include "objidl.h"
# include "shlwapi.h"
2011-05-21 14:24:50 +02:00
2012-10-08 20:54:00 +02:00
static enum AVPixelFormat dshow_pixfmt ( DWORD biCompression , WORD biBitCount )
2011-05-21 14:24:50 +02:00
{
switch ( biCompression ) {
2011-10-08 20:00:00 +02:00
case BI_BITFIELDS :
2011-05-21 14:24:50 +02:00
case BI_RGB :
switch ( biBitCount ) { /* 1-8 are untested */
case 1 :
2012-10-08 20:54:00 +02:00
return AV_PIX_FMT_MONOWHITE ;
2011-05-21 14:24:50 +02:00
case 4 :
2012-10-08 20:54:00 +02:00
return AV_PIX_FMT_RGB4 ;
2011-05-21 14:24:50 +02:00
case 8 :
2012-10-08 20:54:00 +02:00
return AV_PIX_FMT_RGB8 ;
2011-05-21 14:24:50 +02:00
case 16 :
2012-10-08 20:54:00 +02:00
return AV_PIX_FMT_RGB555 ;
2011-05-21 14:24:50 +02:00
case 24 :
2012-10-08 20:54:00 +02:00
return AV_PIX_FMT_BGR24 ;
2011-05-21 14:24:50 +02:00
case 32 :
2014-03-08 13:46:05 +01:00
return AV_PIX_FMT_0RGB32 ;
2011-05-21 14:24:50 +02:00
}
}
2014-08-07 06:22:48 +02:00
return avpriv_find_pix_fmt ( avpriv_get_raw_pix_fmt_tags ( ) , biCompression ) ; // all others
2011-05-21 14:24:50 +02:00
}
static int
dshow_read_close ( AVFormatContext * s )
{
struct dshow_ctx * ctx = s - > priv_data ;
AVPacketList * pktl ;
if ( ctx - > control ) {
IMediaControl_Stop ( ctx - > control ) ;
IMediaControl_Release ( ctx - > control ) ;
}
2012-12-17 06:44:12 +01:00
if ( ctx - > media_event )
IMediaEvent_Release ( ctx - > media_event ) ;
2011-09-13 19:56:28 +02:00
if ( ctx - > graph ) {
IEnumFilters * fenum ;
int r ;
r = IGraphBuilder_EnumFilters ( ctx - > graph , & fenum ) ;
if ( r = = S_OK ) {
IBaseFilter * f ;
IEnumFilters_Reset ( fenum ) ;
2011-09-30 22:49:43 +02:00
while ( IEnumFilters_Next ( fenum , 1 , & f , NULL ) = = S_OK ) {
2011-09-13 19:56:28 +02:00
if ( IGraphBuilder_RemoveFilter ( ctx - > graph , f ) = = S_OK )
IEnumFilters_Reset ( fenum ) ; /* When a filter is removed,
* the list must be reset . */
2011-09-30 22:49:43 +02:00
IBaseFilter_Release ( f ) ;
}
2011-09-13 19:56:28 +02:00
IEnumFilters_Release ( fenum ) ;
}
IGraphBuilder_Release ( ctx - > graph ) ;
}
2011-05-21 14:24:50 +02:00
if ( ctx - > capture_pin [ VideoDevice ] )
libAVPin_Release ( ctx - > capture_pin [ VideoDevice ] ) ;
if ( ctx - > capture_pin [ AudioDevice ] )
libAVPin_Release ( ctx - > capture_pin [ AudioDevice ] ) ;
if ( ctx - > capture_filter [ VideoDevice ] )
libAVFilter_Release ( ctx - > capture_filter [ VideoDevice ] ) ;
if ( ctx - > capture_filter [ AudioDevice ] )
libAVFilter_Release ( ctx - > capture_filter [ AudioDevice ] ) ;
if ( ctx - > device_pin [ VideoDevice ] )
IPin_Release ( ctx - > device_pin [ VideoDevice ] ) ;
if ( ctx - > device_pin [ AudioDevice ] )
IPin_Release ( ctx - > device_pin [ AudioDevice ] ) ;
if ( ctx - > device_filter [ VideoDevice ] )
IBaseFilter_Release ( ctx - > device_filter [ VideoDevice ] ) ;
if ( ctx - > device_filter [ AudioDevice ] )
IBaseFilter_Release ( ctx - > device_filter [ AudioDevice ] ) ;
if ( ctx - > device_name [ 0 ] )
2014-12-28 18:16:37 +01:00
av_freep ( & ctx - > device_name [ 0 ] ) ;
2011-05-21 14:24:50 +02:00
if ( ctx - > device_name [ 1 ] )
2014-12-28 18:16:37 +01:00
av_freep ( & ctx - > device_name [ 1 ] ) ;
2011-05-21 14:24:50 +02:00
if ( ctx - > mutex )
CloseHandle ( ctx - > mutex ) ;
2012-12-17 06:44:12 +01:00
if ( ctx - > event [ 0 ] )
CloseHandle ( ctx - > event [ 0 ] ) ;
if ( ctx - > event [ 1 ] )
CloseHandle ( ctx - > event [ 1 ] ) ;
2011-05-21 14:24:50 +02:00
pktl = ctx - > pktl ;
while ( pktl ) {
AVPacketList * next = pktl - > next ;
2015-10-27 14:35:30 +01:00
av_packet_unref ( & pktl - > pkt ) ;
2011-05-21 14:24:50 +02:00
av_free ( pktl ) ;
pktl = next ;
}
2012-12-17 06:44:51 +01:00
CoUninitialize ( ) ;
2011-05-21 14:24:50 +02:00
return 0 ;
}
static char * dup_wchar_to_utf8 ( wchar_t * w )
{
char * s = NULL ;
int l = WideCharToMultiByte ( CP_UTF8 , 0 , w , - 1 , 0 , 0 , 0 , 0 ) ;
s = av_malloc ( l ) ;
if ( s )
WideCharToMultiByte ( CP_UTF8 , 0 , w , - 1 , s , l , 0 , 0 ) ;
return s ;
}
2014-03-27 20:44:20 +01:00
static int shall_we_drop ( AVFormatContext * s , int index , enum dshowDeviceType devtype )
2011-05-21 14:24:50 +02:00
{
struct dshow_ctx * ctx = s - > priv_data ;
2013-08-03 22:24:11 +02:00
static const uint8_t dropscore [ ] = { 62 , 75 , 87 , 100 } ;
2011-05-21 14:24:50 +02:00
const int ndropscores = FF_ARRAY_ELEMS ( dropscore ) ;
2014-03-21 23:27:01 +01:00
unsigned int buffer_fullness = ( ctx - > curbufsize [ index ] * 100 ) / s - > max_picture_buffer ;
2015-01-23 14:01:32 +01:00
const char * devtypename = ( devtype = = VideoDevice ) ? " video " : " audio " ;
2011-05-21 14:24:50 +02:00
if ( dropscore [ + + ctx - > video_frame_num % ndropscores ] < = buffer_fullness ) {
av_log ( s , AV_LOG_ERROR ,
2015-01-23 14:01:32 +01:00
" real-time buffer [%s] [%s input] too full or near too full (%d%% of size: %d [rtbufsize parameter])! frame dropped! \n " ,
ctx - > device_name [ devtype ] , devtypename , buffer_fullness , s - > max_picture_buffer ) ;
2011-05-21 14:24:50 +02:00
return 1 ;
}
return 0 ;
}
static void
2014-03-27 20:44:20 +01:00
callback ( void * priv_data , int index , uint8_t * buf , int buf_size , int64_t time , enum dshowDeviceType devtype )
2011-05-21 14:24:50 +02:00
{
AVFormatContext * s = priv_data ;
struct dshow_ctx * ctx = s - > priv_data ;
AVPacketList * * ppktl , * pktl_next ;
// dump_videohdr(s, vdhdr);
WaitForSingleObject ( ctx - > mutex , INFINITE ) ;
2014-03-27 20:44:20 +01:00
if ( shall_we_drop ( s , index , devtype ) )
2012-08-19 16:10:07 +02:00
goto fail ;
2011-05-21 14:24:50 +02:00
pktl_next = av_mallocz ( sizeof ( AVPacketList ) ) ;
if ( ! pktl_next )
goto fail ;
if ( av_new_packet ( & pktl_next - > pkt , buf_size ) < 0 ) {
av_free ( pktl_next ) ;
goto fail ;
}
pktl_next - > pkt . stream_index = index ;
pktl_next - > pkt . pts = time ;
memcpy ( pktl_next - > pkt . data , buf , buf_size ) ;
for ( ppktl = & ctx - > pktl ; * ppktl ; ppktl = & ( * ppktl ) - > next ) ;
* ppktl = pktl_next ;
2014-03-21 23:27:01 +01:00
ctx - > curbufsize [ index ] + = buf_size ;
2011-05-21 14:24:50 +02:00
2012-12-17 06:44:12 +01:00
SetEvent ( ctx - > event [ 1 ] ) ;
2011-05-21 14:24:50 +02:00
ReleaseMutex ( ctx - > mutex ) ;
return ;
fail :
ReleaseMutex ( ctx - > mutex ) ;
return ;
}
2011-09-09 05:08:43 +02:00
/**
* Cycle through available devices using the device enumerator devenum ,
* retrieve the device with type specified by devtype and return the
* pointer to the object found in * pfilter .
2011-09-09 05:09:23 +02:00
* If pfilter is NULL , list all device names .
2011-09-09 05:08:43 +02:00
*/
2011-05-21 14:24:50 +02:00
static int
2011-09-09 05:08:43 +02:00
dshow_cycle_devices ( AVFormatContext * avctx , ICreateDevEnum * devenum ,
2015-01-23 14:54:44 +01:00
enum dshowDeviceType devtype , enum dshowSourceFilterType sourcetype , IBaseFilter * * pfilter )
2011-05-21 14:24:50 +02:00
{
struct dshow_ctx * ctx = avctx - > priv_data ;
IBaseFilter * device_filter = NULL ;
IEnumMoniker * classenum = NULL ;
IMoniker * m = NULL ;
const char * device_name = ctx - > device_name [ devtype ] ;
2011-09-30 22:49:09 +02:00
int skip = ( devtype = = VideoDevice ) ? ctx - > video_device_number
: ctx - > audio_device_number ;
2011-09-02 06:33:24 +02:00
int r ;
2011-05-21 14:24:50 +02:00
const GUID * device_guid [ 2 ] = { & CLSID_VideoInputDeviceCategory ,
& CLSID_AudioInputDeviceCategory } ;
2015-01-23 14:54:44 +01:00
const char * devtypename = ( devtype = = VideoDevice ) ? " video " : " audio only " ;
const char * sourcetypename = ( sourcetype = = VideoSourceDevice ) ? " video " : " audio " ;
2011-05-21 14:24:50 +02:00
2015-01-23 14:54:44 +01:00
r = ICreateDevEnum_CreateClassEnumerator ( devenum , device_guid [ sourcetype ] ,
2011-05-21 14:24:50 +02:00
( IEnumMoniker * * ) & classenum , 0 ) ;
if ( r ! = S_OK ) {
2015-01-23 14:54:44 +01:00
av_log ( avctx , AV_LOG_ERROR , " Could not enumerate %s devices (or none found). \n " ,
2011-05-21 14:24:50 +02:00
devtypename ) ;
2011-09-09 05:08:43 +02:00
return AVERROR ( EIO ) ;
2011-05-21 14:24:50 +02:00
}
2011-09-09 05:16:17 +02:00
while ( ! device_filter & & IEnumMoniker_Next ( classenum , 1 , & m , NULL ) = = S_OK ) {
2011-05-21 14:24:50 +02:00
IPropertyBag * bag = NULL ;
2015-01-23 13:34:30 +01:00
char * friendly_name = NULL ;
char * unique_name = NULL ;
2011-05-21 14:24:50 +02:00
VARIANT var ;
2015-01-23 13:34:30 +01:00
IBindCtx * bind_ctx = NULL ;
LPOLESTR olestr = NULL ;
LPMALLOC co_malloc = NULL ;
int i ;
r = CoGetMalloc ( 1 , & co_malloc ) ;
2016-02-13 16:55:02 +01:00
if ( r ! = S_OK )
2015-01-23 13:34:30 +01:00
goto fail1 ;
r = CreateBindCtx ( 0 , & bind_ctx ) ;
if ( r ! = S_OK )
goto fail1 ;
/* GetDisplayname works for both video and audio, DevicePath doesn't */
r = IMoniker_GetDisplayName ( m , bind_ctx , NULL , & olestr ) ;
if ( r ! = S_OK )
goto fail1 ;
unique_name = dup_wchar_to_utf8 ( olestr ) ;
/* replace ':' with '_' since we use : to delineate between sources */
for ( i = 0 ; i < strlen ( unique_name ) ; i + + ) {
if ( unique_name [ i ] = = ' : ' )
unique_name [ i ] = ' _ ' ;
}
2011-05-21 14:24:50 +02:00
r = IMoniker_BindToStorage ( m , 0 , 0 , & IID_IPropertyBag , ( void * ) & bag ) ;
if ( r ! = S_OK )
goto fail1 ;
var . vt = VT_BSTR ;
r = IPropertyBag_Read ( bag , L " FriendlyName " , & var , NULL ) ;
if ( r ! = S_OK )
goto fail1 ;
2015-01-23 13:34:30 +01:00
friendly_name = dup_wchar_to_utf8 ( var . bstrVal ) ;
2011-05-21 14:24:50 +02:00
2015-01-23 14:54:44 +01:00
if ( pfilter ) {
2015-01-23 13:34:30 +01:00
if ( strcmp ( device_name , friendly_name ) & & strcmp ( device_name , unique_name ) )
2011-09-09 05:09:42 +02:00
goto fail1 ;
2011-05-21 14:24:50 +02:00
2015-01-23 13:34:30 +01:00
if ( ! skip - - ) {
r = IMoniker_BindToObject ( m , 0 , 0 , & IID_IBaseFilter , ( void * ) & device_filter ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Unable to BindToObject for %s \n " , device_name ) ;
goto fail1 ;
}
}
2011-09-09 05:09:23 +02:00
} else {
2015-01-23 13:34:30 +01:00
av_log ( avctx , AV_LOG_INFO , " \" %s \" \n " , friendly_name ) ;
av_log ( avctx , AV_LOG_INFO , " Alternative name \" %s \" \n " , unique_name ) ;
2011-09-09 05:09:23 +02:00
}
2011-05-21 14:24:50 +02:00
fail1 :
2015-01-23 13:34:30 +01:00
if ( olestr & & co_malloc )
IMalloc_Free ( co_malloc , olestr ) ;
if ( bind_ctx )
IBindCtx_Release ( bind_ctx ) ;
av_free ( friendly_name ) ;
av_free ( unique_name ) ;
2011-05-21 14:24:50 +02:00
if ( bag )
IPropertyBag_Release ( bag ) ;
IMoniker_Release ( m ) ;
}
2011-09-09 05:08:43 +02:00
IEnumMoniker_Release ( classenum ) ;
2011-09-09 05:09:23 +02:00
if ( pfilter ) {
2011-09-09 05:09:42 +02:00
if ( ! device_filter ) {
2015-01-23 14:54:44 +01:00
av_log ( avctx , AV_LOG_ERROR , " Could not find %s device with name [%s] among source devices of type %s. \n " ,
devtypename , device_name , sourcetypename ) ;
2011-09-09 05:09:42 +02:00
return AVERROR ( EIO ) ;
}
* pfilter = device_filter ;
2011-09-09 05:09:23 +02:00
}
2011-09-09 05:08:43 +02:00
return 0 ;
}
2011-09-09 05:12:42 +02:00
/**
* Cycle through available formats using the specified pin ,
* try to set parameters specified through AVOptions and if successful
* return 1 in * pformat_set .
2011-09-09 05:15:14 +02:00
* If pformat_set is NULL , list all pin capabilities .
2011-09-09 05:12:42 +02:00
*/
static void
dshow_cycle_formats ( AVFormatContext * avctx , enum dshowDeviceType devtype ,
IPin * pin , int * pformat_set )
{
struct dshow_ctx * ctx = avctx - > priv_data ;
IAMStreamConfig * config = NULL ;
AM_MEDIA_TYPE * type = NULL ;
int format_set = 0 ;
void * caps = NULL ;
2015-01-23 14:01:32 +01:00
int i , n , size , r ;
2011-09-09 05:12:42 +02:00
if ( IPin_QueryInterface ( pin , & IID_IAMStreamConfig , ( void * * ) & config ) ! = S_OK )
return ;
if ( IAMStreamConfig_GetNumberOfCapabilities ( config , & n , & size ) ! = S_OK )
goto end ;
caps = av_malloc ( size ) ;
if ( ! caps )
goto end ;
for ( i = 0 ; i < n & & ! format_set ; i + + ) {
2015-01-23 14:01:32 +01:00
r = IAMStreamConfig_GetStreamCaps ( config , i , & type , ( void * ) caps ) ;
if ( r ! = S_OK )
goto next ;
2011-09-09 05:12:42 +02:00
# if DSHOWDEBUG
ff_print_AM_MEDIA_TYPE ( type ) ;
# endif
if ( devtype = = VideoDevice ) {
VIDEO_STREAM_CONFIG_CAPS * vcaps = caps ;
BITMAPINFOHEADER * bih ;
int64_t * fr ;
2014-08-07 05:43:03 +02:00
const AVCodecTag * const tags [ ] = { avformat_get_riff_video_tags ( ) , NULL } ;
2011-09-09 05:12:42 +02:00
# if DSHOWDEBUG
ff_print_VIDEO_STREAM_CONFIG_CAPS ( vcaps ) ;
# endif
if ( IsEqualGUID ( & type - > formattype , & FORMAT_VideoInfo ) ) {
VIDEOINFOHEADER * v = ( void * ) type - > pbFormat ;
fr = & v - > AvgTimePerFrame ;
bih = & v - > bmiHeader ;
} else if ( IsEqualGUID ( & type - > formattype , & FORMAT_VideoInfo2 ) ) {
VIDEOINFOHEADER2 * v = ( void * ) type - > pbFormat ;
fr = & v - > AvgTimePerFrame ;
bih = & v - > bmiHeader ;
} else {
goto next ;
}
2011-09-09 05:15:14 +02:00
if ( ! pformat_set ) {
2012-10-08 20:54:00 +02:00
enum AVPixelFormat pix_fmt = dshow_pixfmt ( bih - > biCompression , bih - > biBitCount ) ;
if ( pix_fmt = = AV_PIX_FMT_NONE ) {
2014-08-07 05:43:03 +02:00
enum AVCodecID codec_id = av_codec_get_id ( tags , bih - > biCompression ) ;
2012-09-07 01:41:19 +02:00
AVCodec * codec = avcodec_find_decoder ( codec_id ) ;
if ( codec_id = = AV_CODEC_ID_NONE | | ! codec ) {
2012-09-07 23:52:32 +02:00
av_log ( avctx , AV_LOG_INFO , " unknown compression type 0x%X " , ( int ) bih - > biCompression ) ;
2012-09-07 01:41:19 +02:00
} else {
av_log ( avctx , AV_LOG_INFO , " vcodec=%s " , codec - > name ) ;
}
} else {
av_log ( avctx , AV_LOG_INFO , " pixel_format=%s " , av_get_pix_fmt_name ( pix_fmt ) ) ;
}
2011-09-09 05:15:14 +02:00
av_log ( avctx , AV_LOG_INFO , " min s=%ldx%ld fps=%g max s=%ldx%ld fps=%g \n " ,
vcaps - > MinOutputSize . cx , vcaps - > MinOutputSize . cy ,
2011-09-30 23:10:30 +02:00
1e7 / vcaps - > MaxFrameInterval ,
2011-09-09 05:15:14 +02:00
vcaps - > MaxOutputSize . cx , vcaps - > MaxOutputSize . cy ,
2011-09-30 23:10:30 +02:00
1e7 / vcaps - > MinFrameInterval ) ;
2011-09-09 05:15:14 +02:00
continue ;
}
2012-09-07 01:41:19 +02:00
if ( ctx - > video_codec_id ! = AV_CODEC_ID_RAWVIDEO ) {
2014-08-07 05:43:03 +02:00
if ( ctx - > video_codec_id ! = av_codec_get_id ( tags , bih - > biCompression ) )
2012-09-07 01:41:19 +02:00
goto next ;
}
2012-10-08 20:54:00 +02:00
if ( ctx - > pixel_format ! = AV_PIX_FMT_NONE & &
2012-09-07 01:41:19 +02:00
ctx - > pixel_format ! = dshow_pixfmt ( bih - > biCompression , bih - > biBitCount ) ) {
goto next ;
}
2011-09-09 05:12:42 +02:00
if ( ctx - > framerate ) {
int64_t framerate = ( ( int64_t ) ctx - > requested_framerate . den * 10000000 )
/ ctx - > requested_framerate . num ;
if ( framerate > vcaps - > MaxFrameInterval | |
framerate < vcaps - > MinFrameInterval )
goto next ;
* fr = framerate ;
}
2012-09-07 01:50:06 +02:00
if ( ctx - > requested_width & & ctx - > requested_height ) {
2011-09-09 05:12:42 +02:00
if ( ctx - > requested_width > vcaps - > MaxOutputSize . cx | |
ctx - > requested_width < vcaps - > MinOutputSize . cx | |
ctx - > requested_height > vcaps - > MaxOutputSize . cy | |
ctx - > requested_height < vcaps - > MinOutputSize . cy )
goto next ;
bih - > biWidth = ctx - > requested_width ;
bih - > biHeight = ctx - > requested_height ;
}
} else {
AUDIO_STREAM_CONFIG_CAPS * acaps = caps ;
WAVEFORMATEX * fx ;
# if DSHOWDEBUG
ff_print_AUDIO_STREAM_CONFIG_CAPS ( acaps ) ;
# endif
if ( IsEqualGUID ( & type - > formattype , & FORMAT_WaveFormatEx ) ) {
fx = ( void * ) type - > pbFormat ;
} else {
goto next ;
}
2011-09-09 05:15:14 +02:00
if ( ! pformat_set ) {
av_log ( avctx , AV_LOG_INFO , " min ch=%lu bits=%lu rate=%6lu max ch=%lu bits=%lu rate=%6lu \n " ,
acaps - > MinimumChannels , acaps - > MinimumBitsPerSample , acaps - > MinimumSampleFrequency ,
acaps - > MaximumChannels , acaps - > MaximumBitsPerSample , acaps - > MaximumSampleFrequency ) ;
continue ;
}
2011-09-09 05:12:42 +02:00
if ( ctx - > sample_rate ) {
if ( ctx - > sample_rate > acaps - > MaximumSampleFrequency | |
ctx - > sample_rate < acaps - > MinimumSampleFrequency )
goto next ;
fx - > nSamplesPerSec = ctx - > sample_rate ;
}
if ( ctx - > sample_size ) {
if ( ctx - > sample_size > acaps - > MaximumBitsPerSample | |
ctx - > sample_size < acaps - > MinimumBitsPerSample )
goto next ;
fx - > wBitsPerSample = ctx - > sample_size ;
}
if ( ctx - > channels ) {
if ( ctx - > channels > acaps - > MaximumChannels | |
ctx - > channels < acaps - > MinimumChannels )
goto next ;
fx - > nChannels = ctx - > channels ;
}
}
if ( IAMStreamConfig_SetFormat ( config , type ) ! = S_OK )
goto next ;
format_set = 1 ;
next :
if ( type - > pbFormat )
CoTaskMemFree ( type - > pbFormat ) ;
CoTaskMemFree ( type ) ;
}
end :
IAMStreamConfig_Release ( config ) ;
2014-12-28 18:17:12 +01:00
av_free ( caps ) ;
2011-09-09 05:15:14 +02:00
if ( pformat_set )
2011-09-09 05:15:22 +02:00
* pformat_set = format_set ;
2011-09-09 05:12:42 +02:00
}
2012-08-25 10:09:37 +02:00
/**
* Set audio device buffer size in milliseconds ( which can directly impact
* latency , depending on the device ) .
*/
static int
dshow_set_audio_buffer_size ( AVFormatContext * avctx , IPin * pin )
{
struct dshow_ctx * ctx = avctx - > priv_data ;
IAMBufferNegotiation * buffer_negotiation = NULL ;
ALLOCATOR_PROPERTIES props = { - 1 , - 1 , - 1 , - 1 } ;
IAMStreamConfig * config = NULL ;
AM_MEDIA_TYPE * type = NULL ;
int ret = AVERROR ( EIO ) ;
if ( IPin_QueryInterface ( pin , & IID_IAMStreamConfig , ( void * * ) & config ) ! = S_OK )
goto end ;
if ( IAMStreamConfig_GetFormat ( config , & type ) ! = S_OK )
goto end ;
if ( ! IsEqualGUID ( & type - > formattype , & FORMAT_WaveFormatEx ) )
goto end ;
props . cbBuffer = ( ( ( WAVEFORMATEX * ) type - > pbFormat ) - > nAvgBytesPerSec )
* ctx - > audio_buffer_size / 1000 ;
if ( IPin_QueryInterface ( pin , & IID_IAMBufferNegotiation , ( void * * ) & buffer_negotiation ) ! = S_OK )
goto end ;
if ( IAMBufferNegotiation_SuggestAllocatorProperties ( buffer_negotiation , & props ) ! = S_OK )
goto end ;
ret = 0 ;
end :
if ( buffer_negotiation )
IAMBufferNegotiation_Release ( buffer_negotiation ) ;
if ( type ) {
if ( type - > pbFormat )
CoTaskMemFree ( type - > pbFormat ) ;
CoTaskMemFree ( type ) ;
}
if ( config )
IAMStreamConfig_Release ( config ) ;
return ret ;
}
2015-01-23 14:49:37 +01:00
/**
* Pops up a user dialog allowing them to adjust properties for the given filter , if possible .
*/
void
dshow_show_filter_properties ( IBaseFilter * device_filter , AVFormatContext * avctx ) {
ISpecifyPropertyPages * property_pages = NULL ;
IUnknown * device_filter_iunknown = NULL ;
HRESULT hr ;
2015-01-23 14:54:44 +01:00
FILTER_INFO filter_info = { 0 } ; /* a warning on this line is false positive GCC bug 53119 AFAICT */
2015-01-23 14:49:37 +01:00
CAUUID ca_guid = { 0 } ;
hr = IBaseFilter_QueryInterface ( device_filter , & IID_ISpecifyPropertyPages , ( void * * ) & property_pages ) ;
if ( hr ! = S_OK ) {
av_log ( avctx , AV_LOG_WARNING , " requested filter does not have a property page to show " ) ;
goto end ;
}
hr = IBaseFilter_QueryFilterInfo ( device_filter , & filter_info ) ;
if ( hr ! = S_OK ) {
goto fail ;
}
hr = IBaseFilter_QueryInterface ( device_filter , & IID_IUnknown , ( void * * ) & device_filter_iunknown ) ;
if ( hr ! = S_OK ) {
goto fail ;
}
hr = ISpecifyPropertyPages_GetPages ( property_pages , & ca_guid ) ;
if ( hr ! = S_OK ) {
goto fail ;
}
hr = OleCreatePropertyFrame ( NULL , 0 , 0 , filter_info . achName , 1 , & device_filter_iunknown , ca_guid . cElems ,
ca_guid . pElems , 0 , 0 , NULL ) ;
if ( hr ! = S_OK ) {
goto fail ;
}
goto end ;
fail :
av_log ( avctx , AV_LOG_ERROR , " Failure showing property pages for filter " ) ;
end :
if ( property_pages )
ISpecifyPropertyPages_Release ( property_pages ) ;
if ( device_filter_iunknown )
IUnknown_Release ( device_filter_iunknown ) ;
if ( filter_info . pGraph )
IFilterGraph_Release ( filter_info . pGraph ) ;
if ( ca_guid . pElems )
CoTaskMemFree ( ca_guid . pElems ) ;
}
2011-09-09 05:10:07 +02:00
/**
* Cycle through available pins using the device_filter device , of type
* devtype , retrieve the first output pin and return the pointer to the
* object found in * ppin .
2011-09-09 05:15:14 +02:00
* If ppin is NULL , cycle through all pins listing audio / video capabilities .
2011-09-09 05:10:07 +02:00
*/
2011-09-09 05:08:43 +02:00
static int
2011-09-09 05:10:07 +02:00
dshow_cycle_pins ( AVFormatContext * avctx , enum dshowDeviceType devtype ,
2015-01-23 14:54:44 +01:00
enum dshowSourceFilterType sourcetype , IBaseFilter * device_filter , IPin * * ppin )
2011-09-09 05:08:43 +02:00
{
2011-09-09 05:12:42 +02:00
struct dshow_ctx * ctx = avctx - > priv_data ;
2011-09-09 05:08:43 +02:00
IEnumPins * pins = 0 ;
IPin * device_pin = NULL ;
IPin * pin ;
int r ;
const GUID * mediatype [ 2 ] = { & MEDIATYPE_Video , & MEDIATYPE_Audio } ;
2015-01-23 14:54:44 +01:00
const char * devtypename = ( devtype = = VideoDevice ) ? " video " : " audio only " ;
const char * sourcetypename = ( sourcetype = = VideoSourceDevice ) ? " video " : " audio " ;
2011-05-21 14:24:50 +02:00
2012-09-07 01:50:06 +02:00
int set_format = ( devtype = = VideoDevice & & ( ctx - > framerate | |
( ctx - > requested_width & & ctx - > requested_height ) | |
2012-10-08 20:54:00 +02:00
ctx - > pixel_format ! = AV_PIX_FMT_NONE | |
2012-09-07 01:41:19 +02:00
ctx - > video_codec_id ! = AV_CODEC_ID_RAWVIDEO ) )
2011-09-09 05:12:42 +02:00
| | ( devtype = = AudioDevice & & ( ctx - > channels | | ctx - > sample_rate ) ) ;
int format_set = 0 ;
2015-01-23 14:49:37 +01:00
int should_show_properties = ( devtype = = VideoDevice ) ? ctx - > show_video_device_dialog : ctx - > show_audio_device_dialog ;
if ( should_show_properties )
dshow_show_filter_properties ( device_filter , avctx ) ;
2011-09-09 05:12:42 +02:00
2011-05-21 14:24:50 +02:00
r = IBaseFilter_EnumPins ( device_filter , & pins ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not enumerate pins. \n " ) ;
2011-09-09 05:10:07 +02:00
return AVERROR ( EIO ) ;
2011-05-21 14:24:50 +02:00
}
2011-09-09 05:15:14 +02:00
if ( ! ppin ) {
2015-01-23 14:54:44 +01:00
av_log ( avctx , AV_LOG_INFO , " DirectShow %s device options (from %s devices) \n " ,
devtypename , sourcetypename ) ;
2011-09-09 05:15:14 +02:00
}
2015-01-23 14:54:44 +01:00
2011-09-09 05:16:17 +02:00
while ( ! device_pin & & IEnumPins_Next ( pins , 1 , & pin , NULL ) = = S_OK ) {
2011-05-21 14:24:50 +02:00
IKsPropertySet * p = NULL ;
2011-09-09 05:10:22 +02:00
IEnumMediaTypes * types = NULL ;
2011-05-21 14:24:50 +02:00
PIN_INFO info = { 0 } ;
AM_MEDIA_TYPE * type ;
GUID category ;
DWORD r2 ;
2015-01-23 13:34:30 +01:00
char * name_buf = NULL ;
wchar_t * pin_id = NULL ;
char * pin_buf = NULL ;
char * desired_pin_name = devtype = = VideoDevice ? ctx - > video_pin_name : ctx - > audio_pin_name ;
2011-05-21 14:24:50 +02:00
IPin_QueryPinInfo ( pin , & info ) ;
IBaseFilter_Release ( info . pFilter ) ;
if ( info . dir ! = PINDIR_OUTPUT )
goto next ;
if ( IPin_QueryInterface ( pin , & IID_IKsPropertySet , ( void * * ) & p ) ! = S_OK )
goto next ;
if ( IKsPropertySet_Get ( p , & AMPROPSETID_Pin , AMPROPERTY_PIN_CATEGORY ,
NULL , 0 , & category , sizeof ( GUID ) , & r2 ) ! = S_OK )
goto next ;
if ( ! IsEqualGUID ( & category , & PIN_CATEGORY_CAPTURE ) )
goto next ;
2015-01-23 13:34:30 +01:00
name_buf = dup_wchar_to_utf8 ( info . achName ) ;
r = IPin_QueryId ( pin , & pin_id ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not query pin id \n " ) ;
return AVERROR ( EIO ) ;
}
pin_buf = dup_wchar_to_utf8 ( pin_id ) ;
2011-09-09 05:15:14 +02:00
if ( ! ppin ) {
2015-01-23 13:34:30 +01:00
av_log ( avctx , AV_LOG_INFO , " Pin \" %s \" (alternative pin name \" %s \" ) \n " , name_buf , pin_buf ) ;
2011-09-09 05:15:14 +02:00
dshow_cycle_formats ( avctx , devtype , pin , NULL ) ;
goto next ;
}
2015-01-23 14:54:44 +01:00
2015-01-23 13:34:30 +01:00
if ( desired_pin_name ) {
if ( strcmp ( name_buf , desired_pin_name ) & & strcmp ( pin_buf , desired_pin_name ) ) {
av_log ( avctx , AV_LOG_DEBUG , " skipping pin \" %s \" ( \" %s \" ) != requested \" %s \" \n " ,
name_buf , pin_buf , desired_pin_name ) ;
goto next ;
}
}
2011-09-09 05:12:42 +02:00
if ( set_format ) {
dshow_cycle_formats ( avctx , devtype , pin , & format_set ) ;
if ( ! format_set ) {
goto next ;
}
}
2012-08-25 10:09:37 +02:00
if ( devtype = = AudioDevice & & ctx - > audio_buffer_size ) {
2014-03-25 17:08:49 +01:00
if ( dshow_set_audio_buffer_size ( avctx , pin ) < 0 ) {
av_log ( avctx , AV_LOG_ERROR , " unable to set audio buffer size %d to pin, using pin anyway... " , ctx - > audio_buffer_size ) ;
}
2012-08-25 10:09:37 +02:00
}
2011-09-09 05:12:42 +02:00
2011-05-21 14:24:50 +02:00
if ( IPin_EnumMediaTypes ( pin , & types ) ! = S_OK )
goto next ;
IEnumMediaTypes_Reset ( types ) ;
2015-01-23 14:01:32 +01:00
/* in case format_set was not called, just verify the majortype */
2011-09-09 05:16:17 +02:00
while ( ! device_pin & & IEnumMediaTypes_Next ( types , 1 , & type , NULL ) = = S_OK ) {
2011-05-21 14:24:50 +02:00
if ( IsEqualGUID ( & type - > majortype , mediatype [ devtype ] ) ) {
device_pin = pin ;
2015-01-23 13:34:30 +01:00
av_log ( avctx , AV_LOG_DEBUG , " Selecting pin %s on %s \n " , name_buf , devtypename ) ;
2011-05-21 14:24:50 +02:00
goto next ;
}
CoTaskMemFree ( type ) ;
}
next :
if ( types )
IEnumMediaTypes_Release ( types ) ;
if ( p )
IKsPropertySet_Release ( p ) ;
if ( device_pin ! = pin )
IPin_Release ( pin ) ;
2015-01-23 13:34:30 +01:00
av_free ( name_buf ) ;
av_free ( pin_buf ) ;
if ( pin_id )
CoTaskMemFree ( pin_id ) ;
2011-05-21 14:24:50 +02:00
}
2011-09-09 05:10:07 +02:00
IEnumPins_Release ( pins ) ;
2011-09-09 05:15:14 +02:00
if ( ppin ) {
2011-09-09 05:15:22 +02:00
if ( set_format & & ! format_set ) {
av_log ( avctx , AV_LOG_ERROR , " Could not set %s options \n " , devtypename ) ;
return AVERROR ( EIO ) ;
}
if ( ! device_pin ) {
av_log ( avctx , AV_LOG_ERROR ,
" Could not find output pin from %s capture device. \n " , devtypename ) ;
return AVERROR ( EIO ) ;
}
* ppin = device_pin ;
2011-09-09 05:15:14 +02:00
}
return 0 ;
}
/**
2015-01-23 14:54:44 +01:00
* List options for device with type devtype , source filter type sourcetype
2011-09-09 05:15:14 +02:00
*
* @ param devenum device enumerator used for accessing the device
*/
static int
dshow_list_device_options ( AVFormatContext * avctx , ICreateDevEnum * devenum ,
2015-01-23 14:54:44 +01:00
enum dshowDeviceType devtype , enum dshowSourceFilterType sourcetype )
2011-09-09 05:15:14 +02:00
{
2011-09-30 22:50:00 +02:00
struct dshow_ctx * ctx = avctx - > priv_data ;
2011-09-09 05:15:14 +02:00
IBaseFilter * device_filter = NULL ;
int r ;
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_cycle_devices ( avctx , devenum , devtype , sourcetype , & device_filter ) ) < 0 )
2011-09-09 05:15:14 +02:00
return r ;
2011-09-30 22:50:00 +02:00
ctx - > device_filter [ devtype ] = device_filter ;
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_cycle_pins ( avctx , devtype , sourcetype , device_filter , NULL ) ) < 0 )
2011-09-09 05:15:14 +02:00
return r ;
2011-09-09 05:10:07 +02:00
return 0 ;
}
static int
2015-01-23 14:54:44 +01:00
dshow_open_device ( AVFormatContext * avctx , ICreateDevEnum * devenum ,
enum dshowDeviceType devtype , enum dshowSourceFilterType sourcetype )
2011-09-09 05:10:07 +02:00
{
struct dshow_ctx * ctx = avctx - > priv_data ;
IBaseFilter * device_filter = NULL ;
IGraphBuilder * graph = ctx - > graph ;
IPin * device_pin = NULL ;
libAVPin * capture_pin = NULL ;
libAVFilter * capture_filter = NULL ;
2015-01-23 14:35:16 +01:00
ICaptureGraphBuilder2 * graph_builder2 = NULL ;
2011-09-09 05:10:07 +02:00
int ret = AVERROR ( EIO ) ;
int r ;
2015-04-22 18:38:38 +02:00
IStream * ifile_stream = NULL ;
IStream * ofile_stream = NULL ;
IPersistStream * pers_stream = NULL ;
2011-09-09 05:10:07 +02:00
const wchar_t * filter_name [ 2 ] = { L " Audio capture filter " , L " Video capture filter " } ;
2015-04-22 18:38:38 +02:00
if ( ( ( ctx - > audio_filter_load_file ) & & ( strlen ( ctx - > audio_filter_load_file ) > 0 ) & & ( sourcetype = = AudioSourceDevice ) ) | |
( ( ctx - > video_filter_load_file ) & & ( strlen ( ctx - > video_filter_load_file ) > 0 ) & & ( sourcetype = = VideoSourceDevice ) ) ) {
HRESULT hr ;
char * filename = NULL ;
if ( sourcetype = = AudioSourceDevice )
filename = ctx - > audio_filter_load_file ;
else
filename = ctx - > video_filter_load_file ;
hr = SHCreateStreamOnFile ( ( LPCSTR ) filename , STGM_READ , & ifile_stream ) ;
if ( S_OK ! = hr ) {
av_log ( avctx , AV_LOG_ERROR , " Could not open capture filter description file. \n " ) ;
goto error ;
}
hr = OleLoadFromStream ( ifile_stream , & IID_IBaseFilter , ( void * * ) & device_filter ) ;
if ( hr ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not load capture filter from file. \n " ) ;
goto error ;
}
if ( sourcetype = = AudioSourceDevice )
av_log ( avctx , AV_LOG_INFO , " Audio- " ) ;
else
av_log ( avctx , AV_LOG_INFO , " Video- " ) ;
av_log ( avctx , AV_LOG_INFO , " Capture filter loaded successfully from file \" %s \" . \n " , filename ) ;
} else {
if ( ( r = dshow_cycle_devices ( avctx , devenum , devtype , sourcetype , & device_filter ) ) < 0 ) {
ret = r ;
goto error ;
}
2011-09-09 05:10:07 +02:00
}
ctx - > device_filter [ devtype ] = device_filter ;
r = IGraphBuilder_AddFilter ( graph , device_filter , NULL ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not add device filter to graph. \n " ) ;
goto error ;
}
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_cycle_pins ( avctx , devtype , sourcetype , device_filter , & device_pin ) ) < 0 ) {
2011-09-09 05:10:07 +02:00
ret = r ;
2011-05-21 14:24:50 +02:00
goto error ;
}
2015-01-23 14:35:16 +01:00
2011-05-21 14:24:50 +02:00
ctx - > device_pin [ devtype ] = device_pin ;
capture_filter = libAVFilter_Create ( avctx , callback , devtype ) ;
if ( ! capture_filter ) {
av_log ( avctx , AV_LOG_ERROR , " Could not create grabber filter. \n " ) ;
goto error ;
}
ctx - > capture_filter [ devtype ] = capture_filter ;
2015-04-22 18:38:38 +02:00
if ( ( ( ctx - > audio_filter_save_file ) & & ( strlen ( ctx - > audio_filter_save_file ) > 0 ) & & ( sourcetype = = AudioSourceDevice ) ) | |
( ( ctx - > video_filter_save_file ) & & ( strlen ( ctx - > video_filter_save_file ) > 0 ) & & ( sourcetype = = VideoSourceDevice ) ) ) {
HRESULT hr ;
char * filename = NULL ;
if ( sourcetype = = AudioSourceDevice )
filename = ctx - > audio_filter_save_file ;
else
filename = ctx - > video_filter_save_file ;
hr = SHCreateStreamOnFile ( ( LPCSTR ) filename , STGM_CREATE | STGM_READWRITE , & ofile_stream ) ;
if ( S_OK ! = hr ) {
av_log ( avctx , AV_LOG_ERROR , " Could not create capture filter description file. \n " ) ;
goto error ;
}
hr = IBaseFilter_QueryInterface ( device_filter , & IID_IPersistStream , ( void * * ) & pers_stream ) ;
if ( hr ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Query for IPersistStream failed. \n " ) ;
goto error ;
}
hr = OleSaveToStream ( pers_stream , ofile_stream ) ;
if ( hr ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not save capture filter \n " ) ;
goto error ;
}
hr = IStream_Commit ( ofile_stream , STGC_DEFAULT ) ;
if ( S_OK ! = hr ) {
av_log ( avctx , AV_LOG_ERROR , " Could not commit capture filter data to file. \n " ) ;
goto error ;
}
if ( sourcetype = = AudioSourceDevice )
av_log ( avctx , AV_LOG_INFO , " Audio- " ) ;
else
av_log ( avctx , AV_LOG_INFO , " Video- " ) ;
av_log ( avctx , AV_LOG_INFO , " Capture filter saved successfully to file \" %s \" . \n " , filename ) ;
}
2011-05-21 14:24:50 +02:00
r = IGraphBuilder_AddFilter ( graph , ( IBaseFilter * ) capture_filter ,
filter_name [ devtype ] ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not add capture filter to graph \n " ) ;
goto error ;
}
libAVPin_AddRef ( capture_filter - > pin ) ;
capture_pin = capture_filter - > pin ;
ctx - > capture_pin [ devtype ] = capture_pin ;
2015-01-23 14:35:16 +01:00
r = CoCreateInstance ( & CLSID_CaptureGraphBuilder2 , NULL , CLSCTX_INPROC_SERVER ,
& IID_ICaptureGraphBuilder2 , ( void * * ) & graph_builder2 ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not create CaptureGraphBuilder2 \n " ) ;
goto error ;
}
ICaptureGraphBuilder2_SetFiltergraph ( graph_builder2 , graph ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not set graph for CaptureGraphBuilder2 \n " ) ;
goto error ;
}
r = ICaptureGraphBuilder2_RenderStream ( graph_builder2 , NULL , NULL , ( IUnknown * ) device_pin , NULL /* no intermediate filter */ ,
( IBaseFilter * ) capture_filter ) ; /* connect pins, optionally insert intermediate filters like crossbar if necessary */
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not RenderStream to connect pins \n " ) ;
goto error ;
}
r = dshow_try_setup_crossbar_options ( graph_builder2 , device_filter , devtype , avctx ) ;
2011-05-21 14:24:50 +02:00
if ( r ! = S_OK ) {
2015-01-23 14:35:16 +01:00
av_log ( avctx , AV_LOG_ERROR , " Could not setup CrossBar \n " ) ;
2011-05-21 14:24:50 +02:00
goto error ;
}
ret = 0 ;
error :
2015-01-23 14:35:16 +01:00
if ( graph_builder2 ! = NULL )
ICaptureGraphBuilder2_Release ( graph_builder2 ) ;
2015-04-22 18:38:38 +02:00
if ( pers_stream )
IPersistStream_Release ( pers_stream ) ;
if ( ifile_stream )
IStream_Release ( ifile_stream ) ;
if ( ofile_stream )
IStream_Release ( ofile_stream ) ;
2011-05-21 14:24:50 +02:00
return ret ;
}
2012-08-07 23:57:21 +02:00
static enum AVCodecID waveform_codec_id ( enum AVSampleFormat sample_fmt )
2011-05-21 14:24:50 +02:00
{
switch ( sample_fmt ) {
2012-08-07 22:45:46 +02:00
case AV_SAMPLE_FMT_U8 : return AV_CODEC_ID_PCM_U8 ;
case AV_SAMPLE_FMT_S16 : return AV_CODEC_ID_PCM_S16LE ;
case AV_SAMPLE_FMT_S32 : return AV_CODEC_ID_PCM_S32LE ;
default : return AV_CODEC_ID_NONE ; /* Should never happen. */
2011-05-21 14:24:50 +02:00
}
}
2012-01-28 04:23:26 +01:00
static enum AVSampleFormat sample_fmt_bits_per_sample ( int bits )
2011-05-21 14:24:50 +02:00
{
switch ( bits ) {
case 8 : return AV_SAMPLE_FMT_U8 ;
case 16 : return AV_SAMPLE_FMT_S16 ;
case 32 : return AV_SAMPLE_FMT_S32 ;
default : return AV_SAMPLE_FMT_NONE ; /* Should never happen. */
}
}
static int
2012-01-28 04:23:26 +01:00
dshow_add_device ( AVFormatContext * avctx ,
2011-05-21 14:24:50 +02:00
enum dshowDeviceType devtype )
{
struct dshow_ctx * ctx = avctx - > priv_data ;
AM_MEDIA_TYPE type ;
AVCodecContext * codec ;
AVStream * st ;
int ret = AVERROR ( EIO ) ;
2011-11-05 13:11:18 +01:00
st = avformat_new_stream ( avctx , NULL ) ;
2011-05-21 14:24:50 +02:00
if ( ! st ) {
ret = AVERROR ( ENOMEM ) ;
goto error ;
}
2011-11-05 13:11:18 +01:00
st - > id = devtype ;
2011-05-21 14:24:50 +02:00
ctx - > capture_filter [ devtype ] - > stream_index = st - > index ;
libAVPin_ConnectionMediaType ( ctx - > capture_pin [ devtype ] , & type ) ;
codec = st - > codec ;
if ( devtype = = VideoDevice ) {
BITMAPINFOHEADER * bih = NULL ;
2012-02-03 17:55:20 +01:00
AVRational time_base ;
2011-05-21 14:24:50 +02:00
if ( IsEqualGUID ( & type . formattype , & FORMAT_VideoInfo ) ) {
VIDEOINFOHEADER * v = ( void * ) type . pbFormat ;
2012-02-03 17:55:20 +01:00
time_base = ( AVRational ) { v - > AvgTimePerFrame , 10000000 } ;
2011-05-21 14:24:50 +02:00
bih = & v - > bmiHeader ;
} else if ( IsEqualGUID ( & type . formattype , & FORMAT_VideoInfo2 ) ) {
VIDEOINFOHEADER2 * v = ( void * ) type . pbFormat ;
2012-02-03 17:55:20 +01:00
time_base = ( AVRational ) { v - > AvgTimePerFrame , 10000000 } ;
2011-05-21 14:24:50 +02:00
bih = & v - > bmiHeader ;
}
if ( ! bih ) {
av_log ( avctx , AV_LOG_ERROR , " Could not get media type. \n " ) ;
goto error ;
}
2012-02-03 17:55:20 +01:00
codec - > time_base = time_base ;
2011-05-21 14:24:50 +02:00
codec - > codec_type = AVMEDIA_TYPE_VIDEO ;
codec - > width = bih - > biWidth ;
codec - > height = bih - > biHeight ;
2014-03-08 13:42:47 +01:00
codec - > codec_tag = bih - > biCompression ;
2011-05-21 14:24:50 +02:00
codec - > pix_fmt = dshow_pixfmt ( bih - > biCompression , bih - > biBitCount ) ;
2013-03-19 21:25:49 +01:00
if ( bih - > biCompression = = MKTAG ( ' H ' , ' D ' , ' Y ' , ' C ' ) ) {
av_log ( avctx , AV_LOG_DEBUG , " attempt to use full range for HDYC... \n " ) ;
codec - > color_range = AVCOL_RANGE_MPEG ; // just in case it needs this...
2013-01-16 02:37:30 +01:00
}
2012-10-08 20:54:00 +02:00
if ( codec - > pix_fmt = = AV_PIX_FMT_NONE ) {
2014-08-07 05:43:03 +02:00
const AVCodecTag * const tags [ ] = { avformat_get_riff_video_tags ( ) , NULL } ;
codec - > codec_id = av_codec_get_id ( tags , bih - > biCompression ) ;
2012-08-07 22:45:46 +02:00
if ( codec - > codec_id = = AV_CODEC_ID_NONE ) {
2011-05-21 14:24:50 +02:00
av_log ( avctx , AV_LOG_ERROR , " Unknown compression type. "
2013-01-15 23:08:23 +01:00
" Please report type 0x%X. \n " , ( int ) bih - > biCompression ) ;
2011-05-21 14:24:50 +02:00
return AVERROR_PATCHWELCOME ;
}
codec - > bits_per_coded_sample = bih - > biBitCount ;
} else {
2012-08-07 22:45:46 +02:00
codec - > codec_id = AV_CODEC_ID_RAWVIDEO ;
2011-10-08 20:00:00 +02:00
if ( bih - > biCompression = = BI_RGB | | bih - > biCompression = = BI_BITFIELDS ) {
2011-05-21 14:24:50 +02:00
codec - > bits_per_coded_sample = bih - > biBitCount ;
2015-07-27 22:53:16 +02:00
codec - > extradata = av_malloc ( 9 + AV_INPUT_BUFFER_PADDING_SIZE ) ;
2011-05-21 14:24:50 +02:00
if ( codec - > extradata ) {
codec - > extradata_size = 9 ;
memcpy ( codec - > extradata , " BottomUp " , 9 ) ;
}
}
}
} else {
WAVEFORMATEX * fx = NULL ;
if ( IsEqualGUID ( & type . formattype , & FORMAT_WaveFormatEx ) ) {
fx = ( void * ) type . pbFormat ;
}
if ( ! fx ) {
av_log ( avctx , AV_LOG_ERROR , " Could not get media type. \n " ) ;
goto error ;
}
2011-05-21 21:53:15 +02:00
codec - > codec_type = AVMEDIA_TYPE_AUDIO ;
2011-05-21 14:24:50 +02:00
codec - > sample_fmt = sample_fmt_bits_per_sample ( fx - > wBitsPerSample ) ;
codec - > codec_id = waveform_codec_id ( codec - > sample_fmt ) ;
codec - > sample_rate = fx - > nSamplesPerSec ;
codec - > channels = fx - > nChannels ;
}
2011-12-03 00:45:46 +01:00
avpriv_set_pts_info ( st , 64 , 1 , 10000000 ) ;
2011-05-21 14:24:50 +02:00
ret = 0 ;
error :
return ret ;
}
static int parse_device_name ( AVFormatContext * avctx )
{
struct dshow_ctx * ctx = avctx - > priv_data ;
char * * device_name = ctx - > device_name ;
char * name = av_strdup ( avctx - > filename ) ;
char * tmp = name ;
int ret = 1 ;
char * type ;
while ( ( type = strtok ( tmp , " = " ) ) ) {
char * token = strtok ( NULL , " : " ) ;
tmp = NULL ;
if ( ! strcmp ( type , " video " ) ) {
device_name [ 0 ] = token ;
} else if ( ! strcmp ( type , " audio " ) ) {
device_name [ 1 ] = token ;
} else {
device_name [ 0 ] = NULL ;
device_name [ 1 ] = NULL ;
break ;
}
}
if ( ! device_name [ 0 ] & & ! device_name [ 1 ] ) {
ret = 0 ;
} else {
if ( device_name [ 0 ] )
device_name [ 0 ] = av_strdup ( device_name [ 0 ] ) ;
if ( device_name [ 1 ] )
device_name [ 1 ] = av_strdup ( device_name [ 1 ] ) ;
}
av_free ( name ) ;
return ret ;
}
2012-01-28 04:23:26 +01:00
static int dshow_read_header ( AVFormatContext * avctx )
2011-05-21 14:24:50 +02:00
{
struct dshow_ctx * ctx = avctx - > priv_data ;
IGraphBuilder * graph = NULL ;
ICreateDevEnum * devenum = NULL ;
IMediaControl * control = NULL ;
2012-12-17 06:44:12 +01:00
IMediaEvent * media_event = NULL ;
HANDLE media_event_handle ;
HANDLE proc ;
2011-05-21 14:24:50 +02:00
int ret = AVERROR ( EIO ) ;
int r ;
2012-12-17 06:44:51 +01:00
CoInitialize ( 0 ) ;
2011-09-09 05:09:23 +02:00
if ( ! ctx - > list_devices & & ! parse_device_name ( avctx ) ) {
2011-05-21 14:24:50 +02:00
av_log ( avctx , AV_LOG_ERROR , " Malformed dshow input string. \n " ) ;
goto error ;
}
2012-09-07 01:41:19 +02:00
ctx - > video_codec_id = avctx - > video_codec_id ? avctx - > video_codec_id
: AV_CODEC_ID_RAWVIDEO ;
2012-10-08 20:54:00 +02:00
if ( ctx - > pixel_format ! = AV_PIX_FMT_NONE ) {
2012-09-07 01:41:19 +02:00
if ( ctx - > video_codec_id ! = AV_CODEC_ID_RAWVIDEO ) {
av_log ( avctx , AV_LOG_ERROR , " Pixel format may only be set when "
" video codec is not set or set to rawvideo \n " ) ;
ret = AVERROR ( EINVAL ) ;
goto error ;
}
}
2011-09-09 05:12:42 +02:00
if ( ctx - > framerate ) {
r = av_parse_video_rate ( & ctx - > requested_framerate , ctx - > framerate ) ;
if ( r < 0 ) {
av_log ( avctx , AV_LOG_ERROR , " Could not parse framerate '%s'. \n " , ctx - > framerate ) ;
goto error ;
}
}
2011-05-21 14:24:50 +02:00
r = CoCreateInstance ( & CLSID_FilterGraph , NULL , CLSCTX_INPROC_SERVER ,
& IID_IGraphBuilder , ( void * * ) & graph ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not create capture graph. \n " ) ;
goto error ;
}
ctx - > graph = graph ;
r = CoCreateInstance ( & CLSID_SystemDeviceEnum , NULL , CLSCTX_INPROC_SERVER ,
& IID_ICreateDevEnum , ( void * * ) & devenum ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not enumerate system devices. \n " ) ;
goto error ;
}
2011-09-09 05:09:23 +02:00
if ( ctx - > list_devices ) {
2015-01-23 14:54:44 +01:00
av_log ( avctx , AV_LOG_INFO , " DirectShow video devices (some may be both video and audio devices) \n " ) ;
dshow_cycle_devices ( avctx , devenum , VideoDevice , VideoSourceDevice , NULL ) ;
2011-09-09 05:09:23 +02:00
av_log ( avctx , AV_LOG_INFO , " DirectShow audio devices \n " ) ;
2015-01-23 14:54:44 +01:00
dshow_cycle_devices ( avctx , devenum , AudioDevice , AudioSourceDevice , NULL ) ;
2011-09-09 05:09:23 +02:00
ret = AVERROR_EXIT ;
goto error ;
}
2011-09-09 05:15:14 +02:00
if ( ctx - > list_options ) {
if ( ctx - > device_name [ VideoDevice ] )
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_list_device_options ( avctx , devenum , VideoDevice , VideoSourceDevice ) ) ) {
2015-01-23 14:35:16 +01:00
ret = r ;
goto error ;
}
2015-01-23 14:54:44 +01:00
if ( ctx - > device_name [ AudioDevice ] ) {
if ( dshow_list_device_options ( avctx , devenum , AudioDevice , AudioSourceDevice ) ) {
/* show audio options from combined video+audio sources as fallback */
if ( ( r = dshow_list_device_options ( avctx , devenum , AudioDevice , VideoSourceDevice ) ) ) {
ret = r ;
goto error ;
}
2015-01-23 14:35:16 +01:00
}
2015-01-23 14:54:44 +01:00
}
2011-09-09 05:15:14 +02:00
}
2011-05-21 14:24:50 +02:00
if ( ctx - > device_name [ VideoDevice ] ) {
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_open_device ( avctx , devenum , VideoDevice , VideoSourceDevice ) ) < 0 | |
2012-12-06 05:32:47 +01:00
( r = dshow_add_device ( avctx , VideoDevice ) ) < 0 ) {
ret = r ;
2011-05-21 14:24:50 +02:00
goto error ;
2012-12-06 05:32:47 +01:00
}
2011-05-21 14:24:50 +02:00
}
if ( ctx - > device_name [ AudioDevice ] ) {
2015-01-23 14:54:44 +01:00
if ( ( r = dshow_open_device ( avctx , devenum , AudioDevice , AudioSourceDevice ) ) < 0 | |
2012-12-06 05:32:47 +01:00
( r = dshow_add_device ( avctx , AudioDevice ) ) < 0 ) {
2015-01-26 10:03:48 +01:00
av_log ( avctx , AV_LOG_INFO , " Searching for audio device within video devices for %s \n " , ctx - > device_name [ AudioDevice ] ) ;
2015-01-23 14:54:44 +01:00
/* see if there's a video source with an audio pin with the given audio name */
if ( ( r = dshow_open_device ( avctx , devenum , AudioDevice , VideoSourceDevice ) ) < 0 | |
( r = dshow_add_device ( avctx , AudioDevice ) ) < 0 ) {
ret = r ;
goto error ;
}
2012-12-06 05:32:47 +01:00
}
2011-05-21 14:24:50 +02:00
}
2015-01-23 14:35:16 +01:00
if ( ctx - > list_options ) {
/* allow it to list crossbar options in dshow_open_device */
ret = AVERROR_EXIT ;
goto error ;
}
2014-03-21 23:27:01 +01:00
ctx - > curbufsize [ 0 ] = 0 ;
ctx - > curbufsize [ 1 ] = 0 ;
2011-05-21 14:24:50 +02:00
ctx - > mutex = CreateMutex ( NULL , 0 , NULL ) ;
if ( ! ctx - > mutex ) {
av_log ( avctx , AV_LOG_ERROR , " Could not create Mutex \n " ) ;
goto error ;
}
2012-12-17 06:44:12 +01:00
ctx - > event [ 1 ] = CreateEvent ( NULL , 1 , 0 , NULL ) ;
if ( ! ctx - > event [ 1 ] ) {
2011-05-21 14:24:50 +02:00
av_log ( avctx , AV_LOG_ERROR , " Could not create Event \n " ) ;
goto error ;
}
r = IGraphBuilder_QueryInterface ( graph , & IID_IMediaControl , ( void * * ) & control ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not get media control. \n " ) ;
goto error ;
}
ctx - > control = control ;
2012-12-17 06:44:12 +01:00
r = IGraphBuilder_QueryInterface ( graph , & IID_IMediaEvent , ( void * * ) & media_event ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not get media event. \n " ) ;
goto error ;
}
ctx - > media_event = media_event ;
r = IMediaEvent_GetEventHandle ( media_event , ( void * ) & media_event_handle ) ;
if ( r ! = S_OK ) {
av_log ( avctx , AV_LOG_ERROR , " Could not get media event handle. \n " ) ;
goto error ;
}
proc = GetCurrentProcess ( ) ;
r = DuplicateHandle ( proc , media_event_handle , proc , & ctx - > event [ 0 ] ,
0 , 0 , DUPLICATE_SAME_ACCESS ) ;
if ( ! r ) {
av_log ( avctx , AV_LOG_ERROR , " Could not duplicate media event handle. \n " ) ;
goto error ;
}
2011-05-21 14:24:50 +02:00
r = IMediaControl_Run ( control ) ;
if ( r = = S_FALSE ) {
OAFilterState pfs ;
r = IMediaControl_GetState ( control , 0 , & pfs ) ;
}
if ( r ! = S_OK ) {
2015-01-26 10:03:48 +01:00
av_log ( avctx , AV_LOG_ERROR , " Could not run graph (sometimes caused by a device already in use by other application) \n " ) ;
2011-05-21 14:24:50 +02:00
goto error ;
}
ret = 0 ;
error :
if ( devenum )
ICreateDevEnum_Release ( devenum ) ;
2012-12-17 06:44:51 +01:00
if ( ret < 0 )
dshow_read_close ( avctx ) ;
2011-05-21 14:24:50 +02:00
return ret ;
}
2012-12-17 06:44:12 +01:00
/**
* Checks media events from DirectShow and returns - 1 on error or EOF . Also
* purges all events that might be in the event queue to stop the trigger
* of event notification .
*/
static int dshow_check_event_queue ( IMediaEvent * media_event )
{
LONG_PTR p1 , p2 ;
long code ;
int ret = 0 ;
while ( IMediaEvent_GetEvent ( media_event , & code , & p1 , & p2 , 0 ) ! = E_ABORT ) {
if ( code = = EC_COMPLETE | | code = = EC_DEVICE_LOST | | code = = EC_ERRORABORT )
ret = - 1 ;
IMediaEvent_FreeEventParams ( media_event , code , p1 , p2 ) ;
}
return ret ;
}
2011-05-21 14:24:50 +02:00
static int dshow_read_packet ( AVFormatContext * s , AVPacket * pkt )
{
struct dshow_ctx * ctx = s - > priv_data ;
AVPacketList * pktl = NULL ;
2012-12-17 06:44:12 +01:00
while ( ! ctx - > eof & & ! pktl ) {
2011-05-21 14:24:50 +02:00
WaitForSingleObject ( ctx - > mutex , INFINITE ) ;
pktl = ctx - > pktl ;
2012-08-16 18:42:46 +02:00
if ( pktl ) {
* pkt = pktl - > pkt ;
2011-05-21 14:24:50 +02:00
ctx - > pktl = ctx - > pktl - > next ;
av_free ( pktl ) ;
2014-03-21 23:27:01 +01:00
ctx - > curbufsize [ pkt - > stream_index ] - = pkt - > size ;
2011-05-21 14:24:50 +02:00
}
2012-12-17 06:44:12 +01:00
ResetEvent ( ctx - > event [ 1 ] ) ;
2011-05-21 14:24:50 +02:00
ReleaseMutex ( ctx - > mutex ) ;
if ( ! pktl ) {
2012-12-17 06:44:12 +01:00
if ( dshow_check_event_queue ( ctx - > media_event ) < 0 ) {
ctx - > eof = 1 ;
} else if ( s - > flags & AVFMT_FLAG_NONBLOCK ) {
2011-05-21 14:24:50 +02:00
return AVERROR ( EAGAIN ) ;
} else {
2012-12-17 06:44:12 +01:00
WaitForMultipleObjects ( 2 , ctx - > event , 0 , INFINITE ) ;
2011-05-21 14:24:50 +02:00
}
}
}
2012-12-17 06:44:12 +01:00
return ctx - > eof ? AVERROR ( EIO ) : pkt - > size ;
2011-05-21 14:24:50 +02:00
}
2011-09-09 05:09:23 +02:00
# define OFFSET(x) offsetof(struct dshow_ctx, x)
# define DEC AV_OPT_FLAG_DECODING_PARAM
static const AVOption options [ ] = {
2012-09-07 01:50:06 +02:00
{ " video_size " , " set video size given a string such as 640x480 or hd720. " , OFFSET ( requested_width ) , AV_OPT_TYPE_IMAGE_SIZE , { . str = NULL } , 0 , 0 , DEC } ,
2013-12-16 16:11:49 +01:00
{ " pixel_format " , " set video pixel format " , OFFSET ( pixel_format ) , AV_OPT_TYPE_PIXEL_FMT , { . i64 = AV_PIX_FMT_NONE } , - 1 , INT_MAX , DEC } ,
2011-10-17 07:33:10 +02:00
{ " framerate " , " set video frame rate " , OFFSET ( framerate ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , DEC } ,
2012-09-05 14:26:01 +02:00
{ " sample_rate " , " set audio sample rate " , OFFSET ( sample_rate ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , DEC } ,
{ " sample_size " , " set audio sample size " , OFFSET ( sample_size ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 16 , DEC } ,
{ " channels " , " set number of audio channels, such as 1 or 2 " , OFFSET ( channels ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , DEC } ,
2015-01-23 13:34:30 +01:00
{ " audio_buffer_size " , " set audio device buffer latency size in milliseconds (default is the device's default) " , OFFSET ( audio_buffer_size ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , DEC } ,
2015-09-06 19:23:08 +02:00
{ " list_devices " , " list available devices " , OFFSET ( list_devices ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " list_options " , " list available options for specified device " , OFFSET ( list_options ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
2012-09-05 14:26:01 +02:00
{ " video_device_number " , " set video device number for devices with same name (starts at 0) " , OFFSET ( video_device_number ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , DEC } ,
{ " audio_device_number " , " set audio device number for devices with same name (starts at 0) " , OFFSET ( audio_device_number ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , DEC } ,
2015-01-23 13:34:30 +01:00
{ " video_pin_name " , " select video capture pin by name " , OFFSET ( video_pin_name ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , AV_OPT_FLAG_ENCODING_PARAM } ,
{ " audio_pin_name " , " select audio capture pin by name " , OFFSET ( audio_pin_name ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , AV_OPT_FLAG_ENCODING_PARAM } ,
2015-01-23 14:35:16 +01:00
{ " crossbar_video_input_pin_number " , " set video input pin number for crossbar device " , OFFSET ( crossbar_video_input_pin_number ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , DEC } ,
{ " crossbar_audio_input_pin_number " , " set audio input pin number for crossbar device " , OFFSET ( crossbar_audio_input_pin_number ) , AV_OPT_TYPE_INT , { . i64 = - 1 } , - 1 , INT_MAX , DEC } ,
2015-09-06 19:23:08 +02:00
{ " show_video_device_dialog " , " display property dialog for video capture device " , OFFSET ( show_video_device_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " show_audio_device_dialog " , " display property dialog for audio capture device " , OFFSET ( show_audio_device_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " show_video_crossbar_connection_dialog " , " display property dialog for crossbar connecting pins filter on video device " , OFFSET ( show_video_crossbar_connection_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " show_audio_crossbar_connection_dialog " , " display property dialog for crossbar connecting pins filter on audio device " , OFFSET ( show_audio_crossbar_connection_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " show_analog_tv_tuner_dialog " , " display property dialog for analog tuner filter " , OFFSET ( show_analog_tv_tuner_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
{ " show_analog_tv_tuner_audio_dialog " , " display property dialog for analog tuner audio filter " , OFFSET ( show_analog_tv_tuner_audio_dialog ) , AV_OPT_TYPE_BOOL , { . i64 = 0 } , 0 , 1 , DEC } ,
2015-04-22 18:38:38 +02:00
{ " audio_device_load " , " load audio capture filter device (and properties) from file " , OFFSET ( audio_filter_load_file ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , DEC } ,
{ " audio_device_save " , " save audio capture filter device (and properties) to file " , OFFSET ( audio_filter_save_file ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , DEC } ,
{ " video_device_load " , " load video capture filter device (and properties) from file " , OFFSET ( video_filter_load_file ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , DEC } ,
{ " video_device_save " , " save video capture filter device (and properties) to file " , OFFSET ( video_filter_save_file ) , AV_OPT_TYPE_STRING , { . str = NULL } , 0 , 0 , DEC } ,
2011-09-09 05:09:23 +02:00
{ NULL } ,
} ;
static const AVClass dshow_class = {
2012-12-13 21:33:59 +01:00
. class_name = " dshow indev " ,
2011-09-09 05:09:23 +02:00
. item_name = av_default_item_name ,
. option = options ,
. version = LIBAVUTIL_VERSION_INT ,
2014-02-22 23:32:51 +01:00
. category = AV_CLASS_CATEGORY_DEVICE_VIDEO_INPUT ,
2011-09-09 05:09:23 +02:00
} ;
2011-05-21 21:53:15 +02:00
AVInputFormat ff_dshow_demuxer = {
2012-02-03 17:50:19 +01:00
. name = " dshow " ,
. long_name = NULL_IF_CONFIG_SMALL ( " DirectShow capture " ) ,
. priv_data_size = sizeof ( struct dshow_ctx ) ,
. read_header = dshow_read_header ,
. read_packet = dshow_read_packet ,
. read_close = dshow_read_close ,
. flags = AVFMT_NOFILE ,
. priv_class = & dshow_class ,
2011-05-21 14:24:50 +02:00
} ;