]> git.sesse.net Git - vlc/blob - modules/access/pvr.c
Export Content-Type out of HTTP access using Control.
[vlc] / modules / access / pvr.c
1 /*****************************************************************************
2  * pvr.c
3  *****************************************************************************
4  * Copyright (C) 2001, 2002 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Eric Petit <titer@videolan.org>
8  *          Paul Corke <paulc@datatote.co.uk>
9  *
10  * This program is free software; you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation; either version 2 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23  *****************************************************************************/
24
25 /*****************************************************************************
26  * Preamble
27  *****************************************************************************/
28 #include <vlc/vlc.h>
29 #include <vlc_access.h>
30
31 #include <sys/types.h>
32 #include <sys/stat.h>
33 #include <fcntl.h>
34 #include <unistd.h>
35 #include <errno.h>
36 #include <linux/types.h>
37 #include <sys/ioctl.h>
38 #include <sys/poll.h>
39 #ifdef HAVE_NEW_LINUX_VIDEODEV2_H
40 #   ifdef VIDEODEV2_H_FILE
41 #   include VIDEODEV2_H_FILE
42 #   else
43 #   include <linux/videodev2.h>
44 #   endif
45 #else
46 #include "videodev2.h"
47 #endif
48
49 /*****************************************************************************
50  * Module descriptor
51  *****************************************************************************/
52 static int  Open ( vlc_object_t * );
53 static void Close( vlc_object_t * );
54
55 #define CACHING_TEXT N_("Caching value in ms")
56 #define CACHING_LONGTEXT N_( \
57     "Default caching value for PVR streams. This " \
58     "value should be set in milliseconds." )
59
60 #define DEVICE_TEXT N_( "Device" )
61 #define DEVICE_LONGTEXT N_( "PVR video device" )
62
63 #define RADIO_DEVICE_TEXT N_( "Radio device" )
64 #define RADIO_DEVICE_LONGTEXT N_( "PVR radio device" )
65
66 #define NORM_TEXT N_( "Norm" )
67 #define NORM_LONGTEXT N_( "Norm of the stream " \
68     "(Automatic, SECAM, PAL, or NTSC)." )
69
70 #define WIDTH_TEXT N_( "Width" )
71 #define WIDTH_LONGTEXT N_( "Width of the stream to capture " \
72     "(-1 for autodetection)." )
73
74 #define HEIGHT_TEXT N_( "Height" )
75 #define HEIGHT_LONGTEXT N_( "Height of the stream to capture " \
76     "(-1 for autodetection)." )
77
78 #define FREQUENCY_TEXT N_( "Frequency" )
79 #define FREQUENCY_LONGTEXT N_( "Frequency to capture (in kHz), if applicable." )
80
81 #define FRAMERATE_TEXT N_( "Framerate" )
82 #define FRAMERATE_LONGTEXT N_( "Framerate to capture, if applicable " \
83     "(-1 for autodetect)." )
84
85 #define KEYINT_TEXT N_( "Key interval" )
86 #define KEYINT_LONGTEXT N_( "Interval between keyframes (-1 for autodetect)." )
87
88 #define BFRAMES_TEXT N_( "B Frames" )
89 #define BFRAMES_LONGTEXT N_("If this option is set, B-Frames will be used. " \
90     "Use this option to set the number of B-Frames.")
91
92 #define BITRATE_TEXT N_( "Bitrate" )
93 #define BITRATE_LONGTEXT N_( "Bitrate to use (-1 for default)." )
94
95 #define BITRATE_PEAK_TEXT N_( "Bitrate peak" )
96 #define BITRATE_PEAK_LONGTEXT N_( "Peak bitrate in VBR mode." )
97
98 #define BITRATE_MODE_TEXT N_( "Bitrate mode" )
99 #define BITRATE_MODE_LONGTEXT N_( "Bitrate mode to use (VBR or CBR)." )
100
101 #define BITMASK_TEXT N_( "Audio bitmask" )
102 #define BITMASK_LONGTEXT N_("Bitmask that will "\
103     "get used by the audio part of the card." )
104
105 #define VOLUME_TEXT N_( "Volume" )
106 #define VOLUME_LONGTEXT N_("Audio volume (0-65535)." )
107
108 #define CHAN_TEXT N_( "Channel" )
109 #define CHAN_LONGTEXT N_( "Channel of the card to use (Usually, 0 = tuner, " \
110     "1 = composite, 2 = svideo)" )
111
112 static int i_norm_list[] =
113     { V4L2_STD_UNKNOWN, V4L2_STD_SECAM, V4L2_STD_PAL, V4L2_STD_NTSC };
114 static const char *psz_norm_list_text[] =
115     { N_("Automatic"), N_("SECAM"), N_("PAL"),  N_("NTSC") };
116
117 static int i_bitrates[] = { 0, 1 };
118 static const char *psz_bitrates_list_text[] = { N_("vbr"), N_("cbr") };
119
120 static int pi_radio_range[2] = { 65000, 108000 };
121
122 vlc_module_begin();
123     set_shortname( _("PVR") );
124     set_description( _("IVTV MPEG Encoding cards input") );
125     set_category( CAT_INPUT );
126     set_subcategory( SUBCAT_INPUT_ACCESS );
127     set_capability( "access2", 0 );
128     add_shortcut( "pvr" );
129
130     add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT,
131                  CACHING_LONGTEXT, VLC_TRUE );
132     add_string( "pvr-device", "/dev/video0", NULL, DEVICE_TEXT,
133                  DEVICE_LONGTEXT, VLC_FALSE );
134     add_string( "pvr-radio-device", "/dev/radio0", NULL, RADIO_DEVICE_TEXT,
135                  RADIO_DEVICE_LONGTEXT, VLC_FALSE );
136     add_integer( "pvr-norm", V4L2_STD_UNKNOWN , NULL, NORM_TEXT,
137                  NORM_LONGTEXT, VLC_FALSE );
138        change_integer_list( i_norm_list, psz_norm_list_text, 0 );
139     add_integer( "pvr-width", -1, NULL, WIDTH_TEXT, WIDTH_LONGTEXT, VLC_TRUE );
140     add_integer( "pvr-height", -1, NULL, HEIGHT_TEXT, HEIGHT_LONGTEXT,
141                  VLC_TRUE );
142     add_integer( "pvr-frequency", -1, NULL, FREQUENCY_TEXT, FREQUENCY_LONGTEXT,
143                  VLC_FALSE );
144     add_integer( "pvr-framerate", -1, NULL, FRAMERATE_TEXT, FRAMERATE_LONGTEXT,
145                  VLC_TRUE );
146     add_integer( "pvr-keyint", -1, NULL, KEYINT_TEXT, KEYINT_LONGTEXT,
147                  VLC_TRUE );
148     add_integer( "pvr-bframes", -1, NULL, FRAMERATE_TEXT, FRAMERATE_LONGTEXT,
149                  VLC_TRUE );
150     add_integer( "pvr-bitrate", -1, NULL, BITRATE_TEXT, BITRATE_LONGTEXT,
151                  VLC_FALSE );
152     add_integer( "pvr-bitrate-peak", -1, NULL, BITRATE_PEAK_TEXT,
153                  BITRATE_PEAK_LONGTEXT, VLC_TRUE );
154     add_integer( "pvr-bitrate-mode", -1, NULL, BITRATE_MODE_TEXT,
155                  BITRATE_MODE_LONGTEXT, VLC_TRUE );
156         change_integer_list( i_bitrates, psz_bitrates_list_text, 0 );
157     add_integer( "pvr-audio-bitmask", -1, NULL, BITMASK_TEXT,
158                  BITMASK_LONGTEXT, VLC_TRUE );
159     add_integer( "pvr-audio-volume", -1, NULL, VOLUME_TEXT,
160                  VOLUME_LONGTEXT, VLC_TRUE );
161     add_integer( "pvr-channel", -1, NULL, CHAN_TEXT, CHAN_LONGTEXT, VLC_TRUE );
162
163     set_callbacks( Open, Close );
164 vlc_module_end();
165
166 /*****************************************************************************
167  * Prototypes
168  *****************************************************************************/
169 static int Read   ( access_t *, uint8_t *, int );
170 static int Control( access_t *, int, va_list );
171
172 /* ivtv specific ioctls */
173 #define IVTV_IOC_G_CODEC    0xFFEE7703
174 #define IVTV_IOC_S_CODEC    0xFFEE7704
175
176 /* for use with IVTV_IOC_G_CODEC and IVTV_IOC_S_CODEC */
177
178 struct ivtv_ioctl_codec {
179     uint32_t aspect;
180     uint32_t audio_bitmask;
181     uint32_t bframes;
182     uint32_t bitrate_mode;
183     uint32_t bitrate;
184     uint32_t bitrate_peak;
185     uint32_t dnr_mode;
186     uint32_t dnr_spatial;
187     uint32_t dnr_temporal;
188     uint32_t dnr_type;
189     uint32_t framerate;
190     uint32_t framespergop;
191     uint32_t gop_closure;
192     uint32_t pulldown;
193     uint32_t stream_type;
194 };
195
196 struct access_sys_t
197 {
198     /* file descriptor */
199     int i_fd;
200     int i_radio_fd;
201
202     char *psz_videodev;
203     char *psz_radiodev;
204
205     /* options */
206     int i_standard;
207     int i_width;
208     int i_height;
209     int i_frequency;
210     int i_framerate;
211     int i_keyint;
212     int i_bframes;
213     int i_bitrate;
214     int i_bitrate_peak;
215     int i_bitrate_mode;
216     int i_audio_bitmask;
217     int i_input;
218     int i_volume;
219
220     /* driver version */
221     vlc_bool_t b_v4l2_api;
222 };
223
224 /*****************************************************************************
225  * ConfigureIVTV: set up codec parameters using the old ivtv api
226  *****************************************************************************/
227 static int ConfigureIVTV( access_t * p_access )
228 {
229     access_sys_t *p_sys = (access_sys_t *) p_access->p_sys;
230     struct ivtv_ioctl_codec codec;
231     int result;
232
233     memset( &codec, 0, sizeof(struct ivtv_ioctl_codec) );
234
235     result = ioctl( p_sys->i_fd, IVTV_IOC_G_CODEC, &codec );
236     if( result < 0 )
237     {
238         msg_Err( p_access, "Failed to read current capture card settings." );
239         return VLC_EGENERIC;
240     }
241
242     if( p_sys->i_framerate != -1 )
243     {
244         switch( p_sys->i_framerate )
245         {
246             case 30:
247                 codec.framerate = 0;
248                 break;
249
250             case 25:
251                 codec.framerate = 1;
252                 break;
253
254             default:
255                 msg_Warn( p_access, "Invalid framerate, reverting to 25." );
256                 codec.framerate = 1;
257                 break;
258         }
259     }
260
261     if( p_sys->i_bitrate != -1 )
262     {
263         codec.bitrate = p_sys->i_bitrate;
264     }
265
266     if( p_sys->i_bitrate_peak != -1 )
267     {
268         codec.bitrate_peak = p_sys->i_bitrate_peak;
269     }
270
271     if( p_sys->i_bitrate_mode != -1 )
272     {
273         codec.bitrate_mode = p_sys->i_bitrate_mode;
274     }
275
276     if( p_sys->i_audio_bitmask != -1 )
277     {
278         codec.audio_bitmask = p_sys->i_audio_bitmask;
279     }
280
281     if( p_sys->i_keyint != -1 )
282     {
283         codec.framespergop = p_sys->i_keyint;
284     }
285
286     if( p_sys->i_bframes != -1 )
287     {
288         codec.bframes = p_sys->i_bframes;
289     }
290
291     result = ioctl( p_sys->i_fd, IVTV_IOC_S_CODEC, &codec );
292     if( result  < 0 )
293     {
294         msg_Err( p_access, "Failed to write new capture card settings." );
295         return VLC_EGENERIC;
296     }
297
298     msg_Dbg( p_access, "Setting codec parameters to:  framerate: "
299                         "%d, bitrate: %d/%d/%d",
300                         codec.framerate, codec.bitrate,
301                         codec.bitrate_peak, codec.bitrate_mode );
302     return VLC_SUCCESS;
303 }
304
305 #ifdef HAVE_NEW_LINUX_VIDEODEV2_H
306
307 #define MAX_V4L2_CTRLS (6)
308 /*****************************************************************************
309  * AddV4L2Ctrl: adds a control to the v4l2 controls list
310  *****************************************************************************/
311 static void AddV4L2Ctrl( access_t * p_access,
312                          struct v4l2_ext_controls * p_controls,
313                          uint32_t i_id, uint32_t i_value )
314 {
315     if( p_controls->count >= MAX_V4L2_CTRLS )
316     {
317         msg_Err( p_access, "Tried to set too many v4l2 controls at once." );
318         return;
319     }
320
321     p_controls->controls[p_controls->count].id    = i_id;
322     p_controls->controls[p_controls->count].value = i_value;
323     p_controls->count++;
324 }
325
326 /*****************************************************************************
327  * V4L2SampleRate: calculate v4l2 sample rate from pvr-audio-bitmask
328  *****************************************************************************/
329 static uint32_t V4L2SampleRate( uint32_t i_bitmask )
330 {
331     switch( i_bitmask & 0x0003 )
332     {
333         case 0x0001: return V4L2_MPEG_AUDIO_SAMPLING_FREQ_48000;
334         case 0x0002: return V4L2_MPEG_AUDIO_SAMPLING_FREQ_32000;
335     }
336     return V4L2_MPEG_AUDIO_SAMPLING_FREQ_44100;
337 }
338
339 /*****************************************************************************
340  * V4L2AudioEncoding: calculate v4l2 audio encoding level from pvr-audio-bitmask
341  *****************************************************************************/
342 static uint32_t V4L2AudioEncoding( uint32_t i_bitmask )
343 {
344     switch( i_bitmask & 0x000c )
345     {
346         case 0x0004: return V4L2_MPEG_AUDIO_ENCODING_LAYER_1;
347         case 0x0008: return V4L2_MPEG_AUDIO_ENCODING_LAYER_2;
348     }
349     return 0xffffffff;
350 }
351
352 /*****************************************************************************
353  * V4L2AudioL1Bitrate: calculate v4l2 audio bitrate for layer-1 audio from pvr-audio-bitmask
354  *****************************************************************************/
355 static uint32_t V4L2AudioL1Bitrate( uint32_t i_bitmask )
356 {
357     switch( i_bitmask & 0x00f0 )
358     {
359         case 0x0010: return V4L2_MPEG_AUDIO_L1_BITRATE_32K;
360         case 0x0020: return V4L2_MPEG_AUDIO_L1_BITRATE_64K;
361         case 0x0030: return V4L2_MPEG_AUDIO_L1_BITRATE_96K;
362         case 0x0040: return V4L2_MPEG_AUDIO_L1_BITRATE_128K;
363         case 0x0050: return V4L2_MPEG_AUDIO_L1_BITRATE_160K;
364         case 0x0060: return V4L2_MPEG_AUDIO_L1_BITRATE_192K;
365         case 0x0070: return V4L2_MPEG_AUDIO_L1_BITRATE_224K;
366         case 0x0080: return V4L2_MPEG_AUDIO_L1_BITRATE_256K;
367         case 0x0090: return V4L2_MPEG_AUDIO_L1_BITRATE_288K;
368         case 0x00a0: return V4L2_MPEG_AUDIO_L1_BITRATE_320K;
369         case 0x00b0: return V4L2_MPEG_AUDIO_L1_BITRATE_352K;
370         case 0x00c0: return V4L2_MPEG_AUDIO_L1_BITRATE_384K;
371         case 0x00d0: return V4L2_MPEG_AUDIO_L1_BITRATE_416K;
372         case 0x00e0: return V4L2_MPEG_AUDIO_L1_BITRATE_448K;
373     }
374     return V4L2_MPEG_AUDIO_L1_BITRATE_320K;
375 }
376
377 /*****************************************************************************
378  * V4L2AudioL2Bitrate: calculate v4l2 audio bitrate for layer-1 audio from pvr-audio-bitmask
379  *****************************************************************************/
380 static uint32_t V4L2AudioL2Bitrate( uint32_t i_bitmask )
381 {
382     switch( i_bitmask & 0x00f0 )
383     {
384         case 0x0010: return V4L2_MPEG_AUDIO_L2_BITRATE_32K;
385         case 0x0020: return V4L2_MPEG_AUDIO_L2_BITRATE_48K;
386         case 0x0030: return V4L2_MPEG_AUDIO_L2_BITRATE_56K;
387         case 0x0040: return V4L2_MPEG_AUDIO_L2_BITRATE_64K;
388         case 0x0050: return V4L2_MPEG_AUDIO_L2_BITRATE_80K;
389         case 0x0060: return V4L2_MPEG_AUDIO_L2_BITRATE_96K;
390         case 0x0070: return V4L2_MPEG_AUDIO_L2_BITRATE_112K;
391         case 0x0080: return V4L2_MPEG_AUDIO_L2_BITRATE_128K;
392         case 0x0090: return V4L2_MPEG_AUDIO_L2_BITRATE_160K;
393         case 0x00a0: return V4L2_MPEG_AUDIO_L2_BITRATE_192K;
394         case 0x00b0: return V4L2_MPEG_AUDIO_L2_BITRATE_224K;
395         case 0x00c0: return V4L2_MPEG_AUDIO_L2_BITRATE_256K;
396         case 0x00d0: return V4L2_MPEG_AUDIO_L2_BITRATE_320K;
397         case 0x00e0: return V4L2_MPEG_AUDIO_L2_BITRATE_384K;
398     }
399     return V4L2_MPEG_AUDIO_L2_BITRATE_192K;
400 }
401
402 /*****************************************************************************
403  * V4L2AudioMode: calculate v4l2 audio mode from pvr-audio-bitmask
404  *****************************************************************************/
405 static uint32_t V4L2AudioMode( uint32_t i_bitmask )
406 {
407     switch( i_bitmask & 0x0300 )
408     {
409         case 0x0100: return V4L2_MPEG_AUDIO_MODE_JOINT_STEREO;
410         case 0x0200: return V4L2_MPEG_AUDIO_MODE_DUAL;
411         case 0x0300: return V4L2_MPEG_AUDIO_MODE_MONO;
412     }
413     return V4L2_MPEG_AUDIO_MODE_STEREO;
414 }
415
416 /*****************************************************************************
417  * ConfigureV4L2: set up codec parameters using the new v4l2 api
418  *****************************************************************************/
419 static int ConfigureV4L2( access_t * p_access )
420 {
421     access_sys_t *p_sys = (access_sys_t *) p_access->p_sys;
422     struct v4l2_ext_controls controls;
423     int result;
424
425     memset( &controls, 0, sizeof(struct v4l2_ext_controls) );
426     controls.ctrl_class  = V4L2_CTRL_CLASS_MPEG;
427     controls.error_idx   = 0;
428     controls.reserved[0] = 0;
429     controls.reserved[1] = 0;
430     controls.count       = 0;
431     controls.controls    = calloc( sizeof( struct v4l2_ext_control ),
432                                    MAX_V4L2_CTRLS );
433
434     if( controls.controls == NULL )
435         return VLC_ENOMEM;
436
437     /* Note: Ignore frame rate.  Doesn't look like it can be changed. */
438     if( p_sys->i_bitrate != -1 )
439     {
440         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_VIDEO_BITRATE,
441                      p_sys->i_bitrate );
442         msg_Dbg( p_access, "Setting [%u] bitrate = %u",
443                  controls.count - 1, p_sys->i_bitrate );
444     }
445
446     if( p_sys->i_bitrate_peak != -1 )
447     {
448         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_VIDEO_BITRATE_PEAK,
449                      p_sys->i_bitrate_peak );
450         msg_Dbg( p_access, "Setting [%u] bitrate_peak = %u",
451                  controls.count - 1, p_sys->i_bitrate_peak );
452     }
453
454     if( p_sys->i_bitrate_mode != -1 )
455     {
456         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_VIDEO_BITRATE_MODE,
457                      p_sys->i_bitrate_mode );
458         msg_Dbg( p_access, "Setting [%u] bitrate_mode = %u",
459                  controls.count - 1, p_sys->i_bitrate_mode );
460     }
461
462     if( p_sys->i_audio_bitmask != -1 )
463     {
464         /* Sample rate */
465         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_AUDIO_SAMPLING_FREQ,
466                     V4L2SampleRate( p_sys->i_audio_bitmask ) );
467
468         /* Encoding layer and bitrate */
469         switch( V4L2AudioEncoding( p_sys->i_audio_bitmask ) )
470         {
471             case V4L2_MPEG_AUDIO_ENCODING_LAYER_1:
472                  AddV4L2Ctrl( p_access, &controls,
473                               V4L2_CID_MPEG_AUDIO_SAMPLING_FREQ,
474                               V4L2_MPEG_AUDIO_ENCODING_LAYER_1 );
475                  AddV4L2Ctrl( p_access, &controls,
476                               V4L2_CID_MPEG_AUDIO_L1_BITRATE,
477                               V4L2AudioL1Bitrate( p_sys->i_audio_bitmask ) );
478                  break;
479
480             case V4L2_MPEG_AUDIO_ENCODING_LAYER_2:
481                  AddV4L2Ctrl( p_access, &controls,
482                               V4L2_CID_MPEG_AUDIO_SAMPLING_FREQ,
483                               V4L2_MPEG_AUDIO_ENCODING_LAYER_2 );
484                  AddV4L2Ctrl( p_access, &controls,
485                               V4L2_CID_MPEG_AUDIO_L2_BITRATE,
486                               V4L2AudioL2Bitrate( p_sys->i_audio_bitmask ) );
487                  break;
488         }
489
490         /* Audio mode - stereo or mono */
491         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_AUDIO_MODE,
492                      V4L2AudioMode( p_sys->i_audio_bitmask ) );
493
494         /* See if the user wants any other audio feature */
495         if( ( p_sys->i_audio_bitmask & 0x1ff00 ) != 0 )
496         {
497             /* It would be possible to support the bits that represent:
498              *   V4L2_CID_MPEG_AUDIO_MODE_EXTENSION
499              *   V4L2_CID_MPEG_AUDIO_EMPHASIS
500              *   V4L2_CID_MPEG_AUDIO_CRC
501              * but they are not currently used.  Tell the user.
502              */
503             msg_Err( p_access, "There were bits in pvr-audio-bitmask that were not used.");
504         }
505         msg_Dbg( p_access, "Setting audio controls");
506     }
507
508     if( p_sys->i_keyint != -1 )
509     {
510         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_VIDEO_GOP_SIZE,
511                      p_sys->i_keyint );
512         msg_Dbg( p_access, "Setting [%u] keyint = %u",
513                  controls.count - 1, p_sys->i_keyint );
514     }
515
516     if( p_sys->i_bframes != -1 )
517     {
518         AddV4L2Ctrl( p_access, &controls, V4L2_CID_MPEG_VIDEO_B_FRAMES,
519                      p_sys->i_bframes );
520         msg_Dbg( p_access, "Setting [%u] bframes = %u",
521                  controls.count - 1, p_sys->i_bframes );
522     }
523
524     result = ioctl( p_sys->i_fd, VIDIOC_S_EXT_CTRLS, &controls );
525     if( result < 0 )
526     {
527         msg_Err( p_access, "Failed to write %u new capture card settings.",
528                             controls.error_idx );
529     }
530     free( controls.controls );
531     return VLC_SUCCESS;
532 }
533
534 #endif /* HAVE_NEW_LINUX_VIDEODEV2_H */
535
536 /*****************************************************************************
537  * Open: open the device
538  *****************************************************************************/
539 static int Open( vlc_object_t * p_this )
540 {
541     access_t *p_access = (access_t*) p_this;
542     access_sys_t * p_sys;
543     char * psz_tofree;
544     char * psz_parser;
545     char * psz_device = NULL;
546     vlc_value_t val;
547     struct v4l2_capability device_capability;
548     int result;
549
550     memset( &device_capability, 0, sizeof(struct v4l2_capability) );
551
552     p_access->pf_read = Read;
553     p_access->pf_block = NULL;
554     p_access->pf_seek = NULL;
555     p_access->pf_control = Control;
556     p_access->info.i_update = 0;
557     p_access->info.i_size = 0;
558     p_access->info.i_pos = 0;
559     p_access->info.b_eof = VLC_FALSE;
560     p_access->info.i_title = 0;
561     p_access->info.i_seekpoint = 0;
562
563     /* create private access data */
564     p_sys = calloc( sizeof( access_sys_t ), 1 );
565     if( !p_sys )
566         return VLC_ENOMEM;
567
568     p_access->p_sys = p_sys;
569
570     /* defaults values */
571     var_Create( p_access, "pvr-caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
572
573     var_Create( p_access, "pvr-device", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
574     var_Get( p_access, "pvr-device" , &val);
575     p_sys->psz_videodev = val.psz_string;
576
577     var_Create( p_access, "pvr-radio-device", VLC_VAR_STRING |
578                                               VLC_VAR_DOINHERIT );
579     var_Get( p_access, "pvr-radio-device" , &val);
580     p_sys->psz_radiodev = val.psz_string;
581
582     var_Create( p_access, "pvr-norm", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
583     var_Get( p_access, "pvr-norm" , &val);
584     p_sys->i_standard = val.i_int;
585
586     var_Create( p_access, "pvr-width", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
587     var_Get( p_access, "pvr-width" , &val);
588     p_sys->i_width = val.i_int;
589
590     var_Create( p_access, "pvr-height", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
591     var_Get( p_access, "pvr-height" , &val);
592     p_sys->i_height = val.i_int;
593
594     var_Create( p_access, "pvr-frequency", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
595     var_Get( p_access, "pvr-frequency" , &val);
596     p_sys->i_frequency = val.i_int;
597
598     var_Create( p_access, "pvr-framerate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
599     var_Get( p_access, "pvr-framerate" , &val);
600     p_sys->i_framerate = val.i_int;
601
602     var_Create( p_access, "pvr-keyint", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
603     var_Get( p_access, "pvr-keyint" , &val);
604     p_sys->i_keyint = val.i_int;
605
606     var_Create( p_access, "pvr-bframes", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
607     var_Get( p_access, "pvr-bframes" , &val);
608     p_sys->i_bframes = val.b_bool;
609
610     var_Create( p_access, "pvr-bitrate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
611     var_Get( p_access, "pvr-bitrate" , &val);
612     p_sys->i_bitrate = val.i_int;
613
614     var_Create( p_access, "pvr-bitrate-peak", VLC_VAR_INTEGER |
615                                               VLC_VAR_DOINHERIT );
616     var_Get( p_access, "pvr-bitrate-peak" , &val);
617     p_sys->i_bitrate_peak = val.i_int;
618
619     var_Create( p_access, "pvr-bitrate-mode", VLC_VAR_INTEGER |
620                                               VLC_VAR_DOINHERIT );
621     var_Get( p_access, "pvr-bitrate-mode" , &val);
622     p_sys->i_bitrate_mode = val.i_int;
623
624     var_Create( p_access, "pvr-audio-bitmask", VLC_VAR_INTEGER |
625                                               VLC_VAR_DOINHERIT );
626     var_Get( p_access, "pvr-audio-bitmask" , &val);
627     p_sys->i_audio_bitmask = val.i_int;
628
629     var_Create( p_access, "pvr-audio-volume", VLC_VAR_INTEGER |
630                                               VLC_VAR_DOINHERIT );
631     var_Get( p_access, "pvr-audio-volume" , &val);
632     p_sys->i_volume = val.i_int;
633
634     var_Create( p_access, "pvr-channel", VLC_VAR_INTEGER |
635                                               VLC_VAR_DOINHERIT );
636     var_Get( p_access, "pvr-channel" , &val);
637     p_sys->i_input = val.i_int;
638
639     /* parse command line options */
640     psz_tofree = strdup( p_access->psz_path );
641     if( !psz_tofree )
642         return VLC_ENOMEM;
643
644     psz_parser = psz_tofree;
645     if( *psz_parser )
646     {
647         for( ;; )
648         {
649             if ( !strncmp( psz_parser, "norm=", strlen( "norm=" ) ) )
650             {
651                 char *psz_parser_init;
652                 psz_parser += strlen( "norm=" );
653                 psz_parser_init = psz_parser;
654                 while ( (*psz_parser != ':')
655                         && (*psz_parser != ',')
656                         && (*psz_parser != '\0') )
657                 {
658                     psz_parser++;
659                 }
660
661                 if ( !strncmp( psz_parser_init, "secam" ,
662                                psz_parser - psz_parser_init ) )
663                 {
664                     p_sys->i_standard = V4L2_STD_SECAM;
665                 }
666                 else if ( !strncmp( psz_parser_init, "pal" ,
667                                     psz_parser - psz_parser_init ) )
668                 {
669                     p_sys->i_standard = V4L2_STD_PAL;
670                 }
671                 else if ( !strncmp( psz_parser_init, "ntsc" ,
672                                     psz_parser - psz_parser_init ) )
673                 {
674                     p_sys->i_standard = V4L2_STD_NTSC;
675                 }
676                 else
677                 {
678                     p_sys->i_standard = strtol( psz_parser_init ,
679                                                 &psz_parser, 0 );
680                 }
681             }
682             else if( !strncmp( psz_parser, "channel=",
683                                strlen( "channel=" ) ) )
684             {
685                 p_sys->i_input =
686                     strtol( psz_parser + strlen( "channel=" ),
687                             &psz_parser, 0 );
688             }
689             else if( !strncmp( psz_parser, "device=", strlen( "device=" ) ) )
690             {
691                 int i_len = strlen( "/dev/videox" );
692                 psz_device = calloc( i_len  + 1, 1 );
693                 if( !psz_device )
694                     return VLC_ENOMEM;
695
696                 snprintf( psz_device, i_len, "/dev/video%ld",
697                             strtol( psz_parser + strlen( "device=" ),
698                             &psz_parser, 0 ) );
699             }
700             else if( !strncmp( psz_parser, "frequency=",
701                                strlen( "frequency=" ) ) )
702             {
703                 p_sys->i_frequency =
704                     strtol( psz_parser + strlen( "frequency=" ),
705                             &psz_parser, 0 );
706             }
707             else if( !strncmp( psz_parser, "framerate=",
708                                strlen( "framerate=" ) ) )
709             {
710                 p_sys->i_framerate =
711                     strtol( psz_parser + strlen( "framerate=" ),
712                             &psz_parser, 0 );
713             }
714             else if( !strncmp( psz_parser, "keyint=",
715                                strlen( "keyint=" ) ) )
716             {
717                 p_sys->i_keyint =
718                     strtol( psz_parser + strlen( "keyint=" ),
719                             &psz_parser, 0 );
720             }
721             else if( !strncmp( psz_parser, "bframes=",
722                                strlen( "bframes=" ) ) )
723             {
724                 p_sys->i_bframes =
725                     strtol( psz_parser + strlen( "bframes=" ),
726                             &psz_parser, 0 );
727             }
728
729             else if( !strncmp( psz_parser, "width=",
730                                strlen( "width=" ) ) )
731             {
732                 p_sys->i_width =
733                     strtol( psz_parser + strlen( "width=" ),
734                             &psz_parser, 0 );
735             }
736             else if( !strncmp( psz_parser, "height=",
737                                strlen( "height=" ) ) )
738             {
739                 p_sys->i_height =
740                     strtol( psz_parser + strlen( "height=" ),
741                             &psz_parser, 0 );
742             }
743             else if( !strncmp( psz_parser, "audio=",
744                                strlen( "audio=" ) ) )
745             {
746                 p_sys->i_audio_bitmask =
747                     strtol( psz_parser + strlen( "audio=" ),
748                             &psz_parser, 0 );
749             }
750             else if( !strncmp( psz_parser, "bitrate=",
751                                strlen( "bitrate=" ) ) )
752             {
753                 p_sys->i_bitrate =
754                     strtol( psz_parser + strlen( "bitrate=" ),
755                             &psz_parser, 0 );
756             }
757             else if( !strncmp( psz_parser, "maxbitrate=",
758                                strlen( "maxbitrate=" ) ) )
759             {
760                 p_sys->i_bitrate_peak =
761                     strtol( psz_parser + strlen( "maxbitrate=" ),
762                             &psz_parser, 0 );
763             }
764             else if( !strncmp( psz_parser, "bitratemode=",
765                                strlen( "bitratemode=" ) ) )
766             {
767                 char *psz_parser_init;
768                 psz_parser += strlen( "bitratemode=" );
769                 psz_parser_init = psz_parser;
770                 while ( (*psz_parser != ':')
771                         && (*psz_parser != ',')
772                         && (*psz_parser != '\0') )
773                 {
774                     psz_parser++;
775                 }
776
777                 if ( !strncmp( psz_parser_init, "vbr" ,
778                                psz_parser - psz_parser_init ) )
779                 {
780                      p_sys->i_bitrate_mode = 0;
781                 }
782                 else if ( !strncmp( psz_parser_init, "cbr" ,
783                                     psz_parser - psz_parser_init ) )
784                 {
785                     p_sys->i_bitrate_mode = 1;
786                 }
787             }
788             else if( !strncmp( psz_parser, "size=",
789                                strlen( "size=" ) ) )
790             {
791                 p_sys->i_width =
792                     strtol( psz_parser + strlen( "size=" ),
793                             &psz_parser, 0 );
794                 p_sys->i_height =
795                     strtol( psz_parser + 1 ,
796                             &psz_parser, 0 );
797             }
798             else
799             {
800                 char *psz_parser_init;
801                 psz_parser_init = psz_parser;
802                 while ( (*psz_parser != ':') &&
803                         (*psz_parser != ',') &&
804                         (*psz_parser != '\0') )
805                 {
806                     psz_parser++;
807                 }
808                 psz_device = calloc( psz_parser - psz_parser_init + 1, 1 );
809                 if( !psz_device )
810                     return VLC_ENOMEM;
811
812                 strncpy( psz_device, psz_parser_init,
813                          psz_parser - psz_parser_init );
814             }
815             if( *psz_parser )
816                 psz_parser++;
817             else
818                 break;
819         }
820     }
821     free( psz_tofree );
822
823     if( psz_device )
824     {
825         if( p_sys->psz_videodev )
826             free( p_sys->psz_videodev );
827         p_sys->psz_videodev = psz_device;
828     }
829
830     /* open the device */
831     p_sys->i_fd = open( p_sys->psz_videodev, O_RDWR );
832     if( p_sys->i_fd < 0 )
833     {
834         msg_Err( p_access, "Cannot open device (%m)." );
835         Close( VLC_OBJECT(p_access) );
836         return VLC_EGENERIC;
837     }
838     msg_Dbg( p_access, "Using video device: %s.", p_sys->psz_videodev);
839
840     /* See what version of ivtvdriver is running */
841     result = ioctl( p_sys->i_fd, VIDIOC_QUERYCAP, &device_capability );
842     if( result < 0 )
843     {
844         msg_Err( p_access, "unknown ivtv driver version in use" );
845         Close( VLC_OBJECT(p_access) );
846         return VLC_EGENERIC;
847     }
848
849     msg_Dbg( p_access, "ivtv driver version %02x.%02x.%02x",
850             ( device_capability.version >> 16 ) & 0xff,
851             ( device_capability.version >>  8 ) & 0xff,
852             ( device_capability.version       ) & 0xff);
853
854     if ( device_capability.version >= 0x000800 )
855     {
856         /* Drivers > 0.8.0 use v4l2 API instead of IVTV ioctls */
857         msg_Dbg( p_access, "this driver uses the v4l2 API" );
858         p_sys->b_v4l2_api = VLC_TRUE;
859     }
860     else
861     {
862         p_sys->b_v4l2_api = VLC_FALSE;
863     }
864
865     /* set the input */
866     if ( p_sys->i_input != -1 )
867     {
868         result = ioctl( p_sys->i_fd, VIDIOC_S_INPUT, &p_sys->i_input );
869         if ( result < 0 )
870             msg_Warn( p_access, "Failed to select the requested input pin." );
871         else
872             msg_Dbg( p_access, "input set to: %d", p_sys->i_input );
873     }
874
875     /* set the video standard */
876     if ( p_sys->i_standard != V4L2_STD_UNKNOWN )
877     {
878         result = ioctl( p_sys->i_fd, VIDIOC_S_STD, &p_sys->i_standard );
879         if ( result  < 0 )
880             msg_Warn( p_access, "Failed to set the requested video standard." );
881         else
882             msg_Dbg( p_access, "video standard set to: %x",
883                      p_sys->i_standard);
884     }
885
886     /* set the picture size */
887     if ( (p_sys->i_width != -1) || (p_sys->i_height != -1) )
888     {
889         struct v4l2_format vfmt;
890
891         memset( &vfmt, 0, sizeof(struct v4l2_format) );
892         vfmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
893
894         result = ioctl( p_sys->i_fd, VIDIOC_G_FMT, &vfmt );
895         if ( result < 0 )
896         {
897             msg_Warn( p_access, "Failed to read current picture size." );
898         }
899         else
900         {
901             if ( p_sys->i_width != -1 )
902             {
903                 vfmt.fmt.pix.width = p_sys->i_width;
904             }
905
906             if ( p_sys->i_height != -1 )
907             {
908                 vfmt.fmt.pix.height = p_sys->i_height;
909             }
910
911             result = ioctl( p_sys->i_fd, VIDIOC_S_FMT, &vfmt );
912             if ( result < 0 )
913             {
914                 msg_Warn( p_access, "Failed to set requested picture size." );
915             }
916             else
917             {
918                 msg_Dbg( p_access, "picture size set to: %dx%d",
919                          vfmt.fmt.pix.width, vfmt.fmt.pix.height );
920             }
921         }
922     }
923
924     /* set the frequency */
925     if ( p_sys->i_frequency != -1 )
926     {
927         int i_fd;
928         struct v4l2_tuner vt;
929
930          /* TODO: let the user choose the tuner */
931         memset( &vt, 0, sizeof(struct v4l2_tuner) );
932
933         if ( (p_sys->i_frequency >= pi_radio_range[0])
934               && (p_sys->i_frequency <= pi_radio_range[1]) )
935         {
936             p_sys->i_radio_fd = open( p_sys->psz_radiodev, O_RDWR );
937             if( p_sys->i_radio_fd < 0 )
938             {
939                 msg_Err( p_access, "Cannot open radio device (%m)." );
940                 Close( VLC_OBJECT(p_access) );
941                 return VLC_EGENERIC;
942             }
943             msg_Dbg( p_access, "using radio device: %s",
944                      p_sys->psz_radiodev );
945             i_fd = p_sys->i_radio_fd;
946         }
947         else
948         {
949             i_fd = p_sys->i_fd;
950             p_sys->i_radio_fd = -1;
951         }
952
953         result = ioctl( i_fd, VIDIOC_G_TUNER, &vt );
954         if ( result < 0 )
955         {
956             msg_Warn( p_access, "Failed to read tuner information (%m)." );
957         }
958         else
959         {
960             struct v4l2_frequency vf;
961
962             memset( &vf, 0, sizeof(struct v4l2_frequency) );
963             vf.tuner = vt.index;
964
965             result = ioctl( i_fd, VIDIOC_G_FREQUENCY, &vf );
966             if ( result < 0 )
967             {
968                 msg_Warn( p_access, "Failed to read tuner frequency (%m)." );
969             }
970             else
971             {
972                 if( vt.capability & V4L2_TUNER_CAP_LOW )
973                     vf.frequency = p_sys->i_frequency * 16;
974                 else
975                     vf.frequency = (p_sys->i_frequency * 16 + 500) / 1000;
976
977                 result = ioctl( i_fd, VIDIOC_S_FREQUENCY, &vf );
978                 if( result < 0 )
979                 {
980                     msg_Warn( p_access, "Failed to set tuner frequency (%m)." );
981                 }
982                 else
983                 {
984                     msg_Dbg( p_access, "tuner frequency set to: %d",
985                              p_sys->i_frequency );
986                 }
987             }
988         }
989     }
990
991     /* control parameters */
992     if ( p_sys->i_volume != -1 )
993     {
994         struct v4l2_control ctrl;
995
996         memset( &ctrl, 0, sizeof(struct v4l2_control) );
997         ctrl.id = V4L2_CID_AUDIO_VOLUME;
998         ctrl.value = p_sys->i_volume;
999
1000         result = ioctl( p_sys->i_fd, VIDIOC_S_CTRL, &ctrl );
1001         if ( result < 0 )
1002         {
1003             msg_Warn( p_access, "Failed to set the volume." );
1004         }
1005     }
1006
1007     /* codec parameters */
1008     if ( (p_sys->i_framerate != -1)
1009             || (p_sys->i_bitrate_mode != -1)
1010             || (p_sys->i_bitrate_peak != -1)
1011             || (p_sys->i_keyint != -1)
1012             || (p_sys->i_bframes != -1)
1013             || (p_sys->i_bitrate != -1)
1014             || (p_sys->i_audio_bitmask != -1) )
1015     {
1016         if( p_sys->b_v4l2_api )
1017         {
1018 #ifdef HAVE_NEW_LINUX_VIDEODEV2_H
1019             result = ConfigureV4L2( p_access );
1020             if( result != VLC_SUCCESS )
1021             {
1022                 Close( VLC_OBJECT(p_access) );
1023                 return result;
1024             }
1025 #else
1026             msg_Warn( p_access, "You have new ivtvdrivers, "
1027                       "but this vlc was built against an old v4l2 version." );
1028 #endif
1029         }
1030         else
1031         {
1032             result = ConfigureIVTV( p_access );
1033             if( result != VLC_SUCCESS )
1034             {
1035                 Close( VLC_OBJECT(p_access) );
1036                 return result;
1037             }
1038         }
1039     }
1040
1041     return VLC_SUCCESS;
1042 }
1043
1044 /*****************************************************************************
1045  * Close: close the device
1046  *****************************************************************************/
1047 static void Close( vlc_object_t * p_this )
1048 {
1049     access_t *p_access = (access_t*) p_this;
1050     access_sys_t *p_sys = (access_sys_t *) p_access->p_sys;
1051
1052     if ( p_sys->i_fd != -1 )
1053         close( p_sys->i_fd );
1054     if ( p_sys->i_radio_fd != -1 )
1055         close( p_sys->i_radio_fd );
1056     if ( p_sys->psz_videodev )
1057         free( p_sys->psz_videodev );
1058     if ( p_sys->psz_radiodev )
1059         free( p_sys->psz_radiodev );
1060     free( p_sys );
1061 }
1062
1063 /*****************************************************************************
1064  * Read
1065  *****************************************************************************/
1066 static int Read( access_t * p_access, uint8_t * p_buffer, int i_len )
1067 {
1068     access_sys_t *p_sys = (access_sys_t *) p_access->p_sys;
1069     struct pollfd ufd;
1070     int i_ret;
1071
1072     ufd.fd = p_sys->i_fd;
1073     ufd.events = POLLIN;
1074
1075     if( p_access->info.b_eof )
1076         return 0;
1077
1078     do
1079     {
1080         if( p_access->b_die )
1081             return 0;
1082
1083         ufd.revents = 0;
1084     }
1085     while( ( i_ret = poll( &ufd, 1, 500 ) ) == 0 );
1086
1087     if( i_ret < 0 )
1088     {
1089         msg_Err( p_access, "Polling error (%m)." );
1090         return -1;
1091     }
1092
1093     i_ret = read( p_sys->i_fd, p_buffer, i_len );
1094     if( i_ret == 0 )
1095     {
1096         p_access->info.b_eof = VLC_TRUE;
1097     }
1098     else if( i_ret > 0 )
1099     {
1100         p_access->info.i_pos += i_ret;
1101     }
1102
1103     return i_ret;
1104 }
1105
1106 /*****************************************************************************
1107  * Control
1108  *****************************************************************************/
1109 static int Control( access_t *p_access, int i_query, va_list args )
1110 {
1111     vlc_bool_t   *pb_bool;
1112     int          *pi_int;
1113     int64_t      *pi_64;
1114
1115     switch( i_query )
1116     {
1117         /* */
1118         case ACCESS_CAN_SEEK:
1119         case ACCESS_CAN_FASTSEEK:
1120             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
1121             *pb_bool = VLC_FALSE;
1122             break;
1123         case ACCESS_CAN_PAUSE:
1124             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
1125             *pb_bool = VLC_FALSE;
1126             break;
1127         case ACCESS_CAN_CONTROL_PACE:
1128             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
1129             *pb_bool = VLC_FALSE;
1130             break;
1131
1132         /* */
1133         case ACCESS_GET_MTU:
1134             pi_int = (int*)va_arg( args, int * );
1135             *pi_int = 0;
1136             break;
1137
1138         case ACCESS_GET_PTS_DELAY:
1139             pi_64 = (int64_t*)va_arg( args, int64_t * );
1140             *pi_64 = (int64_t)var_GetInteger( p_access, "pvr-caching" ) * 1000;
1141             break;
1142
1143         /* */
1144         case ACCESS_SET_PAUSE_STATE:
1145             /* Nothing to do */
1146             break;
1147
1148         case ACCESS_GET_TITLE_INFO:
1149         case ACCESS_SET_TITLE:
1150         case ACCESS_SET_SEEKPOINT:
1151         case ACCESS_SET_PRIVATE_ID_STATE:
1152         case ACCESS_GET_CONTENT_TYPE:
1153             return VLC_EGENERIC;
1154
1155         default:
1156             msg_Warn( p_access, "Unimplemented query in control." );
1157             return VLC_EGENERIC;
1158
1159     }
1160     return VLC_SUCCESS;
1161 }