dshow.cpp 88.8 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 171 172
#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...)." )
173 174
#define COUNTRY_TEXT N_("Tuner country code")
#define COUNTRY_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
175
    "Set the tuner country code that establishes the current " \
176 177 178
    "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
179
    "Select the tuner input type (Cable/Antenna)." )
180 181
#define VIDEO_IN_TEXT N_("Video input pin")
#define VIDEO_IN_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
182
  "Select the video input source, such as composite, s-video, " \
183
  "or tuner. Since these settings are hardware-specific, you should find good " \
Clément Stenac's avatar
Clément Stenac committed
184 185
  "settings in the \"Device config\" area, and use those numbers here. -1 " \
  "means that settings will not be changed.")
186 187
#define AUDIO_IN_TEXT N_("Audio input pin")
#define AUDIO_IN_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
188
  "Select the audio input source. See the \"video input\" option." )
189 190
#define VIDEO_OUT_TEXT N_("Video output pin")
#define VIDEO_OUT_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
191
  "Select the video output type. See the \"video input\" option." )
192 193
#define AUDIO_OUT_TEXT N_("Audio output pin")
#define AUDIO_OUT_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
194
  "Select the audio output type. See the \"video input\" option." )
Gildas Bazin's avatar
 
Gildas Bazin committed
195

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

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

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

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

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

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

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

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

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

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

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

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

247
    add_float( CFG_PREFIX "fps", 0.0f, FPS_TEXT, FPS_LONGTEXT, true )
248
        change_safe()
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
        change_safe()
257

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

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

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

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

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

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

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

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

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

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

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

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

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

313

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

334
    bool      b_pts;
335 336

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

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

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

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

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

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

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

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

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

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

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

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

507

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

773
    return VLC_SUCCESS;
774 775
}

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

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

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

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

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