aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/dvb-frontends
diff options
context:
space:
mode:
authorAntti Palosaari <crope@iki.fi>2014-09-02 03:24:41 -0400
committerMauro Carvalho Chehab <mchehab@osg.samsung.com>2014-09-21 18:44:59 -0400
commit3e41313aeadfc5e3b3f827519f3840bca1b98f6d (patch)
treeb0708ecd916d4655b74f102623b21b282f337384 /drivers/media/dvb-frontends
parent83f1161911c5f32dc4cfa817a73ae028d32c43b7 (diff)
[media] af9033: implement DVBv5 statistics for CNR
Return CNR via DVBv5 statistics API. Signed-off-by: Antti Palosaari <crope@iki.fi> Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
Diffstat (limited to 'drivers/media/dvb-frontends')
-rw-r--r--drivers/media/dvb-frontends/af9033.c54
1 files changed, 52 insertions, 2 deletions
diff --git a/drivers/media/dvb-frontends/af9033.c b/drivers/media/dvb-frontends/af9033.c
index b9a0b009aeda..576e9b5f4bbf 100644
--- a/drivers/media/dvb-frontends/af9033.c
+++ b/drivers/media/dvb-frontends/af9033.c
@@ -1054,11 +1054,12 @@ static void af9033_stat_work(struct work_struct *work)
1054{ 1054{
1055 struct af9033_dev *dev = container_of(work, struct af9033_dev, stat_work.work); 1055 struct af9033_dev *dev = container_of(work, struct af9033_dev, stat_work.work);
1056 struct dtv_frontend_properties *c = &dev->fe.dtv_property_cache; 1056 struct dtv_frontend_properties *c = &dev->fe.dtv_property_cache;
1057 int ret, tmp; 1057 int ret, tmp, i, len;
1058 u8 u8tmp; 1058 u8 u8tmp, buf[3];
1059 1059
1060 dev_dbg(&dev->client->dev, "\n"); 1060 dev_dbg(&dev->client->dev, "\n");
1061 1061
1062 /* signal strength */
1062 if (dev->fe_status & FE_HAS_SIGNAL) { 1063 if (dev->fe_status & FE_HAS_SIGNAL) {
1063 if (dev->is_af9035) { 1064 if (dev->is_af9035) {
1064 ret = af9033_rd_reg(dev, 0x80004a, &u8tmp); 1065 ret = af9033_rd_reg(dev, 0x80004a, &u8tmp);
@@ -1078,6 +1079,55 @@ static void af9033_stat_work(struct work_struct *work)
1078 c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE; 1079 c->strength.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
1079 } 1080 }
1080 1081
1082 /* CNR */
1083 if (dev->fe_status & FE_HAS_VITERBI) {
1084 u32 snr_val;
1085 const struct val_snr *snr_lut;
1086
1087 /* read value */
1088 ret = af9033_rd_regs(dev, 0x80002c, buf, 3);
1089 if (ret)
1090 goto err;
1091
1092 snr_val = (buf[2] << 16) | (buf[1] << 8) | (buf[0] << 0);
1093
1094 /* read current modulation */
1095 ret = af9033_rd_reg(dev, 0x80f903, &u8tmp);
1096 if (ret)
1097 goto err;
1098
1099 switch ((u8tmp >> 0) & 3) {
1100 case 0:
1101 len = ARRAY_SIZE(qpsk_snr_lut);
1102 snr_lut = qpsk_snr_lut;
1103 break;
1104 case 1:
1105 len = ARRAY_SIZE(qam16_snr_lut);
1106 snr_lut = qam16_snr_lut;
1107 break;
1108 case 2:
1109 len = ARRAY_SIZE(qam64_snr_lut);
1110 snr_lut = qam64_snr_lut;
1111 break;
1112 default:
1113 goto err_schedule_delayed_work;
1114 }
1115
1116 for (i = 0; i < len; i++) {
1117 tmp = snr_lut[i].snr * 1000;
1118 if (snr_val < snr_lut[i].val)
1119 break;
1120 }
1121
1122 c->cnr.len = 1;
1123 c->cnr.stat[0].scale = FE_SCALE_DECIBEL;
1124 c->cnr.stat[0].svalue = tmp;
1125 } else {
1126 c->cnr.len = 1;
1127 c->cnr.stat[0].scale = FE_SCALE_NOT_AVAILABLE;
1128 }
1129
1130err_schedule_delayed_work:
1081 schedule_delayed_work(&dev->stat_work, msecs_to_jiffies(2000)); 1131 schedule_delayed_work(&dev->stat_work, msecs_to_jiffies(2000));
1082 return; 1132 return;
1083err: 1133err: