dvb.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:10k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * dvb.c : DVB channel list import (szap/tzap/czap compatible channel lists)
  3.  *****************************************************************************
  4.  * Copyright (C) 2005 the VideoLAN team
  5.  * $Id: 7c36a796f23eace157e9f88c8fe280341f52562a $
  6.  *
  7.  * Authors: Gildas Bazin <gbazin@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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <vlc_common.h>
  30. #include <vlc_demux.h>
  31. #include <vlc_interface.h>
  32. #include <vlc_charset.h>
  33. #include "playlist.h"
  34. #ifndef LONG_MAX
  35. #   define LONG_MAX 2147483647L
  36. #   define LONG_MIN (-LONG_MAX-1)
  37. #endif
  38. /*****************************************************************************
  39.  * Local prototypes
  40.  *****************************************************************************/
  41. static int Demux( demux_t *p_demux);
  42. static int Control( demux_t *p_demux, int i_query, va_list args );
  43. static int ParseLine( char *, char **, char ***, int *);
  44. /*****************************************************************************
  45.  * Import_DVB: main import function
  46.  *****************************************************************************/
  47. int Import_DVB( vlc_object_t *p_this )
  48. {
  49.     demux_t *p_demux = (demux_t *)p_this;
  50.     const uint8_t *p_peek;
  51.     int     i_peek;
  52.     bool b_valid = false;
  53.     if( !demux_IsPathExtension( p_demux, ".conf" ) && !p_demux->b_force )
  54.         return VLC_EGENERIC;
  55.     /* Check if this really is a channels file */
  56.     if( (i_peek = stream_Peek( p_demux->s, &p_peek, 1024 )) > 0 )
  57.     {
  58.         char psz_line[1024+1];
  59.         int i;
  60.         for( i = 0; i < i_peek; i++ )
  61.         {
  62.             if( p_peek[i] == 'n' ) break;
  63.             psz_line[i] = p_peek[i];
  64.         }
  65.         psz_line[i] = 0;
  66.         if( ParseLine( psz_line, 0, 0, 0 ) ) b_valid = true;
  67.     }
  68.     if( !b_valid ) return VLC_EGENERIC;
  69.     msg_Dbg( p_demux, "found valid DVB conf playlist file");
  70.     p_demux->pf_control = Control;
  71.     p_demux->pf_demux = Demux;
  72.     return VLC_SUCCESS;
  73. }
  74. /*****************************************************************************
  75.  * Deactivate: frees unused data
  76.  *****************************************************************************/
  77. void Close_DVB( vlc_object_t *p_this )
  78. {
  79.     VLC_UNUSED(p_this);
  80. }
  81. /*****************************************************************************
  82.  * Demux: The important stuff
  83.  *****************************************************************************/
  84. static int Demux( demux_t *p_demux )
  85. {
  86.     char       *psz_line;
  87.     input_item_t *p_input;
  88.     INIT_PLAYLIST_STUFF;
  89.     while( (psz_line = stream_ReadLine( p_demux->s )) )
  90.     {
  91.         char **ppsz_options = NULL;
  92.         int  i_options = 0;
  93.         char *psz_name = NULL;
  94.         if( !ParseLine( psz_line, &psz_name, &ppsz_options, &i_options ) )
  95.         {
  96.             free( psz_line );
  97.             continue;
  98.         }
  99.         EnsureUTF8( psz_name );
  100.         for( int i = 0; i< i_options; i++ )
  101.             EnsureUTF8( ppsz_options[i] );
  102.         p_input = input_item_NewExt( p_demux, "dvb://", psz_name,
  103.                                      i_options, (const char**)ppsz_options, VLC_INPUT_OPTION_TRUSTED, -1 );
  104.         input_item_AddSubItem( p_current_input, p_input );
  105.         vlc_gc_decref( p_input );
  106.         while( i_options-- )
  107.             free( ppsz_options[i_options] );
  108.         free( ppsz_options );
  109.         free( psz_line );
  110.     }
  111.     HANDLE_PLAY_AND_RELEASE;
  112.     return 0; /* Needed for correct operation of go back */
  113. }
  114. static const struct
  115. {
  116.     const char *psz_name;
  117.     const char *psz_option;
  118. } dvb_options[] =
  119. {
  120.     { "INVERSION_OFF", "dvb-inversion=0" },
  121.     { "INVERSION_ON", "dvb-inversion=1" },
  122.     { "INVERSION_AUTO", "dvb-inversion=2" },
  123.     { "BANDWIDTH_AUTO", "dvb-bandwidth=0" },
  124.     { "BANDWIDTH_6_MHZ", "dvb-bandwidth=6" },
  125.     { "BANDWIDTH_7_MHZ", "dvb-bandwidth=7" },
  126.     { "BANDWIDTH_8_MHZ", "dvb-bandwidth=8" },
  127.     { "FEC_NONE", "dvb-fec=0" },
  128.     { "FEC_1_2", "dvb-fec=1" },
  129.     { "FEC_2_3", "dvb-fec=2" },
  130.     { "FEC_3_4", "dvb-fec=3" },
  131.     { "FEC_4_5", "dvb-fec=4" },
  132.     { "FEC_5_6", "dvb-fec=5" },
  133.     { "FEC_6_7", "dvb-fec=6" },
  134.     { "FEC_7_8", "dvb-fec=7" },
  135.     { "FEC_8_9", "dvb-fec=8" },
  136.     { "FEC_AUTO", "dvb-fec=9" },
  137.     { "GUARD_INTERVAL_AUTO", "dvb-guard=0" },
  138.     { "GUARD_INTERVAL_1_4", "dvb-guard=4" },
  139.     { "GUARD_INTERVAL_1_8", "dvb-guard=8" },
  140.     { "GUARD_INTERVAL_1_16", "dvb-guard=16" },
  141.     { "GUARD_INTERVAL_1_32", "dvb-guard=32" },
  142.     { "HIERARCHY_NONE", "dvb-hierarchy=-1" },
  143.     { "HIERARCHY_1", "dvb-hierarchy=1" },
  144.     { "HIERARCHY_2", "dvb-hierarchy=2" },
  145.     { "HIERARCHY_4", "dvb-hierarchy=4" },
  146.     { "QPSK", "dvb-modulation=-1" },
  147.     { "QAM_AUTO", "dvb-modulation=0" },
  148.     { "QAM_16", "dvb-modulation=16" },
  149.     { "QAM_32", "dvb-modulation=32" },
  150.     { "QAM_64", "dvb-modulation=64" },
  151.     { "QAM_128", "dvb-modulation=128" },
  152.     { "QAM_256", "dvb-modulation=256" },
  153.     { "8VSB", "dvb-modulation=8"  },
  154.     { "16VSB", "dvb-modulation=16"  },
  155.     { "TRANSMISSION_MODE_AUTO", "dvb-transmission=0" },
  156.     { "TRANSMISSION_MODE_2K", "dvb-transmission=2" },
  157.     { "TRANSMISSION_MODE_8K", "dvb-transmission=8" },
  158.     { 0, 0 }
  159. };
  160. static int ParseLine( char *psz_line, char **ppsz_name,
  161.                       char ***pppsz_options, int *pi_options )
  162. {
  163.     char *psz_name = NULL, *psz_parse = psz_line;
  164.     int i_count = 0, i_program = 0, i_frequency = 0, i_symbolrate = 0;
  165.     bool b_valid = false;
  166.     if( pppsz_options ) *pppsz_options = NULL;
  167.     if( pi_options ) *pi_options = 0;
  168.     if( ppsz_name ) *ppsz_name = NULL;
  169.     /* Skip leading tabs and spaces */
  170.     while( *psz_parse == ' ' || *psz_parse == 't' ||
  171.            *psz_parse == 'n' || *psz_parse == 'r' ) psz_parse++;
  172.     /* Ignore comments */
  173.     if( *psz_parse == '#' ) return false;
  174.     while( psz_parse )
  175.     {
  176.         const char *psz_option = NULL;
  177.         char *psz_end = strchr( psz_parse, ':' );
  178.         if( psz_end ) { *psz_end = 0; psz_end++; }
  179.         if( i_count == 0 )
  180.         {
  181.             /* Channel name */
  182.             psz_name = psz_parse;
  183.         }
  184.         else if( i_count == 1 )
  185.         {
  186.             /* Frequency */
  187.             char *psz_end;
  188.             long i_value;
  189.             i_value = strtol( psz_parse, &psz_end, 10 );
  190.             if( psz_end == psz_parse ||
  191.                 i_value == LONG_MAX || i_value == LONG_MIN ) break;
  192.             i_frequency = i_value;
  193.         }
  194.         else
  195.         {
  196.             int i;
  197.             /* Check option name with our list */
  198.             for( i = 0; dvb_options[i].psz_name; i++ )
  199.             {
  200.                 if( !strcmp( psz_parse, dvb_options[i].psz_name ) )
  201.                 {
  202.                     psz_option = dvb_options[i].psz_option;
  203.                     /* If we recognize one of the strings, then we are sure
  204.                      * the data is really valid (ie. a channels file). */
  205.                     b_valid = true;
  206.                     break;
  207.                 }
  208.             }
  209.             if( !psz_option )
  210.             {
  211.                 /* Option not recognized, test if it is a number */
  212.                 char *psz_end;
  213.                 long i_value;
  214.                 i_value = strtol( psz_parse, &psz_end, 10 );
  215.                 if( psz_end != psz_parse &&
  216.                     i_value != LONG_MAX && i_value != LONG_MIN &&
  217.                     !i_symbolrate )
  218.                 {
  219.                     i_symbolrate = i_value;
  220.                 }
  221.                 else if( psz_end != psz_parse &&
  222.                     i_value != LONG_MAX && i_value != LONG_MIN )
  223.                 {
  224.                     i_program = i_value;
  225.                 }
  226.             }
  227.         }
  228.         if( psz_option && pppsz_options && pi_options )
  229.         {
  230.             char *psz_dup = strdup( psz_option );
  231.             if (psz_dup != NULL)
  232.                 INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
  233.                              psz_dup );
  234.         }
  235.         psz_parse = psz_end;
  236.         i_count++;
  237.     }
  238.     if( !b_valid && pppsz_options && pi_options )
  239.     {
  240.         /* This isn't a valid channels file, cleanup everything */
  241.         while( (*pi_options)-- ) free( (*pppsz_options)[*pi_options] );
  242.         free( *pppsz_options );
  243.         *pppsz_options = NULL; *pi_options = 0;
  244.     }
  245.     if( i_program && pppsz_options && pi_options )
  246.     {
  247.         char *psz_option;
  248.         if( asprintf( &psz_option, "program=%i", i_program ) != -1 )
  249.             INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
  250.                          psz_option );
  251.     }
  252.     if( i_frequency && pppsz_options && pi_options )
  253.     {
  254.         char *psz_option;
  255.         if( asprintf( &psz_option, "dvb-frequency=%i", i_frequency ) != -1 )
  256.             INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
  257.                          psz_option );
  258.     }
  259.     if( i_symbolrate && pppsz_options && pi_options )
  260.     {
  261.         char *psz_option;
  262.         if( asprintf( &psz_option, "dvb-srate=%i", i_symbolrate ) != -1 )
  263.             INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
  264.                          psz_option );
  265.     }
  266.     if( ppsz_name && psz_name ) *ppsz_name = strdup( psz_name );
  267.     return b_valid;
  268. }
  269. static int Control( demux_t *p_demux, int i_query, va_list args )
  270. {
  271.     VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args);
  272.     return VLC_EGENERIC;
  273. }