diff options
author | Antti Palosaari <crope@iki.fi> | 2014-09-02 03:24:41 -0400 |
---|---|---|
committer | Mauro Carvalho Chehab <mchehab@osg.samsung.com> | 2014-09-21 18:44:59 -0400 |
commit | 3e41313aeadfc5e3b3f827519f3840bca1b98f6d (patch) | |
tree | b0708ecd916d4655b74f102623b21b282f337384 /drivers/media/dvb-frontends/af9033.c | |
parent | 83f1161911c5f32dc4cfa817a73ae028d32c43b7 (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/af9033.c')
-rw-r--r-- | drivers/media/dvb-frontends/af9033.c | 54 |
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 | |||
1130 | err_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; |
1083 | err: | 1133 | err: |