2003-12-22 13:29:24 +01:00
|
|
|
/*
|
|
|
|
* pat.c: PAT section filter
|
|
|
|
*
|
|
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
|
|
* how to reach the author.
|
|
|
|
*
|
2021-06-21 20:13:55 +02:00
|
|
|
* $Id: pat.c 5.4 2021/06/21 20:13:55 kls Exp $
|
2003-12-22 13:29:24 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "pat.h"
|
|
|
|
#include <malloc.h>
|
2004-01-04 12:30:00 +01:00
|
|
|
#include "channels.h"
|
2003-12-22 13:29:24 +01:00
|
|
|
#include "libsi/section.h"
|
|
|
|
#include "libsi/descriptor.h"
|
|
|
|
|
2014-02-18 13:12:39 +01:00
|
|
|
#define PMT_SCAN_TIMEOUT 1000 // ms
|
2003-12-22 13:29:24 +01:00
|
|
|
|
|
|
|
// --- cCaDescriptor ---------------------------------------------------------
|
|
|
|
|
|
|
|
class cCaDescriptor : public cListObject {
|
|
|
|
private:
|
|
|
|
int caSystem;
|
2014-01-01 12:37:22 +01:00
|
|
|
int caPid;
|
2009-08-15 22:19:42 +02:00
|
|
|
int esPid;
|
2003-12-22 13:29:24 +01:00
|
|
|
int length;
|
|
|
|
uchar *data;
|
|
|
|
public:
|
2009-08-15 22:19:42 +02:00
|
|
|
cCaDescriptor(int CaSystem, int CaPid, int EsPid, int Length, const uchar *Data);
|
2003-12-22 13:29:24 +01:00
|
|
|
virtual ~cCaDescriptor();
|
2004-01-04 12:30:00 +01:00
|
|
|
bool operator== (const cCaDescriptor &arg) const;
|
|
|
|
int CaSystem(void) { return caSystem; }
|
2014-01-01 12:37:22 +01:00
|
|
|
int CaPid(void) { return caPid; }
|
2009-08-15 22:19:42 +02:00
|
|
|
int EsPid(void) { return esPid; }
|
2003-12-22 13:29:24 +01:00
|
|
|
int Length(void) const { return length; }
|
|
|
|
const uchar *Data(void) const { return data; }
|
|
|
|
};
|
|
|
|
|
2009-08-15 22:19:42 +02:00
|
|
|
cCaDescriptor::cCaDescriptor(int CaSystem, int CaPid, int EsPid, int Length, const uchar *Data)
|
2003-12-22 13:29:24 +01:00
|
|
|
{
|
|
|
|
caSystem = CaSystem;
|
2014-01-01 12:37:22 +01:00
|
|
|
caPid = CaPid;
|
2009-08-15 22:19:42 +02:00
|
|
|
esPid = EsPid;
|
2003-12-22 13:29:24 +01:00
|
|
|
length = Length + 6;
|
|
|
|
data = MALLOC(uchar, length);
|
|
|
|
data[0] = SI::CaDescriptorTag;
|
|
|
|
data[1] = length - 2;
|
|
|
|
data[2] = (caSystem >> 8) & 0xFF;
|
|
|
|
data[3] = caSystem & 0xFF;
|
|
|
|
data[4] = ((CaPid >> 8) & 0x1F) | 0xE0;
|
|
|
|
data[5] = CaPid & 0xFF;
|
|
|
|
if (Length)
|
|
|
|
memcpy(&data[6], Data, Length);
|
|
|
|
}
|
|
|
|
|
|
|
|
cCaDescriptor::~cCaDescriptor()
|
|
|
|
{
|
|
|
|
free(data);
|
|
|
|
}
|
|
|
|
|
2004-01-04 12:30:00 +01:00
|
|
|
bool cCaDescriptor::operator== (const cCaDescriptor &arg) const
|
|
|
|
{
|
2009-08-15 22:19:42 +02:00
|
|
|
return esPid == arg.esPid && length == arg.length && memcmp(data, arg.data, length) == 0;
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
|
2003-12-22 13:29:24 +01:00
|
|
|
// --- cCaDescriptors --------------------------------------------------------
|
|
|
|
|
2004-01-04 12:30:00 +01:00
|
|
|
class cCaDescriptors : public cListObject {
|
2003-12-22 13:29:24 +01:00
|
|
|
private:
|
2004-01-04 12:30:00 +01:00
|
|
|
int source;
|
|
|
|
int transponder;
|
|
|
|
int serviceId;
|
2015-01-04 13:36:46 +01:00
|
|
|
int pmtPid; // needed for OctopusNet - otherwise irrelevant!
|
2004-01-04 12:30:00 +01:00
|
|
|
int numCaIds;
|
|
|
|
int caIds[MAXCAIDS + 1];
|
|
|
|
cList<cCaDescriptor> caDescriptors;
|
|
|
|
void AddCaId(int CaId);
|
2003-12-22 13:29:24 +01:00
|
|
|
public:
|
2015-01-04 13:36:46 +01:00
|
|
|
cCaDescriptors(int Source, int Transponder, int ServiceId, int PmtPid);
|
2004-01-04 12:30:00 +01:00
|
|
|
bool operator== (const cCaDescriptors &arg) const;
|
|
|
|
bool Is(int Source, int Transponder, int ServiceId);
|
|
|
|
bool Is(cCaDescriptors * CaDescriptors);
|
|
|
|
bool Empty(void) { return caDescriptors.Count() == 0; }
|
2009-08-15 22:19:42 +02:00
|
|
|
void AddCaDescriptor(SI::CaDescriptor *d, int EsPid);
|
2016-12-23 14:08:14 +01:00
|
|
|
void GetCaDescriptors(const int *CaSystemIds, cDynamicBuffer &Buffer, int EsPid);
|
2014-01-01 12:37:22 +01:00
|
|
|
int GetCaPids(const int *CaSystemIds, int BufSize, int *Pids);
|
2015-01-04 13:36:46 +01:00
|
|
|
const int GetPmtPid(void) { return pmtPid; };
|
2004-01-04 12:30:00 +01:00
|
|
|
const int *CaIds(void) { return caIds; }
|
2003-12-22 13:29:24 +01:00
|
|
|
};
|
|
|
|
|
2015-01-04 13:36:46 +01:00
|
|
|
cCaDescriptors::cCaDescriptors(int Source, int Transponder, int ServiceId, int PmtPid)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
|
|
|
source = Source;
|
|
|
|
transponder = Transponder;
|
|
|
|
serviceId = ServiceId;
|
2015-01-04 13:36:46 +01:00
|
|
|
pmtPid = PmtPid;
|
2004-01-04 12:30:00 +01:00
|
|
|
numCaIds = 0;
|
|
|
|
caIds[0] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cCaDescriptors::operator== (const cCaDescriptors &arg) const
|
|
|
|
{
|
2015-09-01 11:14:27 +02:00
|
|
|
const cCaDescriptor *ca1 = caDescriptors.First();
|
|
|
|
const cCaDescriptor *ca2 = arg.caDescriptors.First();
|
2004-01-04 12:30:00 +01:00
|
|
|
while (ca1 && ca2) {
|
|
|
|
if (!(*ca1 == *ca2))
|
|
|
|
return false;
|
|
|
|
ca1 = caDescriptors.Next(ca1);
|
|
|
|
ca2 = arg.caDescriptors.Next(ca2);
|
|
|
|
}
|
|
|
|
return !ca1 && !ca2;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cCaDescriptors::Is(int Source, int Transponder, int ServiceId)
|
|
|
|
{
|
|
|
|
return source == Source && transponder == Transponder && serviceId == ServiceId;
|
|
|
|
}
|
|
|
|
|
2004-01-25 15:32:08 +01:00
|
|
|
bool cCaDescriptors::Is(cCaDescriptors *CaDescriptors)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
|
|
|
return Is(CaDescriptors->source, CaDescriptors->transponder, CaDescriptors->serviceId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cCaDescriptors::AddCaId(int CaId)
|
|
|
|
{
|
|
|
|
if (numCaIds < MAXCAIDS) {
|
|
|
|
for (int i = 0; i < numCaIds; i++) {
|
|
|
|
if (caIds[i] == CaId)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
caIds[numCaIds++] = CaId;
|
|
|
|
caIds[numCaIds] = 0;
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
|
2009-08-15 22:19:42 +02:00
|
|
|
void cCaDescriptors::AddCaDescriptor(SI::CaDescriptor *d, int EsPid)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
2009-08-15 22:19:42 +02:00
|
|
|
cCaDescriptor *nca = new cCaDescriptor(d->getCaType(), d->getCaPid(), EsPid, d->privateData.getLength(), d->privateData.getData());
|
2004-01-04 12:30:00 +01:00
|
|
|
for (cCaDescriptor *ca = caDescriptors.First(); ca; ca = caDescriptors.Next(ca)) {
|
|
|
|
if (*ca == *nca) {
|
|
|
|
delete nca;
|
2003-12-22 13:29:24 +01:00
|
|
|
return;
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
AddCaId(nca->CaSystem());
|
|
|
|
caDescriptors.Add(nca);
|
|
|
|
//#define DEBUG_CA_DESCRIPTORS 1
|
|
|
|
#ifdef DEBUG_CA_DESCRIPTORS
|
|
|
|
char buffer[1024];
|
|
|
|
char *q = buffer;
|
2009-08-15 22:19:42 +02:00
|
|
|
q += sprintf(q, "CAM: %04X %5d %5d %04X %04X -", source, transponder, serviceId, d->getCaType(), EsPid);
|
2004-01-04 12:30:00 +01:00
|
|
|
for (int i = 0; i < nca->Length(); i++)
|
|
|
|
q += sprintf(q, " %02X", nca->Data()[i]);
|
2010-01-01 15:44:52 +01:00
|
|
|
dsyslog("%s", buffer);
|
2004-01-04 12:30:00 +01:00
|
|
|
#endif
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
|
2009-08-15 22:19:42 +02:00
|
|
|
// EsPid is to select the "type" of CaDescriptor to be returned
|
|
|
|
// >0 - CaDescriptor for the particular esPid
|
|
|
|
// =0 - common CaDescriptor
|
|
|
|
// <0 - all CaDescriptors regardless of type (old default)
|
|
|
|
|
2016-12-23 14:08:14 +01:00
|
|
|
void cCaDescriptors::GetCaDescriptors(const int *CaSystemIds, cDynamicBuffer &Buffer, int EsPid)
|
2003-12-22 13:29:24 +01:00
|
|
|
{
|
2016-12-23 14:08:14 +01:00
|
|
|
Buffer.Clear();
|
2003-12-22 13:29:24 +01:00
|
|
|
if (!CaSystemIds || !*CaSystemIds)
|
2016-12-23 14:08:14 +01:00
|
|
|
return;
|
|
|
|
for (cCaDescriptor *d = caDescriptors.First(); d; d = caDescriptors.Next(d)) {
|
|
|
|
if (EsPid < 0 || d->EsPid() == EsPid) {
|
|
|
|
const int *caids = CaSystemIds;
|
|
|
|
do {
|
|
|
|
if (*caids == 0xFFFF || d->CaSystem() == *caids)
|
|
|
|
Buffer.Append(d->Data(), d->Length());
|
|
|
|
} while (*++caids);
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
2016-12-23 14:08:14 +01:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
|
2014-01-01 12:37:22 +01:00
|
|
|
int cCaDescriptors::GetCaPids(const int *CaSystemIds, int BufSize, int *Pids)
|
|
|
|
{
|
|
|
|
if (!CaSystemIds || !*CaSystemIds)
|
|
|
|
return 0;
|
|
|
|
if (BufSize > 0 && Pids) {
|
|
|
|
int numPids = 0;
|
|
|
|
for (cCaDescriptor *d = caDescriptors.First(); d; d = caDescriptors.Next(d)) {
|
|
|
|
const int *caids = CaSystemIds;
|
|
|
|
do {
|
2014-01-04 11:17:24 +01:00
|
|
|
if (*caids == 0xFFFF || d->CaSystem() == *caids) {
|
2014-01-01 12:37:22 +01:00
|
|
|
if (numPids + 1 < BufSize) {
|
|
|
|
Pids[numPids++] = d->CaPid();
|
|
|
|
Pids[numPids] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} while (*++caids);
|
|
|
|
}
|
|
|
|
return numPids;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2004-01-04 12:30:00 +01:00
|
|
|
// --- cCaDescriptorHandler --------------------------------------------------
|
|
|
|
|
|
|
|
class cCaDescriptorHandler : public cList<cCaDescriptors> {
|
|
|
|
private:
|
|
|
|
cMutex mutex;
|
|
|
|
public:
|
|
|
|
int AddCaDescriptors(cCaDescriptors *CaDescriptors);
|
|
|
|
// Returns 0 if this is an already known descriptor,
|
|
|
|
// 1 if it is an all new descriptor with actual contents,
|
|
|
|
// and 2 if an existing descriptor was changed.
|
2016-12-23 14:08:14 +01:00
|
|
|
void GetCaDescriptors(int Source, int Transponder, int ServiceId, const int *CaSystemIds, cDynamicBuffer &Buffer, int EsPid);
|
2014-01-01 12:37:22 +01:00
|
|
|
int GetCaPids(int Source, int Transponder, int ServiceId, const int *CaSystemIds, int BufSize, int *Pids);
|
2015-01-04 13:36:46 +01:00
|
|
|
int GetPmtPid(int Source, int Transponder, int ServiceId);
|
2004-01-04 12:30:00 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
int cCaDescriptorHandler::AddCaDescriptors(cCaDescriptors *CaDescriptors)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
for (cCaDescriptors *ca = First(); ca; ca = Next(ca)) {
|
|
|
|
if (ca->Is(CaDescriptors)) {
|
|
|
|
if (*ca == *CaDescriptors) {
|
|
|
|
delete CaDescriptors;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
Del(ca);
|
|
|
|
Add(CaDescriptors);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Add(CaDescriptors);
|
|
|
|
return CaDescriptors->Empty() ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
2016-12-23 14:08:14 +01:00
|
|
|
void cCaDescriptorHandler::GetCaDescriptors(int Source, int Transponder, int ServiceId, const int *CaSystemIds, cDynamicBuffer &Buffer, int EsPid)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
for (cCaDescriptors *ca = First(); ca; ca = Next(ca)) {
|
2016-12-23 14:08:14 +01:00
|
|
|
if (ca->Is(Source, Transponder, ServiceId)) {
|
|
|
|
ca->GetCaDescriptors(CaSystemIds, Buffer, EsPid);
|
|
|
|
break;
|
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-01 12:37:22 +01:00
|
|
|
int cCaDescriptorHandler::GetCaPids(int Source, int Transponder, int ServiceId, const int *CaSystemIds, int BufSize, int *Pids)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
for (cCaDescriptors *ca = First(); ca; ca = Next(ca)) {
|
|
|
|
if (ca->Is(Source, Transponder, ServiceId))
|
|
|
|
return ca->GetCaPids(CaSystemIds, BufSize, Pids);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-01-04 13:36:46 +01:00
|
|
|
int cCaDescriptorHandler::GetPmtPid(int Source, int Transponder, int ServiceId)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
for (cCaDescriptors *ca = First(); ca; ca = Next(ca)) {
|
|
|
|
if (ca->Is(Source, Transponder, ServiceId))
|
|
|
|
return ca->GetPmtPid();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-01-04 12:30:00 +01:00
|
|
|
cCaDescriptorHandler CaDescriptorHandler;
|
2003-12-22 13:29:24 +01:00
|
|
|
|
2016-12-23 14:08:14 +01:00
|
|
|
void GetCaDescriptors(int Source, int Transponder, int ServiceId, const int *CaSystemIds, cDynamicBuffer &Buffer, int EsPid)
|
2003-12-22 13:29:24 +01:00
|
|
|
{
|
2016-12-23 14:08:14 +01:00
|
|
|
CaDescriptorHandler.GetCaDescriptors(Source, Transponder, ServiceId, CaSystemIds, Buffer, EsPid);
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
|
2014-01-01 12:37:22 +01:00
|
|
|
int GetCaPids(int Source, int Transponder, int ServiceId, const int *CaSystemIds, int BufSize, int *Pids)
|
|
|
|
{
|
|
|
|
return CaDescriptorHandler.GetCaPids(Source, Transponder, ServiceId, CaSystemIds, BufSize, Pids);
|
|
|
|
}
|
|
|
|
|
2015-01-04 13:36:46 +01:00
|
|
|
int GetPmtPid(int Source, int Transponder, int ServiceId)
|
|
|
|
{
|
|
|
|
return CaDescriptorHandler.GetPmtPid(Source, Transponder, ServiceId);
|
|
|
|
}
|
|
|
|
|
2020-06-19 12:19:15 +02:00
|
|
|
// --- cPmtPidEntry ----------------------------------------------------------
|
|
|
|
|
|
|
|
class cPmtPidEntry : public cListObject {
|
|
|
|
private:
|
|
|
|
int pid;
|
2021-06-08 14:57:26 +02:00
|
|
|
int count; // the number of SIDs currently requested from this PID
|
|
|
|
int state; // adding/deleting PIDs to/from the filter may only be done from within the Process() function,
|
|
|
|
// otherwise there could be a deadlock between cPatFilter::mutex and cSectionHandler::mutex;
|
|
|
|
// this member tells whether this PID needs to be added to (>0) or deleted from (<0) the filter
|
|
|
|
bool complete; // true if all SIDs on this PID have been received
|
2020-06-19 12:19:15 +02:00
|
|
|
public:
|
|
|
|
cPmtPidEntry(int Pid);
|
|
|
|
int Pid(void) { return pid; }
|
2021-06-08 14:57:26 +02:00
|
|
|
int Count(void) { return count; }
|
|
|
|
int State(void) { int s = state; state = 0; return s; } // returns the current state and resets it
|
2021-06-11 09:38:38 +02:00
|
|
|
void SetState(int State) { state = State; } // 1 = add the PID, -1 = delete the PID, 0 = do nothing
|
2021-06-08 14:57:26 +02:00
|
|
|
void Inc(void) { if (++count == 1) state = 1; }
|
|
|
|
void Dec(void) { if (--count == 0) state = -1; }
|
2020-06-19 12:19:15 +02:00
|
|
|
int Complete(void) { return complete; }
|
|
|
|
void SetComplete(bool State) { complete = State; }
|
|
|
|
};
|
|
|
|
|
|
|
|
cPmtPidEntry::cPmtPidEntry(int Pid)
|
|
|
|
{
|
|
|
|
pid = Pid;
|
2021-06-08 14:57:26 +02:00
|
|
|
count = 0;
|
|
|
|
state = 0;
|
2020-06-19 12:19:15 +02:00
|
|
|
complete = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- cPmtSidEntry ----------------------------------------------------------
|
|
|
|
|
|
|
|
class cPmtSidEntry : public cListObject {
|
|
|
|
private:
|
|
|
|
int sid;
|
|
|
|
int pid;
|
|
|
|
cPmtPidEntry *pidEntry;
|
|
|
|
int version;
|
|
|
|
bool received;
|
|
|
|
public:
|
2021-06-08 14:57:26 +02:00
|
|
|
cPmtSidEntry(int Sid, cPmtPidEntry *PidEntry);
|
2020-06-19 12:19:15 +02:00
|
|
|
int Sid(void) { return sid; }
|
|
|
|
int Pid(void) { return pid; }
|
|
|
|
cPmtPidEntry *PidEntry(void) { return pidEntry; }
|
|
|
|
int Version(void) { return version; }
|
|
|
|
int Received(void) { return received; }
|
|
|
|
void SetVersion(int Version) { version = Version; }
|
|
|
|
void SetReceived(bool State) { received = State; }
|
|
|
|
};
|
|
|
|
|
2021-06-08 14:57:26 +02:00
|
|
|
cPmtSidEntry::cPmtSidEntry(int Sid, cPmtPidEntry *PidEntry)
|
2020-06-19 12:19:15 +02:00
|
|
|
{
|
|
|
|
sid = Sid;
|
2021-06-08 14:57:26 +02:00
|
|
|
pid = PidEntry->Pid();
|
2020-06-19 12:19:15 +02:00
|
|
|
pidEntry = PidEntry;
|
|
|
|
version = -1;
|
|
|
|
received = false;
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:57:26 +02:00
|
|
|
// --- cPmtSidRequest --------------------------------------------------------
|
|
|
|
|
|
|
|
class cPmtSidRequest : public cListObject {
|
|
|
|
private:
|
|
|
|
int sid;
|
|
|
|
int count; // the number of requests for this SID
|
|
|
|
public:
|
|
|
|
cPmtSidRequest(int Sid) { sid = Sid; count = 1; }
|
|
|
|
int Sid(void) { return sid; }
|
|
|
|
int Count(void) { return count; }
|
|
|
|
void Inc(void) { count++; }
|
|
|
|
void Dec(void) { count--; }
|
|
|
|
};
|
|
|
|
|
2003-12-22 13:29:24 +01:00
|
|
|
// --- cPatFilter ------------------------------------------------------------
|
|
|
|
|
2014-02-18 13:12:39 +01:00
|
|
|
//#define DEBUG_PAT_PMT
|
|
|
|
#ifdef DEBUG_PAT_PMT
|
|
|
|
#define DBGLOG(a...) { cString s = cString::sprintf(a); fprintf(stderr, "%s\n", *s); dsyslog("%s", *s); }
|
|
|
|
#else
|
2020-12-18 14:51:57 +01:00
|
|
|
#define DBGLOG(a...) void()
|
2014-02-18 13:12:39 +01:00
|
|
|
#endif
|
|
|
|
|
2003-12-22 13:29:24 +01:00
|
|
|
cPatFilter::cPatFilter(void)
|
|
|
|
{
|
2021-06-08 14:57:26 +02:00
|
|
|
patVersion = -1;
|
|
|
|
activePmt = NULL;
|
|
|
|
transponder = 0;
|
|
|
|
source = 0;
|
2003-12-22 13:29:24 +01:00
|
|
|
Set(0x00, 0x00); // PAT
|
|
|
|
}
|
|
|
|
|
2021-06-11 09:38:38 +02:00
|
|
|
void cPatFilter::SetStatus(bool On)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
if (On) { // restart all requested PMT Pids
|
|
|
|
for (cPmtPidEntry *pPid = pmtPidList.First(); pPid; pPid = pmtPidList.Next(pPid))
|
|
|
|
pPid->SetState(pPid->Count() > 0);
|
|
|
|
if (activePmt && activePmt->Count() == 0) {
|
|
|
|
activePmt->SetState(1);
|
|
|
|
timer.Set(PMT_SCAN_TIMEOUT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBGLOG("PAT filter set status %d", On);
|
|
|
|
cFilter::SetStatus(On);
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:57:26 +02:00
|
|
|
bool cPatFilter::TransponderChanged(void)
|
2003-12-22 13:29:24 +01:00
|
|
|
{
|
2021-06-08 14:57:26 +02:00
|
|
|
if (source != Source() || transponder != Transponder()) {
|
|
|
|
DBGLOG("PAT filter transponder changed from %d/%d to %d/%d", source, transponder, Source(), Transponder());
|
|
|
|
source = Source();
|
|
|
|
transponder = Transponder();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
|
2021-06-08 14:57:26 +02:00
|
|
|
void cPatFilter::Trigger(int)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
2014-02-18 13:12:39 +01:00
|
|
|
cMutexLock MutexLock(&mutex);
|
2021-06-08 14:57:26 +02:00
|
|
|
DBGLOG("PAT filter trigger");
|
2021-06-21 20:13:55 +02:00
|
|
|
patVersion = -1;
|
|
|
|
sectionSyncer.Reset();
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
|
2021-06-08 14:57:26 +02:00
|
|
|
void cPatFilter::Request(int Sid)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
DBGLOG("PAT filter request SID %d", Sid);
|
|
|
|
for (cPmtSidRequest *sr = pmtSidRequestList.First(); sr; sr = pmtSidRequestList.Next(sr)) {
|
|
|
|
if (sr->Sid() == Sid) {
|
|
|
|
sr->Inc();
|
|
|
|
DBGLOG("PAT filter add SID request %d (%d)", Sid, sr->Count());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBGLOG("PAT filter new SID request %d", Sid);
|
|
|
|
pmtSidRequestList.Add(new cPmtSidRequest(Sid));
|
|
|
|
for (cPmtSidEntry *se = pmtSidList.First(); se; se = pmtSidList.Next(se)) {
|
|
|
|
if (se->Sid() == Sid) {
|
|
|
|
cPmtPidEntry *pPid = se->PidEntry();
|
|
|
|
pPid->Inc();
|
|
|
|
DBGLOG(" PMT pid %5d SID %5d (%d)", pPid->Pid(), se->Sid(), pPid->Count());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cPatFilter::Release(int Sid)
|
|
|
|
{
|
|
|
|
cMutexLock MutexLock(&mutex);
|
|
|
|
DBGLOG("PAT filter release SID %d", Sid);
|
|
|
|
for (cPmtSidRequest *sr = pmtSidRequestList.First(); sr; sr = pmtSidRequestList.Next(sr)) {
|
|
|
|
if (sr->Sid() == Sid) {
|
|
|
|
sr->Dec();
|
|
|
|
DBGLOG("PAT filter del SID request %d (%d)", Sid, sr->Count());
|
|
|
|
if (sr->Count() == 0) {
|
|
|
|
pmtSidRequestList.Del(sr);
|
|
|
|
for (cPmtSidEntry *se = pmtSidList.First(); se; se = pmtSidList.Next(se)) {
|
|
|
|
if (se->Sid() == Sid) {
|
|
|
|
cPmtPidEntry *pPid = se->PidEntry();
|
|
|
|
pPid->Dec();
|
|
|
|
DBGLOG(" PMT pid %5d SID %5d (%d)", pPid->Pid(), se->Sid(), pPid->Count());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int cPatFilter::NumSidRequests(int Sid)
|
|
|
|
{
|
|
|
|
for (cPmtSidRequest *sr = pmtSidRequestList.First(); sr; sr = pmtSidRequestList.Next(sr)) {
|
|
|
|
if (sr->Sid() == Sid)
|
|
|
|
return sr->Count();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-06-19 12:19:15 +02:00
|
|
|
bool cPatFilter::PmtPidComplete(int PmtPid)
|
|
|
|
{
|
|
|
|
for (cPmtSidEntry *se = pmtSidList.First(); se; se = pmtSidList.Next(se)) {
|
|
|
|
if (se->Pid() == PmtPid && !se->Received())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cPatFilter::PmtPidReset(int PmtPid)
|
|
|
|
{
|
|
|
|
for (cPmtSidEntry *se = pmtSidList.First(); se; se = pmtSidList.Next(se)) {
|
2021-06-08 14:57:26 +02:00
|
|
|
if (se->Pid() == PmtPid) {
|
2020-06-19 12:19:15 +02:00
|
|
|
se->SetReceived(false);
|
2021-06-08 14:57:26 +02:00
|
|
|
se->PidEntry()->SetComplete(false);
|
|
|
|
}
|
2020-06-19 12:19:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-18 13:12:39 +01:00
|
|
|
bool cPatFilter::PmtVersionChanged(int PmtPid, int Sid, int Version, bool SetNewVersion)
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
2021-06-08 14:57:26 +02:00
|
|
|
for (cPmtSidEntry *se = pmtSidList.First(); se; se = pmtSidList.Next(se)) {
|
2020-06-19 12:19:15 +02:00
|
|
|
if (se->Sid() == Sid && se->Pid() == PmtPid) {
|
|
|
|
if (!se->Received()) {
|
|
|
|
se->SetReceived(true);
|
2020-12-09 21:42:26 +01:00
|
|
|
se->PidEntry()->SetComplete(PmtPidComplete(PmtPid));
|
2020-06-19 12:19:15 +02:00
|
|
|
}
|
|
|
|
if (se->Version() != Version) {
|
2014-02-18 13:12:39 +01:00
|
|
|
if (SetNewVersion)
|
2020-06-19 12:19:15 +02:00
|
|
|
se->SetVersion(Version);
|
2020-12-12 10:31:52 +01:00
|
|
|
else
|
2021-06-08 14:57:26 +02:00
|
|
|
DBGLOG("PMT %d %5d/%5d %2d -> %2d %d", Transponder(), PmtPid, Sid, se->Version(), Version, NumSidRequests(Sid));
|
2014-02-18 13:12:39 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
}
|
2014-02-18 13:12:39 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cPatFilter::SwitchToNextPmtPid(void)
|
|
|
|
{
|
2020-06-19 12:19:15 +02:00
|
|
|
if (activePmt) {
|
2021-06-08 14:57:26 +02:00
|
|
|
if (activePmt->Count() == 0)
|
|
|
|
Del(activePmt->Pid(), SI::TableIdPMT);
|
|
|
|
for (;;) {
|
|
|
|
activePmt = pmtPidList.Next(activePmt);
|
|
|
|
if (!activePmt || activePmt->Count() == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (activePmt) {
|
|
|
|
PmtPidReset(activePmt->Pid());
|
|
|
|
Add(activePmt->Pid(), SI::TableIdPMT);
|
|
|
|
timer.Set(PMT_SCAN_TIMEOUT);
|
|
|
|
}
|
2014-02-18 13:12:39 +01:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void cPatFilter::Process(u_short Pid, u_char Tid, const u_char *Data, int Length)
|
|
|
|
{
|
2014-02-18 13:12:39 +01:00
|
|
|
cMutexLock MutexLock(&mutex);
|
2021-06-08 14:57:26 +02:00
|
|
|
if (TransponderChanged()) {
|
|
|
|
patVersion = -1;
|
|
|
|
sectionSyncer.Reset();
|
|
|
|
}
|
|
|
|
if (patVersion >= 0) {
|
|
|
|
for (cPmtPidEntry *pPid = pmtPidList.First(); pPid; pPid = pmtPidList.Next(pPid)) {
|
|
|
|
int State = pPid->State();
|
|
|
|
if (State > 0)
|
|
|
|
Add(pPid->Pid(), SI::TableIdPMT);
|
|
|
|
else if (State < 0)
|
|
|
|
Del(pPid->Pid(), SI::TableIdPMT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (Pid != 0x00)
|
|
|
|
return;
|
2003-12-22 13:29:24 +01:00
|
|
|
if (Pid == 0x00) {
|
2014-02-18 13:12:39 +01:00
|
|
|
if (Tid == SI::TableIdPAT) {
|
|
|
|
SI::PAT pat(Data, false);
|
|
|
|
if (!pat.CheckCRCAndParse())
|
|
|
|
return;
|
2021-03-16 15:10:54 +01:00
|
|
|
if (sectionSyncer.Check(pat.getVersionNumber(), pat.getSectionNumber())) {
|
2020-06-19 12:19:15 +02:00
|
|
|
DBGLOG("PAT %d %d -> %d %d/%d", Transponder(), patVersion, pat.getVersionNumber(), pat.getSectionNumber(), pat.getLastSectionNumber());
|
2021-06-08 14:57:26 +02:00
|
|
|
bool NeedsSetStatus = patVersion >= 0;
|
2020-06-19 12:19:15 +02:00
|
|
|
if (pat.getVersionNumber() != patVersion) {
|
2021-06-08 14:57:26 +02:00
|
|
|
if (NeedsSetStatus)
|
|
|
|
SetStatus(false); // deletes all PIDs from the filter
|
|
|
|
activePmt = NULL;
|
2020-06-19 12:19:15 +02:00
|
|
|
pmtSidList.Clear();
|
|
|
|
pmtPidList.Clear();
|
|
|
|
patVersion = pat.getVersionNumber();
|
2014-02-18 13:12:39 +01:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
SI::PAT::Association assoc;
|
2004-10-16 10:14:19 +02:00
|
|
|
for (SI::Loop::Iterator it; pat.associationLoop.getNext(assoc, it); ) {
|
2020-06-19 12:19:15 +02:00
|
|
|
if (!assoc.isNITPid()) {
|
|
|
|
int PmtPid = assoc.getPid();
|
2021-06-08 14:57:26 +02:00
|
|
|
int PmtSid = assoc.getServiceId();
|
2020-06-19 12:19:15 +02:00
|
|
|
cPmtPidEntry *pPid = NULL;
|
2021-06-08 14:57:26 +02:00
|
|
|
for (pPid = pmtPidList.First(); pPid; pPid = pmtPidList.Next(pPid)) {
|
|
|
|
if (pPid->Pid() == PmtPid)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
int SidRequest = NumSidRequests(PmtSid);
|
|
|
|
DBGLOG(" PMT pid %5d SID %5d%s%s", PmtPid, PmtSid, SidRequest ? " R" : "", pPid ? " S" : "");
|
2020-06-19 12:19:15 +02:00
|
|
|
if (!pPid) { // new PMT Pid
|
|
|
|
pPid = new cPmtPidEntry(PmtPid);
|
|
|
|
pmtPidList.Add(pPid);
|
|
|
|
}
|
2021-06-08 14:57:26 +02:00
|
|
|
pmtSidList.Add(new cPmtSidEntry(PmtSid, pPid));
|
|
|
|
if (SidRequest > 0)
|
|
|
|
pPid->Inc();
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
}
|
2021-03-16 15:10:54 +01:00
|
|
|
if (sectionSyncer.Processed(pat.getSectionNumber(), pat.getLastSectionNumber())) { // all PAT sections done
|
2021-06-08 14:57:26 +02:00
|
|
|
for (cPmtPidEntry *pPid = pmtPidList.First(); pPid; pPid = pmtPidList.Next(pPid)) {
|
|
|
|
if (pPid->Count() == 0) {
|
2021-06-11 09:38:38 +02:00
|
|
|
pPid->SetState(1);
|
2021-06-08 14:57:26 +02:00
|
|
|
activePmt = pPid;
|
|
|
|
timer.Set(PMT_SCAN_TIMEOUT);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (NeedsSetStatus)
|
|
|
|
SetStatus(true);
|
2020-06-19 12:19:15 +02:00
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-02-18 13:12:39 +01:00
|
|
|
else if (Tid == SI::TableIdPMT && Source() && Transponder()) {
|
2003-12-22 13:29:24 +01:00
|
|
|
SI::PMT pmt(Data, false);
|
|
|
|
if (!pmt.CheckCRCAndParse())
|
|
|
|
return;
|
2020-12-12 10:31:52 +01:00
|
|
|
if (!PmtVersionChanged(Pid, pmt.getTableIdExtension(), pmt.getVersionNumber(), false)) {
|
2020-06-19 12:19:15 +02:00
|
|
|
if (activePmt && activePmt->Complete())
|
|
|
|
SwitchToNextPmtPid();
|
2004-01-04 12:30:00 +01:00
|
|
|
return;
|
|
|
|
}
|
2015-09-01 11:14:27 +02:00
|
|
|
cStateKey StateKey;
|
|
|
|
cChannels *Channels = cChannels::GetChannelsWrite(StateKey, 10);
|
|
|
|
if (!Channels)
|
2004-01-04 12:30:00 +01:00
|
|
|
return;
|
2020-12-12 10:31:52 +01:00
|
|
|
PmtVersionChanged(Pid, pmt.getTableIdExtension(), pmt.getVersionNumber(), true);
|
2015-09-01 11:14:27 +02:00
|
|
|
bool ChannelsModified = false;
|
2020-06-19 12:19:15 +02:00
|
|
|
if (activePmt && activePmt->Complete())
|
|
|
|
SwitchToNextPmtPid();
|
2015-09-01 11:14:27 +02:00
|
|
|
cChannel *Channel = Channels->GetByServiceID(Source(), Transponder(), pmt.getServiceId());
|
2004-01-04 12:30:00 +01:00
|
|
|
if (Channel) {
|
|
|
|
SI::CaDescriptor *d;
|
2015-01-04 13:36:46 +01:00
|
|
|
cCaDescriptors *CaDescriptors = new cCaDescriptors(Channel->Source(), Channel->Transponder(), Channel->Sid(), Pid);
|
2004-01-04 12:30:00 +01:00
|
|
|
// Scan the common loop:
|
|
|
|
for (SI::Loop::Iterator it; (d = (SI::CaDescriptor*)pmt.commonDescriptors.getNext(it, SI::CaDescriptorTag)); ) {
|
2009-08-15 22:19:42 +02:00
|
|
|
CaDescriptors->AddCaDescriptor(d, 0);
|
2004-01-04 12:30:00 +01:00
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
// Scan the stream-specific loop:
|
|
|
|
SI::PMT::Stream stream;
|
|
|
|
int Vpid = 0;
|
2008-08-15 14:49:34 +02:00
|
|
|
int Ppid = 0;
|
|
|
|
int Vtype = 0;
|
2005-08-06 12:29:38 +02:00
|
|
|
int Apids[MAXAPIDS + 1] = { 0 }; // these lists are zero-terminated
|
2010-06-13 11:14:26 +02:00
|
|
|
int Atypes[MAXAPIDS + 1] = { 0 };
|
2005-01-30 12:11:56 +01:00
|
|
|
int Dpids[MAXDPIDS + 1] = { 0 };
|
2010-06-05 13:50:56 +02:00
|
|
|
int Dtypes[MAXDPIDS + 1] = { 0 };
|
2007-10-12 14:52:30 +02:00
|
|
|
int Spids[MAXSPIDS + 1] = { 0 };
|
2009-08-16 15:32:39 +02:00
|
|
|
uchar SubtitlingTypes[MAXSPIDS + 1] = { 0 };
|
|
|
|
uint16_t CompositionPageIds[MAXSPIDS + 1] = { 0 };
|
|
|
|
uint16_t AncillaryPageIds[MAXSPIDS + 1] = { 0 };
|
2005-09-04 14:48:39 +02:00
|
|
|
char ALangs[MAXAPIDS][MAXLANGCODE2] = { "" };
|
|
|
|
char DLangs[MAXDPIDS][MAXLANGCODE2] = { "" };
|
2007-10-12 14:52:30 +02:00
|
|
|
char SLangs[MAXSPIDS][MAXLANGCODE2] = { "" };
|
2004-01-04 12:30:00 +01:00
|
|
|
int Tpid = 0;
|
|
|
|
int NumApids = 0;
|
|
|
|
int NumDpids = 0;
|
2007-10-12 14:52:30 +02:00
|
|
|
int NumSpids = 0;
|
2004-10-16 10:14:19 +02:00
|
|
|
for (SI::Loop::Iterator it; pmt.streamLoop.getNext(stream, it); ) {
|
2009-12-24 13:25:40 +01:00
|
|
|
bool ProcessCaDescriptors = false;
|
2009-08-15 22:19:42 +02:00
|
|
|
int esPid = stream.getPid();
|
2004-01-04 12:30:00 +01:00
|
|
|
switch (stream.getStreamType()) {
|
|
|
|
case 1: // STREAMTYPE_11172_VIDEO
|
|
|
|
case 2: // STREAMTYPE_13818_VIDEO
|
2012-11-25 14:21:15 +01:00
|
|
|
case 0x1B: // H.264
|
2016-12-22 12:22:11 +01:00
|
|
|
case 0x24: // H.265
|
2009-08-15 22:19:42 +02:00
|
|
|
Vpid = esPid;
|
2008-08-15 14:49:34 +02:00
|
|
|
Ppid = pmt.getPCRPid();
|
|
|
|
Vtype = stream.getStreamType();
|
2009-12-24 13:25:40 +01:00
|
|
|
ProcessCaDescriptors = true;
|
2004-01-04 12:30:00 +01:00
|
|
|
break;
|
|
|
|
case 3: // STREAMTYPE_11172_AUDIO
|
|
|
|
case 4: // STREAMTYPE_13818_AUDIO
|
2012-03-02 10:56:49 +01:00
|
|
|
case 0x0F: // ISO/IEC 13818-7 Audio with ADTS transport syntax
|
2010-05-16 13:36:55 +02:00
|
|
|
case 0x11: // ISO/IEC 14496-3 Audio with LATM transport syntax
|
2004-01-04 12:30:00 +01:00
|
|
|
{
|
2004-01-25 15:32:08 +01:00
|
|
|
if (NumApids < MAXAPIDS) {
|
2009-08-15 22:19:42 +02:00
|
|
|
Apids[NumApids] = esPid;
|
2010-05-16 13:36:55 +02:00
|
|
|
Atypes[NumApids] = stream.getStreamType();
|
2004-01-25 15:32:08 +01:00
|
|
|
SI::Descriptor *d;
|
|
|
|
for (SI::Loop::Iterator it; (d = stream.streamDescriptors.getNext(it)); ) {
|
|
|
|
switch (d->getDescriptorTag()) {
|
|
|
|
case SI::ISO639LanguageDescriptorTag: {
|
|
|
|
SI::ISO639LanguageDescriptor *ld = (SI::ISO639LanguageDescriptor *)d;
|
2005-09-04 14:48:39 +02:00
|
|
|
SI::ISO639LanguageDescriptor::Language l;
|
|
|
|
char *s = ALangs[NumApids];
|
|
|
|
int n = 0;
|
|
|
|
for (SI::Loop::Iterator it; ld->languageLoop.getNext(l, it); ) {
|
|
|
|
if (*ld->languageCode != '-') { // some use "---" to indicate "none"
|
|
|
|
if (n > 0)
|
|
|
|
*s++ = '+';
|
|
|
|
strn0cpy(s, I18nNormalizeLanguageCode(l.languageCode), MAXLANGCODE1);
|
|
|
|
s += strlen(s);
|
|
|
|
if (n++ > 1)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2004-01-25 15:32:08 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: ;
|
|
|
|
}
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
NumApids++;
|
|
|
|
}
|
2009-12-24 13:25:40 +01:00
|
|
|
ProcessCaDescriptors = true;
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 5: // STREAMTYPE_13818_PRIVATE
|
|
|
|
case 6: // STREAMTYPE_13818_PES_PRIVATE
|
|
|
|
//XXX case 8: // STREAMTYPE_13818_DSMCC
|
|
|
|
{
|
2004-01-25 15:32:08 +01:00
|
|
|
int dpid = 0;
|
2010-06-05 13:50:56 +02:00
|
|
|
int dtype = 0;
|
2005-09-04 14:48:39 +02:00
|
|
|
char lang[MAXLANGCODE1] = { 0 };
|
2004-01-04 12:30:00 +01:00
|
|
|
SI::Descriptor *d;
|
|
|
|
for (SI::Loop::Iterator it; (d = stream.streamDescriptors.getNext(it)); ) {
|
|
|
|
switch (d->getDescriptorTag()) {
|
|
|
|
case SI::AC3DescriptorTag:
|
2010-03-27 15:19:06 +01:00
|
|
|
case SI::EnhancedAC3DescriptorTag:
|
2009-08-15 22:19:42 +02:00
|
|
|
dpid = esPid;
|
2010-06-05 13:50:56 +02:00
|
|
|
dtype = d->getDescriptorTag();
|
2009-12-24 13:25:40 +01:00
|
|
|
ProcessCaDescriptors = true;
|
2004-01-04 12:30:00 +01:00
|
|
|
break;
|
2007-10-12 14:52:30 +02:00
|
|
|
case SI::SubtitlingDescriptorTag:
|
|
|
|
if (NumSpids < MAXSPIDS) {
|
2009-08-15 22:19:42 +02:00
|
|
|
Spids[NumSpids] = esPid;
|
2007-10-12 14:52:30 +02:00
|
|
|
SI::SubtitlingDescriptor *sd = (SI::SubtitlingDescriptor *)d;
|
|
|
|
SI::SubtitlingDescriptor::Subtitling sub;
|
|
|
|
char *s = SLangs[NumSpids];
|
|
|
|
int n = 0;
|
|
|
|
for (SI::Loop::Iterator it; sd->subtitlingLoop.getNext(sub, it); ) {
|
|
|
|
if (sub.languageCode[0]) {
|
2009-08-16 15:32:39 +02:00
|
|
|
SubtitlingTypes[NumSpids] = sub.getSubtitlingType();
|
|
|
|
CompositionPageIds[NumSpids] = sub.getCompositionPageId();
|
|
|
|
AncillaryPageIds[NumSpids] = sub.getAncillaryPageId();
|
2007-10-12 14:52:30 +02:00
|
|
|
if (n > 0)
|
|
|
|
*s++ = '+';
|
|
|
|
strn0cpy(s, I18nNormalizeLanguageCode(sub.languageCode), MAXLANGCODE1);
|
|
|
|
s += strlen(s);
|
|
|
|
if (n++ > 1)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NumSpids++;
|
|
|
|
}
|
|
|
|
break;
|
2004-01-04 12:30:00 +01:00
|
|
|
case SI::TeletextDescriptorTag:
|
2009-08-15 22:19:42 +02:00
|
|
|
Tpid = esPid;
|
2004-01-04 12:30:00 +01:00
|
|
|
break;
|
2004-01-25 15:32:08 +01:00
|
|
|
case SI::ISO639LanguageDescriptorTag: {
|
|
|
|
SI::ISO639LanguageDescriptor *ld = (SI::ISO639LanguageDescriptor *)d;
|
2005-09-04 14:48:39 +02:00
|
|
|
strn0cpy(lang, I18nNormalizeLanguageCode(ld->languageCode), MAXLANGCODE1);
|
2004-01-25 15:32:08 +01:00
|
|
|
}
|
|
|
|
break;
|
2004-01-04 12:30:00 +01:00
|
|
|
default: ;
|
|
|
|
}
|
|
|
|
delete d;
|
|
|
|
}
|
2004-01-25 15:32:08 +01:00
|
|
|
if (dpid) {
|
2005-01-16 14:40:47 +01:00
|
|
|
if (NumDpids < MAXDPIDS) {
|
2004-01-25 15:32:08 +01:00
|
|
|
Dpids[NumDpids] = dpid;
|
2010-06-05 13:50:56 +02:00
|
|
|
Dtypes[NumDpids] = dtype;
|
2005-09-04 14:48:39 +02:00
|
|
|
strn0cpy(DLangs[NumDpids], lang, MAXLANGCODE1);
|
2004-01-25 15:32:08 +01:00
|
|
|
NumDpids++;
|
|
|
|
}
|
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2010-03-06 12:01:17 +01:00
|
|
|
break;
|
2012-04-15 10:52:34 +02:00
|
|
|
case 0x80: // STREAMTYPE_USER_PRIVATE
|
|
|
|
if (Setup.StandardCompliance == STANDARD_ANSISCTE) { // DigiCipher II VIDEO (ANSI/SCTE 57)
|
|
|
|
Vpid = esPid;
|
|
|
|
Ppid = pmt.getPCRPid();
|
|
|
|
Vtype = 0x02; // compression based upon MPEG-2
|
|
|
|
ProcessCaDescriptors = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// fall through
|
|
|
|
case 0x81: // STREAMTYPE_USER_PRIVATE
|
2019-03-15 13:10:33 +01:00
|
|
|
case 0x87: // eac3
|
2012-04-15 10:52:34 +02:00
|
|
|
if (Setup.StandardCompliance == STANDARD_ANSISCTE) { // ATSC A/53 AUDIO (ANSI/SCTE 57)
|
|
|
|
char lang[MAXLANGCODE1] = { 0 };
|
|
|
|
SI::Descriptor *d;
|
|
|
|
for (SI::Loop::Iterator it; (d = stream.streamDescriptors.getNext(it)); ) {
|
|
|
|
switch (d->getDescriptorTag()) {
|
|
|
|
case SI::ISO639LanguageDescriptorTag: {
|
|
|
|
SI::ISO639LanguageDescriptor *ld = (SI::ISO639LanguageDescriptor *)d;
|
|
|
|
strn0cpy(lang, I18nNormalizeLanguageCode(ld->languageCode), MAXLANGCODE1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: ;
|
|
|
|
}
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
if (NumDpids < MAXDPIDS) {
|
|
|
|
Dpids[NumDpids] = esPid;
|
|
|
|
Dtypes[NumDpids] = SI::AC3DescriptorTag;
|
|
|
|
strn0cpy(DLangs[NumDpids], lang, MAXLANGCODE1);
|
|
|
|
NumDpids++;
|
2011-04-17 13:48:20 +02:00
|
|
|
}
|
2012-04-15 10:52:34 +02:00
|
|
|
ProcessCaDescriptors = true;
|
|
|
|
break;
|
2011-04-17 13:48:20 +02:00
|
|
|
}
|
2012-04-15 10:52:34 +02:00
|
|
|
// fall through
|
|
|
|
case 0x82: // STREAMTYPE_USER_PRIVATE
|
|
|
|
if (Setup.StandardCompliance == STANDARD_ANSISCTE) { // STANDARD SUBTITLE (ANSI/SCTE 27)
|
|
|
|
//TODO
|
|
|
|
break;
|
2011-04-17 13:48:20 +02:00
|
|
|
}
|
2012-04-15 10:52:34 +02:00
|
|
|
// fall through
|
2019-03-15 13:10:33 +01:00
|
|
|
case 0x83 ... 0x86: // STREAMTYPE_USER_PRIVATE
|
|
|
|
case 0x88 ... 0xFF: // STREAMTYPE_USER_PRIVATE
|
2010-11-01 15:47:59 +01:00
|
|
|
{
|
2010-11-07 13:47:16 +01:00
|
|
|
char lang[MAXLANGCODE1] = { 0 };
|
|
|
|
bool IsAc3 = false;
|
2010-11-01 15:47:59 +01:00
|
|
|
SI::Descriptor *d;
|
|
|
|
for (SI::Loop::Iterator it; (d = stream.streamDescriptors.getNext(it)); ) {
|
|
|
|
switch (d->getDescriptorTag()) {
|
|
|
|
case SI::RegistrationDescriptorTag: {
|
|
|
|
SI::RegistrationDescriptor *rd = (SI::RegistrationDescriptor *)d;
|
|
|
|
// http://www.smpte-ra.org/mpegreg/mpegreg.html
|
|
|
|
switch (rd->getFormatIdentifier()) {
|
|
|
|
case 0x41432D33: // 'AC-3'
|
|
|
|
IsAc3 = true;
|
|
|
|
break;
|
|
|
|
default:
|
2011-04-17 13:48:20 +02:00
|
|
|
//printf("Format identifier: 0x%08X (pid: %d)\n", rd->getFormatIdentifier(), esPid);
|
2010-11-01 15:47:59 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SI::ISO639LanguageDescriptorTag: {
|
|
|
|
SI::ISO639LanguageDescriptor *ld = (SI::ISO639LanguageDescriptor *)d;
|
|
|
|
strn0cpy(lang, I18nNormalizeLanguageCode(ld->languageCode), MAXLANGCODE1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default: ;
|
|
|
|
}
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
if (IsAc3) {
|
2010-03-06 12:01:17 +01:00
|
|
|
if (NumDpids < MAXDPIDS) {
|
2011-04-17 13:48:20 +02:00
|
|
|
Dpids[NumDpids] = esPid;
|
|
|
|
Dtypes[NumDpids] = SI::AC3DescriptorTag;
|
|
|
|
strn0cpy(DLangs[NumDpids], lang, MAXLANGCODE1);
|
|
|
|
NumDpids++;
|
|
|
|
}
|
2010-03-06 12:01:17 +01:00
|
|
|
ProcessCaDescriptors = true;
|
|
|
|
}
|
2010-11-01 15:47:59 +01:00
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
break;
|
2009-12-06 12:57:45 +01:00
|
|
|
default: ;//printf("PID: %5d %5d %2d %3d %3d\n", pmt.getServiceId(), stream.getPid(), stream.getStreamType(), pmt.getVersionNumber(), Channel->Number());
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2009-12-24 13:25:40 +01:00
|
|
|
if (ProcessCaDescriptors) {
|
|
|
|
for (SI::Loop::Iterator it; (d = (SI::CaDescriptor*)stream.streamDescriptors.getNext(it, SI::CaDescriptorTag)); ) {
|
|
|
|
CaDescriptors->AddCaDescriptor(d, esPid);
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
}
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2004-01-05 12:08:09 +01:00
|
|
|
if (Setup.UpdateChannels >= 2) {
|
2015-09-01 11:14:27 +02:00
|
|
|
ChannelsModified |= Channel->SetPids(Vpid, Ppid, Vtype, Apids, Atypes, ALangs, Dpids, Dtypes, DLangs, Spids, SLangs, Tpid);
|
|
|
|
ChannelsModified |= Channel->SetCaIds(CaDescriptors->CaIds());
|
|
|
|
ChannelsModified |= Channel->SetSubtitlingDescriptors(SubtitlingTypes, CompositionPageIds, AncillaryPageIds);
|
2004-01-05 12:08:09 +01:00
|
|
|
}
|
2015-09-01 11:14:27 +02:00
|
|
|
ChannelsModified |= Channel->SetCaDescriptors(CaDescriptorHandler.AddCaDescriptors(CaDescriptors));
|
2004-01-04 12:30:00 +01:00
|
|
|
}
|
2015-09-01 11:14:27 +02:00
|
|
|
StateKey.Remove(ChannelsModified);
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|
2014-02-18 13:12:39 +01:00
|
|
|
if (timer.TimedOut()) {
|
2020-06-19 12:19:15 +02:00
|
|
|
if (activePmt)
|
|
|
|
DBGLOG("PMT timeout Pid %d", activePmt->Pid());
|
2014-02-18 13:12:39 +01:00
|
|
|
SwitchToNextPmtPid();
|
|
|
|
}
|
2003-12-22 13:29:24 +01:00
|
|
|
}
|