Commit 6c5637e4 authored by Malcolm Priestley's avatar Malcolm Priestley Committed by Mauro Carvalho Chehab

[media] it913x-fe ver 1.15 read signal strenght using reg VAR_P_INBAND

Read signal strength using VAR_P_INBAND and apply FEC preferred values.

Note this does not work on IT9137 devices even with dvb-usb-it9135-01.fw
firmware.

Config read_sl allows switch between read signal strength and signal
level.
Signed-off-by: default avatarMalcolm Priestley <tvboxspy@gmail.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent aaa589fc
...@@ -412,11 +412,13 @@ static int ite_firmware_select(struct usb_device *udev, ...@@ -412,11 +412,13 @@ static int ite_firmware_select(struct usb_device *udev,
case IT9135_V1_FW: case IT9135_V1_FW:
it913x_config.firmware_ver = 1; it913x_config.firmware_ver = 1;
it913x_config.adc_x2 = 1; it913x_config.adc_x2 = 1;
it913x_config.read_slevel = false;
props->firmware = fw_it9135_v1; props->firmware = fw_it9135_v1;
break; break;
case IT9135_V2_FW: case IT9135_V2_FW:
it913x_config.firmware_ver = 1; it913x_config.firmware_ver = 1;
it913x_config.adc_x2 = 1; it913x_config.adc_x2 = 1;
it913x_config.read_slevel = false;
props->firmware = fw_it9135_v2; props->firmware = fw_it9135_v2;
switch (it913x_config.tuner_id_0) { switch (it913x_config.tuner_id_0) {
case IT9135_61: case IT9135_61:
...@@ -432,6 +434,7 @@ static int ite_firmware_select(struct usb_device *udev, ...@@ -432,6 +434,7 @@ static int ite_firmware_select(struct usb_device *udev,
default: default:
it913x_config.firmware_ver = 0; it913x_config.firmware_ver = 0;
it913x_config.adc_x2 = 0; it913x_config.adc_x2 = 0;
it913x_config.read_slevel = true;
props->firmware = fw_it9137; props->firmware = fw_it9137;
} }
......
...@@ -201,6 +201,11 @@ fe_modulation_t fe_con[] = { ...@@ -201,6 +201,11 @@ fe_modulation_t fe_con[] = {
QAM_64, QAM_64,
}; };
enum {
PRIORITY_HIGH = 0, /* High-priority stream */
PRIORITY_LOW, /* Low-priority stream */
};
/* Standard demodulator functions */ /* Standard demodulator functions */
static struct it913xset set_solo_fe[] = { static struct it913xset set_solo_fe[] = {
{PRO_LINK, GPIOH5_EN, {0x01}, 0x01}, {PRO_LINK, GPIOH5_EN, {0x01}, 0x01},
......
...@@ -57,6 +57,7 @@ struct it913x_fe_state { ...@@ -57,6 +57,7 @@ struct it913x_fe_state {
u32 frequency; u32 frequency;
fe_modulation_t constellation; fe_modulation_t constellation;
fe_transmit_mode_t transmission_mode; fe_transmit_mode_t transmission_mode;
u8 priority;
u32 crystalFrequency; u32 crystalFrequency;
u32 adcFrequency; u32 adcFrequency;
u8 tuner_type; u8 tuner_type;
...@@ -500,19 +501,87 @@ static int it913x_fe_read_status(struct dvb_frontend *fe, fe_status_t *status) ...@@ -500,19 +501,87 @@ static int it913x_fe_read_status(struct dvb_frontend *fe, fe_status_t *status)
return 0; return 0;
} }
/* FEC values based on fe_code_rate_t non supported values 0*/
int it913x_qpsk_pval[] = {0, -93, -91, -90, 0, -89, -88};
int it913x_16qam_pval[] = {0, -87, -85, -84, 0, -83, -82};
int it913x_64qam_pval[] = {0, -82, -80, -78, 0, -77, -76};
static int it913x_get_signal_strength(struct dvb_frontend *fe)
{
struct dtv_frontend_properties *p = &fe->dtv_property_cache;
struct it913x_fe_state *state = fe->demodulator_priv;
u8 code_rate;
int ret, temp;
u8 lna_gain_os;
ret = it913x_read_reg_u8(state, VAR_P_INBAND);
if (ret < 0)
return ret;
/* VHF/UHF gain offset */
if (state->frequency < 300000000)
lna_gain_os = 7;
else
lna_gain_os = 14;
temp = (ret - 100) - lna_gain_os;
if (state->priority == PRIORITY_HIGH)
code_rate = p->code_rate_HP;
else
code_rate = p->code_rate_LP;
if (code_rate >= ARRAY_SIZE(it913x_qpsk_pval))
return -EINVAL;
deb_info("Reg VAR_P_INBAND:%d Calc Offset Value:%d", ret, temp);
/* Apply FEC offset values*/
switch (p->modulation) {
case QPSK:
temp -= it913x_qpsk_pval[code_rate];
break;
case QAM_16:
temp -= it913x_16qam_pval[code_rate];
break;
case QAM_64:
temp -= it913x_64qam_pval[code_rate];
break;
default:
return -EINVAL;
}
if (temp < -15)
ret = 0;
else if ((-15 <= temp) && (temp < 0))
ret = (2 * (temp + 15)) / 3;
else if ((0 <= temp) && (temp < 20))
ret = 4 * temp + 10;
else if ((20 <= temp) && (temp < 35))
ret = (2 * (temp - 20)) / 3 + 90;
else if (temp >= 35)
ret = 100;
deb_info("Signal Strength :%d", ret);
return ret;
}
static int it913x_fe_read_signal_strength(struct dvb_frontend *fe, static int it913x_fe_read_signal_strength(struct dvb_frontend *fe,
u16 *strength) u16 *strength)
{ {
struct it913x_fe_state *state = fe->demodulator_priv; struct it913x_fe_state *state = fe->demodulator_priv;
int ret = it913x_read_reg_u8(state, SIGNAL_LEVEL); int ret = 0;
/*SIGNAL_LEVEL always returns 100%! so using FE_HAS_SIGNAL as switch*/ if (state->config->read_slevel) {
if (state->it913x_status & FE_HAS_SIGNAL) if (state->it913x_status & FE_HAS_SIGNAL)
ret = (ret * 0xff) / 0x64; ret = it913x_read_reg_u8(state, SIGNAL_LEVEL);
else } else
ret = 0x0; ret = it913x_get_signal_strength(fe);
ret |= ret << 0x8;
*strength = ret; if (ret >= 0)
return 0; *strength = (u16)((u32)ret * 0xffff / 0x64);
return (ret < 0) ? -ENODEV : 0;
} }
static int it913x_fe_read_snr(struct dvb_frontend *fe, u16 *snr) static int it913x_fe_read_snr(struct dvb_frontend *fe, u16 *snr)
...@@ -606,6 +675,8 @@ static int it913x_fe_get_frontend(struct dvb_frontend *fe) ...@@ -606,6 +675,8 @@ static int it913x_fe_get_frontend(struct dvb_frontend *fe)
if (reg[2] < 4) if (reg[2] < 4)
p->hierarchy = fe_hi[reg[2]]; p->hierarchy = fe_hi[reg[2]];
state->priority = reg[5];
p->code_rate_HP = (reg[6] < 6) ? fe_code[reg[6]] : FEC_NONE; p->code_rate_HP = (reg[6] < 6) ? fe_code[reg[6]] : FEC_NONE;
p->code_rate_LP = (reg[7] < 6) ? fe_code[reg[7]] : FEC_NONE; p->code_rate_LP = (reg[7] < 6) ? fe_code[reg[7]] : FEC_NONE;
...@@ -972,5 +1043,5 @@ static struct dvb_frontend_ops it913x_fe_ofdm_ops = { ...@@ -972,5 +1043,5 @@ static struct dvb_frontend_ops it913x_fe_ofdm_ops = {
MODULE_DESCRIPTION("it913x Frontend and it9137 tuner"); MODULE_DESCRIPTION("it913x Frontend and it9137 tuner");
MODULE_AUTHOR("Malcolm Priestley tvboxspy@gmail.com"); MODULE_AUTHOR("Malcolm Priestley tvboxspy@gmail.com");
MODULE_VERSION("1.13"); MODULE_VERSION("1.15");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
...@@ -34,6 +34,8 @@ struct ite_config { ...@@ -34,6 +34,8 @@ struct ite_config {
u8 tuner_id_1; u8 tuner_id_1;
u8 dual_mode; u8 dual_mode;
u8 adf; u8 adf;
/* option to read SIGNAL_LEVEL */
u8 read_slevel;
}; };
#if defined(CONFIG_DVB_IT913X_FE) || (defined(CONFIG_DVB_IT913X_FE_MODULE) && \ #if defined(CONFIG_DVB_IT913X_FE) || (defined(CONFIG_DVB_IT913X_FE_MODULE) && \
...@@ -168,6 +170,8 @@ static inline struct dvb_frontend *it913x_fe_attach( ...@@ -168,6 +170,8 @@ static inline struct dvb_frontend *it913x_fe_attach(
#define EST_SIGNAL_LEVEL 0x004a #define EST_SIGNAL_LEVEL 0x004a
#define FREE_BAND 0x004b #define FREE_BAND 0x004b
#define SUSPEND_FLAG 0x004c #define SUSPEND_FLAG 0x004c
#define VAR_P_INBAND 0x00f7
/* Build in tuner types */ /* Build in tuner types */
#define IT9137 0x38 #define IT9137 0x38
#define IT9135_38 0x38 #define IT9135_38 0x38
......
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