pvr.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:25k
源码类别:

多媒体

开发平台:

MultiPlatform

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