dshow.cpp 88.7 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
#include <inttypes.h>
37 38
#include <list>
#include <string>
39

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

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

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

51
#include "access.h"
Gildas Bazin's avatar
 
Gildas Bazin committed
52 53
#include "filter.h"

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

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

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

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

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

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

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

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

108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
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"
    };

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

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

200 201
#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
202
    "Select audio input format with the given number of audio channels (if non 0)" )
203 204 205 206 207 208 209 210 211

#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)" )

212
static int  CommonOpen ( vlc_object_t *, access_sys_t *, bool );
Gildas Bazin's avatar
Gildas Bazin committed
213 214
static void CommonClose( vlc_object_t *, access_sys_t * );

215 216 217 218 219 220
static int  AccessOpen ( vlc_object_t * );
static void AccessClose( vlc_object_t * );

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

221 222 223 224 225
vlc_module_begin ()
    set_shortname( N_("DirectShow") )
    set_description( N_("DirectShow input") )
    set_category( CAT_INPUT )
    set_subcategory( SUBCAT_INPUT_ACCESS )
226

227
    add_string( CFG_PREFIX "vdev", NULL, VDEV_TEXT, VDEV_LONGTEXT, false)
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
228
        change_string_list( ppsz_vdev, ppsz_vdev_text, FindDevicesCallback )
229 230
        change_action_add( FindDevicesCallback, N_("Refresh list") )
        change_action_add( ConfigDevicesCallback, N_("Configure") )
231

232
    add_string( CFG_PREFIX "adev", NULL, ADEV_TEXT, ADEV_LONGTEXT, false)
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
233
        change_string_list( ppsz_adev, ppsz_adev_text, FindDevicesCallback )
234 235
        change_action_add( FindDevicesCallback, N_("Refresh list") )
        change_action_add( ConfigDevicesCallback, N_("Configure") )
236

237
    add_string( CFG_PREFIX "size", NULL, SIZE_TEXT, SIZE_LONGTEXT, false)
238
        change_safe()
239

240
    add_string( CFG_PREFIX "aspect-ratio", "4:3", ASPECT_TEXT, ASPECT_LONGTEXT, false)
241
        change_safe()
242

243
    add_string( CFG_PREFIX "chroma", NULL, CHROMA_TEXT, CHROMA_LONGTEXT, true )
244
        change_safe()
Gildas Bazin's avatar
 
Gildas Bazin committed
245

246
    add_float( CFG_PREFIX "fps", 0.0f, FPS_TEXT, FPS_LONGTEXT, true )
247
        change_safe()
248

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

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

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

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

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

264
    add_integer( CFG_PREFIX "tuner-standard", 0, STANDARD_TEXT, STANDARD_TEXT,
265
                false )
266
        change_integer_list( i_standards_list, ppsz_standards_list_text )
267
        change_safe()
268

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

274
    add_integer( CFG_PREFIX "video-input",  -1, VIDEO_IN_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
275
                 VIDEO_IN_LONGTEXT, true )
276
        change_safe()
277

278
    add_integer( CFG_PREFIX "video-output", -1, VIDEO_OUT_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
279
                 VIDEO_OUT_LONGTEXT, true )
280

281
    add_integer( CFG_PREFIX "audio-input",  -1, AUDIO_IN_TEXT,
282
                 AUDIO_IN_LONGTEXT, true )
283
        change_safe()
284

285
    add_integer( CFG_PREFIX "audio-output", -1, AUDIO_OUT_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
286
                 AUDIO_OUT_LONGTEXT, true )
287

288
    add_integer( CFG_PREFIX "amtuner-mode", AMTUNER_MODE_TV,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
289
                AMTUNER_MODE_TEXT, AMTUNER_MODE_LONGTEXT, false)
290
        change_integer_list( pi_amtuner_mode, ppsz_amtuner_mode_text )
291
        change_safe()
292

293
    add_integer( CFG_PREFIX "audio-channels", 0, AUDIO_CHANNELS_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
294
                 AUDIO_CHANNELS_LONGTEXT, true )
295
    add_integer( CFG_PREFIX "audio-samplerate", 0, AUDIO_SAMPLERATE_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
296
                 AUDIO_SAMPLERATE_LONGTEXT, true )
297
    add_integer( CFG_PREFIX "audio-bitspersample", 0, AUDIO_BITSPERSAMPLE_TEXT,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
298
                 AUDIO_BITSPERSAMPLE_LONGTEXT, true )
299

300 301 302
    add_shortcut( "dshow" )
    set_capability( "access_demux", 0 )
    set_callbacks( DemuxOpen, DemuxClose )
Gildas Bazin's avatar
 
Gildas Bazin committed
303

304 305
    add_submodule ()
    set_description( N_("DirectShow input") )
306
    add_shortcut( "dshow" )
307 308
    set_capability( "access", 0 )
    set_callbacks( AccessOpen, AccessClose )
Gildas Bazin's avatar
 
Gildas Bazin committed
309

310
vlc_module_end ()
Gildas Bazin's avatar
 
Gildas Bazin committed
311

312

Gildas Bazin's avatar
Gildas Bazin committed
313
/*****************************************************************************
Gildas Bazin's avatar
 
Gildas Bazin committed
314
 * DirectShow elementary stream descriptor
Gildas Bazin's avatar
Gildas Bazin committed
315
 *****************************************************************************/
Gildas Bazin's avatar
 
Gildas Bazin committed
316 317 318 319 320 321 322 323 324 325 326 327 328 329
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
330 331 332
    int             i_fourcc;
    es_out_id_t     *p_es;

333
    bool      b_pts;
334 335

    deque<VLCMediaSample> samples_queue;
Gildas Bazin's avatar
 
Gildas Bazin committed
336 337
} dshow_stream_t;

Gildas Bazin's avatar
Gildas Bazin committed
338
/*****************************************************************************
339
 * DirectShow utility functions
Gildas Bazin's avatar
Gildas Bazin committed
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 389 390 391 392 393 394
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
395
/*****************************************************************************
Gildas Bazin's avatar
Gildas Bazin committed
396
 * CommonOpen: open direct show device
Gildas Bazin's avatar
 
Gildas Bazin committed
397
 *****************************************************************************/
Gildas Bazin's avatar
Gildas Bazin committed
398
static int CommonOpen( vlc_object_t *p_this, access_sys_t *p_sys,
399
                       bool b_access_demux )
Gildas Bazin's avatar
 
Gildas Bazin committed
400
{
401
    char *psz_val;
Gildas Bazin's avatar
 
Gildas Bazin committed
402

403
    /* Get/parse options and open device(s) */
Gildas Bazin's avatar
 
Gildas Bazin committed
404
    string vdevname, adevname;
405 406
    int i_width = 0, i_height = 0;
    vlc_fourcc_t i_chroma = 0;
407 408 409 410 411
    bool b_use_audio = true;
    bool b_use_video = true;

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

413 414 415
    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" );
416
    if( psz_val )
417
    {
418
        msg_Dbg( p_this, "dshow-vdev: %s", psz_val ) ;
419
        /* skip none device */
420 421
        if ( strncasecmp( psz_val, "none", 4 ) != 0 )
            vdevname = string( psz_val );
422 423 424
        else
            b_use_video = false ;
    }
425
    free( psz_val );
Gildas Bazin's avatar
 
Gildas Bazin committed
426

427
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "adev" );
428
    if( psz_val )
429
    {
430
        msg_Dbg( p_this, "dshow-adev: %s", psz_val ) ;
431
        /* skip none device */
432 433
        if ( strncasecmp( psz_val, "none", 4 ) != 0 )
            adevname = string( psz_val );
434 435 436
        else
            b_use_audio = false ;
    }
437
    free( psz_val );
438

439 440 441 442 443 444 445 446 447 448 449 450
    /* 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
451 452 453
      { 0, 0, 0 },
    };

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

480
    /* Chroma */
481
    psz_val = var_CreateGetString( p_this, CFG_PREFIX "chroma" );
482
    i_chroma = vlc_fourcc_GetCodecFromString( UNKNOWN_ES, psz_val );
483
    p_sys->b_chroma = i_chroma != 0;
484
    free( psz_val );
Gildas Bazin's avatar
 
Gildas Bazin committed
485

486 487
    var_Create( p_this, CFG_PREFIX "fps", VLC_VAR_FLOAT | VLC_VAR_DOINHERIT );
    var_Create( p_this, CFG_PREFIX "tuner-channel",
488
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
489
    var_Create( p_this, CFG_PREFIX "tuner-frequency",
490
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
491
    var_Create( p_this, CFG_PREFIX "tuner-standard",
492
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
493
    var_Create( p_this, CFG_PREFIX "tuner-country",
494
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
495
    var_Create( p_this, CFG_PREFIX "tuner-input",
496 497
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );

498
    var_Create( p_this, CFG_PREFIX "amtuner-mode",
499 500
                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );

501 502 503 504
    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 );
505

506

Gildas Bazin's avatar
 
Gildas Bazin committed
507 508
    /* Initialize some data */
    p_sys->i_streams = 0;
Rémi Duraffort's avatar
Rémi Duraffort committed
509
    p_sys->pp_streams = NULL;
Gildas Bazin's avatar
 
Gildas Bazin committed
510 511 512
    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
513

514 515 516
    p_sys->p_graph = NULL;
    p_sys->p_capture_graph_builder2 = NULL;
    p_sys->p_control = NULL;
Gildas Bazin's avatar
 
Gildas Bazin committed
517

518 519 520
    /* Build directshow graph */
    CreateDirectShowGraph( p_sys );

521
    vlc_mutex_init( &p_sys->lock );
522
    vlc_cond_init( &p_sys->wait );
523

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

531 532 533 534 535
    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
536
    {
537 538
        msg_Err( p_this, "can't open video device");
        b_err_video = true ;
Gildas Bazin's avatar
 
Gildas Bazin committed
539
    }
540 541

    if ( b_use_video && !b_err_video )
Gildas Bazin's avatar
Gildas Bazin committed
542 543 544 545 546 547 548 549
    {
        /* 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 */
550
                p_stream->i_fourcc == VLC_CODEC_DV ||
Gildas Bazin's avatar
Gildas Bazin committed
551
                /* Raw MPEG video stream */
Laurent Aimar's avatar
Laurent Aimar committed
552
                p_stream->i_fourcc == VLC_CODEC_MPGV )
Gildas Bazin's avatar
Gildas Bazin committed
553
            {
554
                b_use_audio = false;
555 556 557 558

                if( b_access_demux )
                {
                    /* Let the access (only) take care of that */
559
                    return VLC_EGENERIC;
560
                }
Gildas Bazin's avatar
Gildas Bazin committed
561 562
            }
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
563

564 565
        if( p_stream->mt.majortype == MEDIATYPE_Stream )
        {
566
            b_use_audio = false;
567 568 569 570 571 572

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

574
            if( var_GetBool( p_this, CFG_PREFIX "tuner" ) )
575 576 577 578 579 580
            {
                /* 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 );
            }
581
        }
Gildas Bazin's avatar
Gildas Bazin committed
582 583
    }

584 585 586 587 588 589
    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
590
    {
591 592 593 594 595 596 597 598 599
        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
600
        dialog_Fatal( p_this,  _("Capture failed"),
601
                        _("VLC cannot open ANY capture device. "
602 603
                          "Check the error log for details.") );
        return VLC_EGENERIC ;
Gildas Bazin's avatar
 
Gildas Bazin committed
604
    }
Gildas Bazin's avatar
Gildas Bazin committed
605

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

621 622 623 624 625
        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
626

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

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

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

            if( SUCCEEDED( pXbar->QueryInterface( IID_IBaseFilter,
                                                  (void **)&p_XF ) ) )
661 662 663 664
            {
                ShowPropertyPage( p_XF );
                p_XF->Release();
            }
665
        }
666 667
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
668 669
    /* Initialize some data */
    p_sys->i_current_stream = 0;
670

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

Gildas Bazin's avatar
 
Gildas Bazin committed
673 674 675 676
    return VLC_SUCCESS;
}

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

Rémi Duraffort's avatar
Rémi Duraffort committed
684
    p_sys = (access_sys_t*)calloc( 1, sizeof( access_sys_t ) );
Rémi Duraffort's avatar
Rémi Duraffort committed
685 686
    if( !p_sys )
        return VLC_ENOMEM;
Gildas Bazin's avatar
Gildas Bazin committed
687
    p_demux->p_sys = (demux_sys_t *)p_sys;
Gildas Bazin's avatar
 
Gildas Bazin committed
688

689
    if( CommonOpen( p_this, p_sys, true ) != VLC_SUCCESS )
Gildas Bazin's avatar
Gildas Bazin committed
690 691 692 693
    {
        CommonClose( p_this, p_sys );
        return VLC_EGENERIC;
    }
694

Gildas Bazin's avatar
Gildas Bazin committed
695 696 697
    /* Everything is ready. Let's rock baby */
    msg_Dbg( p_this, "Playing...");
    p_sys->p_control->Run();
698

Gildas Bazin's avatar
Gildas Bazin committed
699 700 701 702 703 704
    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;

705
    for( int i = 0; i < p_sys->i_streams; i++ )
706
    {
Gildas Bazin's avatar
Gildas Bazin committed
707 708
        dshow_stream_t *p_stream = p_sys->pp_streams[i];
        es_format_t fmt;
709

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

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

Gildas Bazin's avatar
Gildas Bazin committed
717 718
            fmt.video.i_width  = p_stream->header.video.bmiHeader.biWidth;
            fmt.video.i_height = p_stream->header.video.bmiHeader.biHeight;
719 720 721 722 723 724 725 726 727 728 729 730

            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 );
731

Gildas Bazin's avatar
Gildas Bazin committed
732 733 734 735 736
            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);
            }
737

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

            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
755 756 757 758
        }
        else if( p_stream->mt.majortype == MEDIATYPE_Audio )
        {
            es_format_Init( &fmt, AUDIO_ES, p_stream->i_fourcc );
759

Gildas Bazin's avatar
Gildas Bazin committed
760 761 762 763 764 765 766 767 768 769
            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 );
770
    }
Gildas Bazin's avatar
Gildas Bazin committed
771

772
    return VLC_SUCCESS;
773 774
}

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

Rémi Duraffort's avatar
Rémi Duraffort committed
783
    p_access->p_sys = p_sys = (access_sys_t*)calloc( 1, sizeof( access_sys_t ) );
Rémi Duraffort's avatar
Rémi Duraffort committed
784 785
    if( !p_sys )
        return VLC_ENOMEM;
786

787
    if( CommonOpen( p_this, p_sys, false ) != VLC_SUCCESS )
Gildas Bazin's avatar
Gildas Bazin committed
788 789 790 791
    {
        CommonClose( p_this, p_sys );
        return VLC_EGENERIC;
    }
792

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

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

Gildas Bazin's avatar
Gildas Bazin committed
810
    /* Setup Access */
811 812
    p_access->pf_read = NULL;
    p_access->pf_block = ReadCompressed;
Gildas Bazin's avatar
Gildas Bazin committed
813 814 815 816 817
    p_access->pf_control = AccessControl;
    p_access->pf_seek = NULL;
    p_access->info.i_update = 0;
    p_access->info.i_size = 0;
    p_access->info.i_pos = 0;
818
    p_access->info.b_eof = false;
Gildas Bazin's avatar
Gildas Bazin committed
819 820 821 822 823 824 825 826 827
    p_access->info.i_title = 0;
    p_access->info.i_seekpoint = 0;
    p_access->p_sys = p_sys;

    /* Everything is ready. Let's rock baby */
    msg_Dbg( p_this, "Playing...");
    p_sys->p_control->Run();

    return VLC_SUCCESS;
828
}
Gildas Bazin's avatar
 
Gildas Bazin committed
829

Gildas Bazin's avatar
Gildas Bazin committed
830 831 832 833
/*****************************************************************************
 * CommonClose: close device
 *****************************************************************************/
static void CommonClose( vlc_object_t *p_this, access_sys_t *p_sys )