mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
Original announce message: VDR developer version 1.7.22 is now available at ftp://ftp.tvdr.de/vdr/Developer/vdr-1.7.22.tar.bz2 A 'diff' against the previous version is available at ftp://ftp.tvdr.de/vdr/Developer/vdr-1.7.21-1.7.22.diff MD5 checksums: b9c0fe1aac8e653c0d0234bc72c2bb2c vdr-1.7.22.tar.bz2 868bb332342c9a78beda17cc85e0bb93 vdr-1.7.21-1.7.22.diff WARNING: ======== This is a developer version. Even though I use it in my productive environment. I strongly recommend that you only use it under controlled conditions and for testing and debugging. From the HISTORY file: - Fixed scaling subtitles in case the primary device's GetVideoSize() function doesn't return actual values (thanks to Luca Olivetti). - The DiSEqC codes are now copied in the call to cDiseqc::Execute(). - VDR now supports "Satellite Channel Routing" (SCR) according to EN50494 (based on the "unicable" patch from Lars Hanisch). Since "Unicable" is a registered trademark and stands for only one of many implementations of SCR, the following changes have been made compared to the patch, which need to be taken into account by people who have set up their system using the patch: - The 'U' parameter in the diseqc.conf file has been changed to 'S' ("Scr"). - The configuration file name has been changed from "unicable.conf" to "scr.conf". - Updated sources.conf (thanks to Arthur Konovalov). - The SVDRP command LSTC now also accepts channel IDs (thanks to Dominic Evans). - Fixed handling DVB subtitles and implemented decoding textual DVB subtitles (thanks to Rolf Ahrenberg). - Added cap_net_raw to the capabilities that are not dropped (thanks to Dominic Evans). - Fixed setting the start time of an edited recording (thanks to Christoph Haubrich). - Temporarily switching free devices to transponders in order to have their running status updated is now done by marking the devices as "occupied" for a certain amount of time. - The new setup options "LNB/Device n connected to sat cable" can be used to define which DVB-S devices are connected to the same sat cable and are therefore "bonded". This obsoletes the LNBSHARE patch. Users of the LNBSHARE patch will need to newly set up their sat devices with the above options. - Fixed a crash when deleting a recording while cutting it (thanks to Ville Skyttä). - Fixed several spelling errors (thanks to Ville Skyttä). - The new SVDRP command UPDR can be used to trigger an update of the list of recordings (thanks to Lars Hanisch). - Added generating a pkg-config file to the Makefile (thanks to Ville Skyttä). - Removed the '.pl' suffix from all scripts (thanks to Ville Skyttä). - Changed the default location for the LIRC socket to /var/run/lirc/lircd (thanks to Ville Skyttä). - Added file name and line number to LOG_ERROR_STR() (thanks to Rolf Ahrenberg). - Replaced all calls to sleep() with cCondWait::SleepMs() (thanks to Rolf Ahrenberg). - Fixed a crash with malformed SI data (patch from vdr-portal).
152 lines
5.1 KiB
C++
152 lines
5.1 KiB
C++
/*
|
|
* ringbuffer.h: A ring buffer
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: ringbuffer.h 2.3 2011/12/04 13:38:17 kls Exp $
|
|
*/
|
|
|
|
#ifndef __RINGBUFFER_H
|
|
#define __RINGBUFFER_H
|
|
|
|
#include "thread.h"
|
|
#include "tools.h"
|
|
|
|
class cRingBuffer {
|
|
private:
|
|
cCondWait readyForPut, readyForGet;
|
|
int putTimeout;
|
|
int getTimeout;
|
|
int size;
|
|
time_t lastOverflowReport;
|
|
int overflowCount;
|
|
int overflowBytes;
|
|
protected:
|
|
tThreadId getThreadTid;
|
|
int maxFill;//XXX
|
|
int lastPercent;
|
|
bool statistics;//XXX
|
|
void UpdatePercentage(int Fill);
|
|
void WaitForPut(void);
|
|
void WaitForGet(void);
|
|
void EnablePut(void);
|
|
void EnableGet(void);
|
|
virtual void Clear(void) = 0;
|
|
virtual int Available(void) = 0;
|
|
virtual int Free(void) { return Size() - Available() - 1; }
|
|
int Size(void) { return size; }
|
|
public:
|
|
cRingBuffer(int Size, bool Statistics = false);
|
|
virtual ~cRingBuffer();
|
|
void SetTimeouts(int PutTimeout, int GetTimeout);
|
|
void ReportOverflow(int Bytes);
|
|
};
|
|
|
|
class cRingBufferLinear : public cRingBuffer {
|
|
//#define DEBUGRINGBUFFERS
|
|
#ifdef DEBUGRINGBUFFERS
|
|
private:
|
|
int lastHead, lastTail;
|
|
int lastPut, lastGet;
|
|
static cRingBufferLinear *RBLS[];
|
|
static void AddDebugRBL(cRingBufferLinear *RBL);
|
|
static void DelDebugRBL(cRingBufferLinear *RBL);
|
|
public:
|
|
static void PrintDebugRBL(void);
|
|
#endif
|
|
private:
|
|
int margin, head, tail;
|
|
int gotten;
|
|
uchar *buffer;
|
|
char *description;
|
|
protected:
|
|
virtual int DataReady(const uchar *Data, int Count);
|
|
///< By default a ring buffer has data ready as soon as there are at least
|
|
///< 'margin' bytes available. A derived class can reimplement this function
|
|
///< if it has other conditions that define when data is ready.
|
|
///< The return value is either 0 if there is not yet enough data available,
|
|
///< or the number of bytes from the beginning of Data that are "ready".
|
|
public:
|
|
cRingBufferLinear(int Size, int Margin = 0, bool Statistics = false, const char *Description = NULL);
|
|
///< Creates a linear ring buffer.
|
|
///< The buffer will be able to hold at most Size-Margin-1 bytes of data, and will
|
|
///< be guaranteed to return at least Margin bytes in one consecutive block.
|
|
///< The optional Description is used for debugging only.
|
|
virtual ~cRingBufferLinear();
|
|
virtual int Available(void);
|
|
virtual int Free(void) { return Size() - Available() - 1 - margin; }
|
|
virtual void Clear(void);
|
|
///< Immediately clears the ring buffer.
|
|
int Read(int FileHandle, int Max = 0);
|
|
///< Reads at most Max bytes from FileHandle and stores them in the
|
|
///< ring buffer. If Max is 0, reads as many bytes as possible.
|
|
///< Only one actual read() call is done.
|
|
///< \return Returns the number of bytes actually read and stored, or
|
|
///< an error value from the actual read() call.
|
|
int Read(cUnbufferedFile *File, int Max = 0);
|
|
///< Like Read(int FileHandle, int Max), but reads from a cUnbufferedFile).
|
|
int Put(const uchar *Data, int Count);
|
|
///< Puts at most Count bytes of Data into the ring buffer.
|
|
///< \return Returns the number of bytes actually stored.
|
|
uchar *Get(int &Count);
|
|
///< Gets data from the ring buffer.
|
|
///< The data will remain in the buffer until a call to Del() deletes it.
|
|
///< \return Returns a pointer to the data, and stores the number of bytes
|
|
///< actually available in Count. If the returned pointer is NULL, Count has no meaning.
|
|
void Del(int Count);
|
|
///< Deletes at most Count bytes from the ring buffer.
|
|
///< Count must be less or equal to the number that was returned by a previous
|
|
///< call to Get().
|
|
};
|
|
|
|
enum eFrameType { ftUnknown, ftVideo, ftAudio, ftDolby };
|
|
|
|
class cFrame {
|
|
friend class cRingBufferFrame;
|
|
private:
|
|
cFrame *next;
|
|
uchar *data;
|
|
int count;
|
|
eFrameType type;
|
|
int index;
|
|
uint32_t pts;
|
|
public:
|
|
cFrame(const uchar *Data, int Count, eFrameType = ftUnknown, int Index = -1, uint32_t Pts = 0);
|
|
///< Creates a new cFrame object.
|
|
///< If Count is negative, the cFrame object will take ownership of the given
|
|
///< Data. Otherwise it will allocate Count bytes of memory and copy Data.
|
|
~cFrame();
|
|
uchar *Data(void) const { return data; }
|
|
int Count(void) const { return count; }
|
|
eFrameType Type(void) const { return type; }
|
|
int Index(void) const { return index; }
|
|
uint32_t Pts(void) const { return pts; }
|
|
};
|
|
|
|
class cRingBufferFrame : public cRingBuffer {
|
|
private:
|
|
cMutex mutex;
|
|
cFrame *head;
|
|
int currentFill;
|
|
void Delete(cFrame *Frame);
|
|
void Lock(void) { mutex.Lock(); }
|
|
void Unlock(void) { mutex.Unlock(); }
|
|
public:
|
|
cRingBufferFrame(int Size, bool Statistics = false);
|
|
virtual ~cRingBufferFrame();
|
|
virtual int Available(void);
|
|
virtual void Clear(void);
|
|
// Immediately clears the ring buffer.
|
|
bool Put(cFrame *Frame);
|
|
// Puts the Frame into the ring buffer.
|
|
// Returns true if this was possible.
|
|
cFrame *Get(void);
|
|
// Gets the next frame from the ring buffer.
|
|
// The actual data still remains in the buffer until Drop() is called.
|
|
void Drop(cFrame *Frame);
|
|
// Drops the Frame that has just been fetched with Get().
|
|
};
|
|
|
|
#endif // __RINGBUFFER_H
|