mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- The name of the function cDevice::GetVideoSize() wasn't very well chosen for its purpose of defining the optimum size of the OSD for the current output device. Therefore a new function named cDevice::GetOsdSize() has been introduced (suggested by Rolf Ahrenberg). Plugin authors should implement this function in classes derived from cDevice, if they are able to replay video. cDevice::GetVideoSize() still exists and should return the actual size of the video material that is currently replayed. Note that because of the many possible aspect ratios for video material, the type of the Aspect parameter of GetVideoSize() has been changed to 'double', and the Aspect parameter in both functions is named differently, because it returns different values (suggested by Reinhard Nissl). Thanks to Oliver Endriss for his input on calculating the Aspect factor in GetOsdSize(). - Fixed the way the OSD size is determined on full featured DVB cards (thanks to Oliver Endriss). - Increased MAXOSDHEIGHT to 1200 (suggested by Nicolas Huillard). - Removed limitation to PAL resolution from SPU handling. - Checking fd_video in cDvbDevice::GetVideoSize() to avoid error messages on systems with no real primary replay device (reported by Martin Neuditschko). - Added a note to cTsToPes::GetPes() about having to call it repeatedly, once it has returned a non-NULL value. - Added MPEG 1 handling to remux.c (thanks to Ales Jurik). - Fixed use of time_t in cEIT::cEIT() (thanks to Tobias Bratfisch). - Added missing update of lastOsdSizeUpdate. - EIT events are now only processed if a plausible system time is available, to avoid wrong handling of PDC descriptors (thanks to Tobias Bratfisch). - Removed unused 'synced' member from cTsToPes (reported by Christoph Haubrich). - Added a note to cTsToPes about all TS packets having to belong to the same PID, and that for video data GetPes() may only be called if the next TS packet that will be given to PutTs() has the "payload start" flag set (suggested by Christoph Haubrich). - Added a note about the meaning of PERCENTAGEDELTA in cRingBuffer::UpdatePercentage() (thanks to Rolf Ahrenberg). - The new setup option "Recording/Pause key handling" can be used to define what happens if the Pause key on the remote control is pressed during live tv (thanks to Timo Eskola). - Added a note about cFont::GetFont() not being thread-safe. - Fixed generating PAT/PMT version numbers in case the PIDs change during recording (reported by Reinhard Nissl). - Updated the Ukrainian OSD texts (thanks to Yarema Aka Knedlyk). - Fixed a memory leak when reaching the end of a recording during replay (reported by Reinhard Nissl). - Fixed calling close(-1) in cUnbufferedFile::Close() (reported by Reinhard Nissl). - Added a workaround for the broken linux-dvb driver header files (based on a patch from Tobias Grimm). - Fixed handling the length of DiSEqC command sequences (reported by Reinhard Nissl). - Fixed cOsdMenu::Display() in case the menu size has changed (thanks to Reinhard Nissl). - Added some missing 'const' keywords to avoid compilation errors with gcc 4.4 (thanks to Ville Skyttä and Ludwig Nussel). - Modified cSVDRP::CmdGRAB() to avoid writing into const data (reported by Ludwig Nussel). - Fixed calculating menu colum widths in case the font has a size other than the default size (reported by Reinhard Nissl). - Added a plausibility check for the OSD percentage parameters to avoid problems in case the values are stored in the setup.conf file in a wrong way. - Fixed variable types in cIndexFile (reported by Udo Richter).
448 lines
9.8 KiB
C
448 lines
9.8 KiB
C
/*
|
|
* ringbuffer.c: A ring buffer
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* Parts of this file were inspired by the 'ringbuffy.c' from the
|
|
* LinuxDVB driver (see linuxtv.org).
|
|
*
|
|
* $Id: ringbuffer.c 2.2 2009/05/17 10:05:17 kls Exp $
|
|
*/
|
|
|
|
#include "ringbuffer.h"
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include "tools.h"
|
|
|
|
// --- cRingBuffer -----------------------------------------------------------
|
|
|
|
#define OVERFLOWREPORTDELTA 5 // seconds between reports
|
|
#define PERCENTAGEDELTA 10
|
|
#define PERCENTAGETHRESHOLD 70
|
|
|
|
cRingBuffer::cRingBuffer(int Size, bool Statistics)
|
|
{
|
|
size = Size;
|
|
statistics = Statistics;
|
|
getThreadTid = 0;
|
|
maxFill = 0;
|
|
lastPercent = 0;
|
|
putTimeout = getTimeout = 0;
|
|
lastOverflowReport = 0;
|
|
overflowCount = overflowBytes = 0;
|
|
}
|
|
|
|
cRingBuffer::~cRingBuffer()
|
|
{
|
|
if (statistics)
|
|
dsyslog("buffer stats: %d (%d%%) used", maxFill, maxFill * 100 / (size - 1));
|
|
}
|
|
|
|
void cRingBuffer::UpdatePercentage(int Fill)
|
|
{
|
|
if (Fill > maxFill)
|
|
maxFill = Fill;
|
|
int percent = Fill * 100 / (Size() - 1) / PERCENTAGEDELTA * PERCENTAGEDELTA; // clamp down to nearest quantum
|
|
if (percent != lastPercent) {
|
|
if (percent >= PERCENTAGETHRESHOLD && percent > lastPercent || percent < PERCENTAGETHRESHOLD && lastPercent >= PERCENTAGETHRESHOLD) {
|
|
dsyslog("buffer usage: %d%% (tid=%d)", percent, getThreadTid);
|
|
lastPercent = percent;
|
|
}
|
|
}
|
|
}
|
|
|
|
void cRingBuffer::WaitForPut(void)
|
|
{
|
|
if (putTimeout)
|
|
readyForPut.Wait(putTimeout);
|
|
}
|
|
|
|
void cRingBuffer::WaitForGet(void)
|
|
{
|
|
if (getTimeout)
|
|
readyForGet.Wait(getTimeout);
|
|
}
|
|
|
|
void cRingBuffer::EnablePut(void)
|
|
{
|
|
if (putTimeout && Free() > Size() / 3)
|
|
readyForPut.Signal();
|
|
}
|
|
|
|
void cRingBuffer::EnableGet(void)
|
|
{
|
|
if (getTimeout && Available() > Size() / 3)
|
|
readyForGet.Signal();
|
|
}
|
|
|
|
void cRingBuffer::SetTimeouts(int PutTimeout, int GetTimeout)
|
|
{
|
|
putTimeout = PutTimeout;
|
|
getTimeout = GetTimeout;
|
|
}
|
|
|
|
void cRingBuffer::ReportOverflow(int Bytes)
|
|
{
|
|
overflowCount++;
|
|
overflowBytes += Bytes;
|
|
if (time(NULL) - lastOverflowReport > OVERFLOWREPORTDELTA) {
|
|
esyslog("ERROR: %d ring buffer overflow%s (%d bytes dropped)", overflowCount, overflowCount > 1 ? "s" : "", overflowBytes);
|
|
overflowCount = overflowBytes = 0;
|
|
lastOverflowReport = time(NULL);
|
|
}
|
|
}
|
|
|
|
// --- cRingBufferLinear -----------------------------------------------------
|
|
|
|
#ifdef DEBUGRINGBUFFERS
|
|
#define MAXRBLS 30
|
|
#define DEBUGRBLWIDTH 45
|
|
|
|
cRingBufferLinear *cRingBufferLinear::RBLS[MAXRBLS] = { NULL };
|
|
|
|
void cRingBufferLinear::AddDebugRBL(cRingBufferLinear *RBL)
|
|
{
|
|
for (int i = 0; i < MAXRBLS; i++) {
|
|
if (!RBLS[i]) {
|
|
RBLS[i] = RBL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void cRingBufferLinear::DelDebugRBL(cRingBufferLinear *RBL)
|
|
{
|
|
for (int i = 0; i < MAXRBLS; i++) {
|
|
if (RBLS[i] == RBL) {
|
|
RBLS[i] = NULL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void cRingBufferLinear::PrintDebugRBL(void)
|
|
{
|
|
bool printed = false;
|
|
for (int i = 0; i < MAXRBLS; i++) {
|
|
cRingBufferLinear *p = RBLS[i];
|
|
if (p) {
|
|
printed = true;
|
|
int lh = p->lastHead;
|
|
int lt = p->lastTail;
|
|
int h = lh * DEBUGRBLWIDTH / p->Size();
|
|
int t = lt * DEBUGRBLWIDTH / p->Size();
|
|
char buf[DEBUGRBLWIDTH + 10];
|
|
memset(buf, '-', DEBUGRBLWIDTH);
|
|
if (lt <= lh)
|
|
memset(buf + t, '*', max(h - t, 1));
|
|
else {
|
|
memset(buf, '*', h);
|
|
memset(buf + t, '*', DEBUGRBLWIDTH - t);
|
|
}
|
|
buf[t] = '<';
|
|
buf[h] = '>';
|
|
buf[DEBUGRBLWIDTH] = 0;
|
|
printf("%2d %s %8d %8d %s\n", i, buf, p->lastPut, p->lastGet, p->description);
|
|
}
|
|
}
|
|
if (printed)
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
|
|
cRingBufferLinear::cRingBufferLinear(int Size, int Margin, bool Statistics, const char *Description)
|
|
:cRingBuffer(Size, Statistics)
|
|
{
|
|
description = Description ? strdup(Description) : NULL;
|
|
tail = head = margin = Margin;
|
|
gotten = 0;
|
|
buffer = NULL;
|
|
if (Size > 1) { // 'Size - 1' must not be 0!
|
|
if (Margin <= Size / 2) {
|
|
buffer = MALLOC(uchar, Size);
|
|
if (!buffer)
|
|
esyslog("ERROR: can't allocate ring buffer (size=%d)", Size);
|
|
Clear();
|
|
}
|
|
else
|
|
esyslog("ERROR: invalid margin for ring buffer (%d > %d)", Margin, Size / 2);
|
|
}
|
|
else
|
|
esyslog("ERROR: invalid size for ring buffer (%d)", Size);
|
|
#ifdef DEBUGRINGBUFFERS
|
|
lastHead = head;
|
|
lastTail = tail;
|
|
lastPut = lastGet = -1;
|
|
AddDebugRBL(this);
|
|
#endif
|
|
}
|
|
|
|
cRingBufferLinear::~cRingBufferLinear()
|
|
{
|
|
#ifdef DEBUGRINGBUFFERS
|
|
DelDebugRBL(this);
|
|
#endif
|
|
free(buffer);
|
|
free(description);
|
|
}
|
|
|
|
int cRingBufferLinear::DataReady(const uchar *Data, int Count)
|
|
{
|
|
return Count >= margin ? Count : 0;
|
|
}
|
|
|
|
int cRingBufferLinear::Available(void)
|
|
{
|
|
int diff = head - tail;
|
|
return (diff >= 0) ? diff : Size() + diff - margin;
|
|
}
|
|
|
|
void cRingBufferLinear::Clear(void)
|
|
{
|
|
tail = head;
|
|
#ifdef DEBUGRINGBUFFERS
|
|
lastHead = head;
|
|
lastTail = tail;
|
|
lastPut = lastGet = -1;
|
|
#endif
|
|
maxFill = 0;
|
|
EnablePut();
|
|
}
|
|
|
|
int cRingBufferLinear::Read(int FileHandle, int Max)
|
|
{
|
|
int Tail = tail;
|
|
int diff = Tail - head;
|
|
int free = (diff > 0) ? diff - 1 : Size() - head;
|
|
if (Tail <= margin)
|
|
free--;
|
|
int Count = 0;
|
|
if (free > 0) {
|
|
if (0 < Max && Max < free)
|
|
free = Max;
|
|
Count = safe_read(FileHandle, buffer + head, free);
|
|
if (Count > 0) {
|
|
int Head = head + Count;
|
|
if (Head >= Size())
|
|
Head = margin;
|
|
head = Head;
|
|
if (statistics) {
|
|
int fill = head - Tail;
|
|
if (fill < 0)
|
|
fill = Size() + fill;
|
|
else if (fill >= Size())
|
|
fill = Size() - 1;
|
|
UpdatePercentage(fill);
|
|
}
|
|
}
|
|
}
|
|
#ifdef DEBUGRINGBUFFERS
|
|
lastHead = head;
|
|
lastPut = Count;
|
|
#endif
|
|
EnableGet();
|
|
if (free == 0)
|
|
WaitForPut();
|
|
return Count;
|
|
}
|
|
|
|
int cRingBufferLinear::Put(const uchar *Data, int Count)
|
|
{
|
|
if (Count > 0) {
|
|
int Tail = tail;
|
|
int rest = Size() - head;
|
|
int diff = Tail - head;
|
|
int free = ((Tail < margin) ? rest : (diff > 0) ? diff : Size() + diff - margin) - 1;
|
|
if (statistics) {
|
|
int fill = Size() - free - 1 + Count;
|
|
if (fill >= Size())
|
|
fill = Size() - 1;
|
|
UpdatePercentage(fill);
|
|
}
|
|
if (free > 0) {
|
|
if (free < Count)
|
|
Count = free;
|
|
if (Count >= rest) {
|
|
memcpy(buffer + head, Data, rest);
|
|
if (Count - rest)
|
|
memcpy(buffer + margin, Data + rest, Count - rest);
|
|
head = margin + Count - rest;
|
|
}
|
|
else {
|
|
memcpy(buffer + head, Data, Count);
|
|
head += Count;
|
|
}
|
|
}
|
|
else
|
|
Count = 0;
|
|
#ifdef DEBUGRINGBUFFERS
|
|
lastHead = head;
|
|
lastPut = Count;
|
|
#endif
|
|
EnableGet();
|
|
if (Count == 0)
|
|
WaitForPut();
|
|
}
|
|
return Count;
|
|
}
|
|
|
|
uchar *cRingBufferLinear::Get(int &Count)
|
|
{
|
|
int Head = head;
|
|
if (getThreadTid <= 0)
|
|
getThreadTid = cThread::ThreadId();
|
|
int rest = Size() - tail;
|
|
if (rest < margin && Head < tail) {
|
|
int t = margin - rest;
|
|
memcpy(buffer + t, buffer + tail, rest);
|
|
tail = t;
|
|
rest = Head - tail;
|
|
}
|
|
int diff = Head - tail;
|
|
int cont = (diff >= 0) ? diff : Size() + diff - margin;
|
|
if (cont > rest)
|
|
cont = rest;
|
|
uchar *p = buffer + tail;
|
|
if ((cont = DataReady(p, cont)) > 0) {
|
|
Count = gotten = cont;
|
|
return p;
|
|
}
|
|
WaitForGet();
|
|
return NULL;
|
|
}
|
|
|
|
void cRingBufferLinear::Del(int Count)
|
|
{
|
|
if (Count > gotten) {
|
|
esyslog("ERROR: invalid Count in cRingBufferLinear::Del: %d (limited to %d)", Count, gotten);
|
|
Count = gotten;
|
|
}
|
|
if (Count > 0) {
|
|
int Tail = tail;
|
|
Tail += Count;
|
|
gotten -= Count;
|
|
if (Tail >= Size())
|
|
Tail = margin;
|
|
tail = Tail;
|
|
EnablePut();
|
|
}
|
|
#ifdef DEBUGRINGBUFFERS
|
|
lastTail = tail;
|
|
lastGet = Count;
|
|
#endif
|
|
}
|
|
|
|
// --- cFrame ----------------------------------------------------------------
|
|
|
|
cFrame::cFrame(const uchar *Data, int Count, eFrameType Type, int Index, uint32_t Pts)
|
|
{
|
|
count = abs(Count);
|
|
type = Type;
|
|
index = Index;
|
|
pts = Pts;
|
|
if (Count < 0)
|
|
data = (uchar *)Data;
|
|
else {
|
|
data = MALLOC(uchar, count);
|
|
if (data)
|
|
memcpy(data, Data, count);
|
|
else
|
|
esyslog("ERROR: can't allocate frame buffer (count=%d)", count);
|
|
}
|
|
next = NULL;
|
|
}
|
|
|
|
cFrame::~cFrame()
|
|
{
|
|
free(data);
|
|
}
|
|
|
|
// --- cRingBufferFrame ------------------------------------------------------
|
|
|
|
cRingBufferFrame::cRingBufferFrame(int Size, bool Statistics)
|
|
:cRingBuffer(Size, Statistics)
|
|
{
|
|
head = NULL;
|
|
currentFill = 0;
|
|
}
|
|
|
|
cRingBufferFrame::~cRingBufferFrame()
|
|
{
|
|
Clear();
|
|
}
|
|
|
|
void cRingBufferFrame::Clear(void)
|
|
{
|
|
Lock();
|
|
cFrame *p;
|
|
while ((p = Get()) != NULL)
|
|
Drop(p);
|
|
Unlock();
|
|
EnablePut();
|
|
EnableGet();
|
|
}
|
|
|
|
bool cRingBufferFrame::Put(cFrame *Frame)
|
|
{
|
|
if (Frame->Count() <= Free()) {
|
|
Lock();
|
|
if (head) {
|
|
Frame->next = head->next;
|
|
head->next = Frame;
|
|
head = Frame;
|
|
}
|
|
else {
|
|
head = Frame->next = Frame;
|
|
}
|
|
currentFill += Frame->Count();
|
|
Unlock();
|
|
EnableGet();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
cFrame *cRingBufferFrame::Get(void)
|
|
{
|
|
Lock();
|
|
cFrame *p = head ? head->next : NULL;
|
|
Unlock();
|
|
return p;
|
|
}
|
|
|
|
void cRingBufferFrame::Delete(cFrame *Frame)
|
|
{
|
|
currentFill -= Frame->Count();
|
|
delete Frame;
|
|
}
|
|
|
|
void cRingBufferFrame::Drop(cFrame *Frame)
|
|
{
|
|
Lock();
|
|
if (head) {
|
|
if (Frame == head->next) {
|
|
if (head->next != head) {
|
|
head->next = Frame->next;
|
|
Delete(Frame);
|
|
}
|
|
else {
|
|
Delete(head);
|
|
head = NULL;
|
|
}
|
|
}
|
|
else
|
|
esyslog("ERROR: attempt to drop wrong frame from ring buffer!");
|
|
}
|
|
Unlock();
|
|
EnablePut();
|
|
}
|
|
|
|
int cRingBufferFrame::Available(void)
|
|
{
|
|
Lock();
|
|
int av = currentFill;
|
|
Unlock();
|
|
return av;
|
|
}
|