mirror of
https://projects.vdr-developer.org/git/vdr-plugin-streamdev.git
synced 2023-10-10 19:16:51 +02:00
210 lines
5.3 KiB
C
210 lines
5.3 KiB
C
#include <vdr/ringbuffer.h>
|
|
|
|
#include "server/livestreamer.h"
|
|
#include "remux/ts2ps.h"
|
|
#include "remux/ts2es.h"
|
|
#include "common.h"
|
|
|
|
cStreamdevLiveReceiver::cStreamdevLiveReceiver(cStreamdevLiveStreamer *Streamer, int Ca,
|
|
int Priority, const int *Pids):
|
|
cReceiver(Ca, Priority, 0, Pids),
|
|
m_Streamer(Streamer)
|
|
{
|
|
}
|
|
|
|
cStreamdevLiveReceiver::~cStreamdevLiveReceiver()
|
|
{
|
|
Dprintf("Killing live receiver\n");
|
|
Detach();
|
|
}
|
|
|
|
void cStreamdevLiveReceiver::Receive(uchar *Data, int Length) {
|
|
int p = m_Streamer->Put(Data, Length);
|
|
if (p != Length)
|
|
m_Streamer->ReportOverflow(Length - p);
|
|
}
|
|
|
|
cStreamdevLiveStreamer::cStreamdevLiveStreamer(int Priority):
|
|
cStreamdevStreamer("Live streamer") {
|
|
m_Priority = Priority;
|
|
m_NumPids = 0;
|
|
m_Channel = NULL;
|
|
m_Device = NULL;
|
|
m_Receiver = NULL;
|
|
m_Remux = NULL;
|
|
m_Buffer = NULL;
|
|
m_Sequence = 0;
|
|
#if VDRVERSNUM >= 10300
|
|
m_Filter = NULL;
|
|
#endif
|
|
memset(m_Pids, 0, sizeof(m_Pids));
|
|
}
|
|
|
|
cStreamdevLiveStreamer::~cStreamdevLiveStreamer() {
|
|
Dprintf("Desctructing Live streamer\n");
|
|
delete m_Receiver;
|
|
delete m_Remux;
|
|
#if VDRVERSNUM >= 10300
|
|
delete m_Filter;
|
|
#endif
|
|
free(m_Buffer);
|
|
}
|
|
|
|
void cStreamdevLiveStreamer::Detach(void) {
|
|
m_Device->Detach(m_Receiver);
|
|
}
|
|
|
|
void cStreamdevLiveStreamer::Attach(void) {
|
|
m_Device->AttachReceiver(m_Receiver);
|
|
}
|
|
|
|
void cStreamdevLiveStreamer::Start(cTBSocket *Socket) {
|
|
Dprintf("LIVESTREAMER START\n");
|
|
cStreamdevStreamer::Start(Socket);
|
|
}
|
|
|
|
bool cStreamdevLiveStreamer::SetPid(int Pid, bool On) {
|
|
int idx;
|
|
|
|
if (On) {
|
|
for (idx = 0; idx < m_NumPids; ++idx) {
|
|
if (m_Pids[idx] == Pid)
|
|
return true; // No change needed
|
|
}
|
|
|
|
if (m_NumPids == MAXRECEIVEPIDS) {
|
|
esyslog("ERROR: Streamdev: No free slot to receive pid %d\n", Pid);
|
|
return false;
|
|
}
|
|
|
|
m_Pids[m_NumPids++] = Pid;
|
|
m_Pids[m_NumPids] = 0;
|
|
} else {
|
|
for (idx = 0; idx < m_NumPids; ++idx) {
|
|
if (m_Pids[idx] == Pid) {
|
|
--m_NumPids;
|
|
memmove(&m_Pids[idx], &m_Pids[idx + 1], sizeof(int) * (m_NumPids - idx));
|
|
}
|
|
}
|
|
}
|
|
|
|
DELETENULL(m_Receiver);
|
|
if (m_NumPids > 0) {
|
|
Dprintf("Creating Receiver to respect changed pids\n");
|
|
m_Receiver = new cStreamdevLiveReceiver(this, m_Channel->Ca(), m_Priority, m_Pids);
|
|
if (m_Device != NULL) {
|
|
Dprintf("Attaching new receiver\n");
|
|
m_Device->AttachReceiver(m_Receiver);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool cStreamdevLiveStreamer::SetChannel(const cChannel *Channel, int StreamType,
|
|
bool StreamPIDS) {
|
|
Dprintf("Initializing Remuxer for full channel transfer\n");
|
|
printf("ca pid: %d\n", Channel->Ca());
|
|
m_Channel = Channel;
|
|
switch (StreamType) {
|
|
case stES:
|
|
{
|
|
int pid = ISRADIO(Channel) ? Channel->Apid(0) : Channel->Vpid();
|
|
m_Remux = new cTS2ESRemux(pid);
|
|
return SetPid(pid, true);
|
|
}
|
|
|
|
case stPES:
|
|
m_Remux = new cTS2PSRemux(Channel->Vpid(), Channel->Apid(0), Channel->Apid(1),
|
|
Channel->Dpid(0), 0, false);
|
|
return SetPid(Channel->Vpid(), true)
|
|
&& SetPid(Channel->Apid(0), true)
|
|
&& SetPid(Channel->Apid(1), true)
|
|
&& SetPid(Channel->Dpid(0), true);
|
|
break;
|
|
|
|
case stPS:
|
|
m_Remux = new cTS2PSRemux(Channel->Vpid(), Channel->Apid(0), 0, 0, 0, true);
|
|
return SetPid(Channel->Vpid(), true)
|
|
&& SetPid(Channel->Apid(0), true);
|
|
break;
|
|
|
|
case stTS:
|
|
if (!StreamPIDS) {
|
|
return SetPid(Channel->Vpid(), true)
|
|
&& SetPid(Channel->Apid(0), true)
|
|
&& SetPid(Channel->Apid(1), true)
|
|
&& SetPid(Channel->Dpid(0), true);
|
|
}
|
|
Dprintf("pid streaming mode\n");
|
|
return true;
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cStreamdevLiveStreamer::SetFilter(u_short Pid, u_char Tid, u_char Mask,
|
|
bool On) {
|
|
#if VDRVERSNUM >= 10300
|
|
Dprintf("setting filter\n");
|
|
if (On) {
|
|
if (m_Filter == NULL) {
|
|
m_Filter = new cStreamdevLiveFilter(this);
|
|
Dprintf("attaching filter to device\n");
|
|
m_Device->AttachFilter(m_Filter);
|
|
}
|
|
m_Filter->Set(Pid, Tid, Mask);
|
|
} else if (m_Filter != NULL)
|
|
m_Filter->Del(Pid, Tid, Mask);
|
|
return true;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
uchar *cStreamdevLiveStreamer::Process(const uchar *Data, int &Count,
|
|
int &Result) {
|
|
uchar *remuxed = m_Remux != NULL ? m_Remux->Process(Data, Count, Result)
|
|
: cStreamdevStreamer::Process(Data, Count, Result);
|
|
if (remuxed) {
|
|
/*if (Socket()->Type() == SOCK_DGRAM) {
|
|
free(m_Buffer);
|
|
Result += 12;
|
|
m_Buffer = MALLOC(uchar, Result);
|
|
m_Buffer[0] = 0x01;
|
|
m_Buffer[1] = 0x02;
|
|
m_Buffer[2] = 0x03;
|
|
m_Buffer[3] = 0x04;
|
|
m_Buffer[4] = (Result & 0xff000000) >> 24;
|
|
m_Buffer[5] = (Result & 0xff0000) >> 16;
|
|
m_Buffer[6] = (Result & 0xff00) >> 8;
|
|
m_Buffer[7] = (Result & 0xff);
|
|
m_Buffer[8] = (m_Sequence & 0xff000000) >> 24;
|
|
m_Buffer[9] = (m_Sequence & 0xff0000) >> 16;
|
|
m_Buffer[10] = (m_Sequence & 0xff00) >> 8;
|
|
m_Buffer[11] = (m_Sequence & 0xff);
|
|
memcpy(m_Buffer + 12, Data, Result - 12);
|
|
if (m_Sequence++ == 0x7fffffff)
|
|
m_Sequence = 0;
|
|
return m_Buffer;
|
|
}*/
|
|
return remuxed;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
cTBString cStreamdevLiveStreamer::Report(void) {
|
|
cTBString result;
|
|
|
|
if (m_Device != NULL)
|
|
result += "+- Device is " + cTBString::Number(m_Device->CardIndex()) + "\n";
|
|
if (m_Receiver != NULL)
|
|
result += "+- Receiver is allocated\n";
|
|
|
|
result += "+- Pids are ";
|
|
for (int i = 0; i < MAXRECEIVEPIDS; ++i)
|
|
if (m_Pids[i] != 0)
|
|
result += cTBString::Number(m_Pids[i]) + ", ";
|
|
result += "\n";
|
|
return result;
|
|
}
|