Commit 267897a4 authored by Antti Palosaari's avatar Antti Palosaari Committed by Mauro Carvalho Chehab

[media] tda10071: implement DVBv5 statistics

Implement DVBv5 CNR, signal strength, BER and block errors.

Wrap legacy DVBv3 statistics to DVBv5 internally.
Signed-off-by: default avatarAntti Palosaari <crope@iki.fi>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@osg.samsung.com>
parent 4c4acb7a
...@@ -377,8 +377,11 @@ static int tda10071_read_status(struct dvb_frontend *fe, enum fe_status *status) ...@@ -377,8 +377,11 @@ static int tda10071_read_status(struct dvb_frontend *fe, enum fe_status *status)
{ {
struct tda10071_dev *dev = fe->demodulator_priv; struct tda10071_dev *dev = fe->demodulator_priv;
struct i2c_client *client = dev->client; struct i2c_client *client = dev->client;
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
struct tda10071_cmd cmd;
int ret; int ret;
unsigned int uitmp; unsigned int uitmp;
u8 buf[8];
*status = 0; *status = 0;
...@@ -401,71 +404,105 @@ static int tda10071_read_status(struct dvb_frontend *fe, enum fe_status *status) ...@@ -401,71 +404,105 @@ static int tda10071_read_status(struct dvb_frontend *fe, enum fe_status *status)
dev->fe_status = *status; dev->fe_status = *status;
return ret; /* signal strength */
error: if (dev->fe_status & FE_HAS_SIGNAL) {
dev_dbg(&client->dev, "failed=%d\n", ret); cmd.args[0] = CMD_GET_AGCACC;
return ret; cmd.args[1] = 0;
} cmd.len = 2;
ret = tda10071_cmd_execute(dev, &cmd);
if (ret)
goto error;
static int tda10071_read_snr(struct dvb_frontend *fe, u16 *snr) /* input power estimate dBm */
{ ret = regmap_read(dev->regmap, 0x50, &uitmp);
struct tda10071_dev *dev = fe->demodulator_priv; if (ret)
struct i2c_client *client = dev->client; goto error;
int ret;
u8 buf[2];
if (!dev->warm || !(dev->fe_status & FE_HAS_LOCK)) { c->strength.stat[0].scale = FE_SCALE_DECIBEL;
*snr = 0; c->strength.stat[0].svalue = (int) (uitmp - 256) * 1000;
ret = 0; } else {
goto error; c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
} }
ret = regmap_bulk_read(dev->regmap, 0x3a, buf, 2); /* CNR */
if (ret) if (dev->fe_status & FE_HAS_VITERBI) {
goto error; /* Es/No */
ret = regmap_bulk_read(dev->regmap, 0x3a, buf, 2);
if (ret)
goto error;
/* Es/No dBx10 */ c->cnr.stat[0].scale = FE_SCALE_DECIBEL;
*snr = buf[0] << 8 | buf[1]; c->cnr.stat[0].svalue = (buf[0] << 8 | buf[1] << 0) * 100;
} else {
c->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
}
return ret; /* UCB/PER/BER */
error: if (dev->fe_status & FE_HAS_LOCK) {
dev_dbg(&client->dev, "failed=%d\n", ret); /* TODO: report total bits/packets */
return ret; u8 delivery_system, reg, len;
}
static int tda10071_read_signal_strength(struct dvb_frontend *fe, u16 *strength) switch (dev->delivery_system) {
{ case SYS_DVBS:
struct tda10071_dev *dev = fe->demodulator_priv; reg = 0x4c;
struct i2c_client *client = dev->client; len = 8;
struct tda10071_cmd cmd; delivery_system = 1;
int ret; break;
unsigned int uitmp; case SYS_DVBS2:
reg = 0x4d;
len = 4;
delivery_system = 0;
break;
default:
ret = -EINVAL;
goto error;
}
if (!dev->warm || !(dev->fe_status & FE_HAS_LOCK)) { ret = regmap_read(dev->regmap, reg, &uitmp);
*strength = 0; if (ret)
ret = 0; goto error;
goto error;
}
cmd.args[0] = CMD_GET_AGCACC; if (dev->meas_count == uitmp) {
cmd.args[1] = 0; dev_dbg(&client->dev, "meas not ready=%02x\n", uitmp);
cmd.len = 2; ret = 0;
ret = tda10071_cmd_execute(dev, &cmd); goto error;
if (ret) } else {
goto error; dev->meas_count = uitmp;
}
/* input power estimate dBm */ cmd.args[0] = CMD_BER_UPDATE_COUNTERS;
ret = regmap_read(dev->regmap, 0x50, &uitmp); cmd.args[1] = 0;
if (ret) cmd.args[2] = delivery_system;
goto error; cmd.len = 3;
ret = tda10071_cmd_execute(dev, &cmd);
if (ret)
goto error;
if (uitmp < 181) ret = regmap_bulk_read(dev->regmap, cmd.len, buf, len);
uitmp = 181; /* -75 dBm */ if (ret)
else if (uitmp > 236) goto error;
uitmp = 236; /* -20 dBm */
/* scale value to 0x0000-0xffff */ if (dev->delivery_system == SYS_DVBS) {
*strength = (uitmp-181) * 0xffff / (236-181); dev->dvbv3_ber = buf[0] << 24 | buf[1] << 16 |
buf[2] << 8 | buf[3] << 0;
dev->post_bit_error += buf[0] << 24 | buf[1] << 16 |
buf[2] << 8 | buf[3] << 0;
c->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
c->post_bit_error.stat[0].uvalue = dev->post_bit_error;
dev->block_error += buf[4] << 8 | buf[5] << 0;
c->block_error.stat[0].scale = FE_SCALE_COUNTER;
c->block_error.stat[0].uvalue = dev->block_error;
} else {
dev->dvbv3_ber = buf[0] << 8 | buf[1] << 0;
dev->post_bit_error += buf[0] << 8 | buf[1] << 0;
c->post_bit_error.stat[0].scale = FE_SCALE_COUNTER;
c->post_bit_error.stat[0].uvalue = dev->post_bit_error;
c->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
}
} else {
c->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
c->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
}
return ret; return ret;
error: error:
...@@ -473,94 +510,50 @@ static int tda10071_read_signal_strength(struct dvb_frontend *fe, u16 *strength) ...@@ -473,94 +510,50 @@ static int tda10071_read_signal_strength(struct dvb_frontend *fe, u16 *strength)
return ret; return ret;
} }
static int tda10071_read_ber(struct dvb_frontend *fe, u32 *ber) static int tda10071_read_snr(struct dvb_frontend *fe, u16 *snr)
{ {
struct tda10071_dev *dev = fe->demodulator_priv; struct dtv_frontend_properties *c = &fe->dtv_property_cache;
struct i2c_client *client = dev->client;
struct tda10071_cmd cmd;
int ret, i, len;
unsigned int uitmp;
u8 reg, buf[8];
if (!dev->warm || !(dev->fe_status & FE_HAS_LOCK)) {
*ber = dev->ber = 0;
ret = 0;
goto error;
}
switch (dev->delivery_system) { if (c->cnr.stat[0].scale == FE_SCALE_DECIBEL)
case SYS_DVBS: *snr = div_s64(c->cnr.stat[0].svalue, 100);
reg = 0x4c; else
len = 8; *snr = 0;
i = 1; return 0;
break; }
case SYS_DVBS2:
reg = 0x4d;
len = 4;
i = 0;
break;
default:
*ber = dev->ber = 0;
return 0;
}
ret = regmap_read(dev->regmap, reg, &uitmp); static int tda10071_read_signal_strength(struct dvb_frontend *fe, u16 *strength)
if (ret) {
goto error; struct dtv_frontend_properties *c = &fe->dtv_property_cache;
unsigned int uitmp;
if (dev->meas_count[i] == uitmp) { if (c->strength.stat[0].scale == FE_SCALE_DECIBEL) {
dev_dbg(&client->dev, "meas not ready=%02x\n", uitmp); uitmp = c->strength.stat[0].svalue / 1000 + 256;
*ber = dev->ber; uitmp = clamp(uitmp, 181U, 236U); /* -75dBm - -20dBm */
return 0; /* scale value to 0x0000-0xffff */
*strength = (uitmp-181) * 0xffff / (236-181);
} else { } else {
dev->meas_count[i] = uitmp; *strength = 0;
} }
return 0;
}
cmd.args[0] = CMD_BER_UPDATE_COUNTERS; static int tda10071_read_ber(struct dvb_frontend *fe, u32 *ber)
cmd.args[1] = 0; {
cmd.args[2] = i; struct tda10071_dev *dev = fe->demodulator_priv;
cmd.len = 3;
ret = tda10071_cmd_execute(dev, &cmd);
if (ret)
goto error;
ret = regmap_bulk_read(dev->regmap, cmd.len, buf, len);
if (ret)
goto error;
if (dev->delivery_system == SYS_DVBS) {
*ber = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3];
dev->ucb += (buf[4] << 8) | buf[5];
} else {
*ber = (buf[0] << 8) | buf[1];
}
dev->ber = *ber;
return ret; *ber = dev->dvbv3_ber;
error: return 0;
dev_dbg(&client->dev, "failed=%d\n", ret);
return ret;
} }
static int tda10071_read_ucblocks(struct dvb_frontend *fe, u32 *ucblocks) static int tda10071_read_ucblocks(struct dvb_frontend *fe, u32 *ucblocks)
{ {
struct tda10071_dev *dev = fe->demodulator_priv; struct dtv_frontend_properties *c = &fe->dtv_property_cache;
struct i2c_client *client = dev->client;
int ret = 0;
if (!dev->warm || !(dev->fe_status & FE_HAS_LOCK)) { if (c->block_error.stat[0].scale == FE_SCALE_COUNTER)
*ucblocks = c->block_error.stat[0].uvalue;
else
*ucblocks = 0; *ucblocks = 0;
goto error; return 0;
}
/* UCB is updated when BER is read. Assume BER is read anyway. */
*ucblocks = dev->ucb;
return ret;
error:
dev_dbg(&client->dev, "failed=%d\n", ret);
return ret;
} }
static int tda10071_set_frontend(struct dvb_frontend *fe) static int tda10071_set_frontend(struct dvb_frontend *fe)
...@@ -770,6 +763,7 @@ static int tda10071_init(struct dvb_frontend *fe) ...@@ -770,6 +763,7 @@ static int tda10071_init(struct dvb_frontend *fe)
{ {
struct tda10071_dev *dev = fe->demodulator_priv; struct tda10071_dev *dev = fe->demodulator_priv;
struct i2c_client *client = dev->client; struct i2c_client *client = dev->client;
struct dtv_frontend_properties *c = &fe->dtv_property_cache;
struct tda10071_cmd cmd; struct tda10071_cmd cmd;
int ret, i, len, remaining, fw_size; int ret, i, len, remaining, fw_size;
unsigned int uitmp; unsigned int uitmp;
...@@ -1035,6 +1029,16 @@ static int tda10071_init(struct dvb_frontend *fe) ...@@ -1035,6 +1029,16 @@ static int tda10071_init(struct dvb_frontend *fe)
goto error; goto error;
} }
/* init stats here in order signal app which stats are supported */
c->strength.len = 1;
c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
c->cnr.len = 1;
c->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
c->post_bit_error.len = 1;
c->post_bit_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
c->block_error.len = 1;
c->block_error.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
return ret; return ret;
error_release_firmware: error_release_firmware:
release_firmware(fw); release_firmware(fw);
......
...@@ -38,12 +38,13 @@ struct tda10071_dev { ...@@ -38,12 +38,13 @@ struct tda10071_dev {
u8 pll_multiplier; u8 pll_multiplier;
u8 tuner_i2c_addr; u8 tuner_i2c_addr;
u8 meas_count[2]; u8 meas_count;
u32 ber; u32 dvbv3_ber;
u32 ucb;
enum fe_status fe_status; enum fe_status fe_status;
enum fe_delivery_system delivery_system; enum fe_delivery_system delivery_system;
bool warm; /* FW running */ bool warm; /* FW running */
u64 post_bit_error;
u64 block_error;
}; };
static struct tda10071_modcod { static struct tda10071_modcod {
......
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