dshow.cpp 88.9 KB
Newer Older
Gildas Bazin's avatar
 
Gildas Bazin committed
1
/*****************************************************************************
2
 * dshow.cpp : DirectShow access and access_demux module for vlc
Gildas Bazin's avatar
 
Gildas Bazin committed
3
 *****************************************************************************
4
 * Copyright (C) 2002-2004, 2006, 2008, 2010 the VideoLAN team
5
 * $Id$
Gildas Bazin's avatar
 
Gildas Bazin committed
6
 *
7
 * Author: Gildas Bazin <gbazin@videolan.org>
8
 *         Damien Fouilleul <damienf@videolan.org>
Gildas Bazin's avatar
 
Gildas Bazin committed
9 10 11 12 13 14 15 16 17 18 19 20 21
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
Antoine Cellerier's avatar
Antoine Cellerier committed
22
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
Gildas Bazin's avatar
 
Gildas Bazin committed
23 24 25 26 27
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
28

29 30 31 32
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

33
#define __STDC_CONSTANT_MACROS 1
34
#define __STDC_FORMAT_MACROS 1
35
#define CFG_PREFIX "dshow-"
36 37
#include <inttypes.h>

38
#include <vlc_common.h>
39
#include <vlc_plugin.h>
Clément Stenac's avatar
Clément Stenac committed
40
#include <vlc_access.h>
41
#include <vlc_demux.h>
42 43 44

#include <vlc_dialog.h>      /* dialog_Fatal */
#include <vlc_charset.h>     /* FromWide */
Gildas Bazin's avatar
 
Gildas Bazin committed
45

Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
46 47 48
#include <initguid.h>
#include "vlc_dshow.h"

49
#include "access.h"
Gildas Bazin's avatar
 
Gildas Bazin committed
50 51
#include "filter.h"

Gildas Bazin's avatar
 
Gildas Bazin committed
52 53 54
/*****************************************************************************
 * Access: local prototypes
 *****************************************************************************/
55
static block_t *ReadCompressed( access_t * );
56
static int AccessControl ( access_t *, int, va_list );
Gildas Bazin's avatar
 
Gildas Bazin committed
57

Gildas Bazin's avatar
Gildas Bazin committed
58 59 60
static int Demux       ( demux_t * );
static int DemuxControl( demux_t *, int, va_list );

61
static int OpenDevice( vlc_object_t *, access_sys_t *, string, bool );
Gildas Bazin's avatar
 
Gildas Bazin committed
62
static IBaseFilter *FindCaptureDevice( vlc_object_t *, string *,
63
                                       list<string> *, bool );
64
static size_t EnumDeviceCaps( vlc_object_t *, IBaseFilter *,
65 66
                              int, int, int, int, int, int,
                              AM_MEDIA_TYPE *mt, size_t );
Gildas Bazin's avatar
Gildas Bazin committed
67 68
static bool ConnectFilters( vlc_object_t *, access_sys_t *,
                            IBaseFilter *, CaptureFilter * );
Gildas Bazin's avatar
 
Gildas Bazin committed
69 70
static int FindDevicesCallback( vlc_object_t *, char const *,
                                vlc_value_t, vlc_value_t, void * );
71 72
static int ConfigDevicesCallback( vlc_object_t *, char const *,
                                  vlc_value_t, vlc_value_t, void * );
Gildas Bazin's avatar
 
Gildas Bazin committed
73

74
static void ShowPropertyPage( IUnknown * );
75
static void ShowDeviceProperties( vlc_object_t *, ICaptureGraphBuilder2 *,
76
                                  IBaseFilter *, bool );
77
static void ShowTunerProperties( vlc_object_t *, ICaptureGraphBuilder2 *,
78
                                 IBaseFilter *, bool );
79 80
static void ConfigTuner( vlc_object_t *, ICaptureGraphBuilder2 *,
                         IBaseFilter * );
Gildas Bazin's avatar
 
Gildas Bazin committed
81

Gildas Bazin's avatar
 
Gildas Bazin committed
82
/*****************************************************************************
83
 * Module descriptor
Gildas Bazin's avatar
 
Gildas Bazin committed
84
 *****************************************************************************/
85 86
static const char *const ppsz_vdev[] = { "", "none" };
static const char *const ppsz_vdev_text[] = { N_("Default"), N_("None") };
87

88 89
static const char *const ppsz_adev[] = { "", "none" };
static const char *const ppsz_adev_text[] = { N_("Default"), N_("None") };
90

91 92
static const int pi_tuner_input[] = { 0, 1, 2 };
static const char *const ppsz_tuner_input_text[] =
93
    {N_("Default"), N_("Cable"), N_("Antenna")};
94

95
static const int pi_amtuner_mode[]  = { AMTUNER_MODE_DEFAULT,
96 97 98 99
                                        AMTUNER_MODE_TV,
                                        AMTUNER_MODE_FM_RADIO,
                                        AMTUNER_MODE_AM_RADIO,
                                        AMTUNER_MODE_DSS };
100
static const char *const ppsz_amtuner_mode_text[] = { N_("Default"),
101 102 103 104
                                          N_("TV"),
                                          N_("FM radio"),
                                          N_("AM radio"),
                                          N_("DSS") };
105

106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
static const int i_standards_list[] =
    {
        KS_AnalogVideo_None,
        KS_AnalogVideo_NTSC_M, KS_AnalogVideo_NTSC_M_J, KS_AnalogVideo_NTSC_433,
        KS_AnalogVideo_PAL_B, KS_AnalogVideo_PAL_D, KS_AnalogVideo_PAL_G,
        KS_AnalogVideo_PAL_H, KS_AnalogVideo_PAL_I, KS_AnalogVideo_PAL_M,
        KS_AnalogVideo_PAL_N, KS_AnalogVideo_PAL_60,
        KS_AnalogVideo_SECAM_B, KS_AnalogVideo_SECAM_D, KS_AnalogVideo_SECAM_G,
        KS_AnalogVideo_SECAM_H, KS_AnalogVideo_SECAM_K, KS_AnalogVideo_SECAM_K1,
        KS_AnalogVideo_SECAM_L, KS_AnalogVideo_SECAM_L1,
        KS_AnalogVideo_PAL_N_COMBO
    };
static const char *const ppsz_standards_list_text[] =
    {
        N_("Default"),
        "NTSC_M", "NTSC_M_J", "NTSC_443",
        "PAL_B", "PAL_D", "PAL_G",
        "PAL_H", "PAL_I", "PAL_M",
        "PAL_N", "PAL_60",
        "SECAM_B", "SECAM_D", "SECAM_G",
        "SECAM_H", "SECAM_K", "SECAM_K1",
        "SECAM_L", "SECAM_L1",
        "PAL_N_COMBO"
    };

Gildas Bazin's avatar
 
Gildas Bazin committed
131 132
#define CACHING_TEXT N_("Caching value in ms")
#define CACHING_LONGTEXT N_( \
133
    "Caching value for DirectShow streams. " \
134
    "This value should be set in milliseconds." )
135 136
#define VDEV_TEXT N_("Video device name")
#define VDEV_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
137
    "Name of the video device that will be used by the " \
138 139 140 141
    "DirectShow plugin. If you don't specify anything, the default device " \
    "will be used.")
#define ADEV_TEXT N_("Audio device name")
#define ADEV_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
142
    "Name of the audio device that will be used by the " \
143
    "DirectShow plugin. If you don't specify anything, the default device " \
Christophe Mutricy's avatar
Christophe Mutricy committed
144
    "will be used. ")
145 146
#define SIZE_TEXT N_("Video size")
#define SIZE_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
147
    "Size of the video that will be displayed by the " \
148
    "DirectShow plugin. If you don't specify anything the default size for " \
Christophe Mutricy's avatar
Christophe Mutricy committed
149
    "your device will be used. You can specify a standard size (cif, d1, ...) or <width>x<height>.")
150 151
#define ASPECT_TEXT N_("Picture aspect-ratio n:m")
#define ASPECT_LONGTEXT N_("Define input picture aspect-ratio to use. Default is 4:3" )
152 153 154
#define CHROMA_TEXT N_("Video input chroma format")
#define CHROMA_LONGTEXT N_( \
    "Force the DirectShow video input to use a specific chroma format " \
155
    "(eg. I420 (default), RV24, etc.)")
156 157 158
#define FPS_TEXT N_("Video input frame rate")
#define FPS_LONGTEXT N_( \
    "Force the DirectShow video input to use a specific frame rate" \
Christophe Mutricy's avatar
Christophe Mutricy committed
159
    "(eg. 0 means default, 25, 29.97, 50, 59.94, etc.)")
160
#define CONFIG_TEXT N_("Device properties")
Gildas Bazin's avatar
 
Gildas Bazin committed
161
#define CONFIG_LONGTEXT N_( \
162 163
    "Show the properties dialog of the selected device before starting the " \
    "stream.")
164 165 166
#define TUNER_TEXT N_("Tuner properties")
#define TUNER_LONGTEXT N_( \
    "Show the tuner properties [channel selection] page." )
167 168
#define CHANNEL_TEXT N_("Tuner TV Channel")
#define CHANNEL_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
169
    "Set the TV channel the tuner will set to " \
170
    "(0 means default)." )
171 172 173 174
#define TVFREQ_TEXT N_("Tuner Frequency")
#define TVFREQ_LONGTEXT N_(  "This overrides the channel. Measured in Hz." )
#define STANDARD_TEXT N_( "Standard" )
#define STANDARD_LONGTEXT N_( "Video standard (Default, SECAM_D, PAL_B, NTSC_M, etc...)." )
175 176
#define COUNTRY_TEXT N_("Tuner country code")
#define COUNTRY_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
177
    "Set the tuner country code that establishes the current " \
178 179 180
    "channel-to-frequency mapping (0 means default)." )
#define TUNER_INPUT_TEXT N_("Tuner input type")
#define TUNER_INPUT_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
181
    "Select the tuner input type (Cable/Antenna)." )
182 183
#define VIDEO_IN_TEXT N_("Video input pin")
#define VIDEO_IN_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
184
  "Select the video input source, such as composite, s-video, " \
185
  "or tuner. Since these settings are hardware-specific, you should find good " \
Clément Stenac's avatar
Clément Stenac committed
186 187
  "settings in the \"Device config\" area, and use those numbers here. -1 " \
  "means that settings will not be changed.")
188 189
#define AUDIO_IN_TEXT N_("Audio input pin")
#define AUDIO_IN_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
190
  "Select the audio input source. See the \"video input\" option." )
191 192
#define VIDEO_OUT_TEXT N_("Video output pin")
#define VIDEO_OUT_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
193
  "Select the video output type. See the \"video input\" option." )
194 195
#define AUDIO_OUT_TEXT N_("Audio output pin")
#define AUDIO_OUT_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
196
  "Select the audio output type. See the \"video input\" option." )
Gildas Bazin's avatar
 
Gildas Bazin committed
197

198
#define AMTUNER_MODE_TEXT N_("AM Tuner mode")
199
#define AMTUNER_MODE_LONGTEXT N_( \
200 201
    "AM Tuner mode. Can be one of Default (0), TV (1)," \
     "AM Radio (2), FM Radio (3) or DSS (4).")
202

203 204
#define AUDIO_CHANNELS_TEXT N_("Number of audio channels")
#define AUDIO_CHANNELS_LONGTEXT N_( \
Felix Paul Kühne's avatar
Felix Paul Kühne committed
205
    "Select audio input format with the given number of audio channels (if non 0)" )
206 207 208 209 210 211 212 213 214

#define AUDIO_SAMPLERATE_TEXT N_("Audio sample rate")
#define AUDIO_SAMPLERATE_LONGTEXT N_( \
    "Select audio input format with the given sample rate (if non 0)" )

#define AUDIO_BITSPERSAMPLE_TEXT N_("Audio bits per sample")
#define AUDIO_BITSPERSAMPLE_LONGTEXT N_( \
    "Select audio input format with the given bits/sample (if non 0)" )

215
static int  CommonOpen ( vlc_object_t *, access_sys_t *, bool );
Gildas Bazin's avatar
Gildas Bazin committed
216 217
static void CommonClose( vlc_object_t *, access_sys_t * );

218 219 220 221 222 223
static int  AccessOpen ( vlc_object_t * );
static void AccessClose( vlc_object_t * );

static int  DemuxOpen  ( vlc_object_t * );
static void DemuxClose ( vlc_object_t * );

224 225 226 227 228
vlc_module_begin ()
    set_shortname( N_("DirectShow") )
    set_description( N_("DirectShow input") )
    set_category( CAT_INPUT )
    set_subcategory( SUBCAT_INPUT_ACCESS )
229
    add_integer( CFG_PREFIX "caching", (mtime_t)(0.2*CLOCK_FREQ) / 1000,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
230
                 CACHING_TEXT, CACHING_LONGTEXT, true )
231

232
    add_string( CFG_PREFIX "vdev", NULL, VDEV_TEXT, VDEV_LONGTEXT, false)
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
233
        change_string_list( ppsz_vdev, ppsz_vdev_text, FindDevicesCallback )
234 235
        change_action_add( FindDevicesCallback, N_("Refresh list") )
        change_action_add( ConfigDevicesCallback, N_("Configure") )
236

237
    add_string( CFG_PREFIX "adev", NULL, ADEV_TEXT, ADEV_LONGTEXT, false)
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
238
        change_string_list( ppsz_adev, ppsz_adev_text, FindDevicesCallback )
239 240
        change_action_add( FindDevicesCallback, N_("Refresh list") )
        change_action_add( ConfigDevicesCallback, N_("Configure") )
241

242
    add_string( CFG_PREFIX "size", NULL, SIZE_TEXT, SIZE_LONGTEXT, false)
243

244
    add_string( CFG_PREFIX "aspect-ratio", "4:3", ASPECT_TEXT, ASPECT_LONGTEXT, false)
245

246
    add_string( CFG_PREFIX "chroma", NULL, CHROMA_TEXT, CHROMA_LONGTEXT, true )
Gildas Bazin's avatar
 
Gildas Bazin committed
247

248
    add_float( CFG_PREFIX "fps", 0.0f, FPS_TEXT, FPS_LONGTEXT, true )
249

250
    add_bool( CFG_PREFIX "config", false, CONFIG_TEXT, CONFIG_LONGTEXT, true )
Gildas Bazin's avatar
 
Gildas Bazin committed
251

252
    add_bool( CFG_PREFIX "tuner", false, TUNER_TEXT, TUNER_LONGTEXT, true )
253

254
    add_integer( CFG_PREFIX "tuner-channel", 0, CHANNEL_TEXT, CHANNEL_LONGTEXT,
255
                true )
256

257
    add_integer( CFG_PREFIX "tuner-frequency", 0, TVFREQ_TEXT, TVFREQ_LONGTEXT,
258 259
                true )

260
    add_integer( CFG_PREFIX "tuner-country", 0, COUNTRY_TEXT, COUNTRY_LONGTEXT,
261
                true )
262

263
    add_integer( CFG_PREFIX "tuner-standard", 0, STANDARD_TEXT, STANDARD_LONGTEXT,
264
                false )
265
        change_integer_list( i_standards_list, ppsz_standards_list_text )
266

267
    add_integer( CFG_PREFIX "tuner-input", 0, TUNER_INPUT_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
268
                 TUNER_INPUT_LONGTEXT, true )
269
        change_integer_list( pi_tuner_input, ppsz_tuner_input_text )
270

271
    add_integer( CFG_PREFIX "video-input",  -1, VIDEO_IN_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
272
                 VIDEO_IN_LONGTEXT, true )
273

274
    add_integer( CFG_PREFIX "video-output", -1, VIDEO_OUT_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
275
                 VIDEO_OUT_LONGTEXT, true )
276

277
    add_integer( CFG_PREFIX "audio-input",  -1, AUDIO_IN_TEXT,
278 279
                 AUDIO_IN_LONGTEXT, true )

280
    add_integer( CFG_PREFIX "audio-output", -1, AUDIO_OUT_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
281
                 AUDIO_OUT_LONGTEXT, true )
282

283
    add_integer( CFG_PREFIX "amtuner-mode", AMTUNER_MODE_TV,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
284
                AMTUNER_MODE_TEXT, AMTUNER_MODE_LONGTEXT, false)
285
        change_integer_list( pi_amtuner_mode, ppsz_amtuner_mode_text )
286

287
    add_integer( CFG_PREFIX "audio-channels", 0, AUDIO_CHANNELS_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
288
                 AUDIO_CHANNELS_LONGTEXT, true )
289
    add_integer( CFG_PREFIX "audio-samplerate", 0, AUDIO_SAMPLERATE_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
290
                 AUDIO_SAMPLERATE_LONGTEXT, true )
291
    add_integer( CFG_PREFIX "audio-bitspersample", 0, AUDIO_BITSPERSAMPLE_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
292
                 AUDIO_BITSPERSAMPLE_LONGTEXT, true )
293

294 295 296
    add_shortcut( "dshow" )
    set_capability( "access_demux", 0 )
    set_callbacks( DemuxOpen, DemuxClose )
Gildas Bazin's avatar
 
Gildas Bazin committed
297

298 299
    add_submodule ()
    set_description( N_("DirectShow input") )
300
    add_shortcut( "dshow" )
301 302
    set_capability( "access", 0 )
    set_callbacks( AccessOpen, AccessClose )
Gildas Bazin's avatar
 
Gildas Bazin committed
303

304
vlc_module_end ()
Gildas Bazin's avatar
 
Gildas Bazin committed
305

306

Gildas Bazin's avatar
Gildas Bazin committed
307
/*****************************************************************************
Gildas Bazin's avatar
 
Gildas Bazin committed
308
 * DirectShow elementary stream descriptor
Gildas Bazin's avatar
Gildas Bazin committed
309
 *****************************************************************************/
Gildas Bazin's avatar
 
Gildas Bazin committed
310 311 312 313 314 315 316 317 318 319 320 321 322 323
typedef struct dshow_stream_t
{
    string          devicename;
    IBaseFilter     *p_device_filter;
    CaptureFilter   *p_capture_filter;
    AM_MEDIA_TYPE   mt;

    union
    {
      VIDEOINFOHEADER video;
      WAVEFORMATEX    audio;

    } header;

Gildas Bazin's avatar
Gildas Bazin committed
324 325 326
    int             i_fourcc;
    es_out_id_t     *p_es;

327
    bool      b_pts;
328 329

    deque<VLCMediaSample> samples_queue;
Gildas Bazin's avatar
 
Gildas Bazin committed
330 331
} dshow_stream_t;

Gildas Bazin's avatar
Gildas Bazin committed
332
/*****************************************************************************
333
 * DirectShow utility functions
Gildas Bazin's avatar
Gildas Bazin committed
334
 *****************************************************************************/
335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388
static void CreateDirectShowGraph( access_sys_t *p_sys )
{
    p_sys->i_crossbar_route_depth = 0;

    /* Create directshow filter graph */
    if( SUCCEEDED( CoCreateInstance( CLSID_FilterGraph, 0, CLSCTX_INPROC,
                       (REFIID)IID_IFilterGraph, (void **)&p_sys->p_graph) ) )
    {
        /* Create directshow capture graph builder if available */
        if( SUCCEEDED( CoCreateInstance( CLSID_CaptureGraphBuilder2, 0,
                         CLSCTX_INPROC, (REFIID)IID_ICaptureGraphBuilder2,
                         (void **)&p_sys->p_capture_graph_builder2 ) ) )
        {
            p_sys->p_capture_graph_builder2->
                SetFiltergraph((IGraphBuilder *)p_sys->p_graph);
        }

        p_sys->p_graph->QueryInterface( IID_IMediaControl,
                                        (void **)&p_sys->p_control );
    }
}

static void DeleteDirectShowGraph( access_sys_t *p_sys )
{
    DeleteCrossbarRoutes( p_sys );

    /* Remove filters from graph */
    for( int i = 0; i < p_sys->i_streams; i++ )
    {
        p_sys->p_graph->RemoveFilter( p_sys->pp_streams[i]->p_capture_filter );
        p_sys->p_graph->RemoveFilter( p_sys->pp_streams[i]->p_device_filter );
        p_sys->pp_streams[i]->p_capture_filter->Release();
        p_sys->pp_streams[i]->p_device_filter->Release();
    }

    /* Release directshow objects */
    if( p_sys->p_control )
    {
        p_sys->p_control->Release();
        p_sys->p_control = NULL;
    }
    if( p_sys->p_capture_graph_builder2 )
    {
        p_sys->p_capture_graph_builder2->Release();
        p_sys->p_capture_graph_builder2 = NULL;
    }

    if( p_sys->p_graph )
    {
        p_sys->p_graph->Release();
        p_sys->p_graph = NULL;
    }
}

Gildas Bazin's avatar
 
Gildas Bazin committed
389
/*****************************************************************************
Gildas Bazin's avatar
Gildas Bazin committed
390
 * CommonOpen: open direct show device
Gildas Bazin's avatar
 
Gildas Bazin committed
391
 *****************************************************************************/
Gildas Bazin's avatar
Gildas Bazin committed
392
static int CommonOpen( vlc_object_t *p_this, access_sys_t *p_sys,
393
                       bool b_access_demux )
Gildas Bazin's avatar
 
Gildas Bazin committed
394
{
395
    char *psz_val;
Gildas Bazin's avatar
 
Gildas Bazin committed
396

397
    /* Get/parse options and open device(s) */
Gildas Bazin's avatar
 
Gildas Bazin committed
398
    string vdevname, adevname;
399 400
    int i_width = 0, i_height = 0;
    vlc_fourcc_t i_chroma = 0;
401 402 403 404 405
    bool b_use_audio = true;
    bool b_use_video = true;

    /* Initialize OLE/COM */
    CoInitialize( 0 );
Gildas Bazin's avatar
 
Gildas Bazin committed
406

407 408 409
    var_Create( p_this,  CFG_PREFIX "config", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
    var_Create( p_this,  CFG_PREFIX "tuner", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "vdev" );
410
    if( psz_val )
411
    {
412
        msg_Dbg( p_this, "dshow-vdev: %s", psz_val ) ;
413
        /* skip none device */
414 415
        if ( strncasecmp( psz_val, "none", 4 ) != 0 )
            vdevname = string( psz_val );
416 417 418
        else
            b_use_video = false ;
    }
419
    free( psz_val );
Gildas Bazin's avatar
 
Gildas Bazin committed
420

421
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "adev" );
422
    if( psz_val )
423
    {
424
        msg_Dbg( p_this, "dshow-adev: %s", psz_val ) ;
425
        /* skip none device */
426 427
        if ( strncasecmp( psz_val, "none", 4 ) != 0 )
            adevname = string( psz_val );
428 429 430
        else
            b_use_audio = false ;
    }
431
    free( psz_val );
432

433 434 435 436 437 438 439 440 441 442 443 444
    /* DShow Size */
    static struct {
        const char *psz_size;
        int  i_width;
        int  i_height;
    } size_table[] =
    { { "subqcif", 128, 96  },
      {    "qsif", 160, 120 },
      {    "qcif", 176, 144 },
      {     "sif", 320, 240 },
      {     "cif", 352, 288 },
      {      "d1", 640, 480 },
Gildas Bazin's avatar
Gildas Bazin committed
445 446 447
      { 0, 0, 0 },
    };

448
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "size" );
449
    if( !EMPTY_STR(psz_val) )
Gildas Bazin's avatar
 
Gildas Bazin committed
450
    {
451
        int i;
Gildas Bazin's avatar
Gildas Bazin committed
452
        for( i = 0; size_table[i].psz_size; i++ )
453
        {
454
            if( !strcmp( psz_val, size_table[i].psz_size ) )
Gildas Bazin's avatar
Gildas Bazin committed
455 456 457 458 459
            {
                i_width = size_table[i].i_width;
                i_height = size_table[i].i_height;
                break;
            }
460
        }
Gildas Bazin's avatar
Gildas Bazin committed
461
        if( !size_table[i].psz_size ) /* Try to parse "WidthxHeight" */
462 463
        {
            char *psz_parser;
464
            i_width = strtol( psz_val, &psz_parser, 0 );
465
            if( *psz_parser == 'x' || *psz_parser == 'X')
Gildas Bazin's avatar
 
Gildas Bazin committed
466
            {
467
                i_height = strtol( psz_parser + 1, &psz_parser, 0 );
Gildas Bazin's avatar
 
Gildas Bazin committed
468
            }
Clément Stenac's avatar
Clément Stenac committed
469
            msg_Dbg( p_this, "width x height %dx%d", i_width, i_height );
Gildas Bazin's avatar
 
Gildas Bazin committed
470 471
        }
    }
472
    free( psz_val );
Gildas Bazin's avatar
 
Gildas Bazin committed
473

474
    /* Chroma */
475
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "chroma" );
476
    i_chroma = vlc_fourcc_GetCodecFromString( UNKNOWN_ES, psz_val );
477
    p_sys->b_chroma = i_chroma != 0;
478
    free( psz_val );
Gildas Bazin's avatar
 
Gildas Bazin committed
479

480 481
    var_Create( p_this, CFG_PREFIX "fps", VLC_VAR_FLOAT | VLC_VAR_DOINHERIT );
    var_Create( p_this, CFG_PREFIX "tuner-channel",
482
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
483
    var_Create( p_this, CFG_PREFIX "tuner-frequency",
484
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
485
    var_Create( p_this, CFG_PREFIX "tuner-standard",
486
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
487
    var_Create( p_this, CFG_PREFIX "tuner-country",
488
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
489
    var_Create( p_this, CFG_PREFIX "tuner-input",
490 491
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );

492
    var_Create( p_this, CFG_PREFIX "amtuner-mode",
493 494
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );

495
    var_Create( p_this, CFG_PREFIX "caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
Gildas Bazin's avatar
 
Gildas Bazin committed
496

497 498 499 500
    var_Create( p_this, CFG_PREFIX "video-input", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
    var_Create( p_this, CFG_PREFIX "audio-input", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
    var_Create( p_this, CFG_PREFIX "video-output", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
    var_Create( p_this, CFG_PREFIX "audio-output", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
501

502

Gildas Bazin's avatar
 
Gildas Bazin committed
503 504
    /* Initialize some data */
    p_sys->i_streams = 0;
Rémi Duraffort's avatar
Rémi Duraffort committed
505
    p_sys->pp_streams = NULL;
Gildas Bazin's avatar
 
Gildas Bazin committed
506 507 508
    p_sys->i_width = i_width;
    p_sys->i_height = i_height;
    p_sys->i_chroma = i_chroma;
Gildas Bazin's avatar
 
Gildas Bazin committed
509

510 511 512
    p_sys->p_graph = NULL;
    p_sys->p_capture_graph_builder2 = NULL;
    p_sys->p_control = NULL;
Gildas Bazin's avatar
 
Gildas Bazin committed
513

514 515 516
    /* Build directshow graph */
    CreateDirectShowGraph( p_sys );

517
    vlc_mutex_init( &p_sys->lock );
518
    vlc_cond_init( &p_sys->wait );
519

520 521
    if( !b_use_video && !b_use_audio )
    {
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
522
        dialog_Fatal( p_this, _("Capture failed"),
basos g's avatar
basos g committed
523
                        _("No video or audio device selected.") );
524 525
        return VLC_EGENERIC ;
    }
Gildas Bazin's avatar
 
Gildas Bazin committed
526

527 528 529 530 531
    if( !b_use_video )
        msg_Dbg( p_this, "skipping video device" ) ;
    bool b_err_video = false ;

    if( b_use_video && OpenDevice( p_this, p_sys, vdevname, 0 ) != VLC_SUCCESS )
Gildas Bazin's avatar
 
Gildas Bazin committed
532
    {
533 534
        msg_Err( p_this, "can't open video device");
        b_err_video = true ;
Gildas Bazin's avatar
 
Gildas Bazin committed
535
    }
536 537

    if ( b_use_video && !b_err_video )
Gildas Bazin's avatar
Gildas Bazin committed
538 539 540 541 542 543 544 545
    {
        /* Check if we can handle the demuxing ourselves or need to spawn
         * a demuxer module */
        dshow_stream_t *p_stream = p_sys->pp_streams[p_sys->i_streams-1];

        if( p_stream->mt.majortype == MEDIATYPE_Video )
        {
            if( /* Raw DV stream */
546
                p_stream->i_fourcc == VLC_CODEC_DV ||
Gildas Bazin's avatar
Gildas Bazin committed
547
                /* Raw MPEG video stream */
Laurent Aimar's avatar
Laurent Aimar committed
548
                p_stream->i_fourcc == VLC_CODEC_MPGV )
Gildas Bazin's avatar
Gildas Bazin committed
549
            {
550
                b_use_audio = false;
551 552 553 554

                if( b_access_demux )
                {
                    /* Let the access (only) take care of that */
555
                    return VLC_EGENERIC;
556
                }
Gildas Bazin's avatar
Gildas Bazin committed
557 558
            }
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
559

560 561
        if( p_stream->mt.majortype == MEDIATYPE_Stream )
        {
562
            b_use_audio = false;
563 564 565 566 567 568

            if( b_access_demux )
            {
                /* Let the access (only) take care of that */
                return VLC_EGENERIC;
            }
569

570
            if( var_GetBool( p_this, CFG_PREFIX "tuner" ) )
571 572 573 574 575 576
            {
                /* FIXME: we do MEDIATYPE_Stream here so we don't do
                 * it twice. */
                ShowTunerProperties( p_this, p_sys->p_capture_graph_builder2,
                                     p_stream->p_device_filter, 0 );
            }
577
        }
Gildas Bazin's avatar
Gildas Bazin committed
578 579
    }

580 581 582 583 584 585
    if( !b_use_audio )
        msg_Dbg( p_this, "skipping audio device") ;

    bool b_err_audio = false ;

    if( b_use_audio && OpenDevice( p_this, p_sys, adevname, 1 ) != VLC_SUCCESS )
Gildas Bazin's avatar
 
Gildas Bazin committed
586
    {
587 588 589 590 591 592 593 594 595
        msg_Err( p_this, "can't open audio device");
        b_err_audio = true ;
    }

    if( ( b_use_video && b_err_video && b_use_audio && b_err_audio ) ||
        ( !b_use_video && b_use_audio && b_err_audio ) ||
        ( b_use_video && !b_use_audio && b_err_video ) )
    {
        msg_Err( p_this, "FATAL: could not open ANY device" ) ;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
596
        dialog_Fatal( p_this,  _("Capture failed"),
597 598 599
                        _("VLC cannot open ANY capture device."
                          "Check the error log for details.") );
        return VLC_EGENERIC ;
Gildas Bazin's avatar
 
Gildas Bazin committed
600
    }
Gildas Bazin's avatar
Gildas Bazin committed
601

602
    for( int i = p_sys->i_crossbar_route_depth-1; i >= 0 ; --i )
Gildas Bazin's avatar
 
Gildas Bazin committed
603
    {
604
        int i_val = var_GetInteger( p_this, CFG_PREFIX "video-input" );
605 606
        if( i_val >= 0 )
            p_sys->crossbar_routes[i].VideoInputIndex = i_val;
607
        i_val = var_GetInteger( p_this, CFG_PREFIX "video-output" );
608 609
        if( i_val >= 0 )
            p_sys->crossbar_routes[i].VideoOutputIndex = i_val;
610
        i_val = var_GetInteger( p_this, CFG_PREFIX "audio-input" );
611 612
        if( i_val >= 0 )
            p_sys->crossbar_routes[i].AudioInputIndex = i_val;
613
        i_val = var_GetInteger( p_this, CFG_PREFIX "audio-output" );
614 615
        if( i_val >= 0 )
            p_sys->crossbar_routes[i].AudioOutputIndex = i_val;
616

617 618 619 620 621
        IAMCrossbar *pXbar = p_sys->crossbar_routes[i].pXbar;
        LONG VideoInputIndex = p_sys->crossbar_routes[i].VideoInputIndex;
        LONG VideoOutputIndex = p_sys->crossbar_routes[i].VideoOutputIndex;
        LONG AudioInputIndex = p_sys->crossbar_routes[i].AudioInputIndex;
        LONG AudioOutputIndex = p_sys->crossbar_routes[i].AudioOutputIndex;
Gildas Bazin's avatar
 
Gildas Bazin committed
622

623 624
        if( SUCCEEDED(pXbar->Route(VideoOutputIndex, VideoInputIndex)) )
        {
Clément Stenac's avatar
Clément Stenac committed
625 626
            msg_Dbg( p_this, "crossbar at depth %d, routed video "
                     "output %ld to video input %ld", i, VideoOutputIndex,
627
                     VideoInputIndex );
Gildas Bazin's avatar
 
Gildas Bazin committed
628

629 630 631 632 633
            if( AudioOutputIndex != -1 && AudioInputIndex != -1 )
            {
                if( SUCCEEDED( pXbar->Route(AudioOutputIndex,
                                            AudioInputIndex)) )
                {
Clément Stenac's avatar
Clément Stenac committed
634 635
                    msg_Dbg(p_this, "crossbar at depth %d, routed audio "
                            "output %ld to audio input %ld", i,
636 637 638 639
                            AudioOutputIndex, AudioInputIndex );
                }
            }
        }
640 641 642
        else
            msg_Err( p_this, "crossbar at depth %d could not route video "
                     "output %ld to input %ld", i, VideoOutputIndex, VideoInputIndex );
643 644
    }

645 646 647
    /*
    ** Show properties pages from other filters in graph
    */
648
    if( var_GetBool( p_this, CFG_PREFIX "config" ) )
649
    {
650
        for( int i = p_sys->i_crossbar_route_depth-1; i >= 0 ; --i )
651 652 653
        {
            IAMCrossbar *pXbar = p_sys->crossbar_routes[i].pXbar;
            IBaseFilter *p_XF;
654 655 656

            if( SUCCEEDED( pXbar->QueryInterface( IID_IBaseFilter,
                                                  (void **)&p_XF ) ) )
657 658 659 660
            {
                ShowPropertyPage( p_XF );
                p_XF->Release();
            }
661
        }
662 663
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
664 665
    /* Initialize some data */
    p_sys->i_current_stream = 0;
666

Gildas Bazin's avatar
Gildas Bazin committed
667
    if( !p_sys->i_streams ) return VLC_EGENERIC;
Gildas Bazin's avatar
 
Gildas Bazin committed
668

Gildas Bazin's avatar
 
Gildas Bazin committed
669 670 671 672
    return VLC_SUCCESS;
}

/*****************************************************************************
Gildas Bazin's avatar
Gildas Bazin committed
673
 * DemuxOpen: open direct show device as an access_demux module
Gildas Bazin's avatar
 
Gildas Bazin committed
674
 *****************************************************************************/
Gildas Bazin's avatar
Gildas Bazin committed
675
static int DemuxOpen( vlc_object_t *p_this )
Gildas Bazin's avatar
 
Gildas Bazin committed
676
{
Gildas Bazin's avatar
Gildas Bazin committed
677 678
    demux_t      *p_demux = (demux_t *)p_this;
    access_sys_t *p_sys;
Gildas Bazin's avatar
 
Gildas Bazin committed
679

Rémi Duraffort's avatar
Rémi Duraffort committed
680
    p_sys = (access_sys_t*)calloc( 1, sizeof( access_sys_t ) );
Rémi Duraffort's avatar
Rémi Duraffort committed
681 682
    if( !p_sys )
        return VLC_ENOMEM;
Gildas Bazin's avatar
Gildas Bazin committed
683
    p_demux->p_sys = (demux_sys_t *)p_sys;
Gildas Bazin's avatar
 
Gildas Bazin committed
684

685
    if( CommonOpen( p_this, p_sys, true ) != VLC_SUCCESS )
Gildas Bazin's avatar
Gildas Bazin committed
686 687 688 689
    {
        CommonClose( p_this, p_sys );
        return VLC_EGENERIC;
    }
690

Gildas Bazin's avatar
Gildas Bazin committed
691 692 693
    /* Everything is ready. Let's rock baby */
    msg_Dbg( p_this, "Playing...");
    p_sys->p_control->Run();
694

Gildas Bazin's avatar
Gildas Bazin committed
695 696 697 698 699 700
    p_demux->pf_demux   = Demux;
    p_demux->pf_control = DemuxControl;
    p_demux->info.i_update = 0;
    p_demux->info.i_title = 0;
    p_demux->info.i_seekpoint = 0;

701
    for( int i = 0; i < p_sys->i_streams; i++ )
702
    {
Gildas Bazin's avatar
Gildas Bazin committed
703 704
        dshow_stream_t *p_stream = p_sys->pp_streams[i];
        es_format_t fmt;
705

Gildas Bazin's avatar
Gildas Bazin committed
706 707
        if( p_stream->mt.majortype == MEDIATYPE_Video )
        {
708
            char *psz_aspect = var_CreateGetString( p_this, CFG_PREFIX "aspect-ratio" );
709 710
            char *psz_delim = !EMPTY_STR( psz_aspect ) ? strchr( psz_aspect, ':' ) : NULL;

Gildas Bazin's avatar
Gildas Bazin committed
711
            es_format_Init( &fmt, VIDEO_ES, p_stream->i_fourcc );
712

Gildas Bazin's avatar
Gildas Bazin committed
713 714
            fmt.video.i_width  = p_stream->header.video.bmiHeader.biWidth;
            fmt.video.i_height = p_stream->header.video.bmiHeader.biHeight;
715 716 717 718 719 720 721 722 723 724 725 726

            if( psz_delim )
            {
                fmt.video.i_sar_num = atoi( psz_aspect ) * fmt.video.i_height;
                fmt.video.i_sar_den = atoi( psz_delim + 1 ) * fmt.video.i_width;
            }
            else
            {
                fmt.video.i_sar_num = 4 * fmt.video.i_height;
                fmt.video.i_sar_den = 3 * fmt.video.i_width;
            }
            free( psz_aspect );
727

Gildas Bazin's avatar
Gildas Bazin committed
728 729 730 731 732
            if( !p_stream->header.video.bmiHeader.biCompression )
            {
                /* RGB DIB are coded from bottom to top */
                fmt.video.i_height = (unsigned int)(-(int)fmt.video.i_height);
            }
733

Gildas Bazin's avatar
Gildas Bazin committed
734
            /* Setup rgb mask for RGB formats */
735
            if( p_stream->i_fourcc == VLC_CODEC_RGB24 )
Gildas Bazin's avatar
Gildas Bazin committed
736
            {
737 738 739 740
                /* This is in RGB format
            http://msdn.microsoft.com/en-us/library/dd407253%28VS.85%29.aspx?ppud=4
                 */
                fmt.video.i_rmask = 0x00ff0000;
741
                fmt.video.i_gmask = 0x0000ff00;
742
                fmt.video.i_bmask = 0x000000ff;
Gildas Bazin's avatar
Gildas Bazin committed
743
            }
744 745 746 747 748 749 750

            if( p_stream->header.video.AvgTimePerFrame )
            {
                fmt.video.i_frame_rate = 10000000;
                fmt.video.i_frame_rate_base =
                    p_stream->header.video.AvgTimePerFrame;
            }
Gildas Bazin's avatar
Gildas Bazin committed
751 752 753 754
        }
        else if( p_stream->mt.majortype == MEDIATYPE_Audio )
        {
            es_format_Init( &fmt, AUDIO_ES, p_stream->i_fourcc );
755

Gildas Bazin's avatar
Gildas Bazin committed
756 757 758 759 760 761 762 763 764 765
            fmt.audio.i_channels = p_stream->header.audio.nChannels;
            fmt.audio.i_rate = p_stream->header.audio.nSamplesPerSec;
            fmt.audio.i_bitspersample = p_stream->header.audio.wBitsPerSample;
            fmt.audio.i_blockalign = fmt.audio.i_channels *
                fmt.audio.i_bitspersample / 8;
            fmt.i_bitrate = fmt.audio.i_channels * fmt.audio.i_rate *
                fmt.audio.i_bitspersample;
        }

        p_stream->p_es = es_out_Add( p_demux->out, &fmt );
766
    }
Gildas Bazin's avatar
Gildas Bazin committed
767

768
    return VLC_SUCCESS;
769 770
}

Gildas Bazin's avatar
Gildas Bazin committed
771 772 773 774
/*****************************************************************************
 * AccessOpen: open direct show device as an access module
 *****************************************************************************/
static int AccessOpen( vlc_object_t *p_this )
775
{
Gildas Bazin's avatar
Gildas Bazin committed
776 777
    access_t     *p_access = (access_t*)p_this;
    access_sys_t *p_sys;
778

Rémi Duraffort's avatar
Rémi Duraffort committed
779
    p_access->p_sys = p_sys = (access_sys_t*)calloc( 1, sizeof( access_sys_t ) );
Rémi Duraffort's avatar
Rémi Duraffort committed
780 781
    if( !p_sys )
        return VLC_ENOMEM;
782

783
    if( CommonOpen( p_this, p_sys, false ) != VLC_SUCCESS )
Gildas Bazin's avatar
Gildas Bazin committed
784 785 786 787
    {
        CommonClose( p_this, p_sys );
        return VLC_EGENERIC;
    }
788

Gildas Bazin's avatar
Gildas Bazin committed
789
    dshow_stream_t *p_stream = p_sys->pp_streams[0];
790

Gildas Bazin's avatar
Gildas Bazin committed
791 792
    /* Check if we need to force demuxers */
    if( !p_access->psz_demux || !*p_access->psz_demux )
Gildas Bazin's avatar
 
Gildas Bazin committed
793
    {
794
        if( p_stream->i_fourcc == VLC_CODEC_DV )
795
        {
Rafaël Carré's avatar
Rafaël Carré committed
796
            free( p_access->psz_demux );
Gildas Bazin's avatar
Gildas Bazin committed
797 798
            p_access->psz_demux = strdup( "rawdv" );
        }
Laurent Aimar's avatar
Laurent Aimar committed
799
        else if( p_stream->i_fourcc == VLC_CODEC_MPGV )
Gildas Bazin's avatar
Gildas Bazin committed
800
        {
Rafaël Carré's avatar
Rafaël Carré committed
801 802
            free( p_access->psz_demux );
            p_access->psz_demux = strdup( "mpgv" );
803
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
804
    }
Gildas Bazin's avatar
 
Gildas Bazin committed
805

Gildas Bazin's avatar
Gildas Bazin committed
806
    /* Setup Access */
807 808
    p_access->pf_read = NULL;
    p_access->pf_block = ReadCompressed;