]> git.sesse.net Git - vlc/blob - modules/access/pvr/pvr.c
* modules/access/pvr/pvr.c: Fixed width/height selection with latest ivtv
[vlc] / modules / access / pvr / pvr.c
1 /*****************************************************************************
2  * pvr.c
3  *****************************************************************************
4  * Copyright (C) 2001, 2002 VideoLAN
5  * $Id$
6  *
7  * Authors: Eric Petit <titer@videolan.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #include <vlc/vlc.h>
28 #include <vlc/input.h>
29
30 #include <stdlib.h>
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 "videodev2.h"
39
40 /*****************************************************************************
41  * Module descriptor
42  *****************************************************************************/
43 static int  Open ( vlc_object_t * );
44 static void Close( vlc_object_t * );
45
46 #define CACHING_TEXT N_("Caching value in ms")
47 #define CACHING_LONGTEXT N_( \
48     "Allows you to modify the default caching value for pvr streams. This " \
49     "value should be set in millisecond units." )
50
51 #define DEVICE_TEXT N_( "Device" )
52 #define DEVICE_LONGTEXT N_( "PVR video device" )
53
54 #define NORM_TEXT N_( "Norm" )
55 #define NORM_LONGTEXT N_( "Defines the norm of the stream " \
56     "(Automatic, SECAM, PAL, or NTSC)" )
57
58 #define WIDTH_TEXT N_( "Width" )
59 #define WIDTH_LONGTEXT N_( "Width of the stream to capture " \
60     "(-1 for autodetect)" )
61
62 #define HEIGHT_TEXT N_( "Height" )
63 #define HEIGHT_LONGTEXT N_( "Height of the stream to capture " \
64     "(-1 for autodetect)" )
65
66 #define FREQUENCY_TEXT N_( "Frequency" )
67 #define FREQUENCY_LONGTEXT N_( "Frequency to capture (in kHz), if applicable" )
68
69 #define FRAMERATE_TEXT N_( "Framerate" )
70 #define FRAMERATE_LONGTEXT N_( "Framerate to capture, if applicable " \
71     "(-1 for auto)" )
72
73 #define KEYINT_TEXT N_( "Key interval" )
74 #define KEYINT_LONGTEXT N_( "Interval between keyframes (-1 for auto)" )
75
76 #define BFRAMES_TEXT N_( "B Frames" )
77 #define BFRAMES_LONGTEXT N_("If this option is set, B-Frames will be used. " \
78     "Use this option to set the number of B-Frames.")
79
80 #define BITRATE_TEXT N_( "Bitrate" )
81 #define BITRATE_LONGTEXT N_( "Bitrate to use (-1 for default)" )
82
83 #define BITRATE_PEAK_TEXT N_( "Bitrate peak" )
84 #define BITRATE_PEAK_LONGTEXT N_( "Peak bitrate in VBR mode" )
85
86 #define BITRATE_MODE_TEXT N_( "Bitrate mode (vbr or cbr)" )
87 #define BITRATE_MODE_LONGTEXT N_( "Bitrate mode to use" )
88
89 #define BITMASK_TEXT N_( "Audio bitmask" )
90 #define BITMASK_LONGTEXT N_("This option allows setting of bitmask that will "\
91     "get used by the audio part of the card." )
92
93 #define CHAN_TEXT N_( "Channel" )
94 #define CHAN_LONGTEXT N_( "Channel of the card to use (Usually, 0 = tuner, " \
95     "1 = composite, 2 = svideo)" )
96
97 #define CACHING_TEXT N_("Caching value in ms")
98 #define CACHING_LONGTEXT N_( \
99     "Allows you to modify the default caching value for pvr streams. This " \
100     "value should be set in millisecond units." )
101
102 static int i_norm_list[] =
103     { V4L2_STD_UNKNOWN, V4L2_STD_SECAM, V4L2_STD_PAL, V4L2_STD_NTSC };
104 static char *psz_norm_list_text[] =
105     { N_("Automatic"), N_("SECAM"), N_("PAL"),  N_("NTSC") };
106
107 static int i_bitrates[] = { 0, 1 };
108 static char *psz_bitrates_list_text[] = { N_("vbr"), N_("cbr") };
109
110 vlc_module_begin();
111     set_shortname( _("PVR") );
112     set_description( _("MPEG Encoding cards input (with ivtv drivers)") );
113     set_category( CAT_INPUT );
114     set_subcategory( SUBCAT_INPUT_ACCESS );
115     set_capability( "access2", 0 );
116     add_shortcut( "pvr" );
117
118     add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE );
119     add_string( "pvr-device", "/dev/video0", NULL, DEVICE_TEXT,
120                  DEVICE_LONGTEXT, VLC_FALSE );
121     add_integer( "pvr-norm", V4L2_STD_UNKNOWN , NULL, NORM_TEXT,
122                  NORM_LONGTEXT, VLC_FALSE );
123        change_integer_list( i_norm_list, psz_norm_list_text, 0 );
124     add_integer( "pvr-width", -1, NULL, WIDTH_TEXT, WIDTH_LONGTEXT, VLC_TRUE );
125     add_integer( "pvr-height", -1, NULL, HEIGHT_TEXT, HEIGHT_LONGTEXT,
126                  VLC_TRUE );
127     add_integer( "pvr-frequency", -1, NULL, FREQUENCY_TEXT, FREQUENCY_LONGTEXT,
128                  VLC_FALSE );
129     add_integer( "pvr-framerate", -1, NULL, FRAMERATE_TEXT, FRAMERATE_LONGTEXT,
130                  VLC_TRUE );
131     add_integer( "pvr-keyint", -1, NULL, KEYINT_TEXT, KEYINT_LONGTEXT,
132                  VLC_TRUE );
133     add_integer( "pvr-bframes", -1, NULL, FRAMERATE_TEXT, FRAMERATE_LONGTEXT,
134                  VLC_TRUE );
135     add_integer( "pvr-bitrate", -1, NULL, BITRATE_TEXT, BITRATE_LONGTEXT,
136                  VLC_FALSE );
137     add_integer( "pvr-bitrate-peak", -1, NULL, BITRATE_PEAK_TEXT,
138                  BITRATE_PEAK_LONGTEXT, VLC_TRUE );
139     add_integer( "pvr-bitrate-mode", -1, NULL, BITRATE_MODE_TEXT,
140                  BITRATE_MODE_LONGTEXT, VLC_TRUE );
141         change_integer_list( i_bitrates, psz_bitrates_list_text, 0 );
142     add_integer( "pvr-audio-bitmask", -1, NULL, BITMASK_TEXT,
143                  BITMASK_LONGTEXT, VLC_TRUE );
144     add_integer( "pvr-channel", -1, NULL, CHAN_TEXT, CHAN_LONGTEXT, VLC_TRUE );
145     add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL,
146                  CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE );
147
148     set_callbacks( Open, Close );
149 vlc_module_end();
150
151 /*****************************************************************************
152  * Prototypes
153  *****************************************************************************/
154 static int Read   ( access_t *, uint8_t *, int );
155 static int Control( access_t *, int, va_list );
156
157 /* ivtv specific ioctls */
158 #define IVTV_IOC_G_CODEC    0xFFEE7703
159 #define IVTV_IOC_S_CODEC    0xFFEE7704
160
161 /* for use with IVTV_IOC_G_CODEC and IVTV_IOC_S_CODEC */
162
163 struct ivtv_ioctl_codec {
164         uint32_t aspect;
165         uint32_t audio_bitmask;
166         uint32_t bframes;
167         uint32_t bitrate_mode;
168         uint32_t bitrate;
169         uint32_t bitrate_peak;
170         uint32_t dnr_mode;
171         uint32_t dnr_spatial;
172         uint32_t dnr_temporal;
173         uint32_t dnr_type;
174         uint32_t framerate;
175         uint32_t framespergop;
176         uint32_t gop_closure;
177         uint32_t pulldown;
178         uint32_t stream_type;
179 };
180
181 struct access_sys_t
182 {
183     /* file descriptor */
184     int i_fd;
185
186     /* options */
187     int i_standard;
188     int i_width;
189     int i_height;
190     int i_frequency;
191     int i_framerate;
192     int i_keyint;
193     int i_bframes;
194     int i_bitrate;
195     int i_bitrate_peak;
196     int i_bitrate_mode;
197     int i_audio_bitmask;
198     int i_input;
199 };
200
201 /*****************************************************************************
202  * Open: open the device
203  *****************************************************************************/
204 static int Open( vlc_object_t * p_this )
205 {
206     access_t *p_access = (access_t*) p_this;
207     access_sys_t * p_sys;
208     char * psz_tofree, * psz_parser, * psz_device;
209     vlc_value_t val;
210
211     struct v4l2_format vfmt;
212     struct v4l2_frequency vf;
213     struct ivtv_ioctl_codec codec;
214
215     //psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 );
216
217     p_access->pf_read = Read;
218     p_access->pf_block = NULL;
219     p_access->pf_seek = NULL;
220     p_access->pf_control = Control;
221     p_access->info.i_update = 0;
222     p_access->info.i_size = 0;
223     p_access->info.i_pos = 0;
224     p_access->info.b_eof = VLC_FALSE;
225     p_access->info.i_title = 0;
226     p_access->info.i_seekpoint = 0;
227
228     /* create private access data */
229     p_sys = calloc( sizeof( access_sys_t ), 1 );
230     p_access->p_sys = p_sys;
231
232     /* defaults values */
233     var_Create( p_access, "pvr-caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
234
235     var_Create( p_access, "pvr-device", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
236     var_Get( p_access, "pvr-device" , &val);
237     psz_device = val.psz_string;
238
239     var_Create( p_access, "pvr-norm", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
240     var_Get( p_access, "pvr-norm" , &val);
241     p_sys->i_standard = val.i_int;
242
243     var_Create( p_access, "pvr-width", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
244     var_Get( p_access, "pvr-width" , &val);
245     p_sys->i_width = val.i_int;
246
247     var_Create( p_access, "pvr-height", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
248     var_Get( p_access, "pvr-height" , &val);
249     p_sys->i_height = val.i_int;
250
251     var_Create( p_access, "pvr-frequency", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
252     var_Get( p_access, "pvr-frequency" , &val);
253     p_sys->i_frequency = val.i_int;
254
255     var_Create( p_access, "pvr-framerate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
256     var_Get( p_access, "pvr-framerate" , &val);
257     p_sys->i_framerate = val.i_int;
258
259     var_Create( p_access, "pvr-keyint", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
260     var_Get( p_access, "pvr-keyint" , &val);
261     p_sys->i_keyint = val.i_int;
262
263     var_Create( p_access, "pvr-bframes", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
264     var_Get( p_access, "pvr-bframes" , &val);
265     p_sys->i_bframes = val.b_bool;
266
267     var_Create( p_access, "pvr-bitrate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
268     var_Get( p_access, "pvr-bitrate" , &val);
269     p_sys->i_bitrate = val.i_int;
270
271     var_Create( p_access, "pvr-bitrate-peak", VLC_VAR_INTEGER |
272                                               VLC_VAR_DOINHERIT );
273     var_Get( p_access, "pvr-bitrate-peak" , &val);
274     p_sys->i_bitrate_peak = val.i_int;
275
276     var_Create( p_access, "pvr-bitrate-mode", VLC_VAR_INTEGER |
277                                               VLC_VAR_DOINHERIT );
278     var_Get( p_access, "pvr-bitrate-mode" , &val);
279     p_sys->i_bitrate_mode = val.i_int;
280
281     var_Create( p_access, "pvr-audio-bitmask", VLC_VAR_INTEGER |
282                                               VLC_VAR_DOINHERIT );
283     var_Get( p_access, "pvr-audio-bitmask" , &val);
284     p_sys->i_audio_bitmask = val.i_int;
285     var_Create( p_access, "pvr-channel", VLC_VAR_INTEGER |
286                                               VLC_VAR_DOINHERIT );
287     var_Get( p_access, "pvr-channel" , &val);
288     p_sys->i_input = val.i_int;
289
290     /* parse command line options */
291     psz_tofree = strdup( p_access->psz_path );
292     psz_parser = psz_tofree;
293
294     if( *psz_parser )
295     {
296         for( ;; )
297         {
298             if ( !strncmp( psz_parser, "norm=", strlen( "norm=" ) ) )
299             {
300                 char *psz_parser_init;
301                 psz_parser += strlen( "norm=" );
302                 psz_parser_init = psz_parser;
303                 while ( *psz_parser != ':' && *psz_parser != ','
304                                                     && *psz_parser != '\0' )
305                 {
306                     psz_parser++;
307                 }
308
309                 if ( !strncmp( psz_parser_init, "secam" ,
310                                psz_parser - psz_parser_init ) )
311                 {
312                     p_sys->i_standard = V4L2_STD_SECAM;
313                 }
314                 else if ( !strncmp( psz_parser_init, "pal" ,
315                                     psz_parser - psz_parser_init ) )
316                 {
317                     p_sys->i_standard = V4L2_STD_PAL;
318                 }
319                 else if ( !strncmp( psz_parser_init, "ntsc" ,
320                                     psz_parser - psz_parser_init ) )
321                 {
322                     p_sys->i_standard = V4L2_STD_NTSC;
323                 }
324                 else
325                 {
326                     p_sys->i_standard = strtol( psz_parser_init ,
327                                                 &psz_parser, 0 );
328                 }
329             }
330             else if( !strncmp( psz_parser, "channel=",
331                                strlen( "channel=" ) ) )
332             {
333                 p_sys->i_input =
334                   strtol( psz_parser + strlen( "channel=" ),
335                             &psz_parser, 0 );
336             }
337             else if( !strncmp( psz_parser, "device=", strlen( "device=" ) ) )
338             {
339                 psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 );
340                 sprintf( psz_device, "/dev/video%ld",
341                             strtol( psz_parser + strlen( "device=" ),
342                             &psz_parser, 0 ) );
343             }
344             else if( !strncmp( psz_parser, "frequency=",
345                                strlen( "frequency=" ) ) )
346             {
347                 p_sys->i_frequency =
348                   strtol( psz_parser + strlen( "frequency=" ),
349                             &psz_parser, 0 );
350             }
351             else if( !strncmp( psz_parser, "framerate=",
352                                strlen( "framerate=" ) ) )
353             {
354                 p_sys->i_framerate =
355                     strtol( psz_parser + strlen( "framerate=" ),
356                             &psz_parser, 0 );
357             }
358             else if( !strncmp( psz_parser, "keyint=",
359                                strlen( "keyint=" ) ) )
360             {
361                 p_sys->i_keyint =
362                     strtol( psz_parser + strlen( "keyint=" ),
363                             &psz_parser, 0 );
364             }
365             else if( !strncmp( psz_parser, "bframes=",
366                                strlen( "bframes=" ) ) )
367             {
368                 p_sys->i_bframes =
369                     strtol( psz_parser + strlen( "bframes=" ),
370                             &psz_parser, 0 );
371             }
372
373             else if( !strncmp( psz_parser, "width=",
374                                strlen( "width=" ) ) )
375             {
376                 p_sys->i_width =
377                     strtol( psz_parser + strlen( "width=" ),
378                             &psz_parser, 0 );
379             }
380             else if( !strncmp( psz_parser, "height=",
381                                strlen( "height=" ) ) )
382             {
383                 p_sys->i_height =
384                     strtol( psz_parser + strlen( "height=" ),
385                             &psz_parser, 0 );
386             }
387             else if( !strncmp( psz_parser, "audio=",
388                                strlen( "audio=" ) ) )
389             {
390                 p_sys->i_audio_bitmask =
391                     strtol( psz_parser + strlen( "audio=" ),
392                             &psz_parser, 0 );
393             }
394             else if( !strncmp( psz_parser, "bitrate=",
395                                strlen( "bitrate=" ) ) )
396             {
397                 p_sys->i_bitrate =
398                     strtol( psz_parser + strlen( "bitrate=" ),
399                             &psz_parser, 0 );
400             }
401             else if( !strncmp( psz_parser, "maxbitrate=",
402                                strlen( "maxbitrate=" ) ) )
403             {
404                 p_sys->i_bitrate_peak =
405                     strtol( psz_parser + strlen( "maxbitrate=" ),
406                             &psz_parser, 0 );
407             }
408             else if( !strncmp( psz_parser, "bitratemode=",
409                                strlen( "bitratemode=" ) ) )
410             {
411                 char *psz_parser_init;
412                 psz_parser += strlen( "bitratemode=" );
413                 psz_parser_init = psz_parser;
414                 while ( *psz_parser != ':' && *psz_parser != ','
415                          && *psz_parser != '\0' )
416                 {
417                     psz_parser++;
418                 }
419
420                 if ( !strncmp( psz_parser_init, "vbr" ,
421                                psz_parser - psz_parser_init ) )
422                 {
423                      p_sys->i_bitrate_mode = 0;
424                 }
425                 else if ( !strncmp( psz_parser_init, "cbr" ,
426                                     psz_parser - psz_parser_init ) )
427                 {
428                     p_sys->i_bitrate_mode = 1;
429                 }
430             }
431             else if( !strncmp( psz_parser, "size=",
432                                strlen( "size=" ) ) )
433             {
434                 p_sys->i_width =
435                     strtol( psz_parser + strlen( "size=" ),
436                             &psz_parser, 0 );
437                 p_sys->i_height =
438                     strtol( psz_parser + 1 ,
439                             &psz_parser, 0 );
440             }
441             else
442             {
443                 char *psz_parser_init;
444                 psz_parser_init = psz_parser;
445                 while ( *psz_parser != ':' && *psz_parser != ',' && *psz_parser != '\0' )
446                 {
447                     psz_parser++;
448                 }
449                 psz_device = calloc( psz_parser - psz_parser_init + 1, 1 );
450                 strncpy( psz_device, psz_parser_init,
451                          psz_parser - psz_parser_init );
452             }
453             if( *psz_parser )
454                 psz_parser++;
455             else
456                 break;
457         }
458     }
459
460     //give a default value to psz_device if none has bee specified
461
462     if (!psz_device)
463     {
464         psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 );
465         strcpy( psz_device, "/dev/video0" );
466     }
467
468     free( psz_tofree );
469
470     /* open the device */
471     if( ( p_sys->i_fd = open( psz_device, O_RDWR ) ) < 0 )
472     {
473         msg_Err( p_access, "cannot open device (%s)", strerror( errno ) );
474         return VLC_EGENERIC;
475     }
476     else
477     {
478         msg_Dbg( p_access, "using video device: %s",psz_device);
479     }
480
481     free( psz_device );
482
483     /* set the input */
484     if ( p_sys->i_input != -1 )
485     {
486         if ( ioctl( p_sys->i_fd, VIDIOC_S_INPUT, &p_sys->i_input ) < 0 )
487         {
488             msg_Warn( p_access, "VIDIOC_S_INPUT failed" );
489         }
490         else
491         {
492             msg_Dbg( p_access, "input set to:%d", p_sys->i_input);
493         }
494     }
495
496     /* set the video standard */
497     if ( p_sys->i_standard != V4L2_STD_UNKNOWN )
498     {
499         if ( ioctl( p_sys->i_fd, VIDIOC_S_STD, &p_sys->i_standard ) < 0 )
500         {
501             msg_Warn( p_access, "VIDIOC_S_STD failed" );
502         }
503         else
504         {
505             msg_Dbg( p_access, "video standard set to:%x", p_sys->i_standard);
506         }
507     }
508
509     /* set the picture size */
510     if ( p_sys->i_width != -1 || p_sys->i_height != -1 )
511     {
512         vfmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
513         if ( ioctl( p_sys->i_fd, VIDIOC_G_FMT, &vfmt ) < 0 )
514         {
515             msg_Warn( p_access, "VIDIOC_G_FMT failed" );
516         }
517         else
518         {
519             if ( p_sys->i_width != -1 )
520             {
521                 vfmt.fmt.pix.width = p_sys->i_width;
522             }
523
524             if ( p_sys->i_height != -1 )
525             {
526                 vfmt.fmt.pix.height = p_sys->i_height;
527             }
528
529             if ( ioctl( p_sys->i_fd, VIDIOC_S_FMT, &vfmt ) < 0 )
530             {
531                 msg_Warn( p_access, "VIDIOC_S_FMT failed" );
532             }
533             else
534             {
535                 msg_Dbg( p_access, "picture size set to:%dx%d",
536                          vfmt.fmt.pix.width, vfmt.fmt.pix.height );
537             }
538         }
539     }
540
541     /* set the frequency */
542     if ( p_sys->i_frequency != -1 )
543     {
544         vf.tuner = 0; /* TODO: let the user choose the tuner */
545         if ( ioctl( p_sys->i_fd, VIDIOC_G_FREQUENCY, &vf ) < 0 )
546         {
547             msg_Warn( p_access, "VIDIOC_G_FREQUENCY failed (%s)",
548                       strerror( errno ) );
549         }
550         else
551         {
552             vf.frequency = p_sys->i_frequency * 16 / 1000;
553             if( ioctl( p_sys->i_fd, VIDIOC_S_FREQUENCY, &vf ) < 0 )
554             {
555                 msg_Warn( p_access, "VIDIOC_S_FREQUENCY failed (%s)",
556                           strerror( errno ) );
557             }
558             else
559             {
560                 msg_Dbg( p_access, "Tuner frequency set to:%d",
561                          p_sys->i_frequency);
562             }
563         }
564     }
565
566     /* codec parameters */
567     if ( p_sys->i_framerate != -1
568             || p_sys->i_bitrate_mode != -1
569             || p_sys->i_bitrate_peak != -1
570             || p_sys->i_keyint != -1
571             || p_sys->i_bframes != -1
572             || p_sys->i_bitrate != -1
573             || p_sys->i_audio_bitmask != -1 )
574     {
575         if ( ioctl( p_sys->i_fd, IVTV_IOC_G_CODEC, &codec ) < 0 )
576         {
577             msg_Warn( p_access, "IVTV_IOC_G_CODEC failed" );
578         }
579         else
580         {
581             if ( p_sys->i_framerate != -1 )
582             {
583                 switch ( p_sys->i_framerate )
584                 {
585                     case 30:
586                         codec.framerate = 0;
587                         break;
588
589                     case 25:
590                         codec.framerate = 1;
591                         break;
592
593                     default:
594                         msg_Warn( p_access, "invalid framerate, reverting to 25" );
595                         codec.framerate = 1;
596                         break;
597                 }
598             }
599
600             if ( p_sys->i_bitrate != -1 )
601             {
602                 codec.bitrate = p_sys->i_bitrate;
603             }
604
605             if ( p_sys->i_bitrate_peak != -1 )
606             {
607                 codec.bitrate_peak = p_sys->i_bitrate_peak;
608             }
609
610             if ( p_sys->i_bitrate_mode != -1 )
611             {
612                 codec.bitrate_mode = p_sys->i_bitrate_mode;
613             }
614
615             if ( p_sys->i_audio_bitmask != -1 )
616             {
617                 codec.audio_bitmask = p_sys->i_audio_bitmask;
618             }
619             if ( p_sys->i_keyint != -1 )
620             {
621                 codec.framespergop = p_sys->i_keyint;
622             }
623
624             if ( p_sys->i_bframes != -1 )
625             {
626                 codec.bframes = p_sys->i_bframes;
627             }
628             if( ioctl( p_sys->i_fd, IVTV_IOC_S_CODEC, &codec ) < 0 )
629             {
630                 msg_Warn( p_access, "IVTV_IOC_S_CODEC failed" );
631             }
632             else
633             {
634                 msg_Dbg( p_access, "Setting codec parameters to:  framerate: %d, bitrate: %d/%d/%d",
635                codec.framerate, codec.bitrate, codec.bitrate_peak, codec.bitrate_mode );
636             }
637         }
638     }
639
640     /* do a quick read */
641 #if 0
642     if ( p_sys->i_fd )
643     {
644         if ( read( p_sys->i_fd, psz_tmp, 1 ) )
645         {
646             msg_Dbg(p_input, "Could read byte from device");
647         }
648         else
649         {
650             msg_Warn(p_input, "Could not read byte from device");
651         }
652     }
653 #endif
654
655     return VLC_SUCCESS;
656 }
657
658 /*****************************************************************************
659  * Close: close the device
660  *****************************************************************************/
661 static void Close( vlc_object_t * p_this )
662 {
663     access_t *p_access = (access_t*) p_this;
664     access_sys_t * p_sys = p_access->p_sys;
665
666     close( p_sys->i_fd );
667     free( p_sys );
668 }
669
670 /*****************************************************************************
671  * Read
672  *****************************************************************************/
673 static int Read( access_t * p_access, uint8_t * p_buffer, int i_len )
674 {
675     access_sys_t * p_sys = p_access->p_sys;
676
677     int i_ret;
678
679     struct timeval timeout;
680     fd_set fds;
681
682     FD_ZERO( &fds );
683     FD_SET( p_sys->i_fd, &fds );
684     timeout.tv_sec = 0;
685     timeout.tv_usec = 500000;
686
687     if( p_access->info.b_eof )
688         return 0;
689
690     while( !( i_ret = select( p_sys->i_fd + 1, &fds, NULL, NULL, &timeout) ) )
691     {
692         FD_ZERO( &fds );
693         FD_SET( p_sys->i_fd, &fds );
694         timeout.tv_sec = 0;
695         timeout.tv_usec = 500000;
696
697         if( p_access->b_die )
698             return 0;
699     }
700
701     if( i_ret < 0 )
702     {
703         msg_Err( p_access, "select error (%s)", strerror( errno ) );
704         return -1;
705     }
706
707     i_ret = read( p_sys->i_fd, p_buffer, i_len );
708     if( i_ret == 0 )
709     {
710         p_access->info.b_eof = VLC_TRUE;
711     }
712     else if( i_ret > 0 )
713     {
714         p_access->info.i_pos += i_ret;
715     }
716
717     return i_ret;
718 }
719
720 /*****************************************************************************
721  * Control
722  *****************************************************************************/
723 static int Control( access_t *p_access, int i_query, va_list args )
724 {
725     vlc_bool_t   *pb_bool;
726     int          *pi_int;
727     int64_t      *pi_64;
728
729     switch( i_query )
730     {
731         /* */
732         case ACCESS_CAN_SEEK:
733         case ACCESS_CAN_FASTSEEK:
734             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
735             *pb_bool = VLC_FALSE;
736             break;
737         case ACCESS_CAN_PAUSE:
738             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
739             *pb_bool = VLC_FALSE;
740             break;
741         case ACCESS_CAN_CONTROL_PACE:
742             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
743             *pb_bool = VLC_FALSE;
744             break;
745
746         /* */
747         case ACCESS_GET_MTU:
748             pi_int = (int*)va_arg( args, int * );
749             *pi_int = 0;
750             break;
751
752         case ACCESS_GET_PTS_DELAY:
753             pi_64 = (int64_t*)va_arg( args, int64_t * );
754             *pi_64 = (int64_t)var_GetInteger( p_access, "pvr-caching" ) * 1000;
755             break;
756
757         /* */
758         case ACCESS_SET_PAUSE_STATE:
759             /* Nothing to do */
760             break;
761
762         case ACCESS_GET_TITLE_INFO:
763         case ACCESS_SET_TITLE:
764         case ACCESS_SET_SEEKPOINT:
765         case ACCESS_SET_PRIVATE_ID_STATE:
766             return VLC_EGENERIC;
767
768         default:
769             msg_Warn( p_access, "unimplemented query in control" );
770             return VLC_EGENERIC;
771
772     }
773     return VLC_SUCCESS;
774 }