diff options
author | Klaus Schmidinger <vdr@tvdr.de> | 2017-04-20 14:43:54 +0200 |
---|---|---|
committer | Klaus Schmidinger <vdr@tvdr.de> | 2017-04-20 14:43:54 +0200 |
commit | 9cd28b2ded3c8db71a857a2c3b12a4d48204e996 (patch) | |
tree | 52aa6bfc135fd604a8d28411aef3e93d092bd123 | |
parent | 686831caf5e314c424e1411f2c33df7e25c9b695 (diff) | |
download | vdr-9cd28b2ded3c8db71a857a2c3b12a4d48204e996.tar.gz vdr-9cd28b2ded3c8db71a857a2c3b12a4d48204e996.tar.bz2 |
Fixed handling the uncorrected block counter for DVB API 3 devices when calculating signal quality
-rw-r--r-- | HISTORY | 2 | ||||
-rw-r--r-- | dvbdevice.c | 37 |
2 files changed, 37 insertions, 2 deletions
@@ -8983,3 +8983,5 @@ Video Disk Recorder Revision History - Updated the Finnish OSD texts (thanks to Rolf Ahrenberg). - Fixed displaying remote timers in the main menu of skin LCARS. - Fixed editing a remote timer immediately after it has been created. +- Fixed handling the uncorrected block counter for DVB API 3 devices when calculating + signal quality. diff --git a/dvbdevice.c b/dvbdevice.c index 1c259059..3a900a3a 100644 --- a/dvbdevice.c +++ b/dvbdevice.c @@ -4,7 +4,7 @@ * See the main source file 'vdr.c' for copyright information and * how to reach the author. * - * $Id: dvbdevice.c 4.8 2017/04/18 13:11:07 kls Exp $ + * $Id: dvbdevice.c 4.9 2017/04/20 14:42:35 kls Exp $ */ #include "dvbdevice.h" @@ -309,6 +309,9 @@ private: int tuneTimeout; int lockTimeout; time_t lastTimeoutReport; + mutable uint32_t lastUncValue; + mutable uint32_t lastUncDelta; + mutable time_t lastUncChange; cChannel channel; const cDiseqc *lastDiseqc; int diseqcOffset; @@ -364,6 +367,9 @@ cDvbTuner::cDvbTuner(const cDvbDevice *Device, int Fd_Frontend, int Adapter, int tuneTimeout = 0; lockTimeout = 0; lastTimeoutReport = 0; + lastUncValue = 0; + lastUncDelta = 0; + lastUncChange = 0; lastDiseqc = NULL; diseqcOffset = 0; lastSource = 0; @@ -818,8 +824,35 @@ int cDvbTuner::GetSignalQuality(void) const #endif uint32_t Unc; while (1) { - if (ioctl(fd_frontend, FE_READ_UNCORRECTED_BLOCKS, &Unc) != -1) + if (ioctl(fd_frontend, FE_READ_UNCORRECTED_BLOCKS, &Unc) != -1) { + if (Unc != lastUncValue) { +#ifdef DEBUG_SIGNALQUALITY + fprintf(stderr, "FE %d/%d: API3 UNC = %u %u %u\n", adapter, frontend, Unc, lastUncValue, lastUncDelta); +#endif + lastUncDelta = (Unc >= lastUncValue) ? Unc - lastUncValue : lastUncValue - Unc; + lastUncValue = Unc; + lastUncChange = time(NULL); + } + // The number of uncorrected blocks is a counter, which is normally + // at a constant value and only increases if there are new uncorrected + // blocks. So a change in the Unc value indicates reduced signal quality. + // Whenever the Unc counter changes, we take the delta between the old + // and new value into account for calculating the overall signal quality. + // The impact of Unc is considered for 2 seconds, and after that it is + // bisected with every passing second in order to phase it out. Otherwise + // once one or more uncorrected blocks occur, the signal quality would + // be considered low even if there haven't been any more uncorrected bocks + // for quite a while. + Unc = lastUncDelta; + int t = time(NULL) - lastUncChange - 2; + if (t > 0) + Unc >>= min(t, 32); +#ifdef DEBUG_SIGNALQUALITY + if (Unc > 0) + fprintf(stderr, "FE %d/%d: API3 UNC = %u\n", adapter, frontend, Unc); +#endif break; + } if (errno != EINTR) { Unc = 0; #ifdef DEBUG_SIGNALQUALITY |