Use DVB API v5 aka S2API instead of obsolete v3 API

Patch is based on work of Tony Houghton <h@realh.co.uk>

https://bugzilla.gnome.org/show_bug.cgi?id=654479
This commit is contained in:
Sebastian Pölsterl 2012-10-26 18:00:02 +02:00 committed by Edward Hervey
parent 43fea50c8a
commit 5d406192c3

View File

@ -68,6 +68,8 @@ GST_DEBUG_CATEGORY_STATIC (gstdvbsrc_debug);
#define LOF1 (9750*1000UL) #define LOF1 (9750*1000UL)
#define LOF2 (10600*1000UL) #define LOF2 (10600*1000UL)
#define NUM_DTV_PROPS 16
/* Arguments */ /* Arguments */
enum enum
{ {
@ -1260,21 +1262,32 @@ diseqc (int secfd, int sat_no, int voltage, int tone)
} }
inline static void
set_prop (struct dtv_property *props, int *n, guint32 cmd, guint32 data)
{
if (*n == NUM_DTV_PROPS) {
g_critical ("Index out of bounds");
} else {
props[*n].cmd = cmd;
props[(*n)++].u.data = data;
}
}
static gboolean static gboolean
gst_dvbsrc_tune (GstDvbSrc * object) gst_dvbsrc_tune (GstDvbSrc * object)
{ {
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 struct dtv_properties props;
struct dvbfe_params feparams; struct dtv_property dvb_prop[NUM_DTV_PROPS];
#else
struct dvb_frontend_parameters feparams;
#endif
fe_sec_voltage_t voltage; fe_sec_voltage_t voltage;
fe_status_t status; fe_status_t status;
int n;
int i; int i;
int j; int j;
unsigned int del_sys = SYS_UNDEFINED;
unsigned int freq = object->freq; unsigned int freq = object->freq;
unsigned int sym_rate = object->sym_rate * 1000; unsigned int sym_rate = object->sym_rate * 1000;
unsigned int bandwidth;
int inversion = object->inversion;
/* found in mail archive on linuxtv.org /* found in mail archive on linuxtv.org
* What works well for us is: * What works well for us is:
@ -1291,37 +1304,42 @@ gst_dvbsrc_tune (GstDvbSrc * object)
return FALSE; return FALSE;
} }
GST_DEBUG_OBJECT (object, "api version %d.%d", DVB_API_VERSION,
DVB_API_VERSION_MINOR);
gst_dvbsrc_unset_pes_filters (object); gst_dvbsrc_unset_pes_filters (object);
for (j = 0; j < 5; j++) { for (j = 0; j < 5; j++) {
memset (dvb_prop, 0, sizeof (dvb_prop));
dvb_prop[0].cmd = DTV_CLEAR;
props.num = 1;
props.props = dvb_prop;
if (ioctl (object->fd_frontend, FE_SET_PROPERTY, &props) < 0) {
g_warning ("Error resetting tuner: %s", strerror (errno));
}
/* First three entries are reserved */
n = 3;
switch (object->adapter_type) { switch (object->adapter_type) {
case FE_QPSK: case FE_QPSK:
object->tone = SEC_TONE_OFF; object->tone = SEC_TONE_OFF;
if (freq > 2200000) { if (freq > 2200000) {
// this must be an absolute frequency // this must be an absolute frequency
if (freq < SLOF) { if (freq < SLOF) {
feparams.frequency = (freq - LOF1); freq -= LOF1;
} else { } else {
feparams.frequency = (freq - LOF2); freq -= LOF2;
object->tone = SEC_TONE_ON; object->tone = SEC_TONE_ON;
} }
} else {
// this is an L-Band frequency
feparams.frequency = freq;
} }
feparams.inversion = INVERSION_AUTO;
GST_DEBUG_OBJECT (object, "api version %d.%d", DVB_API_VERSION, inversion = INVERSION_AUTO;
DVB_API_VERSION_MINOR); set_prop (dvb_prop, &n, DTV_SYMBOL_RATE, sym_rate);
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 set_prop (dvb_prop, &n, DTV_INNER_FEC, object->code_rate_hp);
GST_DEBUG_OBJECT (object, "using multiproto driver"); /* TODO add dvbs2 */
feparams.delsys.dvbs.symbol_rate = sym_rate; del_sys = SYS_DVBS;
feparams.delsys.dvbs.fec = object->code_rate_hp;
#else
feparams.u.qpsk.symbol_rate = sym_rate;
feparams.u.qpsk.fec_inner = object->code_rate_hp;
#endif
GST_INFO_OBJECT (object, GST_INFO_OBJECT (object,
"tuning DVB-S to L-Band:%u, Pol:%d, srate=%u, 22kHz=%s", "tuning DVB-S to L-Band:%u, Pol:%d, srate=%u, 22kHz=%s",
feparams.frequency, object->pol, sym_rate, freq, object->pol, sym_rate,
object->tone == SEC_TONE_ON ? "on" : "off"); object->tone == SEC_TONE_ON ? "on" : "off");
if (object->pol == DVB_POL_H) if (object->pol == DVB_POL_H)
@ -1330,13 +1348,10 @@ gst_dvbsrc_tune (GstDvbSrc * object)
voltage = SEC_VOLTAGE_13; voltage = SEC_VOLTAGE_13;
if (object->diseqc_src == -1 || object->send_diseqc == FALSE) { if (object->diseqc_src == -1 || object->send_diseqc == FALSE) {
if (ioctl (object->fd_frontend, FE_SET_VOLTAGE, voltage) < 0) { set_prop (dvb_prop, &n, DTV_VOLTAGE, voltage);
g_warning ("Unable to set voltage on dvb frontend device");
}
if (ioctl (object->fd_frontend, FE_SET_TONE, object->tone) < 0) { // DTV_TONE not yet implemented
g_warning ("Error setting tone: %s", strerror (errno)); // set_prop (fe_props_array, &n, DTV_TONE, object->tone)
}
} else { } else {
GST_DEBUG_OBJECT (object, "Sending DISEqC"); GST_DEBUG_OBJECT (object, "Sending DISEqC");
diseqc (object->fd_frontend, object->diseqc_src, voltage, diseqc (object->fd_frontend, object->diseqc_src, voltage,
@ -1348,48 +1363,52 @@ gst_dvbsrc_tune (GstDvbSrc * object)
break; break;
case FE_OFDM: case FE_OFDM:
del_sys = SYS_DVBT;
feparams.frequency = freq; bandwidth = 0;
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 if (object->bandwidth != BANDWIDTH_AUTO) {
feparams.delsys.dvbs.fec = object->code_rate_hp; /* Presumably not specifying bandwidth with s2api is equivalent
feparams.delsys.dvbs.modulation = object->modulation; * to BANDWIDTH_AUTO.
feparams.delsys.dvbs.symbol_rate = sym_rate; */
#else switch (object->bandwidth) {
feparams.u.ofdm.bandwidth = object->bandwidth; case BANDWIDTH_8_MHZ:
feparams.u.ofdm.code_rate_HP = object->code_rate_hp; bandwidth = 8000000;
feparams.u.ofdm.code_rate_LP = object->code_rate_lp; break;
feparams.u.ofdm.constellation = object->modulation; case BANDWIDTH_7_MHZ:
feparams.u.ofdm.transmission_mode = object->transmission_mode; bandwidth = 7000000;
feparams.u.ofdm.guard_interval = object->guard_interval; break;
feparams.u.ofdm.hierarchy_information = object->hierarchy_information; case BANDWIDTH_6_MHZ:
#endif bandwidth = 6000000;
feparams.inversion = object->inversion; break;
default:
break;
}
}
if (bandwidth) {
set_prop (dvb_prop, &n, DTV_BANDWIDTH_HZ, bandwidth);
}
set_prop (dvb_prop, &n, DTV_CODE_RATE_HP, object->code_rate_hp);
set_prop (dvb_prop, &n, DTV_CODE_RATE_LP, object->code_rate_lp);
set_prop (dvb_prop, &n, DTV_MODULATION, object->modulation);
set_prop (dvb_prop, &n, DTV_TRANSMISSION_MODE,
object->transmission_mode);
set_prop (dvb_prop, &n, DTV_GUARD_INTERVAL, object->guard_interval);
set_prop (dvb_prop, &n, DTV_HIERARCHY, object->hierarchy_information);
GST_INFO_OBJECT (object, "tuning DVB-T to %d Hz", freq); GST_INFO_OBJECT (object, "tuning DVB-T to %d Hz", freq);
break; break;
case FE_QAM: case FE_QAM:
GST_INFO_OBJECT (object, "Tuning DVB-C to %d, srate=%d", freq, GST_INFO_OBJECT (object, "Tuning DVB-C to %d, srate=%d", freq,
sym_rate); sym_rate);
feparams.frequency = freq;
feparams.inversion = object->inversion; del_sys = SYS_DVBC_ANNEX_AC;
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 set_prop (dvb_prop, &n, DTV_INNER_FEC, object->code_rate_hp);
feparams.delsys.dvbs.fec = object->code_rate_hp; set_prop (dvb_prop, &n, DTV_MODULATION, object->modulation);
feparams.delsys.dvbs.modulation = object->modulation; set_prop (dvb_prop, &n, DTV_SYMBOL_RATE, sym_rate);
feparams.delsys.dvbs.symbol_rate = sym_rate;
#else
feparams.u.qam.fec_inner = object->code_rate_hp;
feparams.u.qam.modulation = object->modulation;
feparams.u.qam.symbol_rate = sym_rate;
#endif
break; break;
case FE_ATSC: case FE_ATSC:
GST_INFO_OBJECT (object, "Tuning ATSC to %d", freq); GST_INFO_OBJECT (object, "Tuning ATSC to %d", freq);
feparams.frequency = freq; del_sys = SYS_ATSC;
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 set_prop (dvb_prop, &n, DTV_MODULATION, object->modulation);
feparams.delsys.atsc.modulation = object->modulation;
#else
feparams.u.vsb.modulation = object->modulation;
#endif
break; break;
default: default:
g_error ("Unknown frontend type: %d", object->adapter_type); g_error ("Unknown frontend type: %d", object->adapter_type);
@ -1397,11 +1416,17 @@ gst_dvbsrc_tune (GstDvbSrc * object)
} }
usleep (100000); usleep (100000);
/* now tune the frontend */ /* now tune the frontend */
#if DVB_API_VERSION == 3 && DVB_API_VERSION_MINOR == 3 set_prop (dvb_prop, &n, DTV_TUNE, 0);
if (ioctl (object->fd_frontend, DVBFE_SET_PARAMS, &feparams) < 0) { props.num = n;
#else props.props = dvb_prop;
if (ioctl (object->fd_frontend, FE_SET_FRONTEND, &feparams) < 0) { /* set first three entries */
#endif n = 0;
set_prop (dvb_prop, &n, DTV_DELIVERY_SYSTEM, del_sys);
set_prop (dvb_prop, &n, DTV_FREQUENCY, freq);
set_prop (dvb_prop, &n, DTV_INVERSION, inversion);
GST_DEBUG_OBJECT (object, "Setting %d properties", props.num);
if (ioctl (object->fd_frontend, FE_SET_PROPERTY, &props) < 0) {
g_warning ("Error tuning channel: %s", strerror (errno)); g_warning ("Error tuning channel: %s", strerror (errno));
} }
for (i = 0; i < 50; i++) { for (i = 0; i < 50; i++) {