Commit ff09a9da authored by Rémi Denis-Courmont's avatar Rémi Denis-Courmont

channels.conf: rewrite and fix

 - add support for DVB-S and ATSC
 - add new parameter values
 - fix code rates for DVB-T ("fec" != "code-rate-{h,l}p")
parent 960b7bb4
/*****************************************************************************
* dvb.c : DVB channel list import (szap/tzap/czap compatible channel lists)
* dvb.c: LinuxTV channels list
*****************************************************************************
* Copyright (C) 2005-20009 the VideoLAN team
* $Id$
......@@ -21,294 +21,323 @@
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
/*****************************************************************************
* Preamble
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <limits.h>
#include <assert.h>
#include <vlc_common.h>
#include <vlc_demux.h>
#include <vlc_charset.h>
#include "playlist.h"
#ifndef LONG_MAX
# define LONG_MAX 2147483647L
# define LONG_MIN (-LONG_MAX-1)
#endif
/*****************************************************************************
* Local prototypes
*****************************************************************************/
static int Demux( demux_t *p_demux);
static int Demux(demux_t *);
static input_item_t *ParseLine(char *line);
static int ParseLine( char *, char **, char ***, int *);
/*****************************************************************************
* Import_DVB: main import function
*****************************************************************************/
int Import_DVB( vlc_object_t *p_this )
/** Detect dvb-utils zap channels.conf format */
int Import_DVB(vlc_object_t *obj)
{
demux_t *p_demux = (demux_t *)p_this;
const uint8_t *p_peek;
int i_peek;
bool b_valid = false;
demux_t *demux = (demux_t *)obj;
if( !demux_IsPathExtension( p_demux, ".conf" ) && !p_demux->b_force )
if (!demux_IsPathExtension(demux, ".conf" ) && !demux->b_force )
return VLC_EGENERIC;
/* Check if this really is a channels file */
if( (i_peek = stream_Peek( p_demux->s, &p_peek, 1024 )) > 0 )
{
char psz_line[1024+1];
int i;
const uint8_t *peek;
int len = stream_Peek(demux->s, &peek, 1023);
if (len <= 0)
return VLC_EGENERIC;
for( i = 0; i < i_peek; i++ )
{
if( p_peek[i] == '\n' ) break;
psz_line[i] = p_peek[i];
}
psz_line[i] = 0;
const uint8_t *eol = memchr(peek, '\n', len);
if (eol == NULL)
return VLC_EGENERIC;
len = eol - peek;
if( ParseLine( psz_line, 0, 0, 0 ) ) b_valid = true;
}
char line[len + 1];
memcpy(line, peek, len);
line[len] = '\0';
if( !b_valid ) return VLC_EGENERIC;
input_item_t *item = ParseLine(line);
if (item == NULL)
return VLC_EGENERIC;
vlc_gc_decref(item);
msg_Dbg( p_demux, "found valid DVB conf playlist file");
p_demux->pf_control = Control;
p_demux->pf_demux = Demux;
msg_Dbg(demux, "found valid channels.conf file");
demux->pf_control = Control;
demux->pf_demux = Demux;
return VLC_SUCCESS;
}
/*****************************************************************************
* Demux: The important stuff
*****************************************************************************/
static int Demux( demux_t *p_demux )
/** Parses the whole channels.conf file */
static int Demux(demux_t *demux)
{
char *psz_line;
input_item_t *p_input;
input_item_t *p_current_input = GetCurrentItem(p_demux);
input_item_node_t *p_subitems = input_item_node_Create( p_current_input );
input_item_t *input = GetCurrentItem(demux);
input_item_node_t *subitems = input_item_node_Create(input);
char *line;
while( (psz_line = stream_ReadLine( p_demux->s )) )
while ((line = stream_ReadLine(demux->s)) != NULL)
{
char **ppsz_options = NULL;
int i_options = 0;
char *psz_name = NULL;
if( !ParseLine( psz_line, &psz_name, &ppsz_options, &i_options ) )
{
free( psz_line );
input_item_t *item = ParseLine(line);
if (item == NULL)
continue;
}
EnsureUTF8( psz_name );
for( int i = 0; i< i_options; i++ )
EnsureUTF8( ppsz_options[i] );
p_input = input_item_NewExt( "dvb://", psz_name,
i_options, (const char**)ppsz_options, VLC_INPUT_OPTION_TRUSTED, -1 );
input_item_node_AppendItem( p_subitems, p_input );
vlc_gc_decref( p_input );
while( i_options-- )
free( ppsz_options[i_options] );
free( ppsz_options );
free( psz_line );
input_item_node_AppendItem(subitems, item);
vlc_gc_decref(item);
}
input_item_node_PostAndDelete( p_subitems );
input_item_node_PostAndDelete(subitems);
vlc_gc_decref(input);
vlc_gc_decref(p_current_input);
return 0; /* Needed for correct operation of go back */
}
static const struct
static int cmp(const void *k, const void *e)
{
const char *psz_name;
const char *psz_option;
return strcmp(k, e);
}
} dvb_options[] =
static const char *ParseFEC(const char *str)
{
{ "INVERSION_OFF", "dvb-inversion=0" },
{ "INVERSION_ON", "dvb-inversion=1" },
{ "INVERSION_AUTO", "dvb-inversion=-1" },
{ "BANDWIDTH_AUTO", "dvb-bandwidth=0" },
{ "BANDWIDTH_6_MHZ", "dvb-bandwidth=6" },
{ "BANDWIDTH_7_MHZ", "dvb-bandwidth=7" },
{ "BANDWIDTH_8_MHZ", "dvb-bandwidth=8" },
{ "FEC_NONE", "dvb-fec=0" },
{ "FEC_1_2", "dvb-fec=1/2" },
{ "FEC_2_3", "dvb-fec=2/3" },
{ "FEC_3_4", "dvb-fec=3/4" },
{ "FEC_4_5", "dvb-fec=4/5" },
{ "FEC_5_6", "dvb-fec=5/6" },
{ "FEC_6_7", "dvb-fec=6/7" },
{ "FEC_7_8", "dvb-fec=7/8" },
{ "FEC_8_9", "dvb-fec=8/9" },
{ "FEC_AUTO", "dvb-fec=" },
{ "GUARD_INTERVAL_AUTO", "dvb-guard=" },
{ "GUARD_INTERVAL_1_4", "dvb-guard=1/4" },
{ "GUARD_INTERVAL_1_8", "dvb-guard=1/8" },
{ "GUARD_INTERVAL_1_16", "dvb-guard=1/16" },
{ "GUARD_INTERVAL_1_32", "dvb-guard=1/32" },
{ "HIERARCHY_NONE", "dvb-hierarchy=-1" },
{ "HIERARCHY_1", "dvb-hierarchy=1" },
{ "HIERARCHY_2", "dvb-hierarchy=2" },
{ "HIERARCHY_4", "dvb-hierarchy=4" },
{ "QPSK", "dvb-modulation=QPSK" },
{ "QAM_AUTO", "dvb-modulation=QAM" },
{ "QAM_16", "dvb-modulation=16QAM" },
{ "QAM_32", "dvb-modulation=32QAM" },
{ "QAM_64", "dvb-modulation=64QAM" },
{ "QAM_128", "dvb-modulation=128QAM" },
{ "QAM_256", "dvb-modulation=256QAM" },
{ "8VSB", "dvb-modulation=8VSB" },
{ "16VSB", "dvb-modulation=16VSB" },
{ "TRANSMISSION_MODE_AUTO", "dvb-transmission=-1" },
{ "TRANSMISSION_MODE_2K", "dvb-transmission=2" },
{ "TRANSMISSION_MODE_8K", "dvb-transmission=8" },
{ 0, 0 }
};
static int ParseLine( char *psz_line, char **ppsz_name,
char ***pppsz_options, int *pi_options )
{
char *psz_name = NULL, *psz_parse = psz_line;
int i_count = 0, i_program = 0, i_frequency = 0, i_symbolrate = 0;
bool b_valid = false;
if( pppsz_options ) *pppsz_options = NULL;
if( pi_options ) *pi_options = 0;
if( ppsz_name ) *ppsz_name = NULL;
/* Skip leading tabs and spaces */
while( *psz_parse == ' ' || *psz_parse == '\t' ||
*psz_parse == '\n' || *psz_parse == '\r' ) psz_parse++;
/* Ignore comments */
if( *psz_parse == '#' ) return false;
static const struct fec
{
char dvb[5];
char vlc[5];
} tab[] = {
{ "1_2", "1/2" }, { "2_3", "2/3" }, { "3_4", "3/4" },
{ "4_5", "4/5" }, { "5_6", "5/6" }, { "6_7", "6/7" },
{ "7_8", "7/8" }, { "8_9", "8/9" }, { "9_10", "9/10" },
{ "AUTO", "" }, { "NONE", "0" }
};
if (strncmp(str, "FEC_", 4))
return NULL;
str += 4;
const struct fec *f = bsearch(str, tab, sizeof (tab) / sizeof(tab[0]),
sizeof (tab[0]), cmp);
return (f != NULL) ? f->vlc : NULL;
}
while( psz_parse )
{
const char *psz_option = NULL;
char *psz_option_end = strchr( psz_parse, ':' );
if( psz_option_end ) { *psz_option_end = 0; psz_option_end++; }
if( i_count == 0 )
{
/* Channel name */
psz_name = psz_parse;
}
else if( i_count == 1 )
{
/* Frequency */
char *psz_end;
long i_value;
static const char *ParseModulation(const char *str)
{
static const struct mod
{
char dvb[9];
char vlc[7];
} tab[] = {
{ "APSK_16", "16APSK" }, { "APSK_32", "32APSK" },
{ "DQPSK", "DQPSK" }, { "PSK_8", "8PSK" }, { "QPSK", "QPSK" },
{ "QAM_128", "128QAM" }, { "QAM_16", "16QAM" },
{ "QAM_256", "256QAM" }, { "QAM_32", "32QAM" },
{ "QAM_64", "64QAM" }, { "QAM_AUTO", "QAM" },
{ "VSB_16", "16VSB" }, { "VSB_8", "8VSB" }
};
const struct mod *m = bsearch(str, tab, sizeof (tab) / sizeof(tab[0]),
sizeof (tab[0]), cmp);
return (m != NULL) ? m->vlc : NULL;
}
i_value = strtol( psz_parse, &psz_end, 10 );
if( psz_end == psz_parse ||
i_value == LONG_MAX || i_value == LONG_MIN ) break;
static const char *ParseGuard(const char *str)
{
static const struct guard
{
char dvb[7];
char vlc[7];
} tab[] = {
{ "19_128", "19/128" }, { "19_256", "19/256" }, { "1_128", "1/128" },
{ "1_16", "1/16" }, { "1_32", "1/32" }, { "1_4", "1/4" },
{ "1_8", "1/8" }, { "AUTO", "" },
};
if (strncmp(str, "GUARD_INTERVAL_", 15))
return NULL;
str += 15;
const struct guard *g = bsearch(str, tab, sizeof (tab) / sizeof(tab[0]),
sizeof (tab[0]), cmp);
return (g != NULL) ? g->vlc : NULL;
}
i_frequency = i_value;
}
/* http://www.linuxtv.org/vdrwiki/index.php/Syntax_of_channels.conf or not...
* Read the dvb-apps source code for reference. */
static input_item_t *ParseLine(char *line)
{
char *str, *end;
line += strspn(line, " \t\r"); /* skip leading white spaces */
if (*line == '#')
return NULL; /* skip comments */
/* Extract channel cute name */
char *name = strsep(&line, ":");
assert(name != NULL);
EnsureUTF8(name);
/* Extract central frequency */
str = strsep(&line, ":");
if (str == NULL)
return NULL;
unsigned long freq = strtoul(str, &end, 10);
if (*end)
return NULL;
/* Extract tuning parameters */
str = strsep(&line, ":");
if (str == NULL)
return NULL;
char *mrl;
if (!strcmp(str, "h") || !strcmp(str, "v"))
{ /* DVB-S */
char polarization = toupper(*str);
/* TODO: sat no. */
str = strsep(&line, ":");
if (str == NULL)
return NULL;
/* baud rate */
str = strsep(&line, ":");
if (str == NULL)
return NULL;
unsigned long rate = strtoul(str, &end, 10);
if (*end || rate > (ULONG_MAX / 1000u))
return NULL;
rate *= 1000;
if (asprintf(&mrl,
"dvb-s://frequency=%"PRIu64":polarization=%c:srate=%lu",
freq * UINT64_C(1000000), polarization, rate) == -1)
mrl = NULL;
}
else
if (!strncmp(str, "INVERSION_", 10))
{ /* DVB-C or DVB-T */
int inversion;
str += 10;
if (strcmp(str, "AUTO"))
inversion = -1;
else if (strcmp(str, "OFF"))
inversion = 0;
else if (strcmp(str, "ON"))
inversion = 1;
else
{
int i;
/* Check option name with our list */
for( i = 0; dvb_options[i].psz_name; i++ )
{
if( !strcmp( psz_parse, dvb_options[i].psz_name ) )
{
psz_option = dvb_options[i].psz_option;
/* If we recognize one of the strings, then we are sure
* the data is really valid (ie. a channels file). */
b_valid = true;
break;
}
}
if( !psz_option )
{
/* Option not recognized, test if it is a number */
char *psz_end;
long i_value;
i_value = strtol( psz_parse, &psz_end, 10 );
if( psz_end != psz_parse &&
i_value != LONG_MAX && i_value != LONG_MIN &&
!i_symbolrate )
{
i_symbolrate = i_value;
}
else if( psz_end != psz_parse &&
i_value != LONG_MAX && i_value != LONG_MIN )
{
i_program = i_value;
}
}
return NULL;
str = strsep(&line, ":");
if (str == NULL)
return NULL;
if (strncmp(str, "BANDWIDTH_", 10))
{ /* DVB-C */
unsigned long rate = strtoul(str, &end, 10);
if (*end)
return NULL;
str = strsep(&line, ":");
const char *fec = ParseFEC(str);
str = strsep(&line, ":");
const char *mod = ParseModulation(str);
if (fec == NULL || mod == NULL)
return NULL;
if (asprintf(&mrl, "dvb-c://frequency=%lu:inversion:%d:srate=%lu:"
"fec=%s:modulation=%s", freq, inversion, rate, fec,
mod) == -1)
mrl = NULL;
}
if( psz_option && pppsz_options && pi_options )
{
char *psz_dup = strdup( psz_option );
if (psz_dup != NULL)
INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
psz_dup );
else
{ /* DVB-T */
unsigned bandwidth = atoi(str + 10);
str = strsep(&line, ":");
const char *hp = ParseFEC(str);
str = strsep(&line, ":");
const char *lp = ParseFEC(str);
str = strsep(&line, ":");
const char *mod = ParseModulation(str);
if (hp == NULL || lp == NULL || mod == NULL)
return NULL;
str = strsep(&line, ":");
if (str == NULL || strncmp(str, "TRANSMISSION_MODE_", 18))
return NULL;
int xmit = atoi(str);
if (xmit == 0)
xmit = -1; /* AUTO */
str = strsep(&line, ":");
const char *guard = ParseGuard(str);
if (guard == NULL)
return NULL;
str = strsep(&line, ":");
if (str == NULL || strncmp(str, "HIERARCHY_", 10))
return NULL;
str += 10;
int hierarchy = atoi(str);
if (!strcmp(str, "AUTO"))
hierarchy = -1;
if (asprintf(&mrl, "dvb-t://frequency=%lu:inversion=%d:"
"bandwidth=%u:code-rate-hp=%s:code-rate-lp=%s:"
"modulation=%s:transmission=%d:guard=%s:"
"hierarchy=%d", freq, inversion, bandwidth, hp, lp,
mod, xmit, guard, hierarchy) == -1)
mrl = NULL;
}
psz_parse = psz_option_end;
i_count++;
}
if( !b_valid && pppsz_options && pi_options )
{
/* This isn't a valid channels file, cleanup everything */
while( (*pi_options)-- ) free( (*pppsz_options)[*pi_options] );
free( *pppsz_options );
*pppsz_options = NULL; *pi_options = 0;
else
{ /* ATSC */
const char *mod = ParseModulation(str);
if (mod == NULL)
return NULL;
if (asprintf(&mrl, "atsc://frequency=%lu:modulation=%s", freq,
mod) == -1)
mrl = NULL;
}
if( i_program && pppsz_options && pi_options )
{
char *psz_option;
if (unlikely(mrl == NULL))
return NULL;
if( asprintf( &psz_option, "program=%i", i_program ) != -1 )
INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
psz_option );
}
if( i_frequency && pppsz_options && pi_options )
/* Video PID (TODO? set video track) */
strsep(&line, ":");
/* Audio PID (TODO? set audio track) */
strsep(&line, ":");
/* Extract SID */
str = strsep(&line, ":");
if (str == NULL)
{
char *psz_option;
if( asprintf( &psz_option, "dvb-frequency=%i", i_frequency ) != -1 )
INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
psz_option );
free(mrl);
return NULL;
}
if( i_symbolrate && pppsz_options && pi_options )
unsigned long sid = strtoul(str, &end, 10);
if (*end || sid > 65535)
{
char *psz_option;
if( asprintf( &psz_option, "dvb-srate=%i", i_symbolrate ) != -1 )
INSERT_ELEM( *pppsz_options, (*pi_options), (*pi_options),
psz_option );
free(mrl);
return NULL;
}
if( ppsz_name && psz_name ) *ppsz_name = strdup( psz_name );
return b_valid;
char sid_opt[sizeof("program=65535")];
snprintf(sid_opt, sizeof(sid_opt), "program=%lu", sid);
const char *opts[] = { sid_opt };
input_item_t *item = input_item_NewWithType(mrl, name, 1, opts, 0, -1,
ITEM_TYPE_CARD);
free(mrl);
return item;
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment