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

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * access.c: DVB card input v4l2 only
  3.  *****************************************************************************
  4.  * Copyright (C) 1998-2004 VideoLAN
  5.  *
  6.  * Authors: Johan Bilien <jobi@via.ecp.fr>
  7.  *          Jean-Paul Saman <jpsaman@wxs.nl>
  8.  *          Christophe Massiot <massiot@via.ecp.fr>
  9.  *          Laurent Aimar <fenrir@via.ecp.fr>
  10.  *
  11.  * This program is free software; you can redistribute it and/or modify
  12.  * it under the terms of the GNU General Public License as published by
  13.  * the Free Software Foundation; either version 2 of the License, or
  14.  * (at your option) any later version.
  15.  *
  16.  * This program is distributed in the hope that it will be useful,
  17.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19.  * GNU General Public License for more details.
  20.  *
  21.  * You should have received a copy of the GNU General Public License
  22.  * along with this program; if not, write to the Free Software
  23.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  24.  *****************************************************************************/
  25. /*****************************************************************************
  26.  * Preamble
  27.  *****************************************************************************/
  28. #include <vlc/vlc.h>
  29. #include <vlc/input.h>
  30. #ifdef HAVE_UNISTD_H
  31. #   include <unistd.h>
  32. #endif
  33. #include <fcntl.h>
  34. #include <sys/types.h>
  35. #include <errno.h>
  36. #include "dvb.h"
  37. /*****************************************************************************
  38.  * Module descriptor
  39.  *****************************************************************************/
  40. static int  Open( vlc_object_t *p_this );
  41. static void Close( vlc_object_t *p_this );
  42. #define CACHING_TEXT N_("Caching value in ms")
  43. #define CACHING_LONGTEXT N_( 
  44.     "Allows you to modify the default caching value for dvb streams. This " 
  45.     "value should be set in millisecond units." )
  46. #define ADAPTER_TEXT N_("Adapter card to tune")
  47. #define ADAPTER_LONGTEXT N_("Adapter cards have a device file in directory named /dev/dvb/adapter[n] with n>=0.")
  48. #define DEVICE_TEXT N_("Device number to use on adapter")
  49. #define DEVICE_LONGTEXT ""
  50. #define FREQ_TEXT N_("Transponder/multiplex frequency")
  51. #define FREQ_LONGTEXT N_("In kHz for DVB-S or Hz for DVB-C/T")
  52. #define INVERSION_TEXT N_("Inversion mode")
  53. #define INVERSION_LONGTEXT N_("Inversion mode [0=off, 1=on, 2=auto]")
  54. #define PROBE_TEXT N_("Probe DVB card for capabilities")
  55. #define PROBE_LONGTEXT N_("Some DVB cards do not like to be probed for their capabilities.")
  56. #define LNB_LOF1_TEXT N_("Antenna lnb_lof1 (kHz)")
  57. #define LNB_LOF1_LONGTEXT ""
  58. #define LNB_LOF2_TEXT N_("Antenna lnb_lof2 (kHz)")
  59. #define LNB_LOF2_LONGTEXT ""
  60. #define LNB_SLOF_TEXT N_("Antenna lnb_slof (kHz)")
  61. #define LNB_SLOF_LONGTEXT ""
  62. /* Satellite */
  63. #define BUDGET_TEXT N_("Budget mode")
  64. #define BUDGET_LONGTEXT N_("This allows you to stream an entire transponder with a budget card.")
  65. #define SATNO_TEXT N_("Satellite number in the Diseqc system")
  66. #define SATNO_LONGTEXT N_("[0=no diseqc, 1-4=normal diseqc, -1=A, -2=B simple diseqc]")
  67. #define VOLTAGE_TEXT N_("LNB voltage")
  68. #define VOLTAGE_LONGTEXT N_("In Volts [0, 13=vertical, 18=horizontal]")
  69. #define TONE_TEXT N_("22 kHz tone")
  70. #define TONE_LONGTEXT N_("[0=off, 1=on, -1=auto]")
  71. #define FEC_TEXT N_("Transponder FEC")
  72. #define FEC_LONGTEXT N_("FEC=Forward Error Correction mode [9=auto]")
  73. #define SRATE_TEXT N_("Transponder symbol rate in kHz")
  74. #define SRATE_LONGTEXT ""
  75. /* Cable */
  76. #define MODULATION_TEXT N_("Modulation type")
  77. #define MODULATION_LONGTEXT N_("Modulation type for front-end device.")
  78. /* Terrestrial */
  79. #define CODE_RATE_HP_TEXT N_("Terrestrial high priority stream code rate (FEC)")
  80. #define CODE_RATE_HP_LONGTEXT ""
  81. #define CODE_RATE_LP_TEXT N_("Terrestrial low priority stream code rate (FEC)")
  82. #define CODE_RATE_LP_LONGTEXT ""
  83. #define BANDWIDTH_TEXT N_("Terrestrial bandwidth")
  84. #define BANDWIDTH_LONGTEXT N_("Terrestrial bandwidth [0=auto,6,7,8 in MHz]")
  85. #define GUARD_TEXT N_("Terrestrial guard interval")
  86. #define GUARD_LONGTEXT ""
  87. #define TRANSMISSION_TEXT N_("Terrestrial transmission mode")
  88. #define TRANSMISSION_LONGTEXT ""
  89. #define HIERARCHY_TEXT N_("Terrestrial hierarchy mode")
  90. #define HIERARCHY_LONGTEXT ""
  91. vlc_module_begin();
  92.     set_shortname( _("DVB") );
  93.     set_description( N_("DVB input with v4l2 support") );
  94.     add_integer( "dvb-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT,
  95.                  CACHING_LONGTEXT, VLC_TRUE );
  96.     add_integer( "dvb-adapter", 0, NULL, ADAPTER_TEXT, ADAPTER_LONGTEXT,
  97.                  VLC_FALSE );
  98.     add_integer( "dvb-device", 0, NULL, DEVICE_TEXT, DEVICE_LONGTEXT,
  99.                  VLC_TRUE );
  100.     add_integer( "dvb-frequency", 11954000, NULL, FREQ_TEXT, FREQ_LONGTEXT,
  101.                  VLC_FALSE );
  102.     add_integer( "dvb-inversion", 2, NULL, INVERSION_TEXT, INVERSION_LONGTEXT,
  103.                  VLC_TRUE );
  104.     add_bool( "dvb-probe", 1, NULL, PROBE_TEXT, PROBE_LONGTEXT, VLC_TRUE );
  105.     add_integer( "dvb-lnb-lof1", 9750000, NULL, LNB_LOF1_TEXT,
  106.                  LNB_LOF1_LONGTEXT, VLC_TRUE );
  107.     add_integer( "dvb-lnb-lof2", 10600000, NULL, LNB_LOF2_TEXT,
  108.                  LNB_LOF2_LONGTEXT, VLC_TRUE );
  109.     add_integer( "dvb-lnb-slof", 11700000, NULL, LNB_SLOF_TEXT,
  110.                  LNB_SLOF_LONGTEXT, VLC_TRUE );
  111.     /* DVB-S (satellite) */
  112.     add_bool( "dvb-budget-mode", 0, NULL, BUDGET_TEXT, BUDGET_LONGTEXT,
  113.               VLC_TRUE );
  114.     add_integer( "dvb-satno", 0, NULL, SATNO_TEXT, SATNO_LONGTEXT,
  115.                  VLC_TRUE );
  116.     add_integer( "dvb-voltage", 13, NULL, VOLTAGE_TEXT, VOLTAGE_LONGTEXT,
  117.                  VLC_TRUE );
  118.     add_integer( "dvb-tone", -1, NULL, TONE_TEXT, TONE_LONGTEXT,
  119.                  VLC_TRUE );
  120.     add_integer( "dvb-fec", 9, NULL, FEC_TEXT, FEC_LONGTEXT, VLC_TRUE );
  121.     add_integer( "dvb-srate", 27500000, NULL, SRATE_TEXT, SRATE_LONGTEXT,
  122.                  VLC_FALSE );
  123.     /* DVB-T (terrestrial) */
  124.     add_integer( "dvb-modulation", 0, NULL, MODULATION_TEXT,
  125.                  MODULATION_LONGTEXT, VLC_TRUE );
  126.     /* DVB-T (terrestrial) */
  127.     add_integer( "dvb-code-rate-hp", 9, NULL, CODE_RATE_HP_TEXT,
  128.                  CODE_RATE_HP_LONGTEXT, VLC_TRUE );
  129.     add_integer( "dvb-code-rate-lp", 9, NULL, CODE_RATE_LP_TEXT,
  130.                  CODE_RATE_LP_LONGTEXT, VLC_TRUE );
  131.     add_integer( "dvb-bandwidth", 0, NULL, BANDWIDTH_TEXT, BANDWIDTH_LONGTEXT,
  132.                  VLC_TRUE );
  133.     add_integer( "dvb-guard", 0, NULL, GUARD_TEXT, GUARD_LONGTEXT, VLC_TRUE );
  134.     add_integer( "dvb-transmission", 0, NULL, TRANSMISSION_TEXT,
  135.                  TRANSMISSION_LONGTEXT, VLC_TRUE );
  136.     add_integer( "dvb-hierarchy", 0, NULL, HIERARCHY_TEXT, HIERARCHY_LONGTEXT,
  137.                  VLC_TRUE );
  138.     set_capability( "access2", 0 );
  139.     add_shortcut( "dvb" );
  140.     add_shortcut( "dvb-s" );
  141.     add_shortcut( "qpsk" );
  142.     add_shortcut( "dvb-c" );
  143.     add_shortcut( "cable" );
  144.     add_shortcut( "dvb-t" );
  145.     add_shortcut( "terrestrial" );
  146.     add_shortcut( "satellite" );    /* compatibility with the interface. */
  147.     set_callbacks( Open, Close );
  148. vlc_module_end();
  149. /*****************************************************************************
  150.  * Local prototypes
  151.  *****************************************************************************/
  152. static block_t *Block( access_t * );
  153. static int Control( access_t *, int, va_list );
  154. #define DVB_READ_ONCE 3
  155. #define TS_PACKET_SIZE 188
  156. static void FilterUnset( access_t *, int i_max );
  157. static void FilterUnsetPID( access_t *, int i_pid );
  158. static void FilterSet( access_t *, int i_pid, int i_type );
  159. static void VarInit( access_t * );
  160. static int  ParseMRL( access_t * );
  161. /*****************************************************************************
  162.  * Open: open the frontend device
  163.  *****************************************************************************/
  164. static int Open( vlc_object_t *p_this )
  165. {
  166.     access_t     *p_access = (access_t*)p_this;
  167.     access_sys_t *p_sys;
  168.     /* Only if selected */
  169.     if( *p_access->psz_access == '' )
  170.         return VLC_EGENERIC;
  171.     /* Set up access */
  172.     p_access->pf_read = NULL;
  173.     p_access->pf_block = Block;
  174.     p_access->pf_control = Control;
  175.     p_access->pf_seek = NULL;
  176.     p_access->info.i_update = 0;
  177.     p_access->info.i_size = 0;
  178.     p_access->info.i_pos = 0;
  179.     p_access->info.b_eof = VLC_FALSE;
  180.     p_access->info.i_title = 0;
  181.     p_access->info.i_seekpoint = 0;
  182.     p_access->p_sys = p_sys = malloc( sizeof( access_sys_t ) );
  183.     memset( p_sys, 0, sizeof( access_sys_t ) );
  184.     /* Create all variables */
  185.     VarInit( p_access );
  186.     /* Parse the command line */
  187.     if( ParseMRL( p_access ) )
  188.     {
  189.         free( p_sys );
  190.         return VLC_EGENERIC;
  191.     }
  192.     /* Getting frontend info */
  193.     if( E_(FrontendOpen)( p_access) )
  194.     {
  195.         free( p_sys );
  196.         return VLC_EGENERIC;
  197.     }
  198.     /* Setting frontend parameters for tuning the hardware */
  199.     msg_Dbg( p_access, "trying to tune the frontend...");
  200.     if( E_(FrontendSet)( p_access ) < 0 )
  201.     {
  202.         E_(FrontendClose)( p_access );
  203.         free( p_sys );
  204.         return VLC_EGENERIC;
  205.     }
  206.     /* Opening DVR device */
  207.     if( E_(DVROpen)( p_access ) < 0 )
  208.     {
  209.         E_(FrontendClose)( p_access );
  210.         free( p_sys );
  211.         return VLC_EGENERIC;
  212.     }
  213.     p_sys->b_budget_mode = var_GetBool( p_access, "dvb-budget-mode" );
  214.     if( p_sys->b_budget_mode )
  215.     {
  216.         msg_Dbg( p_access, "setting filter on all PIDs" );
  217.         FilterSet( p_access, 0x2000, OTHER_TYPE );
  218.     }
  219.     else
  220.     {
  221.         msg_Dbg( p_access, "setting filter on PAT" );
  222.         FilterSet( p_access, 0x0, OTHER_TYPE );
  223.     }
  224.     E_(CAMOpen)( p_access );
  225.     return VLC_SUCCESS;
  226. }
  227. /*****************************************************************************
  228.  * Close : Close the device
  229.  *****************************************************************************/
  230. static void Close( vlc_object_t *p_this )
  231. {
  232.     access_t     *p_access = (access_t*)p_this;
  233.     access_sys_t *p_sys = p_access->p_sys;
  234.     FilterUnset( p_access, p_sys->b_budget_mode ? 1 : MAX_DEMUX );
  235.     E_(DVRClose)( p_access );
  236.     E_(FrontendClose)( p_access );
  237.     E_(CAMClose)( p_access );
  238.     free( p_sys );
  239. }
  240. /*****************************************************************************
  241.  * Block:
  242.  *****************************************************************************/
  243. static block_t *Block( access_t *p_access )
  244. {
  245.     access_sys_t *p_sys = p_access->p_sys;
  246.     block_t *p_block;
  247.     for ( ; ; )
  248.     {
  249.         struct timeval timeout;
  250.         fd_set fds;
  251.         int i_ret;
  252.         /* Initialize file descriptor set */
  253.         FD_ZERO( &fds );
  254.         FD_SET( p_sys->i_handle, &fds );
  255.         /* We'll wait 0.5 second if nothing happens */
  256.         timeout.tv_sec = 0;
  257.         timeout.tv_usec = 500000;
  258.         /* Find if some data is available */
  259.         i_ret = select( p_sys->i_handle + 1, &fds, NULL, NULL, &timeout );
  260.         if ( p_access->b_die )
  261.             return NULL;
  262.         if ( i_ret < 0 && errno == EINTR )
  263.             continue;
  264.         if ( i_ret < 0 )
  265.         {
  266.             msg_Err( p_access, "select error (%s)", strerror(errno) );
  267.             return NULL;
  268.         }
  269.         if ( p_sys->i_ca_handle && mdate() > p_sys->i_ca_next_event )
  270.         {
  271.             E_(CAMPoll)( p_access );
  272.             p_sys->i_ca_next_event = mdate() + p_sys->i_ca_timeout;
  273.         }
  274.         if ( FD_ISSET( p_sys->i_handle, &fds ) )
  275.         {
  276.             break;
  277.         }
  278.     }
  279.     p_block = block_New( p_access, DVB_READ_ONCE * TS_PACKET_SIZE );
  280.     if( ( p_block->i_buffer = read( p_sys->i_handle, p_block->p_buffer,
  281.                                     DVB_READ_ONCE * TS_PACKET_SIZE ) ) <= 0 )
  282.     {
  283.         msg_Err( p_access, "read failed (%s)", strerror(errno) );
  284.         block_Release( p_block );
  285.         return NULL;
  286.     }
  287.     return p_block;
  288. }
  289. /*****************************************************************************
  290.  * Control:
  291.  *****************************************************************************/
  292. static int Control( access_t *p_access, int i_query, va_list args )
  293. {
  294.     access_sys_t *p_sys = p_access->p_sys;
  295.     vlc_bool_t   *pb_bool, b_bool;
  296.     int          *pi_int, i_int;
  297.     int64_t      *pi_64;
  298.     switch( i_query )
  299.     {
  300.         /* */
  301.         case ACCESS_CAN_SEEK:
  302.         case ACCESS_CAN_FASTSEEK:
  303.         case ACCESS_CAN_PAUSE:
  304.         case ACCESS_CAN_CONTROL_PACE:
  305.             pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t* );
  306.             *pb_bool = VLC_FALSE;
  307.             break;
  308.         /* */
  309.         case ACCESS_GET_MTU:
  310.             pi_int = (int*)va_arg( args, int * );
  311.             *pi_int = DVB_READ_ONCE * TS_PACKET_SIZE;
  312.             break;
  313.         case ACCESS_GET_PTS_DELAY:
  314.             pi_64 = (int64_t*)va_arg( args, int64_t * );
  315.             *pi_64 = var_GetInteger( p_access, "dvb-caching" ) * 1000;
  316.             break;
  317.         /* */
  318.         case ACCESS_SET_PAUSE_STATE:
  319.         case ACCESS_GET_TITLE_INFO:
  320.         case ACCESS_SET_TITLE:
  321.         case ACCESS_SET_SEEKPOINT:
  322.             return VLC_EGENERIC;
  323.         case ACCESS_SET_PRIVATE_ID_STATE:
  324.             i_int  = (int)va_arg( args, int );               /* Private data (pid for now)*/
  325.             b_bool = (vlc_bool_t)va_arg( args, vlc_bool_t ); /* b_selected */
  326.             if( !p_sys->b_budget_mode )
  327.             {
  328.                 /* FIXME we may want to give the real type (me ?, I don't ;) */
  329.                 if( b_bool )
  330.                     FilterSet( p_access, i_int, OTHER_TYPE );
  331.                 else
  332.                     FilterUnsetPID( p_access, i_int );
  333.             }
  334.             break;
  335.         case ACCESS_SET_PRIVATE_ID_CA:
  336.         {
  337.             uint8_t **pp_capmts;
  338.             int i_nb_capmts;
  339.             pp_capmts = (uint8_t **)va_arg( args, uint8_t ** );
  340.             i_nb_capmts = (int)va_arg( args, int );
  341.             E_(CAMSet)( p_access, pp_capmts, i_nb_capmts );
  342.             break;
  343.         }
  344.         default:
  345.             msg_Warn( p_access, "unimplemented query in control" );
  346.             return VLC_EGENERIC;
  347.     }
  348.     return VLC_SUCCESS;
  349. }
  350. /*****************************************************************************
  351.  * FilterSet/FilterUnset:
  352.  *****************************************************************************/
  353. static void FilterSet( access_t *p_access, int i_pid, int i_type )
  354. {
  355.     access_sys_t *p_sys = p_access->p_sys;
  356.     int i;
  357.     /* Find first free slot */
  358.     for( i = 0; i < MAX_DEMUX; i++ )
  359.     {
  360.         if( !p_sys->p_demux_handles[i].i_type )
  361.             break;
  362.         if( p_sys->p_demux_handles[i].i_pid == i_pid )
  363.             return; /* Already set */
  364.     }
  365.     if( i >= MAX_DEMUX )
  366.     {
  367.         msg_Err( p_access, "no free p_demux_handles !" );
  368.         return;
  369.     }
  370.     if( E_(DMXSetFilter)( p_access, i_pid,
  371.                            &p_sys->p_demux_handles[i].i_handle, i_type ) )
  372.     {
  373.         msg_Err( p_access, "DMXSetFilter failed" );
  374.         return;
  375.     }
  376.     p_sys->p_demux_handles[i].i_type = i_type;
  377.     p_sys->p_demux_handles[i].i_pid = i_pid;
  378. }
  379. static void FilterUnset( access_t *p_access, int i_max )
  380. {
  381.     access_sys_t *p_sys = p_access->p_sys;
  382.     int i;
  383.     for( i = 0; i < i_max; i++ )
  384.     {
  385.         if( p_sys->p_demux_handles[i].i_type )
  386.         {
  387.             E_(DMXUnsetFilter)( p_access, p_sys->p_demux_handles[i].i_handle );
  388.             p_sys->p_demux_handles[i].i_type = 0;
  389.         }
  390.     }
  391. }
  392. static void FilterUnsetPID( access_t *p_access, int i_pid )
  393. {
  394.     access_sys_t *p_sys = p_access->p_sys;
  395.     int i;
  396.     for( i = 0; i < MAX_DEMUX; i++ )
  397.     {
  398.         if( p_sys->p_demux_handles[i].i_type &&
  399.             p_sys->p_demux_handles[i].i_pid == i_pid )
  400.         {
  401.             E_(DMXUnsetFilter)( p_access, p_sys->p_demux_handles[i].i_handle );
  402.             p_sys->p_demux_handles[i].i_type = 0;
  403.         }
  404.     }
  405. }
  406. /*****************************************************************************
  407.  * VarInit/ParseMRL:
  408.  *****************************************************************************/
  409. static void VarInit( access_t *p_access )
  410. {
  411.     /* */
  412.     var_Create( p_access, "dvb-caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  413.     /* */
  414.     var_Create( p_access, "dvb-adapter", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  415.     var_Create( p_access, "dvb-device", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  416.     var_Create( p_access, "dvb-frequency", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  417.     var_Create( p_access, "dvb-inversion", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  418.     var_Create( p_access, "dvb-probe", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
  419.     var_Create( p_access, "dvb-lnb-lof1", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  420.     var_Create( p_access, "dvb-lnb-lof2", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  421.     var_Create( p_access, "dvb-lnb-slof", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  422.     /* */
  423.     var_Create( p_access, "dvb-budget-mode", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
  424.     var_Create( p_access, "dvb-satno", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  425.     var_Create( p_access, "dvb-voltage", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  426.     var_Create( p_access, "dvb-tone", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  427.     var_Create( p_access, "dvb-fec", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  428.     var_Create( p_access, "dvb-srate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  429.     /* */
  430.     var_Create( p_access, "dvb-modulation", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  431.     /* */
  432.     var_Create( p_access, "dvb-code-rate-hp", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  433.     var_Create( p_access, "dvb-code-rate-lp", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  434.     var_Create( p_access, "dvb-bandwidth", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  435.     var_Create( p_access, "dvb-transmission", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  436.     var_Create( p_access, "dvb-guard", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  437.     var_Create( p_access, "dvb-hierarchy", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
  438. }
  439. /* */
  440. static int ParseMRL( access_t *p_access )
  441. {
  442.     char *psz_dup = strdup( p_access->psz_path );
  443.     char *psz_parser = psz_dup;
  444.     char *psz_next;
  445.     vlc_value_t         val;
  446. #define GET_OPTION_INT( option )                                            
  447.     if ( !strncmp( psz_parser, option "=", strlen(option "=") ) )           
  448.     {                                                                       
  449.         val.i_int = strtol( psz_parser + strlen(option "="), &psz_parser,   
  450.                             0 );                                            
  451.         var_Set( p_access, "dvb-" option, val );                            
  452.     }
  453. #define GET_OPTION_BOOL( option )                                           
  454.     if ( !strncmp( psz_parser, option "=", strlen(option "=") ) )           
  455.     {                                                                       
  456.         val.b_bool = strtol( psz_parser + strlen(option "="), &psz_parser,  
  457.                              0 );                                           
  458.         var_Set( p_access, "dvb-" option, val );                            
  459.     }
  460.     /* Test for old syntax */
  461.     strtol( psz_parser, &psz_next, 10 );
  462.     if( psz_next != psz_parser )
  463.     {
  464.         msg_Err( p_access, "the DVB input old syntax is deprecated, use vlc "
  465.                           "-p dvb to see an explanation of the new syntax" );
  466.         free( psz_dup );
  467.         return VLC_EGENERIC;
  468.     }
  469.     while( *psz_parser )
  470.     {
  471.         GET_OPTION_INT("adapter")
  472.         else GET_OPTION_INT("device")
  473.         else GET_OPTION_INT("frequency")
  474.         else GET_OPTION_INT("inversion")
  475.         else GET_OPTION_BOOL("probe")
  476.         else GET_OPTION_INT("lnb-lof1")
  477.         else GET_OPTION_INT("lnb-lof2")
  478.         else GET_OPTION_INT("lnb-slof")
  479.         else GET_OPTION_BOOL("budget-mode")
  480.         else GET_OPTION_INT("voltage")
  481.         else GET_OPTION_INT("tone")
  482.         else GET_OPTION_INT("fec")
  483.         else GET_OPTION_INT("srate")
  484.         else GET_OPTION_INT("modulation")
  485.         else GET_OPTION_INT("code-rate-hp")
  486.         else GET_OPTION_INT("code-rate-lp")
  487.         else GET_OPTION_INT("bandwidth")
  488.         else GET_OPTION_INT("transmission")
  489.         else GET_OPTION_INT("guard")
  490.         else GET_OPTION_INT("hierarchy")
  491.         else if( !strncmp( psz_parser, "satno=",
  492.                            strlen( "satno=" ) ) )
  493.         {
  494.             psz_parser += strlen( "satno=" );
  495.             if ( *psz_parser == 'A' || *psz_parser == 'a' )
  496.                 val.i_int = -1;
  497.             else if ( *psz_parser == 'B' || *psz_parser == 'b' )
  498.                 val.i_int = -2;
  499.             else
  500.                 val.i_int = strtol( psz_parser, &psz_parser, 0 );
  501.             var_Set( p_access, "dvb-satno", val );
  502.         }
  503.         /* Redundant with voltage but much easier to use */
  504.         else if( !strncmp( psz_parser, "polarization=",
  505.                            strlen( "polarization=" ) ) )
  506.         {
  507.             psz_parser += strlen( "polarization=" );
  508.             if ( *psz_parser == 'V' || *psz_parser == 'v' )
  509.                 val.i_int = 13;
  510.             else if ( *psz_parser == 'H' || *psz_parser == 'h' )
  511.                 val.i_int = 18;
  512.             else
  513.             {
  514.                 msg_Err( p_access, "illegal polarization %c", *psz_parser );
  515.                 free( psz_dup );
  516.                 return VLC_EGENERIC;
  517.             }
  518.             var_Set( p_access, "dvb-voltage", val );
  519.         }
  520.         else
  521.         {
  522.             msg_Err( p_access, "unknown option (%s)", psz_parser );
  523.             free( psz_dup );
  524.             return VLC_EGENERIC;
  525.         }
  526.         if ( *psz_parser )
  527.             psz_parser++;
  528.     }
  529. #undef GET_OPTION_INT
  530. #undef GET_OPTION_BOOL
  531.     free( psz_dup );
  532.     return VLC_SUCCESS;
  533. }