vdr-plugin-satip/tuner.c

750 lines
24 KiB
C
Raw Permalink Normal View History

2014-03-08 12:07:47 +01:00
/*
* tuner.c: SAT>IP plugin for the Video Disk Recorder
*
* See the README file for copyright information and how to reach the author.
*
*/
#define __STDC_FORMAT_MACROS // Required for format specifiers
#include <inttypes.h>
2014-03-08 12:07:47 +01:00
#include "common.h"
#include "config.h"
#include "discover.h"
2014-12-05 22:14:40 +01:00
#include "log.h"
#include "poller.h"
2014-03-08 12:07:47 +01:00
#include "tuner.h"
cSatipTuner::cSatipTuner(cSatipDeviceIf &deviceP, unsigned int packetLenP)
2014-12-14 20:39:13 +01:00
: cThread(cString::sprintf("SATIP#%d tuner", deviceP.GetId())),
2014-03-08 12:07:47 +01:00
sleepM(),
deviceM(&deviceP),
2014-11-10 22:31:30 +01:00
deviceIdM(deviceP.GetId()),
2014-11-16 15:39:20 +01:00
rtspM(*this),
rtpM(*this),
rtcpM(*this),
2014-03-08 12:07:47 +01:00
streamAddrM(""),
streamParamM(""),
lastAddrM(""),
lastParamM(""),
tnrParamM(""),
streamPortM(SATIP_DEFAULT_RTSP_PORT),
currentServerM(NULL, deviceP.GetId(), 0),
nextServerM(NULL, deviceP.GetId(), 0),
2014-03-08 12:07:47 +01:00
mutexM(),
2014-11-16 14:38:23 +01:00
reConnectM(),
2014-03-08 12:07:47 +01:00
keepAliveM(),
statusUpdateM(),
2014-03-08 12:07:47 +01:00
pidUpdateCacheM(),
setupTimeoutM(-1),
sessionM(""),
currentStateM(tsIdle),
2014-11-23 21:22:38 +01:00
internalStateM(),
externalStateM(),
2021-05-26 21:10:34 +02:00
timeoutM(eMinKeepAliveIntervalMs - eKeepAlivePreBufferMs),
2014-03-08 12:07:47 +01:00
hasLockM(false),
signalStrengthDBmM(0.0),
2014-03-08 12:07:47 +01:00
signalStrengthM(-1),
signalQualityM(-1),
frontendIdM(-1),
2014-03-08 12:07:47 +01:00
streamIdM(-1),
pmtPidM(-1),
2014-04-19 16:27:47 +02:00
addPidsM(),
delPidsM(),
2014-03-08 12:07:47 +01:00
pidsM()
{
2014-12-07 16:27:53 +01:00
debug1("%s (, %d) [device %d]", __PRETTY_FUNCTION__, packetLenP, deviceIdM);
2014-11-09 11:51:52 +01:00
// Open sockets
int i = SatipConfig.GetPortRangeStart() ? SatipConfig.GetPortRangeStop() - SatipConfig.GetPortRangeStart() - 1 : 100;
int port = SatipConfig.GetPortRangeStart();
2014-11-09 11:51:52 +01:00
while (i-- > 0) {
2016-12-11 00:18:12 +01:00
// RTP must use an even port number
if (rtpM.Open(port) && (rtpM.Port() % 2 == 0) && rtcpM.Open(rtpM.Port() + 1))
2014-11-09 11:51:52 +01:00
break;
2014-11-16 15:39:20 +01:00
rtpM.Close();
rtcpM.Close();
if (SatipConfig.GetPortRangeStart())
2016-12-11 00:18:12 +01:00
port += 2;
2014-11-09 11:51:52 +01:00
}
2014-11-16 15:39:20 +01:00
if ((rtpM.Port() <= 0) || (rtcpM.Port() <= 0)) {
2014-11-09 11:51:52 +01:00
error("Cannot open required RTP/RTCP ports [device %d]", deviceIdM);
}
// Must be done after socket initialization!
2014-11-16 15:39:20 +01:00
cSatipPoller::GetInstance()->Register(rtpM);
cSatipPoller::GetInstance()->Register(rtcpM);
2014-11-09 19:32:08 +01:00
2014-03-08 12:07:47 +01:00
// Start thread
Start();
}
cSatipTuner::~cSatipTuner()
{
2014-12-06 16:02:45 +01:00
debug1("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
// Stop thread
sleepM.Signal();
if (Running())
Cancel(3);
Close();
2014-11-23 21:22:38 +01:00
currentStateM = tsIdle;
internalStateM.Clear();
externalStateM.Clear();
2014-11-09 11:51:52 +01:00
// Close the listening sockets
2014-11-16 15:39:20 +01:00
cSatipPoller::GetInstance()->Unregister(rtcpM);
cSatipPoller::GetInstance()->Unregister(rtpM);
rtcpM.Close();
rtpM.Close();
2014-03-08 12:07:47 +01:00
}
void cSatipTuner::Action(void)
{
2014-12-06 16:02:45 +01:00
debug1("%s Entering [device %d]", __PRETTY_FUNCTION__, deviceIdM);
bool lastIdleStatus = false;
cTimeMs idleCheck(eIdleCheckTimeoutMs);
cTimeMs tuning(eTuningTimeoutMs);
2014-11-16 14:38:23 +01:00
reConnectM.Set(eConnectTimeoutMs);
2014-03-08 12:07:47 +01:00
// Do the thread loop
while (Running()) {
2014-11-23 21:22:38 +01:00
UpdateCurrentState();
switch (currentStateM) {
case tsIdle:
2014-12-06 16:02:45 +01:00
debug4("%s: tsIdle [device %d]", __PRETTY_FUNCTION__, deviceIdM);
break;
case tsRelease:
2014-12-06 16:02:45 +01:00
debug4("%s: tsRelease [device %d]", __PRETTY_FUNCTION__, deviceIdM);
Disconnect();
2014-11-23 21:22:38 +01:00
RequestState(tsIdle, smInternal);
break;
case tsSet:
2014-12-06 16:02:45 +01:00
debug4("%s: tsSet [device %d]", __PRETTY_FUNCTION__, deviceIdM);
if (currentServerM.IsQuirk(cSatipServer::eSatipQuirkTearAndPlay))
Disconnect();
if (Connect()) {
tuning.Set(eTuningTimeoutMs);
2014-11-23 21:22:38 +01:00
RequestState(tsTuned, smInternal);
UpdatePids(true);
}
2014-12-16 21:35:49 +01:00
else
Disconnect();
break;
case tsTuned:
2014-12-06 16:02:45 +01:00
debug4("%s: tsTuned [device %d]", __PRETTY_FUNCTION__, deviceIdM);
deviceM->SetChannelTuned();
2014-11-16 14:38:23 +01:00
reConnectM.Set(eConnectTimeoutMs);
idleCheck.Set(eIdleCheckTimeoutMs);
lastIdleStatus = false;
2014-11-22 19:06:56 +01:00
// Read reception statistics via DESCRIBE and RTCP
if (hasLockM || ReadReceptionStatus()) {
// Quirk for devices without valid reception data
2015-01-15 22:33:51 +01:00
if (currentServerM.IsQuirk(cSatipServer::eSatipQuirkForceLock)) {
2014-11-22 19:06:56 +01:00
hasLockM = true;
signalStrengthDBmM = eDefaultSignalStrengthDBm;
2014-11-22 19:06:56 +01:00
signalStrengthM = eDefaultSignalStrength;
signalQualityM = eDefaultSignalQuality;
}
if (hasLockM)
2014-11-23 21:22:38 +01:00
RequestState(tsLocked, smInternal);
}
else if (tuning.TimedOut()) {
error("Tuning timeout - retuning [device %d]", deviceIdM);
RequestState(tsSet, smInternal);
}
break;
case tsLocked:
2014-12-06 16:02:45 +01:00
debug4("%s: tsLocked [device %d]", __PRETTY_FUNCTION__, deviceIdM);
if (!UpdatePids()) {
error("Pid update failed - retuning [device %d]", deviceIdM);
2014-11-23 21:22:38 +01:00
RequestState(tsSet, smInternal);
break;
}
if (!KeepAlive()) {
error("Keep-alive failed - retuning [device %d]", deviceIdM);
2014-11-23 21:22:38 +01:00
RequestState(tsSet, smInternal);
break;
}
2014-11-16 14:38:23 +01:00
if (reConnectM.TimedOut()) {
error("Connection timeout - retuning [device %d]", deviceIdM);
2014-11-23 21:22:38 +01:00
RequestState(tsSet, smInternal);
break;
}
if (idleCheck.TimedOut()) {
bool currentIdleStatus = deviceM->IsIdle();
if (lastIdleStatus && currentIdleStatus) {
info("Idle timeout - releasing [device %d]", deviceIdM);
RequestState(tsRelease, smInternal);
}
lastIdleStatus = currentIdleStatus;
idleCheck.Set(eIdleCheckTimeoutMs);
break;
}
2018-10-21 19:21:33 +02:00
Receive();
break;
default:
error("Unknown tuner status %d [device %d]", currentStateM, deviceIdM);
break;
}
2014-11-19 20:01:45 +01:00
if (!StateRequested())
sleepM.Wait(eSleepTimeoutMs); // to avoid busy loop and reduce cpu load
2014-03-08 12:07:47 +01:00
}
2014-12-06 16:02:45 +01:00
debug1("%s Exiting [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
}
bool cSatipTuner::Open(void)
{
cMutexLock MutexLock(&mutexM);
2014-12-06 16:02:45 +01:00
debug1("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
// return always true
return true;
2014-03-08 12:07:47 +01:00
}
bool cSatipTuner::Close(void)
{
cMutexLock MutexLock(&mutexM);
2014-12-06 16:02:45 +01:00
debug1("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
if (setupTimeoutM.TimedOut())
RequestState(tsRelease, smExternal);
// return always true
return true;
2014-03-08 12:07:47 +01:00
}
bool cSatipTuner::Connect(void)
{
cMutexLock MutexLock(&mutexM);
2014-12-06 16:02:45 +01:00
debug1("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
2014-11-09 11:51:52 +01:00
if (!isempty(*streamAddrM)) {
cString connectionUri = GetBaseUrl(*streamAddrM, streamPortM);
tnrParamM = "";
2014-03-08 12:07:47 +01:00
// Just retune
if (streamIdM >= 0) {
2021-02-28 21:15:45 +01:00
if (!strcmp(*streamParamM, *lastParamM) && hasLockM) {
debug1("%s Identical parameters [device %d]", __PRETTY_FUNCTION__, deviceIdM);
//return true; // fall through because detection does not work reliably
}
cString uri = cString::sprintf("%sstream=%d?%s", *connectionUri, streamIdM, *streamParamM);
2014-12-06 16:02:45 +01:00
debug1("%s Retuning [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-12-02 22:17:35 +01:00
if (rtspM.Play(*uri)) {
2014-11-22 22:17:32 +01:00
keepAliveM.Set(timeoutM);
lastParamM = streamParamM;
2014-11-22 22:17:32 +01:00
return true;
}
}
else if (rtspM.SetInterface(nextServerM.IsValid() ? *nextServerM.GetSrcAddress() : NULL) && rtspM.Options(*connectionUri)) {
cString uri = cString::sprintf("%s?%s", *connectionUri, *streamParamM);
2016-12-11 00:18:12 +01:00
bool useTcp = SatipConfig.IsTransportModeRtpOverTcp() && nextServerM.IsValid() && nextServerM.IsQuirk(cSatipServer::eSatipQuirkRtpOverTcp);
2014-03-08 12:07:47 +01:00
// Flush any old content
2015-02-20 19:51:31 +01:00
//rtpM.Flush();
//rtcpM.Flush();
2018-02-06 21:35:48 +01:00
if (useTcp)
debug1("%s Requesting TCP [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2016-06-20 17:14:38 +02:00
if (rtspM.Setup(*uri, rtpM.Port(), rtcpM.Port(), useTcp)) {
lastParamM = streamParamM;
2014-11-22 22:17:32 +01:00
keepAliveM.Set(timeoutM);
2015-01-15 22:33:51 +01:00
if (nextServerM.IsValid()) {
2014-11-09 19:32:08 +01:00
currentServerM = nextServerM;
2015-01-15 22:33:51 +01:00
nextServerM.Reset();
2014-11-09 19:32:08 +01:00
}
lastAddrM = connectionUri;
currentServerM.Attach();
return true;
2014-11-15 21:05:46 +01:00
}
}
rtspM.Reset();
2014-11-22 13:56:20 +01:00
streamIdM = -1;
2014-11-15 21:05:46 +01:00
error("Connect failed [device %d]", deviceIdM);
2014-03-08 12:07:47 +01:00
}
return false;
2014-03-08 12:07:47 +01:00
}
bool cSatipTuner::Disconnect(void)
{
cMutexLock MutexLock(&mutexM);
2014-12-06 16:02:45 +01:00
debug1("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
if (!isempty(*lastAddrM) && (streamIdM >= 0)) {
cString uri = cString::sprintf("%sstream=%d", *lastAddrM, streamIdM);
2014-11-16 15:39:20 +01:00
rtspM.Teardown(*uri);
2015-01-05 19:11:41 +01:00
// some devices requires a teardown for TCP connection also
rtspM.Reset();
2014-11-22 23:13:15 +01:00
streamIdM = -1;
2014-03-08 12:07:47 +01:00
}
// Reset signal parameters
hasLockM = false;
signalStrengthDBmM = 0.0;
2014-03-08 12:07:47 +01:00
signalStrengthM = -1;
signalQualityM = -1;
frontendIdM = -1;
2014-03-08 12:07:47 +01:00
currentServerM.Detach();
statusUpdateM.Set(0);
2021-05-26 21:10:34 +02:00
timeoutM = eMinKeepAliveIntervalMs - eKeepAlivePreBufferMs;
pmtPidM = -1;
2014-04-19 16:27:47 +02:00
addPidsM.Clear();
delPidsM.Clear();
2014-03-08 12:07:47 +01:00
// return always true
2014-03-08 12:07:47 +01:00
return true;
}
2014-11-16 14:38:23 +01:00
void cSatipTuner::ProcessVideoData(u_char *bufferP, int lengthP)
{
debug16("%s (, %d) [device %d]", __PRETTY_FUNCTION__, lengthP, deviceIdM);
if (lengthP > 0) {
uint64_t elapsed;
cTimeMs processing(0);
AddTunerStatistic(lengthP);
elapsed = processing.Elapsed();
if (elapsed > 1)
debug6("%s AddTunerStatistic() took %" PRIu64 " ms [device %d]", __PRETTY_FUNCTION__, elapsed, deviceIdM);
processing.Set(0);
2014-11-16 14:38:23 +01:00
deviceM->WriteData(bufferP, lengthP);
elapsed = processing.Elapsed();
if (elapsed > 1)
debug6("%s WriteData() took %" PRIu64 " ms [device %d]", __FUNCTION__, elapsed, deviceIdM);
}
2014-11-16 14:38:23 +01:00
reConnectM.Set(eConnectTimeoutMs);
}
void cSatipTuner::ProcessRtpData(u_char *bufferP, int lengthP)
{
rtpM.Process(bufferP, lengthP);
}
2014-11-16 14:38:23 +01:00
void cSatipTuner::ProcessApplicationData(u_char *bufferP, int lengthP)
2014-03-08 12:07:47 +01:00
{
debug16("%s (%d) [device %d]", __PRETTY_FUNCTION__, lengthP, deviceIdM);
2014-03-08 12:07:47 +01:00
// DVB-S2:
// ver=<major>.<minor>;src=<srcID>;tuner=<feID>,<level>,<lock>,<quality>,<frequency>,<polarisation>,<system>,<type>,<pilots>,<roll_off>,<symbol_rate>,<fec_inner>;pids=<pid0>,...,<pidn>
// DVB-T2:
// ver=1.1;tuner=<feID>,<level>,<lock>,<quality>,<freq>,<bw>,<msys>,<tmode>,<mtype>,<gi>,<fec>,<plp>,<t2id>,<sm>;pids=<pid0>,...,<pidn>
// DVB-C2:
// ver=1.2;tuner=<feID>,<level>,<lock>,<quality>,<freq>,<bw>,<msys>,<mtype>,<sr>,<c2tft>,<ds>,<plp>,<specinv>;pids=<pid0>,...,<pidn>
if (lengthP > 0) {
char s[lengthP];
memcpy(s, (char *)bufferP, lengthP);
2014-12-14 18:22:52 +01:00
debug10("%s (%s) [device %d]", __PRETTY_FUNCTION__, s, deviceIdM);
2014-03-08 12:07:47 +01:00
char *c = strstr(s, ";tuner=");
if (c) {
int value;
// feID:
frontendIdM = atoi(c + 7);
2014-03-08 12:07:47 +01:00
// level:
// Numerical value between 0 and 255
// An incoming L-band satellite signal of
// -25dBm corresponds to 224
// -65dBm corresponds to 32
// No signal corresponds to 0
c = strstr(c, ",");
2015-01-12 22:58:30 +01:00
value = min(atoi(++c), 255);
signalStrengthDBmM = (value >= 0) ? 40.0 * (value - 32) / 192.0 - 65.0 : 0.0;
2014-03-08 12:07:47 +01:00
// Scale value to 0-100
signalStrengthM = (value >= 0) ? value * 100 / 255 : -1;
2014-03-08 12:07:47 +01:00
// lock:
// lock Set to one of the following values:
// "0" the frontend is not locked
// "1" the frontend is locked
c = strstr(c, ",");
2014-11-10 22:31:30 +01:00
hasLockM = !!atoi(++c);
2014-03-08 12:07:47 +01:00
// quality:
// Numerical value between 0 and 15
// Lowest value corresponds to highest error rate
// The value 15 shall correspond to
// -a BER lower than 2x10-4 after Viterbi for DVB-S
// -a PER lower than 10-7 for DVB-S2
c = strstr(c, ",");
2015-01-12 22:58:30 +01:00
value = min(atoi(++c), 15);
2014-03-08 12:07:47 +01:00
// Scale value to 0-100
signalQualityM = (hasLockM && (value >= 0)) ? (value * 100 / 15) : 0;
}
}
reConnectM.Set(eConnectTimeoutMs);
2014-03-08 12:07:47 +01:00
}
void cSatipTuner::ProcessRtcpData(u_char *bufferP, int lengthP)
{
rtcpM.Process(bufferP, lengthP);
}
void cSatipTuner::SetStreamId(int streamIdP)
{
cMutexLock MutexLock(&mutexM);
2014-12-07 16:27:53 +01:00
debug1("%s (%d) [device %d]", __PRETTY_FUNCTION__, streamIdP, deviceIdM);
streamIdM = streamIdP;
}
void cSatipTuner::SetSessionTimeout(const char *sessionP, int timeoutP)
2014-03-08 12:07:47 +01:00
{
cMutexLock MutexLock(&mutexM);
2014-12-07 16:27:53 +01:00
debug1("%s (%s, %d) [device %d]", __PRETTY_FUNCTION__, sessionP, timeoutP, deviceIdM);
sessionM = sessionP;
2015-01-15 22:33:51 +01:00
if (nextServerM.IsQuirk(cSatipServer::eSatipQuirkSessionId) && !isempty(*sessionM) && startswith(*sessionM, "0"))
2014-12-07 20:58:06 +01:00
rtspM.SetSession(SkipZeroes(*sessionM));
2014-03-31 21:07:15 +02:00
timeoutM = (timeoutP > eMinKeepAliveIntervalMs) ? timeoutP : eMinKeepAliveIntervalMs;
2021-05-26 21:10:34 +02:00
timeoutM -= eKeepAlivePreBufferMs;
2014-03-08 12:07:47 +01:00
}
2016-12-11 00:18:12 +01:00
void cSatipTuner::SetupTransport(int rtpPortP, int rtcpPortP, const char *streamAddrP, const char *sourceAddrP)
{
cMutexLock MutexLock(&mutexM);
debug1("%s (%d, %d, %s, %s) [device %d]", __PRETTY_FUNCTION__, rtpPortP, rtcpPortP, streamAddrP, sourceAddrP, deviceIdM);
bool multicast = !isempty(streamAddrP);
// Adapt RTP to any transport media change
if (multicast != rtpM.IsMulticast() || rtpPortP != rtpM.Port()) {
cSatipPoller::GetInstance()->Unregister(rtpM);
if (rtpPortP >= 0) {
2018-02-06 21:54:36 +01:00
rtpM.Close();
if (multicast)
rtpM.OpenMulticast(rtpPortP, streamAddrP, sourceAddrP);
else
rtpM.Open(rtpPortP);
cSatipPoller::GetInstance()->Register(rtpM);
}
2016-12-11 00:18:12 +01:00
}
// Adapt RTCP to any transport media change
if (multicast != rtcpM.IsMulticast() || rtcpPortP != rtcpM.Port()) {
cSatipPoller::GetInstance()->Unregister(rtcpM);
if (rtcpPortP >= 0) {
2018-02-06 21:54:36 +01:00
rtcpM.Close();
if (multicast)
2018-02-06 21:35:48 +01:00
rtcpM.OpenMulticast(rtcpPortP, streamAddrP, sourceAddrP);
else
2018-02-06 21:35:48 +01:00
rtcpM.Open(rtcpPortP);
cSatipPoller::GetInstance()->Register(rtcpM);
}
2016-12-11 00:18:12 +01:00
}
}
cString cSatipTuner::GetBaseUrl(const char *addressP, const int portP)
{
debug16("%s (%s, %d) [device %d]", __PRETTY_FUNCTION__, addressP, portP, deviceIdM);
if (portP != SATIP_DEFAULT_RTSP_PORT)
return cString::sprintf("rtsp://%s:%d/", addressP, portP);
return cString::sprintf("rtsp://%s/", addressP);
}
2014-11-09 11:51:52 +01:00
int cSatipTuner::GetId(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-11-09 11:51:52 +01:00
return deviceIdM;
}
2015-01-15 22:33:51 +01:00
bool cSatipTuner::SetSource(cSatipServer *serverP, const int transponderP, const char *parameterP, const int indexP)
2014-03-08 12:07:47 +01:00
{
2015-01-15 22:33:51 +01:00
debug1("%s (%d, %s, %d) [device %d]", __PRETTY_FUNCTION__, transponderP, parameterP, indexP, deviceIdM);
cMutexLock MutexLock(&mutexM);
if (serverP) {
2015-01-15 22:33:51 +01:00
nextServerM.Set(serverP, transponderP);
if (!isempty(*nextServerM.GetAddress()) && !isempty(parameterP)) {
// Update stream address and parameter
2015-01-15 22:33:51 +01:00
streamAddrM = rtspM.RtspUnescapeString(*nextServerM.GetAddress());
2014-11-16 15:39:20 +01:00
streamParamM = rtspM.RtspUnescapeString(parameterP);
streamPortM = nextServerM.GetPort();
2016-11-10 15:47:30 +01:00
// Modify parameter if required
if (nextServerM.IsQuirk(cSatipServer::eSatipQuirkForcePilot) && strstr(parameterP, "msys=dvbs2") && !strstr(parameterP, "plts="))
streamParamM = rtspM.RtspUnescapeString(*cString::sprintf("%s&plts=on", parameterP));
// Reconnect
if (!isempty(*lastAddrM)) {
cString connectionUri = GetBaseUrl(*streamAddrM, streamPortM);
if (strcmp(*connectionUri, *lastAddrM))
RequestState(tsRelease, smInternal);
}
RequestState(tsSet, smExternal);
setupTimeoutM.Set(eSetupTimeoutMs);
}
2014-03-08 12:07:47 +01:00
}
2014-11-23 21:22:38 +01:00
else {
streamAddrM = "";
streamParamM = "";
}
2014-03-08 12:07:47 +01:00
return true;
}
bool cSatipTuner::SetPid(int pidP, int typeP, bool onP)
{
debug16("%s (%d, %d, %d) [device %d]", __PRETTY_FUNCTION__, pidP, typeP, onP, deviceIdM);
2014-03-08 12:07:47 +01:00
cMutexLock MutexLock(&mutexM);
2014-04-19 16:27:47 +02:00
if (onP) {
2014-11-09 01:32:50 +01:00
pidsM.AddPid(pidP);
addPidsM.AddPid(pidP);
delPidsM.RemovePid(pidP);
2014-04-19 16:27:47 +02:00
}
else {
2014-11-09 01:32:50 +01:00
pidsM.RemovePid(pidP);
delPidsM.AddPid(pidP);
addPidsM.RemovePid(pidP);
2014-04-19 16:27:47 +02:00
}
2015-01-15 19:03:35 +01:00
debug12("%s (%d, %d, %d) pids=%s [device %d]", __PRETTY_FUNCTION__, pidP, typeP, onP, *pidsM.ListPids(), deviceIdM);
sleepM.Signal();
2014-03-08 12:07:47 +01:00
return true;
}
2014-04-19 16:27:47 +02:00
bool cSatipTuner::UpdatePids(bool forceP)
2014-03-08 12:07:47 +01:00
{
debug16("%s (%d) tunerState=%s [device %d]", __PRETTY_FUNCTION__, forceP, TunerStateString(currentStateM), deviceIdM);
cMutexLock MutexLock(&mutexM);
2014-04-19 16:27:47 +02:00
if (((forceP && pidsM.Size()) || (pidUpdateCacheM.TimedOut() && (addPidsM.Size() || delPidsM.Size()))) &&
!isempty(*streamAddrM) && (streamIdM > 0)) {
cString uri = cString::sprintf("%sstream=%d", *GetBaseUrl(*streamAddrM, streamPortM), streamIdM);
bool useci = (SatipConfig.GetCIExtension() && currentServerM.HasCI());
2015-01-15 22:33:51 +01:00
bool usedummy = currentServerM.IsQuirk(cSatipServer::eSatipQuirkPlayPids);
bool paramadded = false;
if (forceP || usedummy) {
if (pidsM.Size()) {
uri = cString::sprintf("%s%spids=%s", *uri, paramadded ? "&" : "?", *pidsM.ListPids());
if (usedummy && (pidsM.Size() == 1) && (pidsM[0] < 0x20))
uri = cString::sprintf("%s,%d", *uri, eDummyPid);
paramadded = true;
}
2014-04-19 16:27:47 +02:00
}
else {
if (addPidsM.Size()) {
uri = cString::sprintf("%s%saddpids=%s", *uri, paramadded ? "&" : "?", *addPidsM.ListPids());
paramadded = true;
}
if (delPidsM.Size()) {
uri = cString::sprintf("%s%sdelpids=%s", *uri, paramadded ? "&" : "?", *delPidsM.ListPids());
paramadded = true;
}
2014-04-19 16:27:47 +02:00
}
if (useci) {
if (currentServerM.IsQuirk(cSatipServer::eSatipQuirkCiXpmt)) {
// CI extension parameters:
// - x_pmt : specifies the PMT of the service you want the CI to decode
// - x_ci : specfies which CI slot (1..n) to use
// value 0 releases the CI slot
// CI slot released automatically if the stream is released,
// but not when used retuning to another channel
int pid = deviceM->GetPmtPid();
if ((pid > 0) && (pid != pmtPidM)) {
int slot = deviceM->GetCISlot();
uri = cString::sprintf("%s%sx_pmt=%d", *uri, paramadded ? "&" : "?", pid);
if (slot > 0)
uri = cString::sprintf("%s&x_ci=%d", *uri, slot);
paramadded = true;
}
pmtPidM = pid;
}
else if (currentServerM.IsQuirk(cSatipServer::eSatipQuirkCiTnr)) {
// CI extension parameters:
// - tnr : specifies a channel config entry
cString param = deviceM->GetTnrParameterString();
if (!isempty(*param) && strcmp(*tnrParamM, *param) != 0) {
uri = cString::sprintf("%s%stnr=%s", *uri, paramadded ? "&" : "?", *param);
paramadded = true;
}
tnrParamM = param;
}
}
if (paramadded) {
pidUpdateCacheM.Set(ePidUpdateIntervalMs);
if (!rtspM.Play(*uri))
return false;
}
addPidsM.Clear();
delPidsM.Clear();
2014-03-08 12:07:47 +01:00
}
return true;
2014-03-08 12:07:47 +01:00
}
2018-10-21 19:21:33 +02:00
bool cSatipTuner::Receive(void)
{
debug16("%s tunerState=%s [device %d]", __PRETTY_FUNCTION__, TunerStateString(currentStateM), deviceIdM);
cMutexLock MutexLock(&mutexM);
if (!isempty(*streamAddrM)) {
cString uri = GetBaseUrl(*streamAddrM, streamPortM);
if (!rtspM.Receive(*uri))
return false;
}
return true;
}
bool cSatipTuner::KeepAlive(bool forceP)
2014-03-08 12:07:47 +01:00
{
debug16("%s (%d) tunerState=%s [device %d]", __PRETTY_FUNCTION__, forceP, TunerStateString(currentStateM), deviceIdM);
2014-03-08 12:07:47 +01:00
cMutexLock MutexLock(&mutexM);
if (keepAliveM.TimedOut()) {
2014-11-09 19:32:08 +01:00
keepAliveM.Set(timeoutM);
forceP = true;
}
2014-11-22 13:56:20 +01:00
if (forceP && !isempty(*streamAddrM)) {
cString uri = GetBaseUrl(*streamAddrM, streamPortM);
2014-11-16 15:39:20 +01:00
if (!rtspM.Options(*uri))
return false;
2014-03-08 12:07:47 +01:00
}
return true;
2014-03-08 12:07:47 +01:00
}
bool cSatipTuner::ReadReceptionStatus(bool forceP)
{
debug16("%s (%d) tunerState=%s [device %d]", __PRETTY_FUNCTION__, forceP, TunerStateString(currentStateM), deviceIdM);
cMutexLock MutexLock(&mutexM);
if (statusUpdateM.TimedOut()) {
statusUpdateM.Set(eStatusUpdateTimeoutMs);
forceP = true;
}
if (forceP && !isempty(*streamAddrM) && (streamIdM > 0)) {
cString uri = cString::sprintf("%sstream=%d", *GetBaseUrl(*streamAddrM, streamPortM), streamIdM);
2014-11-16 15:39:20 +01:00
if (rtspM.Describe(*uri))
2014-11-09 11:51:52 +01:00
return true;
}
return false;
}
2014-11-23 21:22:38 +01:00
void cSatipTuner::UpdateCurrentState(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-11-23 21:22:38 +01:00
cMutexLock MutexLock(&mutexM);
eTunerState state = currentStateM;
if (internalStateM.Size()) {
state = internalStateM.At(0);
internalStateM.Remove(0);
}
else if (externalStateM.Size()) {
state = externalStateM.At(0);
externalStateM.Remove(0);
}
if (currentStateM != state) {
2014-12-07 16:27:53 +01:00
debug1("%s: Switching from %s to %s [device %d]", __PRETTY_FUNCTION__, TunerStateString(currentStateM), TunerStateString(state), deviceIdM);
2014-11-23 21:22:38 +01:00
currentStateM = state;
}
}
2014-11-19 20:01:45 +01:00
bool cSatipTuner::StateRequested(void)
{
cMutexLock MutexLock(&mutexM);
debug16("%s current=%s internal=%d external=%d [device %d]", __PRETTY_FUNCTION__, TunerStateString(currentStateM), internalStateM.Size(), externalStateM.Size(), deviceIdM);
2014-11-19 20:01:45 +01:00
2014-11-23 21:22:38 +01:00
return (internalStateM.Size() || externalStateM.Size());
2014-11-19 20:01:45 +01:00
}
2014-11-23 21:22:38 +01:00
bool cSatipTuner::RequestState(eTunerState stateP, eStateMode modeP)
{
cMutexLock MutexLock(&mutexM);
2014-12-07 16:27:53 +01:00
debug1("%s (%s, %s) current=%s internal=%d external=%d [device %d]", __PRETTY_FUNCTION__, TunerStateString(stateP), StateModeString(modeP), TunerStateString(currentStateM), internalStateM.Size(), externalStateM.Size(), deviceIdM);
2014-11-23 21:22:38 +01:00
if (modeP == smExternal)
externalStateM.Append(stateP);
else if (modeP == smInternal) {
eTunerState state = internalStateM.Size() ? internalStateM.At(internalStateM.Size() - 1) : currentStateM;
// validate legal state changes
switch (state) {
case tsIdle:
if (stateP == tsRelease)
return false;
case tsRelease:
case tsSet:
case tsLocked:
case tsTuned:
default:
break;
}
internalStateM.Append(stateP);
}
else
return false;
2014-11-22 20:32:55 +01:00
2014-11-23 21:22:38 +01:00
return true;
}
2014-11-22 23:13:15 +01:00
2014-11-23 21:22:38 +01:00
const char *cSatipTuner::StateModeString(eStateMode modeP)
{
switch (modeP) {
case smInternal:
return "smInternal";
case smExternal:
return "smExternal";
default:
2014-11-22 20:32:55 +01:00
break;
}
2014-11-23 21:22:38 +01:00
return "---";
}
const char *cSatipTuner::TunerStateString(eTunerState stateP)
{
switch (stateP) {
case tsIdle:
return "tsIdle";
case tsRelease:
return "tsRelease";
case tsSet:
return "tsSet";
case tsLocked:
return "tsLocked";
case tsTuned:
return "tsTuned";
default:
break;
}
return "---";
}
int cSatipTuner::FrontendId(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
return frontendIdM;
}
2014-03-08 12:07:47 +01:00
int cSatipTuner::SignalStrength(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
return signalStrengthM;
}
double cSatipTuner::SignalStrengthDBm(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
return signalStrengthDBmM;
}
2014-03-08 12:07:47 +01:00
int cSatipTuner::SignalQuality(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
2014-03-08 12:07:47 +01:00
return signalQualityM;
}
bool cSatipTuner::HasLock(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
return (currentStateM >= tsTuned) && hasLockM;
2014-03-08 12:07:47 +01:00
}
cString cSatipTuner::GetSignalStatus(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
return cString::sprintf("lock=%d strength=%d quality=%d frontend=%d", HasLock(), SignalStrength(), SignalQuality(), FrontendId());
2014-03-08 12:07:47 +01:00
}
cString cSatipTuner::GetInformation(void)
{
debug16("%s [device %d]", __PRETTY_FUNCTION__, deviceIdM);
return (currentStateM >= tsTuned) ? cString::sprintf("%s?%s (%s) [stream=%d]", *GetBaseUrl(*streamAddrM, streamPortM), *streamParamM, *rtspM.GetActiveMode(), streamIdM) : "connection failed";
2014-03-08 12:07:47 +01:00
}