mirror of
https://projects.vdr-developer.org/git/vdr-plugin-streamdev.git
synced 2023-10-10 19:16:51 +02:00
Implemented remuxing when replaying recordings
This commit is contained in:
parent
2d919997a8
commit
520adaf3da
@ -3,6 +3,7 @@
|
|||||||
#include "server/connection.h"
|
#include "server/connection.h"
|
||||||
#include "server/streamer.h"
|
#include "server/streamer.h"
|
||||||
#include <vdr/channels.h>
|
#include <vdr/channels.h>
|
||||||
|
#include <vdr/remux.h>
|
||||||
#include <vdr/tools.h>
|
#include <vdr/tools.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/wait.h>
|
#include <sys/wait.h>
|
||||||
@ -25,7 +26,7 @@ protected:
|
|||||||
virtual void Action(void);
|
virtual void Action(void);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connection, const cChannel *Channel, const int *Apids, const int *Dpids);
|
cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connection, const cChannel *Channel, const cPatPmtParser *PatPmt, const int *Apids, const int *Dpids);
|
||||||
virtual ~cTSExt();
|
virtual ~cTSExt();
|
||||||
|
|
||||||
void Put(const uchar *Data, int Count);
|
void Put(const uchar *Data, int Count);
|
||||||
@ -34,7 +35,7 @@ public:
|
|||||||
} // namespace Streamdev
|
} // namespace Streamdev
|
||||||
using namespace Streamdev;
|
using namespace Streamdev;
|
||||||
|
|
||||||
cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connection, const cChannel *Channel, const int *Apids, const int *Dpids):
|
cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connection, const cChannel *Channel, const cPatPmtParser *PatPmt, const int *Apids, const int *Dpids):
|
||||||
m_ResultBuffer(ResultBuffer),
|
m_ResultBuffer(ResultBuffer),
|
||||||
m_Active(false),
|
m_Active(false),
|
||||||
m_Process(-1),
|
m_Process(-1),
|
||||||
@ -73,6 +74,7 @@ cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connect
|
|||||||
#define ADDENV(x...) if (asprintf(&env[i++], x) < 0) i--
|
#define ADDENV(x...) if (asprintf(&env[i++], x) < 0) i--
|
||||||
|
|
||||||
// add channel ID, name and pids to environment
|
// add channel ID, name and pids to environment
|
||||||
|
if (Channel) {
|
||||||
ADDENV("REMUX_CHANNEL_ID=%s", *Channel->GetChannelID().ToString());
|
ADDENV("REMUX_CHANNEL_ID=%s", *Channel->GetChannelID().ToString());
|
||||||
ADDENV("REMUX_CHANNEL_NAME=%s", Channel->Name());
|
ADDENV("REMUX_CHANNEL_NAME=%s", Channel->Name());
|
||||||
ADDENV("REMUX_VTYPE=%d", Channel->Vtype());
|
ADDENV("REMUX_VTYPE=%d", Channel->Vtype());
|
||||||
@ -82,6 +84,14 @@ cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connect
|
|||||||
ADDENV("REMUX_PPID=%d", Channel->Ppid());
|
ADDENV("REMUX_PPID=%d", Channel->Ppid());
|
||||||
if (Channel->Tpid())
|
if (Channel->Tpid())
|
||||||
ADDENV("REMUX_TPID=%d", Channel->Tpid());
|
ADDENV("REMUX_TPID=%d", Channel->Tpid());
|
||||||
|
}
|
||||||
|
else if (PatPmt) {
|
||||||
|
ADDENV("REMUX_VTYPE=%d", PatPmt->Vtype());
|
||||||
|
if (PatPmt->Vpid())
|
||||||
|
ADDENV("REMUX_VPID=%d", PatPmt->Vpid());
|
||||||
|
if (PatPmt->Ppid() != PatPmt->Vpid())
|
||||||
|
ADDENV("REMUX_PPID=%d", PatPmt->Ppid());
|
||||||
|
}
|
||||||
|
|
||||||
std::string buffer;
|
std::string buffer;
|
||||||
if (Apids && *Apids) {
|
if (Apids && *Apids) {
|
||||||
@ -92,10 +102,17 @@ cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connect
|
|||||||
buffer.clear();
|
buffer.clear();
|
||||||
for (const int *pid = Apids; *pid; pid++) {
|
for (const int *pid = Apids; *pid; pid++) {
|
||||||
int j;
|
int j;
|
||||||
|
if (Channel) {
|
||||||
for (j = 0; Channel->Apid(j) && Channel->Apid(j) != *pid; j++)
|
for (j = 0; Channel->Apid(j) && Channel->Apid(j) != *pid; j++)
|
||||||
;
|
;
|
||||||
(buffer += Channel->Alang(j)) += (*(pid + 1) ? " " : "");
|
(buffer += Channel->Alang(j)) += (*(pid + 1) ? " " : "");
|
||||||
}
|
}
|
||||||
|
else if (PatPmt) {
|
||||||
|
for (j = 0; PatPmt->Apid(j) && PatPmt->Apid(j) != *pid; j++)
|
||||||
|
;
|
||||||
|
(buffer += PatPmt->Alang(j)) += (*(pid + 1) ? " " : "");
|
||||||
|
}
|
||||||
|
}
|
||||||
ADDENV("REMUX_ALANG=%s", buffer.c_str());
|
ADDENV("REMUX_ALANG=%s", buffer.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -108,14 +125,21 @@ cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connect
|
|||||||
buffer.clear();
|
buffer.clear();
|
||||||
for (const int *pid = Dpids; *pid; pid++) {
|
for (const int *pid = Dpids; *pid; pid++) {
|
||||||
int j;
|
int j;
|
||||||
|
if (Channel) {
|
||||||
for (j = 0; Channel->Dpid(j) && Channel->Dpid(j) != *pid; j++)
|
for (j = 0; Channel->Dpid(j) && Channel->Dpid(j) != *pid; j++)
|
||||||
;
|
;
|
||||||
(buffer += Channel->Dlang(j)) += (*(pid + 1) ? " " : "");
|
(buffer += Channel->Dlang(j)) += (*(pid + 1) ? " " : "");
|
||||||
}
|
}
|
||||||
|
else if (PatPmt) {
|
||||||
|
for (j = 0; PatPmt->Dpid(j) && PatPmt->Dpid(j) != *pid; j++)
|
||||||
|
;
|
||||||
|
(buffer += PatPmt->Dlang(j)) += (*(pid + 1) ? " " : "");
|
||||||
|
}
|
||||||
|
}
|
||||||
ADDENV("REMUX_DLANG=%s", buffer.c_str());
|
ADDENV("REMUX_DLANG=%s", buffer.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Channel->Spid(0)) {
|
if (Channel && Channel->Spid(0)) {
|
||||||
buffer.clear();
|
buffer.clear();
|
||||||
for (const int *pid = Channel->Spids(); *pid; pid++)
|
for (const int *pid = Channel->Spids(); *pid; pid++)
|
||||||
(buffer += (const char *) itoa(*pid)) += (*(pid + 1) ? " " : "");
|
(buffer += (const char *) itoa(*pid)) += (*(pid + 1) ? " " : "");
|
||||||
@ -126,6 +150,17 @@ cTSExt::cTSExt(cRingBufferLinear *ResultBuffer, const cServerConnection *Connect
|
|||||||
(buffer += Channel->Slang(j)) += (Channel->Spid(j + 1) ? " " : "");
|
(buffer += Channel->Slang(j)) += (Channel->Spid(j + 1) ? " " : "");
|
||||||
ADDENV("REMUX_SLANG=%s", buffer.c_str());
|
ADDENV("REMUX_SLANG=%s", buffer.c_str());
|
||||||
}
|
}
|
||||||
|
else if (PatPmt && PatPmt->Spid(0)) {
|
||||||
|
buffer.clear();
|
||||||
|
for (const int *pid = PatPmt->Spids(); *pid; pid++)
|
||||||
|
(buffer += (const char *) itoa(*pid)) += (*(pid + 1) ? " " : "");
|
||||||
|
ADDENV("REMUX_SPID=%s", buffer.c_str());
|
||||||
|
|
||||||
|
buffer.clear();
|
||||||
|
for (int j = 0; PatPmt->Spid(j); j++)
|
||||||
|
(buffer += PatPmt->Slang(j)) += (PatPmt->Spid(j + 1) ? " " : "");
|
||||||
|
ADDENV("REMUX_SLANG=%s", buffer.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
if (Connection) {
|
if (Connection) {
|
||||||
// add vars for a CGI like interface
|
// add vars for a CGI like interface
|
||||||
@ -296,7 +331,13 @@ void cTSExt::Put(const uchar *Data, int Count)
|
|||||||
|
|
||||||
cExternRemux::cExternRemux(const cServerConnection *Connection, const cChannel *Channel, const int *Apids, const int *Dpids):
|
cExternRemux::cExternRemux(const cServerConnection *Connection, const cChannel *Channel, const int *Apids, const int *Dpids):
|
||||||
m_ResultBuffer(new cRingBufferLinear(WRITERBUFSIZE)),
|
m_ResultBuffer(new cRingBufferLinear(WRITERBUFSIZE)),
|
||||||
m_Remux(new cTSExt(m_ResultBuffer, Connection, Channel, Apids, Dpids))
|
m_Remux(new cTSExt(m_ResultBuffer, Connection, Channel, NULL, Apids, Dpids))
|
||||||
|
{
|
||||||
|
m_ResultBuffer->SetTimeouts(500, 100);
|
||||||
|
}
|
||||||
|
cExternRemux::cExternRemux(const cServerConnection *Connection, const cPatPmtParser *PatPmt, const int *Apids, const int *Dpids):
|
||||||
|
m_ResultBuffer(new cRingBufferLinear(WRITERBUFSIZE)),
|
||||||
|
m_Remux(new cTSExt(m_ResultBuffer, Connection, NULL, PatPmt, Apids, Dpids))
|
||||||
{
|
{
|
||||||
m_ResultBuffer->SetTimeouts(500, 100);
|
m_ResultBuffer->SetTimeouts(500, 100);
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class cChannel;
|
class cChannel;
|
||||||
|
class cPatPmtParser;
|
||||||
class cServerConnection;
|
class cServerConnection;
|
||||||
|
|
||||||
namespace Streamdev {
|
namespace Streamdev {
|
||||||
@ -19,6 +20,7 @@ private:
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
cExternRemux(const cServerConnection *Connection, const cChannel *Channel, const int *APids, const int *Dpids);
|
cExternRemux(const cServerConnection *Connection, const cChannel *Channel, const int *APids, const int *Dpids);
|
||||||
|
cExternRemux(const cServerConnection *Connection, const cPatPmtParser *PatPmt, const int *APids, const int *Dpids);
|
||||||
virtual ~cExternRemux();
|
virtual ~cExternRemux();
|
||||||
|
|
||||||
int Put(const uchar *Data, int Count);
|
int Put(const uchar *Data, int Count);
|
||||||
|
@ -35,6 +35,7 @@ cConnectionHTTP::cConnectionHTTP(void):
|
|||||||
|
|
||||||
cConnectionHTTP::~cConnectionHTTP()
|
cConnectionHTTP::~cConnectionHTTP()
|
||||||
{
|
{
|
||||||
|
SetStreamer(NULL);
|
||||||
delete m_RecPlayer;
|
delete m_RecPlayer;
|
||||||
delete m_MenuList;
|
delete m_MenuList;
|
||||||
}
|
}
|
||||||
@ -164,6 +165,14 @@ bool cConnectionHTTP::ProcessRequest(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tStrStrMap::const_iterator it;
|
||||||
|
it = m_Params.find("apid");
|
||||||
|
if (it != m_Params.end())
|
||||||
|
m_Apid[0] = atoi(it->second.c_str());
|
||||||
|
it = m_Params.find("dpid");
|
||||||
|
if (it != m_Params.end())
|
||||||
|
m_Dpid[0] = atoi(it->second.c_str());
|
||||||
|
|
||||||
tStrStrMap::const_iterator it_method = Headers().find(REQUEST_METHOD);
|
tStrStrMap::const_iterator it_method = Headers().find(REQUEST_METHOD);
|
||||||
tStrStrMap::const_iterator it_pathinfo = Headers().find(PATH_INFO);
|
tStrStrMap::const_iterator it_pathinfo = Headers().find(PATH_INFO);
|
||||||
if (it_method == Headers().end() || it_pathinfo == Headers().end()) {
|
if (it_method == Headers().end() || it_pathinfo == Headers().end()) {
|
||||||
@ -196,30 +205,46 @@ bool cConnectionHTTP::ProcessRequest(void)
|
|||||||
}
|
}
|
||||||
else if (m_RecPlayer != NULL) {
|
else if (m_RecPlayer != NULL) {
|
||||||
Dprintf("GET recording\n");
|
Dprintf("GET recording\n");
|
||||||
|
bool isPes = m_RecPlayer->getCurrentRecording()->IsPesRecording();
|
||||||
|
// no remuxing for old PES recordings
|
||||||
|
if (isPes && m_StreamType != stPES)
|
||||||
|
return HttpResponse(503, true);
|
||||||
|
|
||||||
int64_t from, to;
|
int64_t from, to;
|
||||||
bool hasRange = ParseRange(from, to);
|
bool hasRange = ParseRange(from, to);
|
||||||
|
|
||||||
cStreamdevRecStreamer* recStreamer;
|
cStreamdevRecStreamer* recStreamer;
|
||||||
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
||||||
recStreamer = new cStreamdevRecStreamer(m_RecPlayer, this);
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, (int64_t) 0L, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
||||||
from += m_ReplayPos;
|
from += m_ReplayPos;
|
||||||
if (to >= 0)
|
if (to >= 0)
|
||||||
to += m_ReplayPos;
|
to += m_ReplayPos;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
recStreamer = new cStreamdevRecStreamer(m_RecPlayer, this, m_ReplayPos);
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
||||||
SetStreamer(recStreamer);
|
SetStreamer(recStreamer);
|
||||||
|
|
||||||
|
if (m_StreamType == stEXT)
|
||||||
|
return Respond("HTTP/1.0 200 OK");
|
||||||
|
else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0]))
|
||||||
|
return HttpResponse(200, false, "audio/mpeg");
|
||||||
|
|
||||||
|
const char* contentType = (isPes || m_RecPlayer->getPatPmtData()->Vpid()) ? "video/mpeg" : "audio/mpeg";
|
||||||
|
// range not supported when remuxing
|
||||||
|
if (m_StreamType != stTS && !isPes)
|
||||||
|
return HttpResponse(200, false, contentType);
|
||||||
|
|
||||||
uint64_t total = recStreamer->GetLength();
|
uint64_t total = recStreamer->GetLength();
|
||||||
if (hasRange) {
|
if (hasRange) {
|
||||||
int64_t length = recStreamer->SetRange(from, to);
|
int64_t length = recStreamer->SetRange(from, to);
|
||||||
Dprintf("range response: %lld-%lld/%lld, len %lld\n", (long long)from, (long long)to, (long long)total, (long long)length);
|
Dprintf("range response: %lld-%lld/%lld, len %lld\n", (long long)from, (long long)to, (long long)total, (long long)length);
|
||||||
if (length < 0L)
|
if (length < 0L)
|
||||||
return HttpResponse(416, true, "video/mpeg", "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
return HttpResponse(416, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
||||||
else
|
else
|
||||||
return HttpResponse(206, false, "video/mpeg", "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
return HttpResponse(206, false, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
return HttpResponse(200, false, "video/mpeg", "Accept-Ranges: bytes");
|
return HttpResponse(200, false, contentType, "Accept-Ranges: bytes");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return HttpResponse(404, true);
|
return HttpResponse(404, true);
|
||||||
@ -247,29 +272,45 @@ bool cConnectionHTTP::ProcessRequest(void)
|
|||||||
}
|
}
|
||||||
else if (m_RecPlayer != NULL) {
|
else if (m_RecPlayer != NULL) {
|
||||||
Dprintf("HEAD recording\n");
|
Dprintf("HEAD recording\n");
|
||||||
|
bool isPes = m_RecPlayer->getCurrentRecording()->IsPesRecording();
|
||||||
|
// no remuxing for old PES recordings
|
||||||
|
if (isPes && m_StreamType != stPES)
|
||||||
|
return HttpResponse(503, true);
|
||||||
|
|
||||||
int64_t from, to;
|
int64_t from, to;
|
||||||
bool hasRange = ParseRange(from, to);
|
bool hasRange = ParseRange(from, to);
|
||||||
|
|
||||||
cStreamdevRecStreamer* recStreamer;
|
cStreamdevRecStreamer* recStreamer;
|
||||||
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
||||||
recStreamer = new cStreamdevRecStreamer(m_RecPlayer, this, m_ReplayPos);
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
||||||
from += m_ReplayPos;
|
from += m_ReplayPos;
|
||||||
if (to >= 0)
|
if (to >= 0)
|
||||||
to += m_ReplayPos;
|
to += m_ReplayPos;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
recStreamer = new cStreamdevRecStreamer(m_RecPlayer, this, m_ReplayPos);
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
||||||
SetStreamer(recStreamer);
|
SetStreamer(recStreamer);
|
||||||
|
|
||||||
|
if (m_StreamType == stEXT)
|
||||||
|
return Respond("HTTP/1.0 200 OK");
|
||||||
|
else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0]))
|
||||||
|
return HttpResponse(200, true, "audio/mpeg");
|
||||||
|
|
||||||
|
const char* contentType = (isPes || m_RecPlayer->getPatPmtData()->Vpid()) ? "video/mpeg" : "audio/mpeg";
|
||||||
|
// range not supported when remuxing
|
||||||
|
if (m_StreamType != stTS && !isPes)
|
||||||
|
return HttpResponse(200, false, contentType);
|
||||||
|
|
||||||
uint64_t total = recStreamer->GetLength();
|
uint64_t total = recStreamer->GetLength();
|
||||||
if (hasRange) {
|
if (hasRange) {
|
||||||
int64_t length = recStreamer->SetRange(from, to);
|
int64_t length = recStreamer->SetRange(from, to);
|
||||||
if (length < 0L)
|
if (length < 0L)
|
||||||
return HttpResponse(416, true, "video/mpeg", "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
return HttpResponse(416, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
||||||
else
|
else
|
||||||
return HttpResponse(206, true, "video/mpeg", "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
return HttpResponse(206, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
return HttpResponse(200, true, "video/mpeg", "Accept-Ranges: bytes");
|
return HttpResponse(200, true, contentType, "Accept-Ranges: bytes");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return HttpResponse(404, true);
|
return HttpResponse(404, true);
|
||||||
|
@ -42,6 +42,12 @@ RecPlayer::RecPlayer(const char* FileName)
|
|||||||
if (!indexFile) esyslog("ERROR: Streamdev: Failed to create indexfile!");
|
if (!indexFile) esyslog("ERROR: Streamdev: Failed to create indexfile!");
|
||||||
|
|
||||||
scan();
|
scan();
|
||||||
|
|
||||||
|
parser = new cPatPmtParser();
|
||||||
|
unsigned char buffer[2 * TS_SIZE];
|
||||||
|
unsigned long l = getBlock(buffer, 0UL, sizeof(buffer));
|
||||||
|
if (!l || !parser->ParsePatPmt(buffer, (int) l))
|
||||||
|
esyslog("ERROR: Streamdev: Failed to parse PAT/PMT");
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecPlayer::scan()
|
void RecPlayer::scan()
|
||||||
@ -87,6 +93,7 @@ RecPlayer::~RecPlayer()
|
|||||||
if (file) fclose(file);
|
if (file) fclose(file);
|
||||||
delete indexFile;
|
delete indexFile;
|
||||||
delete recording;
|
delete recording;
|
||||||
|
delete parser;
|
||||||
}
|
}
|
||||||
|
|
||||||
int RecPlayer::openFile(int index)
|
int RecPlayer::openFile(int index)
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <vdr/recording.h>
|
#include <vdr/recording.h>
|
||||||
|
#include <vdr/remux.h>
|
||||||
|
|
||||||
#include "server/streamer.h"
|
#include "server/streamer.h"
|
||||||
|
|
||||||
@ -44,6 +45,7 @@ class RecPlayer
|
|||||||
int openFile(int index);
|
int openFile(int index);
|
||||||
uint64_t getLastPosition();
|
uint64_t getLastPosition();
|
||||||
cRecording* getCurrentRecording();
|
cRecording* getCurrentRecording();
|
||||||
|
const cPatPmtParser* getPatPmtData() { return parser; }
|
||||||
void scan();
|
void scan();
|
||||||
uint64_t positionFromResume(int ResumeID);
|
uint64_t positionFromResume(int ResumeID);
|
||||||
uint64_t positionFromMark(int MarkIndex);
|
uint64_t positionFromMark(int MarkIndex);
|
||||||
@ -56,6 +58,7 @@ class RecPlayer
|
|||||||
private:
|
private:
|
||||||
cRecording* recording;
|
cRecording* recording;
|
||||||
cIndexFile* indexFile;
|
cIndexFile* indexFile;
|
||||||
|
cPatPmtParser* parser;
|
||||||
FILE* file;
|
FILE* file;
|
||||||
int fileOpen;
|
int fileOpen;
|
||||||
Segment* segments[1000];
|
Segment* segments[1000];
|
||||||
|
@ -12,7 +12,7 @@ using namespace Streamdev;
|
|||||||
|
|
||||||
// --- cStreamdevRecStreamer -------------------------------------------------
|
// --- cStreamdevRecStreamer -------------------------------------------------
|
||||||
|
|
||||||
cStreamdevRecStreamer::cStreamdevRecStreamer(RecPlayer *RecPlayer, const cServerConnection *Connection, int64_t StartOffset):
|
cStreamdevRecStreamer::cStreamdevRecStreamer(const cServerConnection *Connection, RecPlayer *RecPlayer, eStreamType StreamType, int64_t StartOffset, const int *Apid, const int *Dpid):
|
||||||
cStreamdevStreamer("streamdev-recstreaming", Connection),
|
cStreamdevStreamer("streamdev-recstreaming", Connection),
|
||||||
m_RecPlayer(RecPlayer),
|
m_RecPlayer(RecPlayer),
|
||||||
m_StartOffset(StartOffset),
|
m_StartOffset(StartOffset),
|
||||||
@ -20,6 +20,36 @@ cStreamdevRecStreamer::cStreamdevRecStreamer(RecPlayer *RecPlayer, const cServer
|
|||||||
{
|
{
|
||||||
Dprintf("New rec streamer\n");
|
Dprintf("New rec streamer\n");
|
||||||
m_To = (int64_t) m_RecPlayer->getLengthBytes() - StartOffset - 1;
|
m_To = (int64_t) m_RecPlayer->getLengthBytes() - StartOffset - 1;
|
||||||
|
|
||||||
|
const cPatPmtParser *parser = RecPlayer->getPatPmtData();
|
||||||
|
const int *Apids = Apid ? Apid : parser->Apids();
|
||||||
|
const int *Dpids = Dpid ? Dpid : parser->Dpids();
|
||||||
|
switch (StreamType) {
|
||||||
|
case stES:
|
||||||
|
{
|
||||||
|
int pid = parser->Vpid();
|
||||||
|
if (Apid && Apid[0])
|
||||||
|
pid = Apid[0];
|
||||||
|
else if (Dpid && Dpid[0])
|
||||||
|
pid = Dpid[0];
|
||||||
|
SetRemux(new cTS2ESRemux(pid));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case stPES:
|
||||||
|
if (!m_RecPlayer->getCurrentRecording()->IsPesRecording())
|
||||||
|
SetRemux(new cTS2PESRemux(parser->Vpid(), Apids, Dpids, parser->Spids()));
|
||||||
|
break;
|
||||||
|
#ifdef STREAMDEV_PS
|
||||||
|
case stPS:
|
||||||
|
SetRemux(new cTS2PSRemux(parser->Vpid(), Apids, Dpids, parser->Spids()));
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
case stEXT:
|
||||||
|
SetRemux(new cExternRemux(Connection, parser, Apids, Dpids));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cStreamdevRecStreamer::~cStreamdevRecStreamer()
|
cStreamdevRecStreamer::~cStreamdevRecStreamer()
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
#ifndef VDR_STREAMDEV_RECSTREAMER_H
|
#ifndef VDR_STREAMDEV_RECSTREAMER_H
|
||||||
#define VDR_STREAMDEV_RECSTREAMER_H
|
#define VDR_STREAMDEV_RECSTREAMER_H
|
||||||
|
|
||||||
|
#include "common.h"
|
||||||
#include "server/streamer.h"
|
#include "server/streamer.h"
|
||||||
#include "server/recplayer.h"
|
#include "server/recplayer.h"
|
||||||
|
|
||||||
@ -26,7 +27,7 @@ public:
|
|||||||
uint64_t GetLength() { return m_RecPlayer->getLengthBytes() - m_StartOffset; }
|
uint64_t GetLength() { return m_RecPlayer->getLengthBytes() - m_StartOffset; }
|
||||||
int64_t SetRange(int64_t &From, int64_t &To);
|
int64_t SetRange(int64_t &From, int64_t &To);
|
||||||
virtual cString ToText() const;
|
virtual cString ToText() const;
|
||||||
cStreamdevRecStreamer(RecPlayer *RecPlayer, const cServerConnection *Connection, int64_t StartOffset = 0L);
|
cStreamdevRecStreamer(const cServerConnection *Connection, RecPlayer *RecPlayer, eStreamType StreamType, int64_t StartOffset = 0L, const int *Apids = NULL, const int *Dpids = NULL);
|
||||||
virtual ~cStreamdevRecStreamer();
|
virtual ~cStreamdevRecStreamer();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user